From a9cfc0805c54e3e23031cde5d06bd8fdb2ed2474 Mon Sep 17 00:00:00 2001 From: dennisppaul Date: Mon, 15 Apr 2024 21:07:35 +0200 Subject: [PATCH] updated libraries --- .../cores/sdl/SDL2.macos.arm/include/SDL.h | 233 + .../sdl/SDL2.macos.arm/include/SDL_assert.h | 322 + .../sdl/SDL2.macos.arm/include/SDL_atomic.h | 414 + .../sdl/SDL2.macos.arm/include/SDL_audio.h | 1500 ++ .../sdl/SDL2.macos.arm/include/SDL_bits.h | 126 + .../SDL2.macos.arm/include/SDL_blendmode.h | 198 + .../SDL2.macos.arm/include/SDL_clipboard.h | 141 + .../sdl/SDL2.macos.arm/include/SDL_config.h | 498 + .../sdl/SDL2.macos.arm/include/SDL_cpuinfo.h | 594 + .../sdl/SDL2.macos.arm/include/SDL_egl.h | 2352 +++ .../sdl/SDL2.macos.arm/include/SDL_endian.h | 348 + .../sdl/SDL2.macos.arm/include/SDL_error.h | 163 + .../sdl/SDL2.macos.arm/include/SDL_events.h | 1166 ++ .../SDL2.macos.arm/include/SDL_filesystem.h | 149 + .../include/SDL_gamecontroller.h | 1074 ++ .../sdl/SDL2.macos.arm/include/SDL_gesture.h | 117 + .../sdl/SDL2.macos.arm/include/SDL_guid.h | 100 + .../sdl/SDL2.macos.arm/include/SDL_haptic.h | 1341 ++ .../sdl/SDL2.macos.arm/include/SDL_hidapi.h | 451 + .../sdl/SDL2.macos.arm/include/SDL_hints.h | 2624 +++ .../sdl/SDL2.macos.arm/include/SDL_joystick.h | 1069 ++ .../sdl/SDL2.macos.arm/include/SDL_keyboard.h | 353 + .../sdl/SDL2.macos.arm/include/SDL_keycode.h | 358 + .../sdl/SDL2.macos.arm/include/SDL_loadso.h | 115 + .../sdl/SDL2.macos.arm/include/SDL_locale.h | 103 + .../sdl/SDL2.macos.arm/include/SDL_log.h | 404 + .../sdl/SDL2.macos.arm/include/SDL_main.h | 282 + .../SDL2.macos.arm/include/SDL_messagebox.h | 193 + .../sdl/SDL2.macos.arm/include/SDL_metal.h | 113 + .../sdl/SDL2.macos.arm/include/SDL_misc.h | 79 + .../sdl/SDL2.macos.arm/include/SDL_mouse.h | 464 + .../sdl/SDL2.macos.arm/include/SDL_mutex.h | 545 + .../sdl/SDL2.macos.arm/include/SDL_name.h | 33 + .../sdl/SDL2.macos.arm/include/SDL_opengl.h | 2132 +++ .../SDL2.macos.arm/include/SDL_opengl_glext.h | 13213 +++++++++++++ .../sdl/SDL2.macos.arm/include/SDL_opengles.h | 39 + .../SDL2.macos.arm/include/SDL_opengles2.h | 52 + .../include/SDL_opengles2_gl2.h | 656 + .../include/SDL_opengles2_gl2ext.h | 4033 ++++ .../include/SDL_opengles2_gl2platform.h | 27 + .../include/SDL_opengles2_khrplatform.h | 311 + .../sdl/SDL2.macos.arm/include/SDL_pixels.h | 644 + .../sdl/SDL2.macos.arm/include/SDL_platform.h | 261 + .../sdl/SDL2.macos.arm/include/SDL_power.h | 87 + .../sdl/SDL2.macos.arm/include/SDL_quit.h | 58 + .../sdl/SDL2.macos.arm/include/SDL_rect.h | 376 + .../sdl/SDL2.macos.arm/include/SDL_render.h | 1924 ++ .../sdl/SDL2.macos.arm/include/SDL_revision.h | 7 + .../sdl/SDL2.macos.arm/include/SDL_rwops.h | 841 + .../sdl/SDL2.macos.arm/include/SDL_scancode.h | 438 + .../sdl/SDL2.macos.arm/include/SDL_sensor.h | 322 + .../sdl/SDL2.macos.arm/include/SDL_shape.h | 155 + .../sdl/SDL2.macos.arm/include/SDL_stdinc.h | 838 + .../sdl/SDL2.macos.arm/include/SDL_surface.h | 997 + .../sdl/SDL2.macos.arm/include/SDL_system.h | 623 + .../sdl/SDL2.macos.arm/include/SDL_syswm.h | 386 + .../sdl/SDL2.macos.arm/include/SDL_test.h | 69 + .../SDL2.macos.arm/include/SDL_test_assert.h | 105 + .../SDL2.macos.arm/include/SDL_test_common.h | 236 + .../SDL2.macos.arm/include/SDL_test_compare.h | 69 + .../SDL2.macos.arm/include/SDL_test_crc32.h | 124 + .../SDL2.macos.arm/include/SDL_test_font.h | 168 + .../SDL2.macos.arm/include/SDL_test_fuzzer.h | 386 + .../SDL2.macos.arm/include/SDL_test_harness.h | 134 + .../SDL2.macos.arm/include/SDL_test_images.h | 78 + .../sdl/SDL2.macos.arm/include/SDL_test_log.h | 67 + .../sdl/SDL2.macos.arm/include/SDL_test_md5.h | 129 + .../SDL2.macos.arm/include/SDL_test_memory.h | 63 + .../SDL2.macos.arm/include/SDL_test_random.h | 115 + .../sdl/SDL2.macos.arm/include/SDL_thread.h | 464 + .../sdl/SDL2.macos.arm/include/SDL_timer.h | 222 + .../sdl/SDL2.macos.arm/include/SDL_touch.h | 150 + .../sdl/SDL2.macos.arm/include/SDL_types.h | 29 + .../sdl/SDL2.macos.arm/include/SDL_version.h | 204 + .../sdl/SDL2.macos.arm/include/SDL_video.h | 2178 +++ .../sdl/SDL2.macos.arm/include/SDL_vulkan.h | 215 + .../sdl/SDL2.macos.arm/include/begin_code.h | 187 + .../sdl/SDL2.macos.arm/include/close_code.h | 40 + .../SDL2.macos.arm/lib/libSDL2-2.0.0.dylib | Bin 0 -> 1540848 bytes .../cores/sdl/SDL2.macos.arm/lib/libSDL2.a | Bin 0 -> 2460728 bytes .../sdl/SDL2.macos.arm/lib/libSDL2.dylib | Bin 0 -> 1540848 bytes .../sdl/SDL2.macos.arm/lib/libSDL2_test.a | Bin 0 -> 259552 bytes .../sdl/SDL2.macos.arm/lib/libSDL2main.a | Bin 0 -> 736 bytes .../include/SDL.h | 0 .../include/SDL_assert.h | 0 .../include/SDL_atomic.h | 0 .../include/SDL_audio.h | 0 .../include/SDL_bits.h | 0 .../include/SDL_blendmode.h | 0 .../include/SDL_clipboard.h | 0 .../include/SDL_config.h | 0 .../include/SDL_cpuinfo.h | 0 .../include/SDL_egl.h | 0 .../include/SDL_endian.h | 0 .../include/SDL_error.h | 0 .../include/SDL_events.h | 0 .../include/SDL_filesystem.h | 0 .../include/SDL_gamecontroller.h | 0 .../include/SDL_gesture.h | 0 .../include/SDL_haptic.h | 0 .../include/SDL_hints.h | 0 .../include/SDL_joystick.h | 0 .../include/SDL_keyboard.h | 0 .../include/SDL_keycode.h | 0 .../include/SDL_loadso.h | 0 .../include/SDL_log.h | 0 .../include/SDL_main.h | 0 .../include/SDL_messagebox.h | 0 .../include/SDL_metal.h | 0 .../include/SDL_mouse.h | 0 .../include/SDL_mutex.h | 0 .../include/SDL_name.h | 0 .../include/SDL_opengl.h | 0 .../include/SDL_opengl_glext.h | 0 .../include/SDL_opengles.h | 0 .../include/SDL_opengles2.h | 0 .../include/SDL_opengles2_gl2.h | 0 .../include/SDL_opengles2_gl2ext.h | 0 .../include/SDL_opengles2_gl2platform.h | 0 .../include/SDL_opengles2_khrplatform.h | 0 .../include/SDL_pixels.h | 0 .../include/SDL_platform.h | 0 .../include/SDL_power.h | 0 .../include/SDL_quit.h | 0 .../include/SDL_rect.h | 0 .../include/SDL_render.h | 0 .../include/SDL_revision.h | 0 .../include/SDL_rwops.h | 0 .../include/SDL_scancode.h | 0 .../include/SDL_sensor.h | 0 .../include/SDL_shape.h | 0 .../include/SDL_stdinc.h | 0 .../include/SDL_surface.h | 0 .../include/SDL_system.h | 0 .../include/SDL_syswm.h | 0 .../include/SDL_test.h | 0 .../include/SDL_test_assert.h | 0 .../include/SDL_test_common.h | 0 .../include/SDL_test_compare.h | 0 .../include/SDL_test_crc32.h | 0 .../include/SDL_test_font.h | 0 .../include/SDL_test_fuzzer.h | 0 .../include/SDL_test_harness.h | 0 .../include/SDL_test_images.h | 0 .../include/SDL_test_log.h | 0 .../include/SDL_test_md5.h | 0 .../include/SDL_test_memory.h | 0 .../include/SDL_test_random.h | 0 .../include/SDL_thread.h | 0 .../include/SDL_timer.h | 0 .../include/SDL_touch.h | 0 .../include/SDL_types.h | 0 .../include/SDL_version.h | 0 .../include/SDL_video.h | 0 .../include/SDL_vulkan.h | 0 .../include/begin_code.h | 0 .../include/close_code.h | 0 .../lib/libSDL2-2.0.0.dylib | Bin .../lib/libSDL2.a | Bin .../lib/libSDL2.dylib | Bin .../lib/libSDL2_test.a | Bin .../lib/libSDL2main.a | Bin .../examples/01.Wavetable/01.Wavetable.ino | 7 +- .../KlangWellen/examples/02.ADSR/02.ADSR.ino | 5 + .../03.LowPassFilter/03.LowPassFilter.ino | 5 + .../examples/04.Reverb/04.Reverb.ino | 5 + .../05.FormantFilter/05.FormantFilter.ino | 5 + .../KlangWellen/examples/06.SAM/06.SAM.ino | 5 + .../examples/07.Sampler/07.Sampler.ino | 9 +- .../examples/08.Delay/08.Delay.ino | 9 +- .../KlangWellen/examples/09.LFO/09.LFO.ino | 5 + .../examples/10.BeatDSP/10.BeatDSP.ino | 5 + .../examples/11.Trigger/11.Trigger.ino | 5 + .../examples/12.Clamp/12.Clamp.ino | 5 + .../examples/13.Vocoder/13.Vocoder.ino | 7 +- .../examples/14.Filters/14.Filters.ino | 5 + .../KlangWellen/examples/15.Ramp/15.Ramp.ino | 5 + .../examples/16.Envelope/16.Envelope.ino | 5 + .../examples/17.Waveshaper/17.Waveshaper.ino | 5 + desktop/libraries/KlangWellen/src/ADSR.h | 6 +- desktop/libraries/KlangWellen/src/BeatDSP.h | 6 +- desktop/libraries/KlangWellen/src/Clamp.h | 6 +- desktop/libraries/KlangWellen/src/Delay.h | 25 +- desktop/libraries/KlangWellen/src/Envelope.h | 11 +- desktop/libraries/KlangWellen/src/Filter.h | 6 +- .../KlangWellen/src/FilterLowPassMoogLadder.h | 6 +- .../KlangWellen/src/FilterVowelFormant.h | 6 +- desktop/libraries/KlangWellen/src/Gain.h | 6 +- .../libraries/KlangWellen/src/KlangWellen.cpp | 31 - .../libraries/KlangWellen/src/KlangWellen.h | 565 +- desktop/libraries/KlangWellen/src/Note.h | 7 +- desktop/libraries/KlangWellen/src/Ramp.h | 6 +- desktop/libraries/KlangWellen/src/Resonator.h | 111 + desktop/libraries/KlangWellen/src/Reverb.h | 6 +- desktop/libraries/KlangWellen/src/SAM.h | 6 +- desktop/libraries/KlangWellen/src/Sampler.h | 246 +- desktop/libraries/KlangWellen/src/Scale.cpp | 14 - desktop/libraries/KlangWellen/src/Scale.h | 19 +- .../KlangWellen/src/ScaleCollection.h | 40 + desktop/libraries/KlangWellen/src/Signal.h | 6 +- desktop/libraries/KlangWellen/src/Stream.h | 201 + desktop/libraries/KlangWellen/src/Trigger.h | 6 +- desktop/libraries/KlangWellen/src/Vocoder.h | 6 +- .../libraries/KlangWellen/src/Waveshaper.h | 6 +- desktop/libraries/KlangWellen/src/Wavetable.h | 26 +- .../examples/01.Basics/Beat/Beat.ino | 5 + .../examples/01.Basics/Blink/Blink.ino | 5 + .../01.Basics/PassThrough/PassThrough.ino | 5 + .../02.Peripherals/Encoders/Encoders.ino | 1 + .../ExternalSDCard/ExternalSDCard.ino | 37 +- .../02.Peripherals/Presets/Presets.ino | 56 + .../examples/02.Peripherals/Serial/Serial.ino | 31 + .../ApplicationTemplate.ino | 11 +- .../QueryButtonStates/QueryButtonStates.ino | 1 + .../SchedulingTasks/SchedulingTasks.ino | 3 +- .../TalkingTerminal/TalkingTerminal.ino | 8 +- .../MaximumMemoryUsage/MaximumMemoryUsage.ino | 8 +- .../ExampleSavingPresets.ino | 87 - .../ExampleListFiles/ExampleListFiles.ino | 4 + .../ExamplePlayWAVE.ino} | 9 +- .../ExampleStreamWAVE/ExampleStreamWAVE.ino | 71 + .../ExampleWAVFileInfo/ExampleWAVFileInfo.ino | 5 +- .../ExampleWriteRawData.ino | 10 +- .../KlangstromCard/src/KlangstromCard.cpp | 14 +- .../src/KlangstromCardBSP_STM32.h | 11 +- .../CommandLineInterface.ino | 5 + .../examples/DrawBuffer/DrawBuffer.ino | 10 +- .../DrawPrimitives/DrawPrimitives.ino | 7 +- .../examples/Fonts/Fonts.ino | 5 + .../FullscreenAnimation.ino | 5 + .../examples/Terminal/Terminal.ino | 4 +- .../USBDeviceKeyboard/USBDeviceKeyboard.ino | 9 +- .../examples/USBDeviceMIDI/USBDeviceMIDI.ino | 9 +- .../USBDeviceMouse/USBDeviceMouse.ino | 9 +- .../examples/USBHostMIDI/USBHostMIDI.ino | 14 +- .../USBHostMouseKeyboard.ino | 12 +- desktop/platform.txt | 14 +- docs/_includes/code/ExampleListFiles.ino | 4 + .../_includes/code/ExamplePlayWAVE.ino | 9 +- docs/_includes/code/ExampleStreamWAVE.ino | 71 + docs/_includes/code/ExampleWAVFileInfo.ino | 5 +- docs/_includes/code/ExampleWriteRawData.ino | 10 +- stm32/boards.txt | 11 +- stm32/cores/arduino/HardwareSerial.cpp | 25 +- stm32/cores/arduino/HardwareSerial.h | 2 +- stm32/cores/arduino/HardwareTimer.h | 23 +- stm32/cores/arduino/Print.cpp | 19 +- stm32/cores/arduino/Print.h | 5 +- stm32/cores/arduino/stm32/LL/stm32yyxx_ll.h | 1 + .../cores/arduino/stm32/LL/stm32yyxx_ll_adc.h | 2 + .../cores/arduino/stm32/LL/stm32yyxx_ll_bus.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_comp.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_cordic.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_cortex.h | 2 + .../cores/arduino/stm32/LL/stm32yyxx_ll_crc.h | 2 + .../cores/arduino/stm32/LL/stm32yyxx_ll_crs.h | 2 + .../cores/arduino/stm32/LL/stm32yyxx_ll_dac.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_dcache.h | 4 +- .../arduino/stm32/LL/stm32yyxx_ll_dlyb.h | 4 +- .../cores/arduino/stm32/LL/stm32yyxx_ll_dma.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_exti.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_fmac.h | 2 + .../cores/arduino/stm32/LL/stm32yyxx_ll_fmc.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_gpio.h | 2 + .../cores/arduino/stm32/LL/stm32yyxx_ll_i2c.h | 2 + .../cores/arduino/stm32/LL/stm32yyxx_ll_i3c.h | 15 + .../arduino/stm32/LL/stm32yyxx_ll_icache.h | 4 +- .../arduino/stm32/LL/stm32yyxx_ll_iwdg.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_lptim.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_lpuart.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_opamp.h | 2 + .../cores/arduino/stm32/LL/stm32yyxx_ll_pka.h | 4 +- .../cores/arduino/stm32/LL/stm32yyxx_ll_pwr.h | 2 + .../cores/arduino/stm32/LL/stm32yyxx_ll_rcc.h | 2 + .../cores/arduino/stm32/LL/stm32yyxx_ll_rng.h | 2 + .../cores/arduino/stm32/LL/stm32yyxx_ll_rtc.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_sdmmc.h | 2 + .../cores/arduino/stm32/LL/stm32yyxx_ll_spi.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_system.h | 2 + .../cores/arduino/stm32/LL/stm32yyxx_ll_tim.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_ucpd.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_usart.h | 2 + .../cores/arduino/stm32/LL/stm32yyxx_ll_usb.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_utils.h | 2 + .../arduino/stm32/LL/stm32yyxx_ll_wwdg.h | 2 + stm32/cores/arduino/stm32/OpenAMP/openamp.c | 2 +- stm32/cores/arduino/stm32/PeripheralPins.h | 20 +- stm32/cores/arduino/stm32/backup.h | 18 +- stm32/cores/arduino/stm32/stm32_def.h | 14 +- stm32/cores/arduino/stm32/stm32_def_build.h | 16 + stm32/cores/arduino/stm32/timer.h | 14 +- stm32/cores/arduino/stm32/uart.h | 2 + stm32/cores/arduino/stm32/usb/cdc/cdc_queue.h | 12 +- stm32/cores/arduino/stm32/usb/cdc/usbd_cdc.c | 300 +- stm32/cores/arduino/stm32/usb/cdc/usbd_cdc.h | 12 +- .../cores/arduino/stm32/usb/cdc/usbd_cdc_if.c | 20 + .../stm32/usb/hid/usbd_hid_composite.c | 178 +- .../stm32/usb/hid/usbd_hid_composite.h | 27 + .../stm32/usb/hid/usbd_hid_composite_if.c | 27 +- stm32/cores/arduino/stm32/usb/usbd_conf.c | 30 +- stm32/cores/arduino/stm32/usb/usbd_conf.h | 11 +- stm32/cores/arduino/stm32/usb/usbd_desc.c | 48 +- stm32/cores/arduino/stm32/usb/usbd_desc.h | 4 +- stm32/cores/arduino/wiring_time.h | 22 + stm32/libraries/CMSIS_DSP/CMakeLists.txt | 15 + .../BasicMathFunctionsF16.c | 1 + .../src/BayesFunctions/BayesFunctionsF16.c | 1 + .../src/CommonTables/CommonTablesF16.c | 1 + .../ComplexMathFunctionsF16.c | 1 + .../DistanceFunctions/DistanceFunctionsF16.c | 1 + .../FastMathFunctions/FastMathFunctionsF16.c | 1 + .../FilteringFunctionsF16.c | 1 + .../InterpolationFunctions.c | 1 + .../InterpolationFunctionsF16.c | 1 + .../src/MatrixFunctions/MatrixFunctionsF16.c | 1 + .../QuaternionMathFunctions.c | 1 + .../src/SVMFunctions/SVMFunctionsF16.c | 1 + .../StatisticsFunctionsF16.c | 1 + .../SupportFunctions/SupportFunctionsF16.c | 1 + .../TransformFunctionsF16.c | 1 + .../EEPROM/src/utility/stm32_eeprom.c | 67 +- .../EEPROM/src/utility/stm32_eeprom.h | 2 +- .../examples/01.Wavetable/01.Wavetable.ino | 7 +- .../KlangWellen/examples/02.ADSR/02.ADSR.ino | 5 + .../03.LowPassFilter/03.LowPassFilter.ino | 5 + .../examples/04.Reverb/04.Reverb.ino | 5 + .../05.FormantFilter/05.FormantFilter.ino | 5 + .../KlangWellen/examples/06.SAM/06.SAM.ino | 5 + .../examples/07.Sampler/07.Sampler.ino | 9 +- .../examples/08.Delay/08.Delay.ino | 9 +- .../KlangWellen/examples/09.LFO/09.LFO.ino | 5 + .../examples/10.BeatDSP/10.BeatDSP.ino | 5 + .../examples/11.Trigger/11.Trigger.ino | 5 + .../examples/12.Clamp/12.Clamp.ino | 5 + .../examples/13.Vocoder/13.Vocoder.ino | 7 +- .../examples/14.Filters/14.Filters.ino | 5 + .../KlangWellen/examples/15.Ramp/15.Ramp.ino | 5 + .../examples/16.Envelope/16.Envelope.ino | 5 + .../examples/17.Waveshaper/17.Waveshaper.ino | 5 + stm32/libraries/KlangWellen/src/ADSR.h | 6 +- stm32/libraries/KlangWellen/src/BeatDSP.h | 6 +- stm32/libraries/KlangWellen/src/Clamp.h | 6 +- stm32/libraries/KlangWellen/src/Delay.h | 25 +- stm32/libraries/KlangWellen/src/Envelope.h | 11 +- stm32/libraries/KlangWellen/src/Filter.h | 6 +- .../KlangWellen/src/FilterLowPassMoogLadder.h | 6 +- .../KlangWellen/src/FilterVowelFormant.h | 6 +- stm32/libraries/KlangWellen/src/Gain.h | 6 +- .../libraries/KlangWellen/src/KlangWellen.cpp | 31 - stm32/libraries/KlangWellen/src/KlangWellen.h | 565 +- stm32/libraries/KlangWellen/src/Note.h | 7 +- stm32/libraries/KlangWellen/src/Ramp.h | 6 +- stm32/libraries/KlangWellen/src/Resonator.h | 111 + stm32/libraries/KlangWellen/src/Reverb.h | 6 +- stm32/libraries/KlangWellen/src/SAM.h | 6 +- stm32/libraries/KlangWellen/src/Sampler.h | 246 +- stm32/libraries/KlangWellen/src/Scale.cpp | 14 - stm32/libraries/KlangWellen/src/Scale.h | 19 +- .../KlangWellen/src/ScaleCollection.h | 40 + stm32/libraries/KlangWellen/src/Signal.h | 6 +- stm32/libraries/KlangWellen/src/Stream.h | 201 + stm32/libraries/KlangWellen/src/Trigger.h | 6 +- stm32/libraries/KlangWellen/src/Vocoder.h | 6 +- stm32/libraries/KlangWellen/src/Waveshaper.h | 6 +- stm32/libraries/KlangWellen/src/Wavetable.h | 26 +- .../examples/01.Basics/Beat/Beat.ino | 5 + .../examples/01.Basics/Blink/Blink.ino | 5 + .../01.Basics/PassThrough/PassThrough.ino | 5 + .../02.Peripherals/Encoders/Encoders.ino | 1 + .../ExternalSDCard/ExternalSDCard.ino | 37 +- .../02.Peripherals/Presets/Presets.ino | 56 + .../examples/02.Peripherals/Serial/Serial.ino | 31 + .../ApplicationTemplate.ino | 11 +- .../QueryButtonStates/QueryButtonStates.ino | 1 + .../SchedulingTasks/SchedulingTasks.ino | 3 +- .../TalkingTerminal/TalkingTerminal.ino | 8 +- .../MaximumMemoryUsage/MaximumMemoryUsage.ino | 8 +- .../ExampleSavingPresets.ino | 87 - .../ExampleListFiles/ExampleListFiles.ino | 4 + .../ExamplePlayWAVE/ExamplePlayWAVE.ino | 81 + .../ExampleStreamWAVE/ExampleStreamWAVE.ino | 71 + .../ExampleWAVFileInfo/ExampleWAVFileInfo.ino | 5 +- .../ExampleWriteRawData.ino | 10 +- .../KlangstromCard/src/KlangstromCard.cpp | 14 +- .../src/KlangstromCardBSP_STM32.h | 11 +- .../CommandLineInterface.ino | 5 + .../examples/DrawBuffer/DrawBuffer.ino | 10 +- .../DrawPrimitives/DrawPrimitives.ino | 7 +- .../examples/Fonts/Fonts.ino | 5 + .../FullscreenAnimation.ino | 5 + .../examples/Terminal/Terminal.ino | 4 +- .../KlangstromTest/library.properties | 11 + .../KlangstromTest/src/KlangstromClass.cpp | 3 + .../KlangstromTest/src/KlangstromClass.h | 6 + .../KlangstromTest/src/KlangstromTest.cpp | 3 + .../KlangstromTest/src/KlangstromTest.h | 6 + stm32/libraries/SPI/README.md | 109 +- stm32/libraries/SPI/library.properties | 4 +- stm32/libraries/SPI/src/SPI.cpp | 393 +- stm32/libraries/SPI/src/SPI.h | 245 +- stm32/libraries/SPI/src/utility/spi_com.c | 146 +- stm32/libraries/SPI/src/utility/spi_com.h | 70 +- stm32/libraries/SrcWrapper/CMakeLists.txt | 3 + .../SrcWrapper/src/HAL/stm32yyxx_hal.c | 4 + .../SrcWrapper/src/HAL/stm32yyxx_hal_adc.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_adc_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_cec.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_comp.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_cordic.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_cortex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_crc.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_crc_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_cryp.c | 2 + .../src/HAL/stm32yyxx_hal_cryp_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_dac.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_dac_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_dcache.c | 4 +- .../SrcWrapper/src/HAL/stm32yyxx_hal_dcmi.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_dma.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_dma_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_dts.c | 4 +- .../SrcWrapper/src/HAL/stm32yyxx_hal_eth.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_eth_ex.c | 4 +- .../SrcWrapper/src/HAL/stm32yyxx_hal_exti.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_fdcan.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_flash.c | 2 + .../src/HAL/stm32yyxx_hal_flash_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_fmac.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_gfxtim.c | 8 + .../SrcWrapper/src/HAL/stm32yyxx_hal_gpio.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_gtzc.c | 4 +- .../SrcWrapper/src/HAL/stm32yyxx_hal_hash.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_hcd.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_i2c.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_i2c_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_i2s.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_i2s_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_i3c.c | 8 + .../SrcWrapper/src/HAL/stm32yyxx_hal_icache.c | 4 +- .../SrcWrapper/src/HAL/stm32yyxx_hal_irda.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_iwdg.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_jpeg.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_lptim.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_mmc.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_mmc_ex.c | 4 +- .../SrcWrapper/src/HAL/stm32yyxx_hal_nand.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_nor.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_opamp.c | 2 + .../src/HAL/stm32yyxx_hal_opamp_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_otfdec.c | 4 +- .../SrcWrapper/src/HAL/stm32yyxx_hal_pcd.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_pcd_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_pka.c | 4 +- .../SrcWrapper/src/HAL/stm32yyxx_hal_pssi.c | 4 +- .../SrcWrapper/src/HAL/stm32yyxx_hal_pwr.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_pwr_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_ramcfg.c | 4 +- .../SrcWrapper/src/HAL/stm32yyxx_hal_rcc.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_rcc_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_rng.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_rng_ex.c | 4 +- .../SrcWrapper/src/HAL/stm32yyxx_hal_rtc.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_rtc_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_sai.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_sai_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_sd.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_sd_ex.c | 4 +- .../SrcWrapper/src/HAL/stm32yyxx_hal_sdram.c | 2 + .../src/HAL/stm32yyxx_hal_smartcard.c | 2 + .../src/HAL/stm32yyxx_hal_smartcard_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_smbus.c | 4 + .../src/HAL/stm32yyxx_hal_smbus_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_spi.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_spi_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_sram.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_tim.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_tim_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_uart.c | 2 + .../src/HAL/stm32yyxx_hal_uart_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_usart.c | 2 + .../src/HAL/stm32yyxx_hal_usart_ex.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_wwdg.c | 2 + .../SrcWrapper/src/HAL/stm32yyxx_hal_xspi.c | 4 +- .../SrcWrapper/src/HardwareTimer.cpp | 8 +- .../SrcWrapper/src/LL/stm32yyxx_ll_adc.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_comp.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_cordic.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_crc.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_crs.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_dac.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_dlyb.c | 4 +- .../SrcWrapper/src/LL/stm32yyxx_ll_dma.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_exti.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_fmac.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_fmc.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_gpio.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_i2c.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_i3c.c | 8 + .../SrcWrapper/src/LL/stm32yyxx_ll_icache.c | 4 +- .../SrcWrapper/src/LL/stm32yyxx_ll_lptim.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_lpuart.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_opamp.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_pka.c | 4 +- .../SrcWrapper/src/LL/stm32yyxx_ll_pwr.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_rcc.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_rng.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_rtc.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_sdmmc.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_spi.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_tim.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_ucpd.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_usart.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_usb.c | 2 + .../SrcWrapper/src/LL/stm32yyxx_ll_utils.c | 2 + .../libraries/SrcWrapper/src/stm32/analog.cpp | 9 +- stm32/libraries/SrcWrapper/src/stm32/clock.c | 19 +- .../SrcWrapper/src/stm32/interrupt.cpp | 9 +- .../SrcWrapper/src/stm32/system_stm32yyxx.c | 2 + .../USBDeviceKeyboard/USBDeviceKeyboard.ino | 9 +- .../examples/USBDeviceMIDI/USBDeviceMIDI.ino | 9 +- .../USBDeviceMouse/USBDeviceMouse.ino | 9 +- .../examples/USBHostMIDI/USBHostMIDI.ino | 14 +- .../USBHostMouseKeyboard.ino | 12 +- stm32/libraries/Wire/src/Wire.cpp | 23 +- stm32/libraries/Wire/src/Wire.h | 1 + stm32/platform.txt | 24 +- .../Device/ST/STM32F4xx/Include/stm32f446xx.h | 56 +- .../Device/ST/STM32F4xx/Include/stm32f4xx.h | 116 +- .../ST/STM32F4xx/Include/system_stm32f4xx.h | 18 +- .../CMSIS/Device/ST/STM32F4xx/README.md | 11 +- .../Device/ST/STM32F4xx/Release_Notes.html | 2818 +-- .../Templates/gcc/startup_stm32f446xx.s | 9 +- .../Device/ST/STM32F4xx/_htmresc/favicon.png | Bin 0 -> 4126 bytes .../ST/STM32F4xx/_htmresc/mini-st_2020.css} | 365 +- .../Device/ST/STM32F4xx/_htmresc/st_logo.png | Bin 18616 -> 0 bytes .../ST/STM32F4xx/_htmresc/st_logo_2020.png | Bin 0 -> 7520 bytes .../Inc/Legacy/stm32_hal_legacy.h | 436 +- .../Inc/stm32f4xx_hal_adc.h | 58 +- .../Inc/stm32f4xx_hal_adc_ex.h | 30 +- .../Inc/stm32f4xx_hal_can.h | 53 +- .../Inc/stm32f4xx_hal_cec.h | 136 +- .../Inc/stm32f4xx_hal_cortex.h | 1 + .../Inc/stm32f4xx_hal_crc.h | 2 +- .../Inc/stm32f4xx_hal_dac.h | 32 +- .../Inc/stm32f4xx_hal_dac_ex.h | 9 +- .../Inc/stm32f4xx_hal_def.h | 2 + .../Inc/stm32f4xx_hal_dfsdm.h | 24 +- .../Inc/stm32f4xx_hal_dsi.h | 12 +- .../Inc/stm32f4xx_hal_eth.h | 42 +- .../Inc/stm32f4xx_hal_fmpi2c.h | 143 +- .../Inc/stm32f4xx_hal_fmpi2c_ex.h | 4 +- .../Inc/stm32f4xx_hal_fmpsmbus.h | 139 +- .../Inc/stm32f4xx_hal_fmpsmbus_ex.h | 6 +- .../Inc/stm32f4xx_hal_hcd.h | 26 +- .../Inc/stm32f4xx_hal_irda.h | 4 +- .../Inc/stm32f4xx_hal_lptim.h | 8 +- .../Inc/stm32f4xx_hal_ltdc.h | 16 +- .../Inc/stm32f4xx_hal_nand.h | 45 +- .../Inc/stm32f4xx_hal_nor.h | 2 +- .../Inc/stm32f4xx_hal_pcd.h | 12 +- .../Inc/stm32f4xx_hal_pcd_ex.h | 14 +- .../Inc/stm32f4xx_hal_pwr.h | 13 +- .../Inc/stm32f4xx_hal_rcc.h | 5 +- .../Inc/stm32f4xx_hal_rcc_ex.h | 108 +- .../Inc/stm32f4xx_hal_rng.h | 6 +- .../Inc/stm32f4xx_hal_rtc.h | 9 +- .../Inc/stm32f4xx_hal_rtc_ex.h | 16 +- .../Inc/stm32f4xx_hal_sai.h | 4 +- .../Inc/stm32f4xx_hal_sai_ex.h | 4 +- .../Inc/stm32f4xx_hal_smartcard.h | 4 +- .../Inc/stm32f4xx_hal_smbus.h | 36 +- .../Inc/stm32f4xx_hal_spdifrx.h | 56 +- .../Inc/stm32f4xx_hal_sram.h | 2 +- .../Inc/stm32f4xx_hal_tim.h | 93 +- .../Inc/stm32f4xx_hal_tim_ex.h | 19 +- .../Inc/stm32f4xx_hal_uart.h | 33 +- .../Inc/stm32f4xx_hal_usart.h | 6 +- .../Inc/stm32f4xx_ll_adc.h | 51 +- .../Inc/stm32f4xx_ll_cortex.h | 10 + .../Inc/stm32f4xx_ll_crc.h | 4 +- .../Inc/stm32f4xx_ll_dac.h | 138 +- .../Inc/stm32f4xx_ll_fmc.h | 4 +- .../Inc/stm32f4xx_ll_fmpi2c.h | 135 +- .../Inc/stm32f4xx_ll_fsmc.h | 2 +- .../Inc/stm32f4xx_ll_i2c.h | 44 +- .../Inc/stm32f4xx_ll_lptim.h | 90 +- .../Inc/stm32f4xx_ll_rng.h | 19 +- .../Inc/stm32f4xx_ll_rtc.h | 22 +- .../Inc/stm32f4xx_ll_sdmmc.h | 8 +- .../Inc/stm32f4xx_ll_tim.h | 177 +- .../Inc/stm32f4xx_ll_usart.h | 220 +- .../Inc/stm32f4xx_ll_usb.h | 250 +- .../STM32F4xx_HAL_Driver/Release_Notes.html | 15646 +++++----------- .../STM32F4xx_HAL_Driver/Src/stm32f4xx_hal.c | 7 +- .../Src/stm32f4xx_hal_adc.c | 260 +- .../Src/stm32f4xx_hal_adc_ex.c | 134 +- .../Src/stm32f4xx_hal_can.c | 92 +- .../Src/stm32f4xx_hal_cec.c | 57 +- .../Src/stm32f4xx_hal_cortex.c | 10 + .../Src/stm32f4xx_hal_crc.c | 4 +- .../Src/stm32f4xx_hal_cryp.c | 126 +- .../Src/stm32f4xx_hal_dac.c | 127 +- .../Src/stm32f4xx_hal_dac_ex.c | 43 +- .../Src/stm32f4xx_hal_dfsdm.c | 28 +- .../Src/stm32f4xx_hal_dma2d.c | 22 +- .../Src/stm32f4xx_hal_dsi.c | 379 +- .../Src/stm32f4xx_hal_eth.c | 85 +- .../Src/stm32f4xx_hal_exti.c | 8 +- .../Src/stm32f4xx_hal_fmpi2c.c | 1509 +- .../Src/stm32f4xx_hal_fmpsmbus.c | 209 +- .../Src/stm32f4xx_hal_hash.c | 2 +- .../Src/stm32f4xx_hal_hcd.c | 757 +- .../Src/stm32f4xx_hal_i2c.c | 251 +- .../Src/stm32f4xx_hal_irda.c | 19 +- .../Src/stm32f4xx_hal_lptim.c | 24 +- .../Src/stm32f4xx_hal_ltdc.c | 8 +- .../Src/stm32f4xx_hal_ltdc_ex.c | 21 +- .../Src/stm32f4xx_hal_nand.c | 104 +- .../Src/stm32f4xx_hal_nor.c | 208 +- .../Src/stm32f4xx_hal_pcd.c | 98 +- .../Src/stm32f4xx_hal_pcd_ex.c | 22 +- .../Src/stm32f4xx_hal_pwr.c | 51 +- .../Src/stm32f4xx_hal_qspi.c | 12 +- .../Src/stm32f4xx_hal_rcc.c | 4 +- .../Src/stm32f4xx_hal_rcc_ex.c | 24 + .../Src/stm32f4xx_hal_rng.c | 29 +- .../Src/stm32f4xx_hal_rtc.c | 94 +- .../Src/stm32f4xx_hal_rtc_ex.c | 41 +- .../Src/stm32f4xx_hal_sai.c | 28 +- .../Src/stm32f4xx_hal_sai_ex.c | 4 +- .../Src/stm32f4xx_hal_sd.c | 22 +- .../Src/stm32f4xx_hal_sdram.c | 36 +- .../Src/stm32f4xx_hal_smartcard.c | 21 +- .../Src/stm32f4xx_hal_smbus.c | 22 +- .../Src/stm32f4xx_hal_spdifrx.c | 56 +- .../Src/stm32f4xx_hal_spi.c | 132 +- .../Src/stm32f4xx_hal_sram.c | 40 +- .../Src/stm32f4xx_hal_tim.c | 189 +- .../Src/stm32f4xx_hal_tim_ex.c | 58 +- ...tm32f4xx_hal_timebase_rtc_alarm_template.c | 8 +- ...m32f4xx_hal_timebase_rtc_wakeup_template.c | 8 +- .../Src/stm32f4xx_hal_timebase_tim_template.c | 2 +- .../Src/stm32f4xx_hal_uart.c | 174 +- .../Src/stm32f4xx_hal_usart.c | 26 +- .../Src/stm32f4xx_ll_adc.c | 116 +- .../Src/stm32f4xx_ll_crc.c | 2 +- .../Src/stm32f4xx_ll_dac.c | 74 +- .../Src/stm32f4xx_ll_fmc.c | 5 +- .../Src/stm32f4xx_ll_fmpi2c.c | 16 +- .../Src/stm32f4xx_ll_fsmc.c | 3 +- .../Src/stm32f4xx_ll_lptim.c | 7 +- .../Src/stm32f4xx_ll_rng.c | 4 +- .../Src/stm32f4xx_ll_rtc.c | 2 +- .../Src/stm32f4xx_ll_tim.c | 71 +- .../Src/stm32f4xx_ll_usart.c | 6 +- .../Src/stm32f4xx_ll_usb.c | 522 +- .../STM32F4xx_HAL_Driver/_htmresc/favicon.png | Bin 0 -> 4126 bytes .../_htmresc/mini-st_2020.css | 1711 ++ .../STM32F4xx_HAL_Driver/_htmresc/st_logo.png | Bin 18616 -> 0 bytes .../_htmresc/st_logo_2020.png | Bin 0 -> 7520 bytes .../OpenAMP/libmetal/.checkpatch.conf | 44 +- .../Middlewares/OpenAMP/libmetal/.gitlint | 196 +- .../Middlewares/OpenAMP/libmetal/VERSION | 6 +- .../OpenAMP/libmetal/scripts/checkpatch.pl | 12988 ++++++------- .../libmetal/scripts/ci/check_compliance.py | 0 .../OpenAMP/libmetal/scripts/do_checkpatch.sh | 0 .../OpenAMP/open-amp/.checkpatch.conf | 42 +- .../Middlewares/OpenAMP/open-amp/.gitlint | 196 +- .../Middlewares/OpenAMP/open-amp/VERSION | 6 +- .../apps/examples/rpc_demo/rpc_demo.c | 0 .../apps/examples/rpc_demo/rpc_demod.c | 0 .../OpenAMP/open-amp/scripts/checkpatch.pl | 12988 ++++++------- .../open-amp/scripts/ci/check_compliance.py | 0 .../OpenAMP/open-amp/scripts/do_checkpatch.sh | 0 .../Class/AUDIO/Inc/usbd_audio.h | 130 +- .../Class/AUDIO/Inc/usbd_audio_if_template.h | 13 +- .../Class/AUDIO/Src/usbd_audio.c | 225 +- .../Class/AUDIO/Src/usbd_audio_if_template.c | 13 +- .../Class/BillBoard/Inc/usbd_billboard.h | 15 +- .../Class/BillBoard/Src/usbd_billboard.c | 38 +- .../Class/CCID/Inc/usbd_ccid.h | 373 + .../Class/CCID/Inc/usbd_ccid_cmd.h | 222 + .../Class/CCID/Inc/usbd_ccid_if_template.h | 76 + .../Class/CCID/Inc/usbd_ccid_sc_if_template.h | 100 + .../CCID/Inc/usbd_ccid_smartcard_template.h | 279 + .../Class/CCID/Src/usbd_ccid.c | 969 + .../Class/CCID/Src/usbd_ccid_cmd.c | 1099 ++ .../Class/CCID/Src/usbd_ccid_if_template.c | 270 + .../Class/CCID/Src/usbd_ccid_sc_if_template.c | 473 + .../CCID/Src/usbd_ccid_smartcard_template.c | 486 + .../Class/CDC/Inc/usbd_cdc.h | 27 +- .../Class/CDC/Inc/usbd_cdc_if_template.h | 12 +- .../Class/CDC/Src/usbd_cdc.c | 491 +- .../Class/CDC/Src/usbd_cdc_if_template.c | 15 +- .../Class/CDC_ECM/Inc/usbd_cdc_ecm.h | 59 +- .../CDC_ECM/Inc/usbd_cdc_ecm_if_template.h | 12 +- .../Class/CDC_ECM/Src/usbd_cdc_ecm.c | 552 +- .../CDC_ECM/Src/usbd_cdc_ecm_if_template.c | 55 +- .../Class/CDC_RNDIS/Inc/usbd_cdc_rndis.h | 40 +- .../Inc/usbd_cdc_rndis_if_template.h | 12 +- .../Class/CDC_RNDIS/Src/usbd_cdc_rndis.c | 568 +- .../Src/usbd_cdc_rndis_if_template.c | 56 +- .../Inc/usbd_composite_builder.h | 289 + .../Src/usbd_composite_builder.c | 1879 ++ .../Class/CustomHID/Inc/usbd_customhid.h | 50 +- .../Inc/usbd_customhid_if_template.h | 12 +- .../Class/CustomHID/Src/usbd_customhid.c | 325 +- .../Src/usbd_customhid_if_template.c | 83 +- .../Class/DFU/Inc/usbd_dfu.h | 59 +- .../Class/DFU/Inc/usbd_dfu_media_template.h | 12 +- .../Class/DFU/Src/usbd_dfu.c | 173 +- .../Class/DFU/Src/usbd_dfu_media_template.c | 25 +- .../Class/HID/Inc/usbd_hid.h | 55 +- .../Class/HID/Src/usbd_hid.c | 258 +- .../Class/MSC/Inc/usbd_msc.h | 22 +- .../Class/MSC/Inc/usbd_msc_bot.h | 18 +- .../Class/MSC/Inc/usbd_msc_data.h | 13 +- .../Class/MSC/Inc/usbd_msc_scsi.h | 15 +- .../Class/MSC/Inc/usbd_msc_storage_template.h | 13 +- .../Class/MSC/Src/usbd_msc.c | 244 +- .../Class/MSC/Src/usbd_msc_bot.c | 115 +- .../Class/MSC/Src/usbd_msc_data.c | 12 +- .../Class/MSC/Src/usbd_msc_scsi.c | 130 +- .../Class/MSC/Src/usbd_msc_storage_template.c | 126 +- .../Class/MTP/Inc/usbd_mtp.h | 358 + .../Class/MTP/Inc/usbd_mtp_if_template.h | 96 + .../Class/MTP/Inc/usbd_mtp_opt.h | 681 + .../Class/MTP/Inc/usbd_mtp_storage.h | 118 + .../Class/MTP/Src/usbd_mtp.c | 701 + .../Class/MTP/Src/usbd_mtp_if_template.c | 347 + .../Class/MTP/Src/usbd_mtp_opt.c | 1267 ++ .../Class/MTP/Src/usbd_mtp_storage.c | 472 + .../Class/Printer/Inc/usbd_printer.h | 15 +- .../Printer/Inc/usbd_printer_if_template.h | 12 +- .../Class/Printer/Src/usbd_printer.c | 301 +- .../Printer/Src/usbd_printer_if_template.c | 44 +- .../Class/Template/Inc/usbd_template.h | 13 +- .../Class/Template/Src/usbd_template.c | 45 +- .../Class/VIDEO/Inc/usbd_video.h | 182 +- .../Class/VIDEO/Inc/usbd_video_if_template.h | 34 +- .../Class/VIDEO/Src/usbd_video.c | 237 +- .../Class/VIDEO/Src/usbd_video_if_template.c | 38 +- .../Core/Inc/usbd_conf_template.h | 25 +- .../Core/Inc/usbd_core.h | 32 +- .../Core/Inc/usbd_ctlreq.h | 12 +- .../Core/Inc/usbd_def.h | 136 +- .../Core/Inc/usbd_desc_template.h | 16 +- .../Core/Inc/usbd_ioreq.h | 13 +- .../Core/Src/usbd_conf_template.c | 31 +- .../Core/Src/usbd_core.c | 709 +- .../Core/Src/usbd_ctlreq.c | 157 +- .../Core/Src/usbd_desc_template.c | 71 +- .../Core/Src/usbd_ioreq.c | 16 +- .../ST/STM32_USB_Device_Library/LICENSE.md | 80 + .../ST/STM32_USB_Device_Library/README.md | 36 + .../Release_Notes.html | 2529 +-- .../_htmresc/favicon.png | Bin 0 -> 4126 bytes .../_htmresc/mini-st.css | 363 +- .../_htmresc/st_logo_2020.png | Bin 0 -> 7520 bytes .../_htmresc/favicon.png | Bin 0 -> 4126 bytes .../_htmresc/mini-st.css | 1711 ++ .../_htmresc/st_logo_2020.png | Bin 0 -> 7520 bytes .../STM32F4xx/stm32f4xx_hal_conf_default.h | 78 + .../STM32H7xx/stm32h7xx_hal_conf_default.h | 96 + stm32/system/STM32H7xx/system_stm32h7xx.c | 5 +- .../KLST_SHEEP/PeripheralPins_KLST_SHEEP.c | 136 +- stm32/variants/KLST_SHEEP/PinNamesVar.h | 4 - stm32/variants/KLST_SHEEP/ldscript.ld | 27 +- .../KLST_TINY/PeripheralPins_KLST_TINY.c | 80 +- stm32/variants/KLST_TINY/ldscript.ld | 12 +- 770 files changed, 99936 insertions(+), 36969 deletions(-) create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_assert.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_atomic.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_audio.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_bits.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_blendmode.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_clipboard.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_config.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_cpuinfo.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_egl.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_endian.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_error.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_events.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_filesystem.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_gamecontroller.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_gesture.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_guid.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_haptic.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_hidapi.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_hints.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_joystick.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_keyboard.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_keycode.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_loadso.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_locale.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_log.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_main.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_messagebox.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_metal.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_misc.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_mouse.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_mutex.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_name.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengl.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengl_glext.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_gl2.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_gl2ext.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_gl2platform.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_khrplatform.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_pixels.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_platform.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_power.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_quit.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_rect.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_render.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_revision.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_rwops.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_scancode.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_sensor.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_shape.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_stdinc.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_surface.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_system.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_syswm.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_test.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_assert.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_common.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_compare.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_crc32.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_font.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_fuzzer.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_harness.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_images.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_log.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_md5.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_memory.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_random.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_thread.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_timer.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_touch.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_types.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_version.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_video.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/SDL_vulkan.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/begin_code.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/include/close_code.h create mode 100644 desktop/cores/sdl/SDL2.macos.arm/lib/libSDL2-2.0.0.dylib create mode 100644 desktop/cores/sdl/SDL2.macos.arm/lib/libSDL2.a create mode 100644 desktop/cores/sdl/SDL2.macos.arm/lib/libSDL2.dylib create mode 100644 desktop/cores/sdl/SDL2.macos.arm/lib/libSDL2_test.a create mode 100644 desktop/cores/sdl/SDL2.macos.arm/lib/libSDL2main.a rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_assert.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_atomic.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_audio.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_bits.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_blendmode.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_clipboard.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_config.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_cpuinfo.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_egl.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_endian.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_error.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_events.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_filesystem.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_gamecontroller.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_gesture.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_haptic.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_hints.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_joystick.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_keyboard.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_keycode.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_loadso.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_log.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_main.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_messagebox.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_metal.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_mouse.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_mutex.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_name.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_opengl.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_opengl_glext.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_opengles.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_opengles2.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_opengles2_gl2.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_opengles2_gl2ext.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_opengles2_gl2platform.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_opengles2_khrplatform.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_pixels.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_platform.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_power.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_quit.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_rect.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_render.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_revision.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_rwops.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_scancode.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_sensor.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_shape.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_stdinc.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_surface.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_system.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_syswm.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_test.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_test_assert.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_test_common.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_test_compare.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_test_crc32.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_test_font.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_test_fuzzer.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_test_harness.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_test_images.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_test_log.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_test_md5.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_test_memory.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_test_random.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_thread.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_timer.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_touch.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_types.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_version.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_video.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/SDL_vulkan.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/begin_code.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/include/close_code.h (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/lib/libSDL2-2.0.0.dylib (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/lib/libSDL2.a (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/lib/libSDL2.dylib (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/lib/libSDL2_test.a (100%) rename desktop/cores/sdl/{SDL2.macosx => SDL2.macos.intel}/lib/libSDL2main.a (100%) delete mode 100644 desktop/libraries/KlangWellen/src/KlangWellen.cpp create mode 100644 desktop/libraries/KlangWellen/src/Resonator.h delete mode 100644 desktop/libraries/KlangWellen/src/Scale.cpp create mode 100644 desktop/libraries/KlangWellen/src/ScaleCollection.h create mode 100644 desktop/libraries/KlangWellen/src/Stream.h rename stm32/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleReadSDCard/ExampleReadSDCard.ino => desktop/libraries/Klangstrom/examples/02.Peripherals/ExternalSDCard/ExternalSDCard.ino (80%) create mode 100644 desktop/libraries/Klangstrom/examples/02.Peripherals/Presets/Presets.ino create mode 100644 desktop/libraries/Klangstrom/examples/02.Peripherals/Serial/Serial.ino delete mode 100644 desktop/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleSavingPresets/ExampleSavingPresets.ino rename desktop/libraries/KlangstromCard/examples/{ExamplePlayWAVFile/ExamplePlayWAVFile.ino => ExamplePlayWAVE/ExamplePlayWAVE.ino} (91%) create mode 100644 desktop/libraries/KlangstromCard/examples/ExampleStreamWAVE/ExampleStreamWAVE.ino rename stm32/libraries/KlangstromCard/examples/ExamplePlayWAVFile/ExamplePlayWAVFile.ino => docs/_includes/code/ExamplePlayWAVE.ino (91%) create mode 100644 docs/_includes/code/ExampleStreamWAVE.ino create mode 100644 stm32/cores/arduino/stm32/LL/stm32yyxx_ll_i3c.h create mode 100644 stm32/libraries/CMSIS_DSP/src/BasicMathFunctions/BasicMathFunctionsF16.c create mode 100644 stm32/libraries/CMSIS_DSP/src/BayesFunctions/BayesFunctionsF16.c create mode 100644 stm32/libraries/CMSIS_DSP/src/CommonTables/CommonTablesF16.c create mode 100644 stm32/libraries/CMSIS_DSP/src/ComplexMathFunctions/ComplexMathFunctionsF16.c create mode 100644 stm32/libraries/CMSIS_DSP/src/DistanceFunctions/DistanceFunctionsF16.c create mode 100644 stm32/libraries/CMSIS_DSP/src/FastMathFunctions/FastMathFunctionsF16.c create mode 100644 stm32/libraries/CMSIS_DSP/src/FilteringFunctions/FilteringFunctionsF16.c create mode 100644 stm32/libraries/CMSIS_DSP/src/InterpolationFunctions/InterpolationFunctions.c create mode 100644 stm32/libraries/CMSIS_DSP/src/InterpolationFunctions/InterpolationFunctionsF16.c create mode 100644 stm32/libraries/CMSIS_DSP/src/MatrixFunctions/MatrixFunctionsF16.c create mode 100644 stm32/libraries/CMSIS_DSP/src/QuaternionMathFunctions/QuaternionMathFunctions.c create mode 100644 stm32/libraries/CMSIS_DSP/src/SVMFunctions/SVMFunctionsF16.c create mode 100644 stm32/libraries/CMSIS_DSP/src/StatisticsFunctions/StatisticsFunctionsF16.c create mode 100644 stm32/libraries/CMSIS_DSP/src/SupportFunctions/SupportFunctionsF16.c create mode 100644 stm32/libraries/CMSIS_DSP/src/TransformFunctions/TransformFunctionsF16.c delete mode 100644 stm32/libraries/KlangWellen/src/KlangWellen.cpp create mode 100644 stm32/libraries/KlangWellen/src/Resonator.h delete mode 100644 stm32/libraries/KlangWellen/src/Scale.cpp create mode 100644 stm32/libraries/KlangWellen/src/ScaleCollection.h create mode 100644 stm32/libraries/KlangWellen/src/Stream.h rename desktop/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleReadSDCard/ExampleReadSDCard.ino => stm32/libraries/Klangstrom/examples/02.Peripherals/ExternalSDCard/ExternalSDCard.ino (80%) create mode 100644 stm32/libraries/Klangstrom/examples/02.Peripherals/Presets/Presets.ino create mode 100644 stm32/libraries/Klangstrom/examples/02.Peripherals/Serial/Serial.ino delete mode 100644 stm32/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleSavingPresets/ExampleSavingPresets.ino create mode 100644 stm32/libraries/KlangstromCard/examples/ExamplePlayWAVE/ExamplePlayWAVE.ino create mode 100644 stm32/libraries/KlangstromCard/examples/ExampleStreamWAVE/ExampleStreamWAVE.ino create mode 100644 stm32/libraries/KlangstromTest/library.properties create mode 100755 stm32/libraries/KlangstromTest/src/KlangstromClass.cpp create mode 100755 stm32/libraries/KlangstromTest/src/KlangstromClass.h create mode 100755 stm32/libraries/KlangstromTest/src/KlangstromTest.cpp create mode 100755 stm32/libraries/KlangstromTest/src/KlangstromTest.h create mode 100644 stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_gfxtim.c create mode 100644 stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i3c.c create mode 100644 stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_i3c.c create mode 100644 stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/_htmresc/favicon.png rename stm32/system/Drivers/{STM32F4xx_HAL_Driver/_htmresc/mini-st.css => CMSIS/Device/ST/STM32F4xx/_htmresc/mini-st_2020.css} (77%) delete mode 100644 stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/_htmresc/st_logo.png create mode 100644 stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/_htmresc/st_logo_2020.png create mode 100644 stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/favicon.png create mode 100644 stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/mini-st_2020.css delete mode 100644 stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/st_logo.png create mode 100644 stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/st_logo_2020.png mode change 100755 => 100644 stm32/system/Middlewares/OpenAMP/libmetal/scripts/checkpatch.pl mode change 100755 => 100644 stm32/system/Middlewares/OpenAMP/libmetal/scripts/ci/check_compliance.py mode change 100755 => 100644 stm32/system/Middlewares/OpenAMP/libmetal/scripts/do_checkpatch.sh mode change 100644 => 100755 stm32/system/Middlewares/OpenAMP/open-amp/apps/examples/rpc_demo/rpc_demo.c mode change 100644 => 100755 stm32/system/Middlewares/OpenAMP/open-amp/apps/examples/rpc_demo/rpc_demod.c mode change 100755 => 100644 stm32/system/Middlewares/OpenAMP/open-amp/scripts/checkpatch.pl mode change 100755 => 100644 stm32/system/Middlewares/OpenAMP/open-amp/scripts/ci/check_compliance.py mode change 100755 => 100644 stm32/system/Middlewares/OpenAMP/open-amp/scripts/do_checkpatch.sh create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid.h create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_cmd.h create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_if_template.h create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_sc_if_template.h create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_smartcard_template.h create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid.c create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_cmd.c create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_if_template.c create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_sc_if_template.c create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_smartcard_template.c create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CompositeBuilder/Inc/usbd_composite_builder.h create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CompositeBuilder/Src/usbd_composite_builder.c create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp.h create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp_if_template.h create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp_opt.h create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp_storage.h create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp.c create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp_if_template.c create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp_opt.c create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp_storage.c create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/LICENSE.md create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/README.md create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/_htmresc/favicon.png rename stm32/system/{Drivers/CMSIS/Device/ST/STM32F4xx => Middlewares/ST/STM32_USB_Device_Library}/_htmresc/mini-st.css (77%) create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Device_Library/_htmresc/st_logo_2020.png create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Host_Library/_htmresc/favicon.png create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Host_Library/_htmresc/mini-st.css create mode 100644 stm32/system/Middlewares/ST/STM32_USB_Host_Library/_htmresc/st_logo_2020.png diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL.h new file mode 100644 index 00000000..9ba8f68c --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL.h @@ -0,0 +1,233 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL.h + * + * Main include header for the SDL library + */ + + +#ifndef SDL_h_ +#define SDL_h_ + +#include "SDL_main.h" +#include "SDL_stdinc.h" +#include "SDL_assert.h" +#include "SDL_atomic.h" +#include "SDL_audio.h" +#include "SDL_clipboard.h" +#include "SDL_cpuinfo.h" +#include "SDL_endian.h" +#include "SDL_error.h" +#include "SDL_events.h" +#include "SDL_filesystem.h" +#include "SDL_gamecontroller.h" +#include "SDL_guid.h" +#include "SDL_haptic.h" +#include "SDL_hidapi.h" +#include "SDL_hints.h" +#include "SDL_joystick.h" +#include "SDL_loadso.h" +#include "SDL_log.h" +#include "SDL_messagebox.h" +#include "SDL_metal.h" +#include "SDL_mutex.h" +#include "SDL_power.h" +#include "SDL_render.h" +#include "SDL_rwops.h" +#include "SDL_sensor.h" +#include "SDL_shape.h" +#include "SDL_system.h" +#include "SDL_thread.h" +#include "SDL_timer.h" +#include "SDL_version.h" +#include "SDL_video.h" +#include "SDL_locale.h" +#include "SDL_misc.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* As of version 0.5, SDL is loaded dynamically into the application */ + +/** + * \name SDL_INIT_* + * + * These are the flags which may be passed to SDL_Init(). You should + * specify the subsystems which you will be using in your application. + */ +/* @{ */ +#define SDL_INIT_TIMER 0x00000001u +#define SDL_INIT_AUDIO 0x00000010u +#define SDL_INIT_VIDEO 0x00000020u /**< SDL_INIT_VIDEO implies SDL_INIT_EVENTS */ +#define SDL_INIT_JOYSTICK 0x00000200u /**< SDL_INIT_JOYSTICK implies SDL_INIT_EVENTS */ +#define SDL_INIT_HAPTIC 0x00001000u +#define SDL_INIT_GAMECONTROLLER 0x00002000u /**< SDL_INIT_GAMECONTROLLER implies SDL_INIT_JOYSTICK */ +#define SDL_INIT_EVENTS 0x00004000u +#define SDL_INIT_SENSOR 0x00008000u +#define SDL_INIT_NOPARACHUTE 0x00100000u /**< compatibility; this flag is ignored. */ +#define SDL_INIT_EVERYTHING ( \ + SDL_INIT_TIMER | SDL_INIT_AUDIO | SDL_INIT_VIDEO | SDL_INIT_EVENTS | \ + SDL_INIT_JOYSTICK | SDL_INIT_HAPTIC | SDL_INIT_GAMECONTROLLER | SDL_INIT_SENSOR \ + ) +/* @} */ + +/** + * Initialize the SDL library. + * + * SDL_Init() simply forwards to calling SDL_InitSubSystem(). Therefore, the + * two may be used interchangeably. Though for readability of your code + * SDL_InitSubSystem() might be preferred. + * + * The file I/O (for example: SDL_RWFromFile) and threading (SDL_CreateThread) + * subsystems are initialized by default. Message boxes + * (SDL_ShowSimpleMessageBox) also attempt to work without initializing the + * video subsystem, in hopes of being useful in showing an error dialog when + * SDL_Init fails. You must specifically initialize other subsystems if you + * use them in your application. + * + * Logging (such as SDL_Log) works without initialization, too. + * + * `flags` may be any of the following OR'd together: + * + * - `SDL_INIT_TIMER`: timer subsystem + * - `SDL_INIT_AUDIO`: audio subsystem + * - `SDL_INIT_VIDEO`: video subsystem; automatically initializes the events + * subsystem + * - `SDL_INIT_JOYSTICK`: joystick subsystem; automatically initializes the + * events subsystem + * - `SDL_INIT_HAPTIC`: haptic (force feedback) subsystem + * - `SDL_INIT_GAMECONTROLLER`: controller subsystem; automatically + * initializes the joystick subsystem + * - `SDL_INIT_EVENTS`: events subsystem + * - `SDL_INIT_EVERYTHING`: all of the above subsystems + * - `SDL_INIT_NOPARACHUTE`: compatibility; this flag is ignored + * + * Subsystem initialization is ref-counted, you must call SDL_QuitSubSystem() + * for each SDL_InitSubSystem() to correctly shutdown a subsystem manually (or + * call SDL_Quit() to force shutdown). If a subsystem is already loaded then + * this call will increase the ref-count and return. + * + * \param flags subsystem initialization flags + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_InitSubSystem + * \sa SDL_Quit + * \sa SDL_SetMainReady + * \sa SDL_WasInit + */ +extern DECLSPEC int SDLCALL SDL_Init(Uint32 flags); + +/** + * Compatibility function to initialize the SDL library. + * + * In SDL2, this function and SDL_Init() are interchangeable. + * + * \param flags any of the flags used by SDL_Init(); see SDL_Init for details. + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Init + * \sa SDL_Quit + * \sa SDL_QuitSubSystem + */ +extern DECLSPEC int SDLCALL SDL_InitSubSystem(Uint32 flags); + +/** + * Shut down specific SDL subsystems. + * + * If you start a subsystem using a call to that subsystem's init function + * (for example SDL_VideoInit()) instead of SDL_Init() or SDL_InitSubSystem(), + * SDL_QuitSubSystem() and SDL_WasInit() will not work. You will need to use + * that subsystem's quit function (SDL_VideoQuit()) directly instead. But + * generally, you should not be using those functions directly anyhow; use + * SDL_Init() instead. + * + * You still need to call SDL_Quit() even if you close all open subsystems + * with SDL_QuitSubSystem(). + * + * \param flags any of the flags used by SDL_Init(); see SDL_Init for details. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_InitSubSystem + * \sa SDL_Quit + */ +extern DECLSPEC void SDLCALL SDL_QuitSubSystem(Uint32 flags); + +/** + * Get a mask of the specified subsystems which are currently initialized. + * + * \param flags any of the flags used by SDL_Init(); see SDL_Init for details. + * \returns a mask of all initialized subsystems if `flags` is 0, otherwise it + * returns the initialization status of the specified subsystems. + * + * The return value does not include SDL_INIT_NOPARACHUTE. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Init + * \sa SDL_InitSubSystem + */ +extern DECLSPEC Uint32 SDLCALL SDL_WasInit(Uint32 flags); + +/** + * Clean up all initialized subsystems. + * + * You should call this function even if you have already shutdown each + * initialized subsystem with SDL_QuitSubSystem(). It is safe to call this + * function even in the case of errors in initialization. + * + * If you start a subsystem using a call to that subsystem's init function + * (for example SDL_VideoInit()) instead of SDL_Init() or SDL_InitSubSystem(), + * then you must use that subsystem's quit function (SDL_VideoQuit()) to shut + * it down before calling SDL_Quit(). But generally, you should not be using + * those functions directly anyhow; use SDL_Init() instead. + * + * You can use this function with atexit() to ensure that it is run when your + * application is shutdown, but it is not wise to do this from a library or + * other dynamically loaded code. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Init + * \sa SDL_QuitSubSystem + */ +extern DECLSPEC void SDLCALL SDL_Quit(void); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_assert.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_assert.h new file mode 100644 index 00000000..7ce823ec --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_assert.h @@ -0,0 +1,322 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +#ifndef SDL_assert_h_ +#define SDL_assert_h_ + +#include "SDL_stdinc.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +#ifndef SDL_ASSERT_LEVEL +#ifdef SDL_DEFAULT_ASSERT_LEVEL +#define SDL_ASSERT_LEVEL SDL_DEFAULT_ASSERT_LEVEL +#elif defined(_DEBUG) || defined(DEBUG) || \ + (defined(__GNUC__) && !defined(__OPTIMIZE__)) +#define SDL_ASSERT_LEVEL 2 +#else +#define SDL_ASSERT_LEVEL 1 +#endif +#endif /* SDL_ASSERT_LEVEL */ + +/* +These are macros and not first class functions so that the debugger breaks +on the assertion line and not in some random guts of SDL, and so each +assert can have unique static variables associated with it. +*/ + +#if defined(_MSC_VER) +/* Don't include intrin.h here because it contains C++ code */ + extern void __cdecl __debugbreak(void); + #define SDL_TriggerBreakpoint() __debugbreak() +#elif _SDL_HAS_BUILTIN(__builtin_debugtrap) + #define SDL_TriggerBreakpoint() __builtin_debugtrap() +#elif ( (!defined(__NACL__)) && ((defined(__GNUC__) || defined(__clang__)) && (defined(__i386__) || defined(__x86_64__))) ) + #define SDL_TriggerBreakpoint() __asm__ __volatile__ ( "int $3\n\t" ) +#elif (defined(__GNUC__) || defined(__clang__)) && defined(__riscv) + #define SDL_TriggerBreakpoint() __asm__ __volatile__ ( "ebreak\n\t" ) +#elif ( defined(__APPLE__) && (defined(__arm64__) || defined(__aarch64__)) ) /* this might work on other ARM targets, but this is a known quantity... */ + #define SDL_TriggerBreakpoint() __asm__ __volatile__ ( "brk #22\n\t" ) +#elif defined(__APPLE__) && defined(__arm__) + #define SDL_TriggerBreakpoint() __asm__ __volatile__ ( "bkpt #22\n\t" ) +#elif defined(__386__) && defined(__WATCOMC__) + #define SDL_TriggerBreakpoint() { _asm { int 0x03 } } +#elif defined(HAVE_SIGNAL_H) && !defined(__WATCOMC__) + #include + #define SDL_TriggerBreakpoint() raise(SIGTRAP) +#else + /* How do we trigger breakpoints on this platform? */ + #define SDL_TriggerBreakpoint() +#endif + +#if defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 supports __func__ as a standard. */ +# define SDL_FUNCTION __func__ +#elif ((defined(__GNUC__) && (__GNUC__ >= 2)) || defined(_MSC_VER) || defined (__WATCOMC__)) +# define SDL_FUNCTION __FUNCTION__ +#else +# define SDL_FUNCTION "???" +#endif +#define SDL_FILE __FILE__ +#define SDL_LINE __LINE__ + +/* +sizeof (x) makes the compiler still parse the expression even without +assertions enabled, so the code is always checked at compile time, but +doesn't actually generate code for it, so there are no side effects or +expensive checks at run time, just the constant size of what x WOULD be, +which presumably gets optimized out as unused. +This also solves the problem of... + + int somevalue = blah(); + SDL_assert(somevalue == 1); + +...which would cause compiles to complain that somevalue is unused if we +disable assertions. +*/ + +/* "while (0,0)" fools Microsoft's compiler's /W4 warning level into thinking + this condition isn't constant. And looks like an owl's face! */ +#ifdef _MSC_VER /* stupid /W4 warnings. */ +#define SDL_NULL_WHILE_LOOP_CONDITION (0,0) +#else +#define SDL_NULL_WHILE_LOOP_CONDITION (0) +#endif + +#define SDL_disabled_assert(condition) \ + do { (void) sizeof ((condition)); } while (SDL_NULL_WHILE_LOOP_CONDITION) + +typedef enum +{ + SDL_ASSERTION_RETRY, /**< Retry the assert immediately. */ + SDL_ASSERTION_BREAK, /**< Make the debugger trigger a breakpoint. */ + SDL_ASSERTION_ABORT, /**< Terminate the program. */ + SDL_ASSERTION_IGNORE, /**< Ignore the assert. */ + SDL_ASSERTION_ALWAYS_IGNORE /**< Ignore the assert from now on. */ +} SDL_AssertState; + +typedef struct SDL_AssertData +{ + int always_ignore; + unsigned int trigger_count; + const char *condition; + const char *filename; + int linenum; + const char *function; + const struct SDL_AssertData *next; +} SDL_AssertData; + +/* Never call this directly. Use the SDL_assert* macros. */ +extern DECLSPEC SDL_AssertState SDLCALL SDL_ReportAssertion(SDL_AssertData *, + const char *, + const char *, int) +#if defined(__clang__) +#if __has_feature(attribute_analyzer_noreturn) +/* this tells Clang's static analysis that we're a custom assert function, + and that the analyzer should assume the condition was always true past this + SDL_assert test. */ + __attribute__((analyzer_noreturn)) +#endif +#endif +; + +/* the do {} while(0) avoids dangling else problems: + if (x) SDL_assert(y); else blah(); + ... without the do/while, the "else" could attach to this macro's "if". + We try to handle just the minimum we need here in a macro...the loop, + the static vars, and break points. The heavy lifting is handled in + SDL_ReportAssertion(), in SDL_assert.c. +*/ +#define SDL_enabled_assert(condition) \ + do { \ + while ( !(condition) ) { \ + static struct SDL_AssertData sdl_assert_data = { 0, 0, #condition, 0, 0, 0, 0 }; \ + const SDL_AssertState sdl_assert_state = SDL_ReportAssertion(&sdl_assert_data, SDL_FUNCTION, SDL_FILE, SDL_LINE); \ + if (sdl_assert_state == SDL_ASSERTION_RETRY) { \ + continue; /* go again. */ \ + } else if (sdl_assert_state == SDL_ASSERTION_BREAK) { \ + SDL_TriggerBreakpoint(); \ + } \ + break; /* not retrying. */ \ + } \ + } while (SDL_NULL_WHILE_LOOP_CONDITION) + +/* Enable various levels of assertions. */ +#if SDL_ASSERT_LEVEL == 0 /* assertions disabled */ +# define SDL_assert(condition) SDL_disabled_assert(condition) +# define SDL_assert_release(condition) SDL_disabled_assert(condition) +# define SDL_assert_paranoid(condition) SDL_disabled_assert(condition) +#elif SDL_ASSERT_LEVEL == 1 /* release settings. */ +# define SDL_assert(condition) SDL_disabled_assert(condition) +# define SDL_assert_release(condition) SDL_enabled_assert(condition) +# define SDL_assert_paranoid(condition) SDL_disabled_assert(condition) +#elif SDL_ASSERT_LEVEL == 2 /* normal settings. */ +# define SDL_assert(condition) SDL_enabled_assert(condition) +# define SDL_assert_release(condition) SDL_enabled_assert(condition) +# define SDL_assert_paranoid(condition) SDL_disabled_assert(condition) +#elif SDL_ASSERT_LEVEL == 3 /* paranoid settings. */ +# define SDL_assert(condition) SDL_enabled_assert(condition) +# define SDL_assert_release(condition) SDL_enabled_assert(condition) +# define SDL_assert_paranoid(condition) SDL_enabled_assert(condition) +#else +# error Unknown assertion level. +#endif + +/* this assertion is never disabled at any level. */ +#define SDL_assert_always(condition) SDL_enabled_assert(condition) + + +/** + * A callback that fires when an SDL assertion fails. + * + * \param data a pointer to the SDL_AssertData structure corresponding to the + * current assertion + * \param userdata what was passed as `userdata` to SDL_SetAssertionHandler() + * \returns an SDL_AssertState value indicating how to handle the failure. + */ +typedef SDL_AssertState (SDLCALL *SDL_AssertionHandler)( + const SDL_AssertData* data, void* userdata); + +/** + * Set an application-defined assertion handler. + * + * This function allows an application to show its own assertion UI and/or + * force the response to an assertion failure. If the application doesn't + * provide this, SDL will try to do the right thing, popping up a + * system-specific GUI dialog, and probably minimizing any fullscreen windows. + * + * This callback may fire from any thread, but it runs wrapped in a mutex, so + * it will only fire from one thread at a time. + * + * This callback is NOT reset to SDL's internal handler upon SDL_Quit()! + * + * \param handler the SDL_AssertionHandler function to call when an assertion + * fails or NULL for the default handler + * \param userdata a pointer that is passed to `handler` + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetAssertionHandler + */ +extern DECLSPEC void SDLCALL SDL_SetAssertionHandler( + SDL_AssertionHandler handler, + void *userdata); + +/** + * Get the default assertion handler. + * + * This returns the function pointer that is called by default when an + * assertion is triggered. This is an internal function provided by SDL, that + * is used for assertions when SDL_SetAssertionHandler() hasn't been used to + * provide a different function. + * + * \returns the default SDL_AssertionHandler that is called when an assert + * triggers. + * + * \since This function is available since SDL 2.0.2. + * + * \sa SDL_GetAssertionHandler + */ +extern DECLSPEC SDL_AssertionHandler SDLCALL SDL_GetDefaultAssertionHandler(void); + +/** + * Get the current assertion handler. + * + * This returns the function pointer that is called when an assertion is + * triggered. This is either the value last passed to + * SDL_SetAssertionHandler(), or if no application-specified function is set, + * is equivalent to calling SDL_GetDefaultAssertionHandler(). + * + * The parameter `puserdata` is a pointer to a void*, which will store the + * "userdata" pointer that was passed to SDL_SetAssertionHandler(). This value + * will always be NULL for the default handler. If you don't care about this + * data, it is safe to pass a NULL pointer to this function to ignore it. + * + * \param puserdata pointer which is filled with the "userdata" pointer that + * was passed to SDL_SetAssertionHandler() + * \returns the SDL_AssertionHandler that is called when an assert triggers. + * + * \since This function is available since SDL 2.0.2. + * + * \sa SDL_SetAssertionHandler + */ +extern DECLSPEC SDL_AssertionHandler SDLCALL SDL_GetAssertionHandler(void **puserdata); + +/** + * Get a list of all assertion failures. + * + * This function gets all assertions triggered since the last call to + * SDL_ResetAssertionReport(), or the start of the program. + * + * The proper way to examine this data looks something like this: + * + * ```c + * const SDL_AssertData *item = SDL_GetAssertionReport(); + * while (item) { + * printf("'%s', %s (%s:%d), triggered %u times, always ignore: %s.\\n", + * item->condition, item->function, item->filename, + * item->linenum, item->trigger_count, + * item->always_ignore ? "yes" : "no"); + * item = item->next; + * } + * ``` + * + * \returns a list of all failed assertions or NULL if the list is empty. This + * memory should not be modified or freed by the application. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ResetAssertionReport + */ +extern DECLSPEC const SDL_AssertData * SDLCALL SDL_GetAssertionReport(void); + +/** + * Clear the list of all assertion failures. + * + * This function will clear the list of all assertions triggered up to that + * point. Immediately following this call, SDL_GetAssertionReport will return + * no items. In addition, any previously-triggered assertions will be reset to + * a trigger_count of zero, and their always_ignore state will be false. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetAssertionReport + */ +extern DECLSPEC void SDLCALL SDL_ResetAssertionReport(void); + + +/* these had wrong naming conventions until 2.0.4. Please update your app! */ +#define SDL_assert_state SDL_AssertState +#define SDL_assert_data SDL_AssertData + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_assert_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_atomic.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_atomic.h new file mode 100644 index 00000000..1dd816a3 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_atomic.h @@ -0,0 +1,414 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_atomic.h + * + * Atomic operations. + * + * IMPORTANT: + * If you are not an expert in concurrent lockless programming, you should + * only be using the atomic lock and reference counting functions in this + * file. In all other cases you should be protecting your data structures + * with full mutexes. + * + * The list of "safe" functions to use are: + * SDL_AtomicLock() + * SDL_AtomicUnlock() + * SDL_AtomicIncRef() + * SDL_AtomicDecRef() + * + * Seriously, here be dragons! + * ^^^^^^^^^^^^^^^^^^^^^^^^^^^ + * + * You can find out a little more about lockless programming and the + * subtle issues that can arise here: + * http://msdn.microsoft.com/en-us/library/ee418650%28v=vs.85%29.aspx + * + * There's also lots of good information here: + * http://www.1024cores.net/home/lock-free-algorithms + * http://preshing.com/ + * + * These operations may or may not actually be implemented using + * processor specific atomic operations. When possible they are + * implemented as true processor specific atomic operations. When that + * is not possible the are implemented using locks that *do* use the + * available atomic operations. + * + * All of the atomic operations that modify memory are full memory barriers. + */ + +#ifndef SDL_atomic_h_ +#define SDL_atomic_h_ + +#include "SDL_stdinc.h" +#include "SDL_platform.h" + +#include "begin_code.h" + +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \name SDL AtomicLock + * + * The atomic locks are efficient spinlocks using CPU instructions, + * but are vulnerable to starvation and can spin forever if a thread + * holding a lock has been terminated. For this reason you should + * minimize the code executed inside an atomic lock and never do + * expensive things like API or system calls while holding them. + * + * The atomic locks are not safe to lock recursively. + * + * Porting Note: + * The spin lock functions and type are required and can not be + * emulated because they are used in the atomic emulation code. + */ +/* @{ */ + +typedef int SDL_SpinLock; + +/** + * Try to lock a spin lock by setting it to a non-zero value. + * + * ***Please note that spinlocks are dangerous if you don't know what you're + * doing. Please be careful using any sort of spinlock!*** + * + * \param lock a pointer to a lock variable + * \returns SDL_TRUE if the lock succeeded, SDL_FALSE if the lock is already + * held. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AtomicLock + * \sa SDL_AtomicUnlock + */ +extern DECLSPEC SDL_bool SDLCALL SDL_AtomicTryLock(SDL_SpinLock *lock); + +/** + * Lock a spin lock by setting it to a non-zero value. + * + * ***Please note that spinlocks are dangerous if you don't know what you're + * doing. Please be careful using any sort of spinlock!*** + * + * \param lock a pointer to a lock variable + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AtomicTryLock + * \sa SDL_AtomicUnlock + */ +extern DECLSPEC void SDLCALL SDL_AtomicLock(SDL_SpinLock *lock); + +/** + * Unlock a spin lock by setting it to 0. + * + * Always returns immediately. + * + * ***Please note that spinlocks are dangerous if you don't know what you're + * doing. Please be careful using any sort of spinlock!*** + * + * \param lock a pointer to a lock variable + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AtomicLock + * \sa SDL_AtomicTryLock + */ +extern DECLSPEC void SDLCALL SDL_AtomicUnlock(SDL_SpinLock *lock); + +/* @} *//* SDL AtomicLock */ + + +/** + * The compiler barrier prevents the compiler from reordering + * reads and writes to globally visible variables across the call. + */ +#if defined(_MSC_VER) && (_MSC_VER > 1200) && !defined(__clang__) +void _ReadWriteBarrier(void); +#pragma intrinsic(_ReadWriteBarrier) +#define SDL_CompilerBarrier() _ReadWriteBarrier() +#elif (defined(__GNUC__) && !defined(__EMSCRIPTEN__)) || (defined(__SUNPRO_C) && (__SUNPRO_C >= 0x5120)) +/* This is correct for all CPUs when using GCC or Solaris Studio 12.1+. */ +#define SDL_CompilerBarrier() __asm__ __volatile__ ("" : : : "memory") +#elif defined(__WATCOMC__) +extern __inline void SDL_CompilerBarrier(void); +#pragma aux SDL_CompilerBarrier = "" parm [] modify exact []; +#else +#define SDL_CompilerBarrier() \ +{ SDL_SpinLock _tmp = 0; SDL_AtomicLock(&_tmp); SDL_AtomicUnlock(&_tmp); } +#endif + +/** + * Memory barriers are designed to prevent reads and writes from being + * reordered by the compiler and being seen out of order on multi-core CPUs. + * + * A typical pattern would be for thread A to write some data and a flag, and + * for thread B to read the flag and get the data. In this case you would + * insert a release barrier between writing the data and the flag, + * guaranteeing that the data write completes no later than the flag is + * written, and you would insert an acquire barrier between reading the flag + * and reading the data, to ensure that all the reads associated with the flag + * have completed. + * + * In this pattern you should always see a release barrier paired with an + * acquire barrier and you should gate the data reads/writes with a single + * flag variable. + * + * For more information on these semantics, take a look at the blog post: + * http://preshing.com/20120913/acquire-and-release-semantics + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC void SDLCALL SDL_MemoryBarrierReleaseFunction(void); +extern DECLSPEC void SDLCALL SDL_MemoryBarrierAcquireFunction(void); + +#if defined(__GNUC__) && (defined(__powerpc__) || defined(__ppc__)) +#define SDL_MemoryBarrierRelease() __asm__ __volatile__ ("lwsync" : : : "memory") +#define SDL_MemoryBarrierAcquire() __asm__ __volatile__ ("lwsync" : : : "memory") +#elif defined(__GNUC__) && defined(__aarch64__) +#define SDL_MemoryBarrierRelease() __asm__ __volatile__ ("dmb ish" : : : "memory") +#define SDL_MemoryBarrierAcquire() __asm__ __volatile__ ("dmb ish" : : : "memory") +#elif defined(__GNUC__) && defined(__arm__) +#if 0 /* defined(__LINUX__) || defined(__ANDROID__) */ +/* Information from: + https://chromium.googlesource.com/chromium/chromium/+/trunk/base/atomicops_internals_arm_gcc.h#19 + + The Linux kernel provides a helper function which provides the right code for a memory barrier, + hard-coded at address 0xffff0fa0 +*/ +typedef void (*SDL_KernelMemoryBarrierFunc)(); +#define SDL_MemoryBarrierRelease() ((SDL_KernelMemoryBarrierFunc)0xffff0fa0)() +#define SDL_MemoryBarrierAcquire() ((SDL_KernelMemoryBarrierFunc)0xffff0fa0)() +#elif 0 /* defined(__QNXNTO__) */ +#include + +#define SDL_MemoryBarrierRelease() __cpu_membarrier() +#define SDL_MemoryBarrierAcquire() __cpu_membarrier() +#else +#if defined(__ARM_ARCH_7__) || defined(__ARM_ARCH_7A__) || defined(__ARM_ARCH_7EM__) || defined(__ARM_ARCH_7R__) || defined(__ARM_ARCH_7M__) || defined(__ARM_ARCH_7S__) || defined(__ARM_ARCH_8A__) +#define SDL_MemoryBarrierRelease() __asm__ __volatile__ ("dmb ish" : : : "memory") +#define SDL_MemoryBarrierAcquire() __asm__ __volatile__ ("dmb ish" : : : "memory") +#elif defined(__ARM_ARCH_6__) || defined(__ARM_ARCH_6J__) || defined(__ARM_ARCH_6K__) || defined(__ARM_ARCH_6T2__) || defined(__ARM_ARCH_6Z__) || defined(__ARM_ARCH_6ZK__) || defined(__ARM_ARCH_5TE__) +#ifdef __thumb__ +/* The mcr instruction isn't available in thumb mode, use real functions */ +#define SDL_MEMORY_BARRIER_USES_FUNCTION +#define SDL_MemoryBarrierRelease() SDL_MemoryBarrierReleaseFunction() +#define SDL_MemoryBarrierAcquire() SDL_MemoryBarrierAcquireFunction() +#else +#define SDL_MemoryBarrierRelease() __asm__ __volatile__ ("mcr p15, 0, %0, c7, c10, 5" : : "r"(0) : "memory") +#define SDL_MemoryBarrierAcquire() __asm__ __volatile__ ("mcr p15, 0, %0, c7, c10, 5" : : "r"(0) : "memory") +#endif /* __thumb__ */ +#else +#define SDL_MemoryBarrierRelease() __asm__ __volatile__ ("" : : : "memory") +#define SDL_MemoryBarrierAcquire() __asm__ __volatile__ ("" : : : "memory") +#endif /* __LINUX__ || __ANDROID__ */ +#endif /* __GNUC__ && __arm__ */ +#else +#if (defined(__SUNPRO_C) && (__SUNPRO_C >= 0x5120)) +/* This is correct for all CPUs on Solaris when using Solaris Studio 12.1+. */ +#include +#define SDL_MemoryBarrierRelease() __machine_rel_barrier() +#define SDL_MemoryBarrierAcquire() __machine_acq_barrier() +#else +/* This is correct for the x86 and x64 CPUs, and we'll expand this over time. */ +#define SDL_MemoryBarrierRelease() SDL_CompilerBarrier() +#define SDL_MemoryBarrierAcquire() SDL_CompilerBarrier() +#endif +#endif + +/* "REP NOP" is PAUSE, coded for tools that don't know it by that name. */ +#if (defined(__GNUC__) || defined(__clang__)) && (defined(__i386__) || defined(__x86_64__)) + #define SDL_CPUPauseInstruction() __asm__ __volatile__("pause\n") /* Some assemblers can't do REP NOP, so go with PAUSE. */ +#elif (defined(__arm__) && defined(__ARM_ARCH) && __ARM_ARCH >= 7) || defined(__aarch64__) + #define SDL_CPUPauseInstruction() __asm__ __volatile__("yield" ::: "memory") +#elif (defined(__powerpc__) || defined(__powerpc64__)) + #define SDL_CPUPauseInstruction() __asm__ __volatile__("or 27,27,27"); +#elif defined(_MSC_VER) && (defined(_M_IX86) || defined(_M_X64)) + #define SDL_CPUPauseInstruction() _mm_pause() /* this is actually "rep nop" and not a SIMD instruction. No inline asm in MSVC x86-64! */ +#elif defined(_MSC_VER) && (defined(_M_ARM) || defined(_M_ARM64)) + #define SDL_CPUPauseInstruction() __yield() +#elif defined(__WATCOMC__) && defined(__386__) + extern __inline void SDL_CPUPauseInstruction(void); + #pragma aux SDL_CPUPauseInstruction = ".686p" ".xmm2" "pause" +#else + #define SDL_CPUPauseInstruction() +#endif + + +/** + * \brief A type representing an atomic integer value. It is a struct + * so people don't accidentally use numeric operations on it. + */ +typedef struct { int value; } SDL_atomic_t; + +/** + * Set an atomic variable to a new value if it is currently an old value. + * + * ***Note: If you don't know what this function is for, you shouldn't use + * it!*** + * + * \param a a pointer to an SDL_atomic_t variable to be modified + * \param oldval the old value + * \param newval the new value + * \returns SDL_TRUE if the atomic variable was set, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AtomicCASPtr + * \sa SDL_AtomicGet + * \sa SDL_AtomicSet + */ +extern DECLSPEC SDL_bool SDLCALL SDL_AtomicCAS(SDL_atomic_t *a, int oldval, int newval); + +/** + * Set an atomic variable to a value. + * + * This function also acts as a full memory barrier. + * + * ***Note: If you don't know what this function is for, you shouldn't use + * it!*** + * + * \param a a pointer to an SDL_atomic_t variable to be modified + * \param v the desired value + * \returns the previous value of the atomic variable. + * + * \since This function is available since SDL 2.0.2. + * + * \sa SDL_AtomicGet + */ +extern DECLSPEC int SDLCALL SDL_AtomicSet(SDL_atomic_t *a, int v); + +/** + * Get the value of an atomic variable. + * + * ***Note: If you don't know what this function is for, you shouldn't use + * it!*** + * + * \param a a pointer to an SDL_atomic_t variable + * \returns the current value of an atomic variable. + * + * \since This function is available since SDL 2.0.2. + * + * \sa SDL_AtomicSet + */ +extern DECLSPEC int SDLCALL SDL_AtomicGet(SDL_atomic_t *a); + +/** + * Add to an atomic variable. + * + * This function also acts as a full memory barrier. + * + * ***Note: If you don't know what this function is for, you shouldn't use + * it!*** + * + * \param a a pointer to an SDL_atomic_t variable to be modified + * \param v the desired value to add + * \returns the previous value of the atomic variable. + * + * \since This function is available since SDL 2.0.2. + * + * \sa SDL_AtomicDecRef + * \sa SDL_AtomicIncRef + */ +extern DECLSPEC int SDLCALL SDL_AtomicAdd(SDL_atomic_t *a, int v); + +/** + * \brief Increment an atomic variable used as a reference count. + */ +#ifndef SDL_AtomicIncRef +#define SDL_AtomicIncRef(a) SDL_AtomicAdd(a, 1) +#endif + +/** + * \brief Decrement an atomic variable used as a reference count. + * + * \return SDL_TRUE if the variable reached zero after decrementing, + * SDL_FALSE otherwise + */ +#ifndef SDL_AtomicDecRef +#define SDL_AtomicDecRef(a) (SDL_AtomicAdd(a, -1) == 1) +#endif + +/** + * Set a pointer to a new value if it is currently an old value. + * + * ***Note: If you don't know what this function is for, you shouldn't use + * it!*** + * + * \param a a pointer to a pointer + * \param oldval the old pointer value + * \param newval the new pointer value + * \returns SDL_TRUE if the pointer was set, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AtomicCAS + * \sa SDL_AtomicGetPtr + * \sa SDL_AtomicSetPtr + */ +extern DECLSPEC SDL_bool SDLCALL SDL_AtomicCASPtr(void **a, void *oldval, void *newval); + +/** + * Set a pointer to a value atomically. + * + * ***Note: If you don't know what this function is for, you shouldn't use + * it!*** + * + * \param a a pointer to a pointer + * \param v the desired pointer value + * \returns the previous value of the pointer. + * + * \since This function is available since SDL 2.0.2. + * + * \sa SDL_AtomicCASPtr + * \sa SDL_AtomicGetPtr + */ +extern DECLSPEC void* SDLCALL SDL_AtomicSetPtr(void **a, void* v); + +/** + * Get the value of a pointer atomically. + * + * ***Note: If you don't know what this function is for, you shouldn't use + * it!*** + * + * \param a a pointer to a pointer + * \returns the current value of a pointer. + * + * \since This function is available since SDL 2.0.2. + * + * \sa SDL_AtomicCASPtr + * \sa SDL_AtomicSetPtr + */ +extern DECLSPEC void* SDLCALL SDL_AtomicGetPtr(void **a); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif + +#include "close_code.h" + +#endif /* SDL_atomic_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_audio.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_audio.h new file mode 100644 index 00000000..ccd35982 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_audio.h @@ -0,0 +1,1500 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/* !!! FIXME: several functions in here need Doxygen comments. */ + +/** + * \file SDL_audio.h + * + * Access to the raw audio mixing buffer for the SDL library. + */ + +#ifndef SDL_audio_h_ +#define SDL_audio_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_endian.h" +#include "SDL_mutex.h" +#include "SDL_thread.h" +#include "SDL_rwops.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \brief Audio format flags. + * + * These are what the 16 bits in SDL_AudioFormat currently mean... + * (Unspecified bits are always zero). + * + * \verbatim + ++-----------------------sample is signed if set + || + || ++-----------sample is bigendian if set + || || + || || ++---sample is float if set + || || || + || || || +---sample bit size---+ + || || || | | + 15 14 13 12 11 10 09 08 07 06 05 04 03 02 01 00 + \endverbatim + * + * There are macros in SDL 2.0 and later to query these bits. + */ +typedef Uint16 SDL_AudioFormat; + +/** + * \name Audio flags + */ +/* @{ */ + +#define SDL_AUDIO_MASK_BITSIZE (0xFF) +#define SDL_AUDIO_MASK_DATATYPE (1<<8) +#define SDL_AUDIO_MASK_ENDIAN (1<<12) +#define SDL_AUDIO_MASK_SIGNED (1<<15) +#define SDL_AUDIO_BITSIZE(x) (x & SDL_AUDIO_MASK_BITSIZE) +#define SDL_AUDIO_ISFLOAT(x) (x & SDL_AUDIO_MASK_DATATYPE) +#define SDL_AUDIO_ISBIGENDIAN(x) (x & SDL_AUDIO_MASK_ENDIAN) +#define SDL_AUDIO_ISSIGNED(x) (x & SDL_AUDIO_MASK_SIGNED) +#define SDL_AUDIO_ISINT(x) (!SDL_AUDIO_ISFLOAT(x)) +#define SDL_AUDIO_ISLITTLEENDIAN(x) (!SDL_AUDIO_ISBIGENDIAN(x)) +#define SDL_AUDIO_ISUNSIGNED(x) (!SDL_AUDIO_ISSIGNED(x)) + +/** + * \name Audio format flags + * + * Defaults to LSB byte order. + */ +/* @{ */ +#define AUDIO_U8 0x0008 /**< Unsigned 8-bit samples */ +#define AUDIO_S8 0x8008 /**< Signed 8-bit samples */ +#define AUDIO_U16LSB 0x0010 /**< Unsigned 16-bit samples */ +#define AUDIO_S16LSB 0x8010 /**< Signed 16-bit samples */ +#define AUDIO_U16MSB 0x1010 /**< As above, but big-endian byte order */ +#define AUDIO_S16MSB 0x9010 /**< As above, but big-endian byte order */ +#define AUDIO_U16 AUDIO_U16LSB +#define AUDIO_S16 AUDIO_S16LSB +/* @} */ + +/** + * \name int32 support + */ +/* @{ */ +#define AUDIO_S32LSB 0x8020 /**< 32-bit integer samples */ +#define AUDIO_S32MSB 0x9020 /**< As above, but big-endian byte order */ +#define AUDIO_S32 AUDIO_S32LSB +/* @} */ + +/** + * \name float32 support + */ +/* @{ */ +#define AUDIO_F32LSB 0x8120 /**< 32-bit floating point samples */ +#define AUDIO_F32MSB 0x9120 /**< As above, but big-endian byte order */ +#define AUDIO_F32 AUDIO_F32LSB +/* @} */ + +/** + * \name Native audio byte ordering + */ +/* @{ */ +#if SDL_BYTEORDER == SDL_LIL_ENDIAN +#define AUDIO_U16SYS AUDIO_U16LSB +#define AUDIO_S16SYS AUDIO_S16LSB +#define AUDIO_S32SYS AUDIO_S32LSB +#define AUDIO_F32SYS AUDIO_F32LSB +#else +#define AUDIO_U16SYS AUDIO_U16MSB +#define AUDIO_S16SYS AUDIO_S16MSB +#define AUDIO_S32SYS AUDIO_S32MSB +#define AUDIO_F32SYS AUDIO_F32MSB +#endif +/* @} */ + +/** + * \name Allow change flags + * + * Which audio format changes are allowed when opening a device. + */ +/* @{ */ +#define SDL_AUDIO_ALLOW_FREQUENCY_CHANGE 0x00000001 +#define SDL_AUDIO_ALLOW_FORMAT_CHANGE 0x00000002 +#define SDL_AUDIO_ALLOW_CHANNELS_CHANGE 0x00000004 +#define SDL_AUDIO_ALLOW_SAMPLES_CHANGE 0x00000008 +#define SDL_AUDIO_ALLOW_ANY_CHANGE (SDL_AUDIO_ALLOW_FREQUENCY_CHANGE|SDL_AUDIO_ALLOW_FORMAT_CHANGE|SDL_AUDIO_ALLOW_CHANNELS_CHANGE|SDL_AUDIO_ALLOW_SAMPLES_CHANGE) +/* @} */ + +/* @} *//* Audio flags */ + +/** + * This function is called when the audio device needs more data. + * + * \param userdata An application-specific parameter saved in + * the SDL_AudioSpec structure + * \param stream A pointer to the audio data buffer. + * \param len The length of that buffer in bytes. + * + * Once the callback returns, the buffer will no longer be valid. + * Stereo samples are stored in a LRLRLR ordering. + * + * You can choose to avoid callbacks and use SDL_QueueAudio() instead, if + * you like. Just open your audio device with a NULL callback. + */ +typedef void (SDLCALL * SDL_AudioCallback) (void *userdata, Uint8 * stream, + int len); + +/** + * The calculated values in this structure are calculated by SDL_OpenAudio(). + * + * For multi-channel audio, the default SDL channel mapping is: + * 2: FL FR (stereo) + * 3: FL FR LFE (2.1 surround) + * 4: FL FR BL BR (quad) + * 5: FL FR LFE BL BR (4.1 surround) + * 6: FL FR FC LFE SL SR (5.1 surround - last two can also be BL BR) + * 7: FL FR FC LFE BC SL SR (6.1 surround) + * 8: FL FR FC LFE BL BR SL SR (7.1 surround) + */ +typedef struct SDL_AudioSpec +{ + int freq; /**< DSP frequency -- samples per second */ + SDL_AudioFormat format; /**< Audio data format */ + Uint8 channels; /**< Number of channels: 1 mono, 2 stereo */ + Uint8 silence; /**< Audio buffer silence value (calculated) */ + Uint16 samples; /**< Audio buffer size in sample FRAMES (total samples divided by channel count) */ + Uint16 padding; /**< Necessary for some compile environments */ + Uint32 size; /**< Audio buffer size in bytes (calculated) */ + SDL_AudioCallback callback; /**< Callback that feeds the audio device (NULL to use SDL_QueueAudio()). */ + void *userdata; /**< Userdata passed to callback (ignored for NULL callbacks). */ +} SDL_AudioSpec; + + +struct SDL_AudioCVT; +typedef void (SDLCALL * SDL_AudioFilter) (struct SDL_AudioCVT * cvt, + SDL_AudioFormat format); + +/** + * \brief Upper limit of filters in SDL_AudioCVT + * + * The maximum number of SDL_AudioFilter functions in SDL_AudioCVT is + * currently limited to 9. The SDL_AudioCVT.filters array has 10 pointers, + * one of which is the terminating NULL pointer. + */ +#define SDL_AUDIOCVT_MAX_FILTERS 9 + +/** + * \struct SDL_AudioCVT + * \brief A structure to hold a set of audio conversion filters and buffers. + * + * Note that various parts of the conversion pipeline can take advantage + * of SIMD operations (like SSE2, for example). SDL_AudioCVT doesn't require + * you to pass it aligned data, but can possibly run much faster if you + * set both its (buf) field to a pointer that is aligned to 16 bytes, and its + * (len) field to something that's a multiple of 16, if possible. + */ +#if defined(__GNUC__) && !defined(__CHERI_PURE_CAPABILITY__) +/* This structure is 84 bytes on 32-bit architectures, make sure GCC doesn't + pad it out to 88 bytes to guarantee ABI compatibility between compilers. + This is not a concern on CHERI architectures, where pointers must be stored + at aligned locations otherwise they will become invalid, and thus structs + containing pointers cannot be packed without giving a warning or error. + vvv + The next time we rev the ABI, make sure to size the ints and add padding. +*/ +#define SDL_AUDIOCVT_PACKED __attribute__((packed)) +#else +#define SDL_AUDIOCVT_PACKED +#endif +/* */ +typedef struct SDL_AudioCVT +{ + int needed; /**< Set to 1 if conversion possible */ + SDL_AudioFormat src_format; /**< Source audio format */ + SDL_AudioFormat dst_format; /**< Target audio format */ + double rate_incr; /**< Rate conversion increment */ + Uint8 *buf; /**< Buffer to hold entire audio data */ + int len; /**< Length of original audio buffer */ + int len_cvt; /**< Length of converted audio buffer */ + int len_mult; /**< buffer must be len*len_mult big */ + double len_ratio; /**< Given len, final size is len*len_ratio */ + SDL_AudioFilter filters[SDL_AUDIOCVT_MAX_FILTERS + 1]; /**< NULL-terminated list of filter functions */ + int filter_index; /**< Current audio conversion function */ +} SDL_AUDIOCVT_PACKED SDL_AudioCVT; + + +/* Function prototypes */ + +/** + * \name Driver discovery functions + * + * These functions return the list of built in audio drivers, in the + * order that they are normally initialized by default. + */ +/* @{ */ + +/** + * Use this function to get the number of built-in audio drivers. + * + * This function returns a hardcoded number. This never returns a negative + * value; if there are no drivers compiled into this build of SDL, this + * function returns zero. The presence of a driver in this list does not mean + * it will function, it just means SDL is capable of interacting with that + * interface. For example, a build of SDL might have esound support, but if + * there's no esound server available, SDL's esound driver would fail if used. + * + * By default, SDL tries all drivers, in its preferred order, until one is + * found to be usable. + * + * \returns the number of built-in audio drivers. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetAudioDriver + */ +extern DECLSPEC int SDLCALL SDL_GetNumAudioDrivers(void); + +/** + * Use this function to get the name of a built in audio driver. + * + * The list of audio drivers is given in the order that they are normally + * initialized by default; the drivers that seem more reasonable to choose + * first (as far as the SDL developers believe) are earlier in the list. + * + * The names of drivers are all simple, low-ASCII identifiers, like "alsa", + * "coreaudio" or "xaudio2". These never have Unicode characters, and are not + * meant to be proper names. + * + * \param index the index of the audio driver; the value ranges from 0 to + * SDL_GetNumAudioDrivers() - 1 + * \returns the name of the audio driver at the requested index, or NULL if an + * invalid index was specified. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetNumAudioDrivers + */ +extern DECLSPEC const char *SDLCALL SDL_GetAudioDriver(int index); +/* @} */ + +/** + * \name Initialization and cleanup + * + * \internal These functions are used internally, and should not be used unless + * you have a specific need to specify the audio driver you want to + * use. You should normally use SDL_Init() or SDL_InitSubSystem(). + */ +/* @{ */ + +/** + * Use this function to initialize a particular audio driver. + * + * This function is used internally, and should not be used unless you have a + * specific need to designate the audio driver you want to use. You should + * normally use SDL_Init() or SDL_InitSubSystem(). + * + * \param driver_name the name of the desired audio driver + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AudioQuit + */ +extern DECLSPEC int SDLCALL SDL_AudioInit(const char *driver_name); + +/** + * Use this function to shut down audio if you initialized it with + * SDL_AudioInit(). + * + * This function is used internally, and should not be used unless you have a + * specific need to specify the audio driver you want to use. You should + * normally use SDL_Quit() or SDL_QuitSubSystem(). + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AudioInit + */ +extern DECLSPEC void SDLCALL SDL_AudioQuit(void); +/* @} */ + +/** + * Get the name of the current audio driver. + * + * The returned string points to internal static memory and thus never becomes + * invalid, even if you quit the audio subsystem and initialize a new driver + * (although such a case would return a different static string from another + * call to this function, of course). As such, you should not modify or free + * the returned string. + * + * \returns the name of the current audio driver or NULL if no driver has been + * initialized. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AudioInit + */ +extern DECLSPEC const char *SDLCALL SDL_GetCurrentAudioDriver(void); + +/** + * This function is a legacy means of opening the audio device. + * + * This function remains for compatibility with SDL 1.2, but also because it's + * slightly easier to use than the new functions in SDL 2.0. The new, more + * powerful, and preferred way to do this is SDL_OpenAudioDevice(). + * + * This function is roughly equivalent to: + * + * ```c + * SDL_OpenAudioDevice(NULL, 0, desired, obtained, SDL_AUDIO_ALLOW_ANY_CHANGE); + * ``` + * + * With two notable exceptions: + * + * - If `obtained` is NULL, we use `desired` (and allow no changes), which + * means desired will be modified to have the correct values for silence, + * etc, and SDL will convert any differences between your app's specific + * request and the hardware behind the scenes. + * - The return value is always success or failure, and not a device ID, which + * means you can only have one device open at a time with this function. + * + * \param desired an SDL_AudioSpec structure representing the desired output + * format. Please refer to the SDL_OpenAudioDevice + * documentation for details on how to prepare this structure. + * \param obtained an SDL_AudioSpec structure filled in with the actual + * parameters, or NULL. + * \returns 0 if successful, placing the actual hardware parameters in the + * structure pointed to by `obtained`. + * + * If `obtained` is NULL, the audio data passed to the callback + * function will be guaranteed to be in the requested format, and + * will be automatically converted to the actual hardware audio + * format if necessary. If `obtained` is NULL, `desired` will have + * fields modified. + * + * This function returns a negative error code on failure to open the + * audio device or failure to set up the audio thread; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CloseAudio + * \sa SDL_LockAudio + * \sa SDL_PauseAudio + * \sa SDL_UnlockAudio + */ +extern DECLSPEC int SDLCALL SDL_OpenAudio(SDL_AudioSpec * desired, + SDL_AudioSpec * obtained); + +/** + * SDL Audio Device IDs. + * + * A successful call to SDL_OpenAudio() is always device id 1, and legacy + * SDL audio APIs assume you want this device ID. SDL_OpenAudioDevice() calls + * always returns devices >= 2 on success. The legacy calls are good both + * for backwards compatibility and when you don't care about multiple, + * specific, or capture devices. + */ +typedef Uint32 SDL_AudioDeviceID; + +/** + * Get the number of built-in audio devices. + * + * This function is only valid after successfully initializing the audio + * subsystem. + * + * Note that audio capture support is not implemented as of SDL 2.0.4, so the + * `iscapture` parameter is for future expansion and should always be zero for + * now. + * + * This function will return -1 if an explicit list of devices can't be + * determined. Returning -1 is not an error. For example, if SDL is set up to + * talk to a remote audio server, it can't list every one available on the + * Internet, but it will still allow a specific host to be specified in + * SDL_OpenAudioDevice(). + * + * In many common cases, when this function returns a value <= 0, it can still + * successfully open the default device (NULL for first argument of + * SDL_OpenAudioDevice()). + * + * This function may trigger a complete redetect of available hardware. It + * should not be called for each iteration of a loop, but rather once at the + * start of a loop: + * + * ```c + * // Don't do this: + * for (int i = 0; i < SDL_GetNumAudioDevices(0); i++) + * + * // do this instead: + * const int count = SDL_GetNumAudioDevices(0); + * for (int i = 0; i < count; ++i) { do_something_here(); } + * ``` + * + * \param iscapture zero to request playback devices, non-zero to request + * recording devices + * \returns the number of available devices exposed by the current driver or + * -1 if an explicit list of devices can't be determined. A return + * value of -1 does not necessarily mean an error condition. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetAudioDeviceName + * \sa SDL_OpenAudioDevice + */ +extern DECLSPEC int SDLCALL SDL_GetNumAudioDevices(int iscapture); + +/** + * Get the human-readable name of a specific audio device. + * + * This function is only valid after successfully initializing the audio + * subsystem. The values returned by this function reflect the latest call to + * SDL_GetNumAudioDevices(); re-call that function to redetect available + * hardware. + * + * The string returned by this function is UTF-8 encoded, read-only, and + * managed internally. You are not to free it. If you need to keep the string + * for any length of time, you should make your own copy of it, as it will be + * invalid next time any of several other SDL functions are called. + * + * \param index the index of the audio device; valid values range from 0 to + * SDL_GetNumAudioDevices() - 1 + * \param iscapture non-zero to query the list of recording devices, zero to + * query the list of output devices. + * \returns the name of the audio device at the requested index, or NULL on + * error. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetNumAudioDevices + * \sa SDL_GetDefaultAudioInfo + */ +extern DECLSPEC const char *SDLCALL SDL_GetAudioDeviceName(int index, + int iscapture); + +/** + * Get the preferred audio format of a specific audio device. + * + * This function is only valid after a successfully initializing the audio + * subsystem. The values returned by this function reflect the latest call to + * SDL_GetNumAudioDevices(); re-call that function to redetect available + * hardware. + * + * `spec` will be filled with the sample rate, sample format, and channel + * count. + * + * \param index the index of the audio device; valid values range from 0 to + * SDL_GetNumAudioDevices() - 1 + * \param iscapture non-zero to query the list of recording devices, zero to + * query the list of output devices. + * \param spec The SDL_AudioSpec to be initialized by this function. + * \returns 0 on success, nonzero on error + * + * \since This function is available since SDL 2.0.16. + * + * \sa SDL_GetNumAudioDevices + * \sa SDL_GetDefaultAudioInfo + */ +extern DECLSPEC int SDLCALL SDL_GetAudioDeviceSpec(int index, + int iscapture, + SDL_AudioSpec *spec); + + +/** + * Get the name and preferred format of the default audio device. + * + * Some (but not all!) platforms have an isolated mechanism to get information + * about the "default" device. This can actually be a completely different + * device that's not in the list you get from SDL_GetAudioDeviceSpec(). It can + * even be a network address! (This is discussed in SDL_OpenAudioDevice().) + * + * As a result, this call is not guaranteed to be performant, as it can query + * the sound server directly every time, unlike the other query functions. You + * should call this function sparingly! + * + * `spec` will be filled with the sample rate, sample format, and channel + * count, if a default device exists on the system. If `name` is provided, + * will be filled with either a dynamically-allocated UTF-8 string or NULL. + * + * \param name A pointer to be filled with the name of the default device (can + * be NULL). Please call SDL_free() when you are done with this + * pointer! + * \param spec The SDL_AudioSpec to be initialized by this function. + * \param iscapture non-zero to query the default recording device, zero to + * query the default output device. + * \returns 0 on success, nonzero on error + * + * \since This function is available since SDL 2.24.0. + * + * \sa SDL_GetAudioDeviceName + * \sa SDL_GetAudioDeviceSpec + * \sa SDL_OpenAudioDevice + */ +extern DECLSPEC int SDLCALL SDL_GetDefaultAudioInfo(char **name, + SDL_AudioSpec *spec, + int iscapture); + + +/** + * Open a specific audio device. + * + * SDL_OpenAudio(), unlike this function, always acts on device ID 1. As such, + * this function will never return a 1 so as not to conflict with the legacy + * function. + * + * Please note that SDL 2.0 before 2.0.5 did not support recording; as such, + * this function would fail if `iscapture` was not zero. Starting with SDL + * 2.0.5, recording is implemented and this value can be non-zero. + * + * Passing in a `device` name of NULL requests the most reasonable default + * (and is equivalent to what SDL_OpenAudio() does to choose a device). The + * `device` name is a UTF-8 string reported by SDL_GetAudioDeviceName(), but + * some drivers allow arbitrary and driver-specific strings, such as a + * hostname/IP address for a remote audio server, or a filename in the + * diskaudio driver. + * + * An opened audio device starts out paused, and should be enabled for playing + * by calling SDL_PauseAudioDevice(devid, 0) when you are ready for your audio + * callback function to be called. Since the audio driver may modify the + * requested size of the audio buffer, you should allocate any local mixing + * buffers after you open the audio device. + * + * The audio callback runs in a separate thread in most cases; you can prevent + * race conditions between your callback and other threads without fully + * pausing playback with SDL_LockAudioDevice(). For more information about the + * callback, see SDL_AudioSpec. + * + * Managing the audio spec via 'desired' and 'obtained': + * + * When filling in the desired audio spec structure: + * + * - `desired->freq` should be the frequency in sample-frames-per-second (Hz). + * - `desired->format` should be the audio format (`AUDIO_S16SYS`, etc). + * - `desired->samples` is the desired size of the audio buffer, in _sample + * frames_ (with stereo output, two samples--left and right--would make a + * single sample frame). This number should be a power of two, and may be + * adjusted by the audio driver to a value more suitable for the hardware. + * Good values seem to range between 512 and 8096 inclusive, depending on + * the application and CPU speed. Smaller values reduce latency, but can + * lead to underflow if the application is doing heavy processing and cannot + * fill the audio buffer in time. Note that the number of sample frames is + * directly related to time by the following formula: `ms = + * (sampleframes*1000)/freq` + * - `desired->size` is the size in _bytes_ of the audio buffer, and is + * calculated by SDL_OpenAudioDevice(). You don't initialize this. + * - `desired->silence` is the value used to set the buffer to silence, and is + * calculated by SDL_OpenAudioDevice(). You don't initialize this. + * - `desired->callback` should be set to a function that will be called when + * the audio device is ready for more data. It is passed a pointer to the + * audio buffer, and the length in bytes of the audio buffer. This function + * usually runs in a separate thread, and so you should protect data + * structures that it accesses by calling SDL_LockAudioDevice() and + * SDL_UnlockAudioDevice() in your code. Alternately, you may pass a NULL + * pointer here, and call SDL_QueueAudio() with some frequency, to queue + * more audio samples to be played (or for capture devices, call + * SDL_DequeueAudio() with some frequency, to obtain audio samples). + * - `desired->userdata` is passed as the first parameter to your callback + * function. If you passed a NULL callback, this value is ignored. + * + * `allowed_changes` can have the following flags OR'd together: + * + * - `SDL_AUDIO_ALLOW_FREQUENCY_CHANGE` + * - `SDL_AUDIO_ALLOW_FORMAT_CHANGE` + * - `SDL_AUDIO_ALLOW_CHANNELS_CHANGE` + * - `SDL_AUDIO_ALLOW_SAMPLES_CHANGE` + * - `SDL_AUDIO_ALLOW_ANY_CHANGE` + * + * These flags specify how SDL should behave when a device cannot offer a + * specific feature. If the application requests a feature that the hardware + * doesn't offer, SDL will always try to get the closest equivalent. + * + * For example, if you ask for float32 audio format, but the sound card only + * supports int16, SDL will set the hardware to int16. If you had set + * SDL_AUDIO_ALLOW_FORMAT_CHANGE, SDL will change the format in the `obtained` + * structure. If that flag was *not* set, SDL will prepare to convert your + * callback's float32 audio to int16 before feeding it to the hardware and + * will keep the originally requested format in the `obtained` structure. + * + * The resulting audio specs, varying depending on hardware and on what + * changes were allowed, will then be written back to `obtained`. + * + * If your application can only handle one specific data format, pass a zero + * for `allowed_changes` and let SDL transparently handle any differences. + * + * \param device a UTF-8 string reported by SDL_GetAudioDeviceName() or a + * driver-specific name as appropriate. NULL requests the most + * reasonable default device. + * \param iscapture non-zero to specify a device should be opened for + * recording, not playback + * \param desired an SDL_AudioSpec structure representing the desired output + * format; see SDL_OpenAudio() for more information + * \param obtained an SDL_AudioSpec structure filled in with the actual output + * format; see SDL_OpenAudio() for more information + * \param allowed_changes 0, or one or more flags OR'd together + * \returns a valid device ID that is > 0 on success or 0 on failure; call + * SDL_GetError() for more information. + * + * For compatibility with SDL 1.2, this will never return 1, since + * SDL reserves that ID for the legacy SDL_OpenAudio() function. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CloseAudioDevice + * \sa SDL_GetAudioDeviceName + * \sa SDL_LockAudioDevice + * \sa SDL_OpenAudio + * \sa SDL_PauseAudioDevice + * \sa SDL_UnlockAudioDevice + */ +extern DECLSPEC SDL_AudioDeviceID SDLCALL SDL_OpenAudioDevice( + const char *device, + int iscapture, + const SDL_AudioSpec *desired, + SDL_AudioSpec *obtained, + int allowed_changes); + + + +/** + * \name Audio state + * + * Get the current audio state. + */ +/* @{ */ +typedef enum +{ + SDL_AUDIO_STOPPED = 0, + SDL_AUDIO_PLAYING, + SDL_AUDIO_PAUSED +} SDL_AudioStatus; + +/** + * This function is a legacy means of querying the audio device. + * + * New programs might want to use SDL_GetAudioDeviceStatus() instead. This + * function is equivalent to calling... + * + * ```c + * SDL_GetAudioDeviceStatus(1); + * ``` + * + * ...and is only useful if you used the legacy SDL_OpenAudio() function. + * + * \returns the SDL_AudioStatus of the audio device opened by SDL_OpenAudio(). + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetAudioDeviceStatus + */ +extern DECLSPEC SDL_AudioStatus SDLCALL SDL_GetAudioStatus(void); + +/** + * Use this function to get the current audio state of an audio device. + * + * \param dev the ID of an audio device previously opened with + * SDL_OpenAudioDevice() + * \returns the SDL_AudioStatus of the specified audio device. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_PauseAudioDevice + */ +extern DECLSPEC SDL_AudioStatus SDLCALL SDL_GetAudioDeviceStatus(SDL_AudioDeviceID dev); +/* @} *//* Audio State */ + +/** + * \name Pause audio functions + * + * These functions pause and unpause the audio callback processing. + * They should be called with a parameter of 0 after opening the audio + * device to start playing sound. This is so you can safely initialize + * data for your callback function after opening the audio device. + * Silence will be written to the audio device during the pause. + */ +/* @{ */ + +/** + * This function is a legacy means of pausing the audio device. + * + * New programs might want to use SDL_PauseAudioDevice() instead. This + * function is equivalent to calling... + * + * ```c + * SDL_PauseAudioDevice(1, pause_on); + * ``` + * + * ...and is only useful if you used the legacy SDL_OpenAudio() function. + * + * \param pause_on non-zero to pause, 0 to unpause + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetAudioStatus + * \sa SDL_PauseAudioDevice + */ +extern DECLSPEC void SDLCALL SDL_PauseAudio(int pause_on); + +/** + * Use this function to pause and unpause audio playback on a specified + * device. + * + * This function pauses and unpauses the audio callback processing for a given + * device. Newly-opened audio devices start in the paused state, so you must + * call this function with **pause_on**=0 after opening the specified audio + * device to start playing sound. This allows you to safely initialize data + * for your callback function after opening the audio device. Silence will be + * written to the audio device while paused, and the audio callback is + * guaranteed to not be called. Pausing one device does not prevent other + * unpaused devices from running their callbacks. + * + * Pausing state does not stack; even if you pause a device several times, a + * single unpause will start the device playing again, and vice versa. This is + * different from how SDL_LockAudioDevice() works. + * + * If you just need to protect a few variables from race conditions vs your + * callback, you shouldn't pause the audio device, as it will lead to dropouts + * in the audio playback. Instead, you should use SDL_LockAudioDevice(). + * + * \param dev a device opened by SDL_OpenAudioDevice() + * \param pause_on non-zero to pause, 0 to unpause + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LockAudioDevice + */ +extern DECLSPEC void SDLCALL SDL_PauseAudioDevice(SDL_AudioDeviceID dev, + int pause_on); +/* @} *//* Pause audio functions */ + +/** + * Load the audio data of a WAVE file into memory. + * + * Loading a WAVE file requires `src`, `spec`, `audio_buf` and `audio_len` to + * be valid pointers. The entire data portion of the file is then loaded into + * memory and decoded if necessary. + * + * If `freesrc` is non-zero, the data source gets automatically closed and + * freed before the function returns. + * + * Supported formats are RIFF WAVE files with the formats PCM (8, 16, 24, and + * 32 bits), IEEE Float (32 bits), Microsoft ADPCM and IMA ADPCM (4 bits), and + * A-law and mu-law (8 bits). Other formats are currently unsupported and + * cause an error. + * + * If this function succeeds, the pointer returned by it is equal to `spec` + * and the pointer to the audio data allocated by the function is written to + * `audio_buf` and its length in bytes to `audio_len`. The SDL_AudioSpec + * members `freq`, `channels`, and `format` are set to the values of the audio + * data in the buffer. The `samples` member is set to a sane default and all + * others are set to zero. + * + * It's necessary to use SDL_FreeWAV() to free the audio data returned in + * `audio_buf` when it is no longer used. + * + * Because of the underspecification of the .WAV format, there are many + * problematic files in the wild that cause issues with strict decoders. To + * provide compatibility with these files, this decoder is lenient in regards + * to the truncation of the file, the fact chunk, and the size of the RIFF + * chunk. The hints `SDL_HINT_WAVE_RIFF_CHUNK_SIZE`, + * `SDL_HINT_WAVE_TRUNCATION`, and `SDL_HINT_WAVE_FACT_CHUNK` can be used to + * tune the behavior of the loading process. + * + * Any file that is invalid (due to truncation, corruption, or wrong values in + * the headers), too big, or unsupported causes an error. Additionally, any + * critical I/O error from the data source will terminate the loading process + * with an error. The function returns NULL on error and in all cases (with + * the exception of `src` being NULL), an appropriate error message will be + * set. + * + * It is required that the data source supports seeking. + * + * Example: + * + * ```c + * SDL_LoadWAV_RW(SDL_RWFromFile("sample.wav", "rb"), 1, &spec, &buf, &len); + * ``` + * + * Note that the SDL_LoadWAV macro does this same thing for you, but in a less + * messy way: + * + * ```c + * SDL_LoadWAV("sample.wav", &spec, &buf, &len); + * ``` + * + * \param src The data source for the WAVE data + * \param freesrc If non-zero, SDL will _always_ free the data source + * \param spec An SDL_AudioSpec that will be filled in with the wave file's + * format details + * \param audio_buf A pointer filled with the audio data, allocated by the + * function. + * \param audio_len A pointer filled with the length of the audio data buffer + * in bytes + * \returns This function, if successfully called, returns `spec`, which will + * be filled with the audio data format of the wave source data. + * `audio_buf` will be filled with a pointer to an allocated buffer + * containing the audio data, and `audio_len` is filled with the + * length of that audio buffer in bytes. + * + * This function returns NULL if the .WAV file cannot be opened, uses + * an unknown data format, or is corrupt; call SDL_GetError() for + * more information. + * + * When the application is done with the data returned in + * `audio_buf`, it should call SDL_FreeWAV() to dispose of it. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_FreeWAV + * \sa SDL_LoadWAV + */ +extern DECLSPEC SDL_AudioSpec *SDLCALL SDL_LoadWAV_RW(SDL_RWops * src, + int freesrc, + SDL_AudioSpec * spec, + Uint8 ** audio_buf, + Uint32 * audio_len); + +/** + * Loads a WAV from a file. + * Compatibility convenience function. + */ +#define SDL_LoadWAV(file, spec, audio_buf, audio_len) \ + SDL_LoadWAV_RW(SDL_RWFromFile(file, "rb"),1, spec,audio_buf,audio_len) + +/** + * Free data previously allocated with SDL_LoadWAV() or SDL_LoadWAV_RW(). + * + * After a WAVE file has been opened with SDL_LoadWAV() or SDL_LoadWAV_RW() + * its data can eventually be freed with SDL_FreeWAV(). It is safe to call + * this function with a NULL pointer. + * + * \param audio_buf a pointer to the buffer created by SDL_LoadWAV() or + * SDL_LoadWAV_RW() + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LoadWAV + * \sa SDL_LoadWAV_RW + */ +extern DECLSPEC void SDLCALL SDL_FreeWAV(Uint8 * audio_buf); + +/** + * Initialize an SDL_AudioCVT structure for conversion. + * + * Before an SDL_AudioCVT structure can be used to convert audio data it must + * be initialized with source and destination information. + * + * This function will zero out every field of the SDL_AudioCVT, so it must be + * called before the application fills in the final buffer information. + * + * Once this function has returned successfully, and reported that a + * conversion is necessary, the application fills in the rest of the fields in + * SDL_AudioCVT, now that it knows how large a buffer it needs to allocate, + * and then can call SDL_ConvertAudio() to complete the conversion. + * + * \param cvt an SDL_AudioCVT structure filled in with audio conversion + * information + * \param src_format the source format of the audio data; for more info see + * SDL_AudioFormat + * \param src_channels the number of channels in the source + * \param src_rate the frequency (sample-frames-per-second) of the source + * \param dst_format the destination format of the audio data; for more info + * see SDL_AudioFormat + * \param dst_channels the number of channels in the destination + * \param dst_rate the frequency (sample-frames-per-second) of the destination + * \returns 1 if the audio filter is prepared, 0 if no conversion is needed, + * or a negative error code on failure; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ConvertAudio + */ +extern DECLSPEC int SDLCALL SDL_BuildAudioCVT(SDL_AudioCVT * cvt, + SDL_AudioFormat src_format, + Uint8 src_channels, + int src_rate, + SDL_AudioFormat dst_format, + Uint8 dst_channels, + int dst_rate); + +/** + * Convert audio data to a desired audio format. + * + * This function does the actual audio data conversion, after the application + * has called SDL_BuildAudioCVT() to prepare the conversion information and + * then filled in the buffer details. + * + * Once the application has initialized the `cvt` structure using + * SDL_BuildAudioCVT(), allocated an audio buffer and filled it with audio + * data in the source format, this function will convert the buffer, in-place, + * to the desired format. + * + * The data conversion may go through several passes; any given pass may + * possibly temporarily increase the size of the data. For example, SDL might + * expand 16-bit data to 32 bits before resampling to a lower frequency, + * shrinking the data size after having grown it briefly. Since the supplied + * buffer will be both the source and destination, converting as necessary + * in-place, the application must allocate a buffer that will fully contain + * the data during its largest conversion pass. After SDL_BuildAudioCVT() + * returns, the application should set the `cvt->len` field to the size, in + * bytes, of the source data, and allocate a buffer that is `cvt->len * + * cvt->len_mult` bytes long for the `buf` field. + * + * The source data should be copied into this buffer before the call to + * SDL_ConvertAudio(). Upon successful return, this buffer will contain the + * converted audio, and `cvt->len_cvt` will be the size of the converted data, + * in bytes. Any bytes in the buffer past `cvt->len_cvt` are undefined once + * this function returns. + * + * \param cvt an SDL_AudioCVT structure that was previously set up by + * SDL_BuildAudioCVT(). + * \returns 0 if the conversion was completed successfully or a negative error + * code on failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_BuildAudioCVT + */ +extern DECLSPEC int SDLCALL SDL_ConvertAudio(SDL_AudioCVT * cvt); + +/* SDL_AudioStream is a new audio conversion interface. + The benefits vs SDL_AudioCVT: + - it can handle resampling data in chunks without generating + artifacts, when it doesn't have the complete buffer available. + - it can handle incoming data in any variable size. + - You push data as you have it, and pull it when you need it + */ +/* this is opaque to the outside world. */ +struct _SDL_AudioStream; +typedef struct _SDL_AudioStream SDL_AudioStream; + +/** + * Create a new audio stream. + * + * \param src_format The format of the source audio + * \param src_channels The number of channels of the source audio + * \param src_rate The sampling rate of the source audio + * \param dst_format The format of the desired audio output + * \param dst_channels The number of channels of the desired audio output + * \param dst_rate The sampling rate of the desired audio output + * \returns 0 on success, or -1 on error. + * + * \since This function is available since SDL 2.0.7. + * + * \sa SDL_AudioStreamPut + * \sa SDL_AudioStreamGet + * \sa SDL_AudioStreamAvailable + * \sa SDL_AudioStreamFlush + * \sa SDL_AudioStreamClear + * \sa SDL_FreeAudioStream + */ +extern DECLSPEC SDL_AudioStream * SDLCALL SDL_NewAudioStream(const SDL_AudioFormat src_format, + const Uint8 src_channels, + const int src_rate, + const SDL_AudioFormat dst_format, + const Uint8 dst_channels, + const int dst_rate); + +/** + * Add data to be converted/resampled to the stream. + * + * \param stream The stream the audio data is being added to + * \param buf A pointer to the audio data to add + * \param len The number of bytes to write to the stream + * \returns 0 on success, or -1 on error. + * + * \since This function is available since SDL 2.0.7. + * + * \sa SDL_NewAudioStream + * \sa SDL_AudioStreamGet + * \sa SDL_AudioStreamAvailable + * \sa SDL_AudioStreamFlush + * \sa SDL_AudioStreamClear + * \sa SDL_FreeAudioStream + */ +extern DECLSPEC int SDLCALL SDL_AudioStreamPut(SDL_AudioStream *stream, const void *buf, int len); + +/** + * Get converted/resampled data from the stream + * + * \param stream The stream the audio is being requested from + * \param buf A buffer to fill with audio data + * \param len The maximum number of bytes to fill + * \returns the number of bytes read from the stream, or -1 on error + * + * \since This function is available since SDL 2.0.7. + * + * \sa SDL_NewAudioStream + * \sa SDL_AudioStreamPut + * \sa SDL_AudioStreamAvailable + * \sa SDL_AudioStreamFlush + * \sa SDL_AudioStreamClear + * \sa SDL_FreeAudioStream + */ +extern DECLSPEC int SDLCALL SDL_AudioStreamGet(SDL_AudioStream *stream, void *buf, int len); + +/** + * Get the number of converted/resampled bytes available. + * + * The stream may be buffering data behind the scenes until it has enough to + * resample correctly, so this number might be lower than what you expect, or + * even be zero. Add more data or flush the stream if you need the data now. + * + * \since This function is available since SDL 2.0.7. + * + * \sa SDL_NewAudioStream + * \sa SDL_AudioStreamPut + * \sa SDL_AudioStreamGet + * \sa SDL_AudioStreamFlush + * \sa SDL_AudioStreamClear + * \sa SDL_FreeAudioStream + */ +extern DECLSPEC int SDLCALL SDL_AudioStreamAvailable(SDL_AudioStream *stream); + +/** + * Tell the stream that you're done sending data, and anything being buffered + * should be converted/resampled and made available immediately. + * + * It is legal to add more data to a stream after flushing, but there will be + * audio gaps in the output. Generally this is intended to signal the end of + * input, so the complete output becomes available. + * + * \since This function is available since SDL 2.0.7. + * + * \sa SDL_NewAudioStream + * \sa SDL_AudioStreamPut + * \sa SDL_AudioStreamGet + * \sa SDL_AudioStreamAvailable + * \sa SDL_AudioStreamClear + * \sa SDL_FreeAudioStream + */ +extern DECLSPEC int SDLCALL SDL_AudioStreamFlush(SDL_AudioStream *stream); + +/** + * Clear any pending data in the stream without converting it + * + * \since This function is available since SDL 2.0.7. + * + * \sa SDL_NewAudioStream + * \sa SDL_AudioStreamPut + * \sa SDL_AudioStreamGet + * \sa SDL_AudioStreamAvailable + * \sa SDL_AudioStreamFlush + * \sa SDL_FreeAudioStream + */ +extern DECLSPEC void SDLCALL SDL_AudioStreamClear(SDL_AudioStream *stream); + +/** + * Free an audio stream + * + * \since This function is available since SDL 2.0.7. + * + * \sa SDL_NewAudioStream + * \sa SDL_AudioStreamPut + * \sa SDL_AudioStreamGet + * \sa SDL_AudioStreamAvailable + * \sa SDL_AudioStreamFlush + * \sa SDL_AudioStreamClear + */ +extern DECLSPEC void SDLCALL SDL_FreeAudioStream(SDL_AudioStream *stream); + +#define SDL_MIX_MAXVOLUME 128 + +/** + * This function is a legacy means of mixing audio. + * + * This function is equivalent to calling... + * + * ```c + * SDL_MixAudioFormat(dst, src, format, len, volume); + * ``` + * + * ...where `format` is the obtained format of the audio device from the + * legacy SDL_OpenAudio() function. + * + * \param dst the destination for the mixed audio + * \param src the source audio buffer to be mixed + * \param len the length of the audio buffer in bytes + * \param volume ranges from 0 - 128, and should be set to SDL_MIX_MAXVOLUME + * for full audio volume + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_MixAudioFormat + */ +extern DECLSPEC void SDLCALL SDL_MixAudio(Uint8 * dst, const Uint8 * src, + Uint32 len, int volume); + +/** + * Mix audio data in a specified format. + * + * This takes an audio buffer `src` of `len` bytes of `format` data and mixes + * it into `dst`, performing addition, volume adjustment, and overflow + * clipping. The buffer pointed to by `dst` must also be `len` bytes of + * `format` data. + * + * This is provided for convenience -- you can mix your own audio data. + * + * Do not use this function for mixing together more than two streams of + * sample data. The output from repeated application of this function may be + * distorted by clipping, because there is no accumulator with greater range + * than the input (not to mention this being an inefficient way of doing it). + * + * It is a common misconception that this function is required to write audio + * data to an output stream in an audio callback. While you can do that, + * SDL_MixAudioFormat() is really only needed when you're mixing a single + * audio stream with a volume adjustment. + * + * \param dst the destination for the mixed audio + * \param src the source audio buffer to be mixed + * \param format the SDL_AudioFormat structure representing the desired audio + * format + * \param len the length of the audio buffer in bytes + * \param volume ranges from 0 - 128, and should be set to SDL_MIX_MAXVOLUME + * for full audio volume + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC void SDLCALL SDL_MixAudioFormat(Uint8 * dst, + const Uint8 * src, + SDL_AudioFormat format, + Uint32 len, int volume); + +/** + * Queue more audio on non-callback devices. + * + * If you are looking to retrieve queued audio from a non-callback capture + * device, you want SDL_DequeueAudio() instead. SDL_QueueAudio() will return + * -1 to signify an error if you use it with capture devices. + * + * SDL offers two ways to feed audio to the device: you can either supply a + * callback that SDL triggers with some frequency to obtain more audio (pull + * method), or you can supply no callback, and then SDL will expect you to + * supply data at regular intervals (push method) with this function. + * + * There are no limits on the amount of data you can queue, short of + * exhaustion of address space. Queued data will drain to the device as + * necessary without further intervention from you. If the device needs audio + * but there is not enough queued, it will play silence to make up the + * difference. This means you will have skips in your audio playback if you + * aren't routinely queueing sufficient data. + * + * This function copies the supplied data, so you are safe to free it when the + * function returns. This function is thread-safe, but queueing to the same + * device from two threads at once does not promise which buffer will be + * queued first. + * + * You may not queue audio on a device that is using an application-supplied + * callback; doing so returns an error. You have to use the audio callback or + * queue audio with this function, but not both. + * + * You should not call SDL_LockAudio() on the device before queueing; SDL + * handles locking internally for this function. + * + * Note that SDL2 does not support planar audio. You will need to resample + * from planar audio formats into a non-planar one (see SDL_AudioFormat) + * before queuing audio. + * + * \param dev the device ID to which we will queue audio + * \param data the data to queue to the device for later playback + * \param len the number of bytes (not samples!) to which `data` points + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.4. + * + * \sa SDL_ClearQueuedAudio + * \sa SDL_GetQueuedAudioSize + */ +extern DECLSPEC int SDLCALL SDL_QueueAudio(SDL_AudioDeviceID dev, const void *data, Uint32 len); + +/** + * Dequeue more audio on non-callback devices. + * + * If you are looking to queue audio for output on a non-callback playback + * device, you want SDL_QueueAudio() instead. SDL_DequeueAudio() will always + * return 0 if you use it with playback devices. + * + * SDL offers two ways to retrieve audio from a capture device: you can either + * supply a callback that SDL triggers with some frequency as the device + * records more audio data, (push method), or you can supply no callback, and + * then SDL will expect you to retrieve data at regular intervals (pull + * method) with this function. + * + * There are no limits on the amount of data you can queue, short of + * exhaustion of address space. Data from the device will keep queuing as + * necessary without further intervention from you. This means you will + * eventually run out of memory if you aren't routinely dequeueing data. + * + * Capture devices will not queue data when paused; if you are expecting to + * not need captured audio for some length of time, use SDL_PauseAudioDevice() + * to stop the capture device from queueing more data. This can be useful + * during, say, level loading times. When unpaused, capture devices will start + * queueing data from that point, having flushed any capturable data available + * while paused. + * + * This function is thread-safe, but dequeueing from the same device from two + * threads at once does not promise which thread will dequeue data first. + * + * You may not dequeue audio from a device that is using an + * application-supplied callback; doing so returns an error. You have to use + * the audio callback, or dequeue audio with this function, but not both. + * + * You should not call SDL_LockAudio() on the device before dequeueing; SDL + * handles locking internally for this function. + * + * \param dev the device ID from which we will dequeue audio + * \param data a pointer into where audio data should be copied + * \param len the number of bytes (not samples!) to which (data) points + * \returns the number of bytes dequeued, which could be less than requested; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.5. + * + * \sa SDL_ClearQueuedAudio + * \sa SDL_GetQueuedAudioSize + */ +extern DECLSPEC Uint32 SDLCALL SDL_DequeueAudio(SDL_AudioDeviceID dev, void *data, Uint32 len); + +/** + * Get the number of bytes of still-queued audio. + * + * For playback devices: this is the number of bytes that have been queued for + * playback with SDL_QueueAudio(), but have not yet been sent to the hardware. + * + * Once we've sent it to the hardware, this function can not decide the exact + * byte boundary of what has been played. It's possible that we just gave the + * hardware several kilobytes right before you called this function, but it + * hasn't played any of it yet, or maybe half of it, etc. + * + * For capture devices, this is the number of bytes that have been captured by + * the device and are waiting for you to dequeue. This number may grow at any + * time, so this only informs of the lower-bound of available data. + * + * You may not queue or dequeue audio on a device that is using an + * application-supplied callback; calling this function on such a device + * always returns 0. You have to use the audio callback or queue audio, but + * not both. + * + * You should not call SDL_LockAudio() on the device before querying; SDL + * handles locking internally for this function. + * + * \param dev the device ID of which we will query queued audio size + * \returns the number of bytes (not samples!) of queued audio. + * + * \since This function is available since SDL 2.0.4. + * + * \sa SDL_ClearQueuedAudio + * \sa SDL_QueueAudio + * \sa SDL_DequeueAudio + */ +extern DECLSPEC Uint32 SDLCALL SDL_GetQueuedAudioSize(SDL_AudioDeviceID dev); + +/** + * Drop any queued audio data waiting to be sent to the hardware. + * + * Immediately after this call, SDL_GetQueuedAudioSize() will return 0. For + * output devices, the hardware will start playing silence if more audio isn't + * queued. For capture devices, the hardware will start filling the empty + * queue with new data if the capture device isn't paused. + * + * This will not prevent playback of queued audio that's already been sent to + * the hardware, as we can not undo that, so expect there to be some fraction + * of a second of audio that might still be heard. This can be useful if you + * want to, say, drop any pending music or any unprocessed microphone input + * during a level change in your game. + * + * You may not queue or dequeue audio on a device that is using an + * application-supplied callback; calling this function on such a device + * always returns 0. You have to use the audio callback or queue audio, but + * not both. + * + * You should not call SDL_LockAudio() on the device before clearing the + * queue; SDL handles locking internally for this function. + * + * This function always succeeds and thus returns void. + * + * \param dev the device ID of which to clear the audio queue + * + * \since This function is available since SDL 2.0.4. + * + * \sa SDL_GetQueuedAudioSize + * \sa SDL_QueueAudio + * \sa SDL_DequeueAudio + */ +extern DECLSPEC void SDLCALL SDL_ClearQueuedAudio(SDL_AudioDeviceID dev); + + +/** + * \name Audio lock functions + * + * The lock manipulated by these functions protects the callback function. + * During a SDL_LockAudio()/SDL_UnlockAudio() pair, you can be guaranteed that + * the callback function is not running. Do not call these from the callback + * function or you will cause deadlock. + */ +/* @{ */ + +/** + * This function is a legacy means of locking the audio device. + * + * New programs might want to use SDL_LockAudioDevice() instead. This function + * is equivalent to calling... + * + * ```c + * SDL_LockAudioDevice(1); + * ``` + * + * ...and is only useful if you used the legacy SDL_OpenAudio() function. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LockAudioDevice + * \sa SDL_UnlockAudio + * \sa SDL_UnlockAudioDevice + */ +extern DECLSPEC void SDLCALL SDL_LockAudio(void); + +/** + * Use this function to lock out the audio callback function for a specified + * device. + * + * The lock manipulated by these functions protects the audio callback + * function specified in SDL_OpenAudioDevice(). During a + * SDL_LockAudioDevice()/SDL_UnlockAudioDevice() pair, you can be guaranteed + * that the callback function for that device is not running, even if the + * device is not paused. While a device is locked, any other unpaused, + * unlocked devices may still run their callbacks. + * + * Calling this function from inside your audio callback is unnecessary. SDL + * obtains this lock before calling your function, and releases it when the + * function returns. + * + * You should not hold the lock longer than absolutely necessary. If you hold + * it too long, you'll experience dropouts in your audio playback. Ideally, + * your application locks the device, sets a few variables and unlocks again. + * Do not do heavy work while holding the lock for a device. + * + * It is safe to lock the audio device multiple times, as long as you unlock + * it an equivalent number of times. The callback will not run until the + * device has been unlocked completely in this way. If your application fails + * to unlock the device appropriately, your callback will never run, you might + * hear repeating bursts of audio, and SDL_CloseAudioDevice() will probably + * deadlock. + * + * Internally, the audio device lock is a mutex; if you lock from two threads + * at once, not only will you block the audio callback, you'll block the other + * thread. + * + * \param dev the ID of the device to be locked + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_UnlockAudioDevice + */ +extern DECLSPEC void SDLCALL SDL_LockAudioDevice(SDL_AudioDeviceID dev); + +/** + * This function is a legacy means of unlocking the audio device. + * + * New programs might want to use SDL_UnlockAudioDevice() instead. This + * function is equivalent to calling... + * + * ```c + * SDL_UnlockAudioDevice(1); + * ``` + * + * ...and is only useful if you used the legacy SDL_OpenAudio() function. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LockAudio + * \sa SDL_UnlockAudioDevice + */ +extern DECLSPEC void SDLCALL SDL_UnlockAudio(void); + +/** + * Use this function to unlock the audio callback function for a specified + * device. + * + * This function should be paired with a previous SDL_LockAudioDevice() call. + * + * \param dev the ID of the device to be unlocked + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LockAudioDevice + */ +extern DECLSPEC void SDLCALL SDL_UnlockAudioDevice(SDL_AudioDeviceID dev); +/* @} *//* Audio lock functions */ + +/** + * This function is a legacy means of closing the audio device. + * + * This function is equivalent to calling... + * + * ```c + * SDL_CloseAudioDevice(1); + * ``` + * + * ...and is only useful if you used the legacy SDL_OpenAudio() function. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_OpenAudio + */ +extern DECLSPEC void SDLCALL SDL_CloseAudio(void); + +/** + * Use this function to shut down audio processing and close the audio device. + * + * The application should close open audio devices once they are no longer + * needed. Calling this function will wait until the device's audio callback + * is not running, release the audio hardware and then clean up internal + * state. No further audio will play from this device once this function + * returns. + * + * This function may block briefly while pending audio data is played by the + * hardware, so that applications don't drop the last buffer of data they + * supplied. + * + * The device ID is invalid as soon as the device is closed, and is eligible + * for reuse in a new SDL_OpenAudioDevice() call immediately. + * + * \param dev an audio device previously opened with SDL_OpenAudioDevice() + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_OpenAudioDevice + */ +extern DECLSPEC void SDLCALL SDL_CloseAudioDevice(SDL_AudioDeviceID dev); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_audio_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_bits.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_bits.h new file mode 100644 index 00000000..81161ae5 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_bits.h @@ -0,0 +1,126 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_bits.h + * + * Functions for fiddling with bits and bitmasks. + */ + +#ifndef SDL_bits_h_ +#define SDL_bits_h_ + +#include "SDL_stdinc.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \file SDL_bits.h + */ + +/** + * Get the index of the most significant bit. Result is undefined when called + * with 0. This operation can also be stated as "count leading zeroes" and + * "log base 2". + * + * \return the index of the most significant bit, or -1 if the value is 0. + */ +#if defined(__WATCOMC__) && defined(__386__) +extern __inline int _SDL_bsr_watcom(Uint32); +#pragma aux _SDL_bsr_watcom = \ + "bsr eax, eax" \ + parm [eax] nomemory \ + value [eax] \ + modify exact [eax] nomemory; +#endif + +SDL_FORCE_INLINE int +SDL_MostSignificantBitIndex32(Uint32 x) +{ +#if defined(__GNUC__) && (__GNUC__ >= 4 || (__GNUC__ == 3 && __GNUC_MINOR__ >= 4)) + /* Count Leading Zeroes builtin in GCC. + * http://gcc.gnu.org/onlinedocs/gcc-4.3.4/gcc/Other-Builtins.html + */ + if (x == 0) { + return -1; + } + return 31 - __builtin_clz(x); +#elif defined(__WATCOMC__) && defined(__386__) + if (x == 0) { + return -1; + } + return _SDL_bsr_watcom(x); +#elif defined(_MSC_VER) + unsigned long index; + if (_BitScanReverse(&index, x)) { + return index; + } + return -1; +#else + /* Based off of Bit Twiddling Hacks by Sean Eron Anderson + * , released in the public domain. + * http://graphics.stanford.edu/~seander/bithacks.html#IntegerLog + */ + const Uint32 b[] = {0x2, 0xC, 0xF0, 0xFF00, 0xFFFF0000}; + const int S[] = {1, 2, 4, 8, 16}; + + int msbIndex = 0; + int i; + + if (x == 0) { + return -1; + } + + for (i = 4; i >= 0; i--) + { + if (x & b[i]) + { + x >>= S[i]; + msbIndex |= S[i]; + } + } + + return msbIndex; +#endif +} + +SDL_FORCE_INLINE SDL_bool +SDL_HasExactlyOneBitSet32(Uint32 x) +{ + if (x && !(x & (x - 1))) { + return SDL_TRUE; + } + return SDL_FALSE; +} + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_bits_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_blendmode.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_blendmode.h new file mode 100644 index 00000000..4ecbe507 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_blendmode.h @@ -0,0 +1,198 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_blendmode.h + * + * Header file declaring the SDL_BlendMode enumeration + */ + +#ifndef SDL_blendmode_h_ +#define SDL_blendmode_h_ + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \brief The blend mode used in SDL_RenderCopy() and drawing operations. + */ +typedef enum +{ + SDL_BLENDMODE_NONE = 0x00000000, /**< no blending + dstRGBA = srcRGBA */ + SDL_BLENDMODE_BLEND = 0x00000001, /**< alpha blending + dstRGB = (srcRGB * srcA) + (dstRGB * (1-srcA)) + dstA = srcA + (dstA * (1-srcA)) */ + SDL_BLENDMODE_ADD = 0x00000002, /**< additive blending + dstRGB = (srcRGB * srcA) + dstRGB + dstA = dstA */ + SDL_BLENDMODE_MOD = 0x00000004, /**< color modulate + dstRGB = srcRGB * dstRGB + dstA = dstA */ + SDL_BLENDMODE_MUL = 0x00000008, /**< color multiply + dstRGB = (srcRGB * dstRGB) + (dstRGB * (1-srcA)) + dstA = dstA */ + SDL_BLENDMODE_INVALID = 0x7FFFFFFF + + /* Additional custom blend modes can be returned by SDL_ComposeCustomBlendMode() */ + +} SDL_BlendMode; + +/** + * \brief The blend operation used when combining source and destination pixel components + */ +typedef enum +{ + SDL_BLENDOPERATION_ADD = 0x1, /**< dst + src: supported by all renderers */ + SDL_BLENDOPERATION_SUBTRACT = 0x2, /**< dst - src : supported by D3D9, D3D11, OpenGL, OpenGLES */ + SDL_BLENDOPERATION_REV_SUBTRACT = 0x3, /**< src - dst : supported by D3D9, D3D11, OpenGL, OpenGLES */ + SDL_BLENDOPERATION_MINIMUM = 0x4, /**< min(dst, src) : supported by D3D9, D3D11 */ + SDL_BLENDOPERATION_MAXIMUM = 0x5 /**< max(dst, src) : supported by D3D9, D3D11 */ +} SDL_BlendOperation; + +/** + * \brief The normalized factor used to multiply pixel components + */ +typedef enum +{ + SDL_BLENDFACTOR_ZERO = 0x1, /**< 0, 0, 0, 0 */ + SDL_BLENDFACTOR_ONE = 0x2, /**< 1, 1, 1, 1 */ + SDL_BLENDFACTOR_SRC_COLOR = 0x3, /**< srcR, srcG, srcB, srcA */ + SDL_BLENDFACTOR_ONE_MINUS_SRC_COLOR = 0x4, /**< 1-srcR, 1-srcG, 1-srcB, 1-srcA */ + SDL_BLENDFACTOR_SRC_ALPHA = 0x5, /**< srcA, srcA, srcA, srcA */ + SDL_BLENDFACTOR_ONE_MINUS_SRC_ALPHA = 0x6, /**< 1-srcA, 1-srcA, 1-srcA, 1-srcA */ + SDL_BLENDFACTOR_DST_COLOR = 0x7, /**< dstR, dstG, dstB, dstA */ + SDL_BLENDFACTOR_ONE_MINUS_DST_COLOR = 0x8, /**< 1-dstR, 1-dstG, 1-dstB, 1-dstA */ + SDL_BLENDFACTOR_DST_ALPHA = 0x9, /**< dstA, dstA, dstA, dstA */ + SDL_BLENDFACTOR_ONE_MINUS_DST_ALPHA = 0xA /**< 1-dstA, 1-dstA, 1-dstA, 1-dstA */ +} SDL_BlendFactor; + +/** + * Compose a custom blend mode for renderers. + * + * The functions SDL_SetRenderDrawBlendMode and SDL_SetTextureBlendMode accept + * the SDL_BlendMode returned by this function if the renderer supports it. + * + * A blend mode controls how the pixels from a drawing operation (source) get + * combined with the pixels from the render target (destination). First, the + * components of the source and destination pixels get multiplied with their + * blend factors. Then, the blend operation takes the two products and + * calculates the result that will get stored in the render target. + * + * Expressed in pseudocode, it would look like this: + * + * ```c + * dstRGB = colorOperation(srcRGB * srcColorFactor, dstRGB * dstColorFactor); + * dstA = alphaOperation(srcA * srcAlphaFactor, dstA * dstAlphaFactor); + * ``` + * + * Where the functions `colorOperation(src, dst)` and `alphaOperation(src, + * dst)` can return one of the following: + * + * - `src + dst` + * - `src - dst` + * - `dst - src` + * - `min(src, dst)` + * - `max(src, dst)` + * + * The red, green, and blue components are always multiplied with the first, + * second, and third components of the SDL_BlendFactor, respectively. The + * fourth component is not used. + * + * The alpha component is always multiplied with the fourth component of the + * SDL_BlendFactor. The other components are not used in the alpha + * calculation. + * + * Support for these blend modes varies for each renderer. To check if a + * specific SDL_BlendMode is supported, create a renderer and pass it to + * either SDL_SetRenderDrawBlendMode or SDL_SetTextureBlendMode. They will + * return with an error if the blend mode is not supported. + * + * This list describes the support of custom blend modes for each renderer in + * SDL 2.0.6. All renderers support the four blend modes listed in the + * SDL_BlendMode enumeration. + * + * - **direct3d**: Supports all operations with all factors. However, some + * factors produce unexpected results with `SDL_BLENDOPERATION_MINIMUM` and + * `SDL_BLENDOPERATION_MAXIMUM`. + * - **direct3d11**: Same as Direct3D 9. + * - **opengl**: Supports the `SDL_BLENDOPERATION_ADD` operation with all + * factors. OpenGL versions 1.1, 1.2, and 1.3 do not work correctly with SDL + * 2.0.6. + * - **opengles**: Supports the `SDL_BLENDOPERATION_ADD` operation with all + * factors. Color and alpha factors need to be the same. OpenGL ES 1 + * implementation specific: May also support `SDL_BLENDOPERATION_SUBTRACT` + * and `SDL_BLENDOPERATION_REV_SUBTRACT`. May support color and alpha + * operations being different from each other. May support color and alpha + * factors being different from each other. + * - **opengles2**: Supports the `SDL_BLENDOPERATION_ADD`, + * `SDL_BLENDOPERATION_SUBTRACT`, `SDL_BLENDOPERATION_REV_SUBTRACT` + * operations with all factors. + * - **psp**: No custom blend mode support. + * - **software**: No custom blend mode support. + * + * Some renderers do not provide an alpha component for the default render + * target. The `SDL_BLENDFACTOR_DST_ALPHA` and + * `SDL_BLENDFACTOR_ONE_MINUS_DST_ALPHA` factors do not have an effect in this + * case. + * + * \param srcColorFactor the SDL_BlendFactor applied to the red, green, and + * blue components of the source pixels + * \param dstColorFactor the SDL_BlendFactor applied to the red, green, and + * blue components of the destination pixels + * \param colorOperation the SDL_BlendOperation used to combine the red, + * green, and blue components of the source and + * destination pixels + * \param srcAlphaFactor the SDL_BlendFactor applied to the alpha component of + * the source pixels + * \param dstAlphaFactor the SDL_BlendFactor applied to the alpha component of + * the destination pixels + * \param alphaOperation the SDL_BlendOperation used to combine the alpha + * component of the source and destination pixels + * \returns an SDL_BlendMode that represents the chosen factors and + * operations. + * + * \since This function is available since SDL 2.0.6. + * + * \sa SDL_SetRenderDrawBlendMode + * \sa SDL_GetRenderDrawBlendMode + * \sa SDL_SetTextureBlendMode + * \sa SDL_GetTextureBlendMode + */ +extern DECLSPEC SDL_BlendMode SDLCALL SDL_ComposeCustomBlendMode(SDL_BlendFactor srcColorFactor, + SDL_BlendFactor dstColorFactor, + SDL_BlendOperation colorOperation, + SDL_BlendFactor srcAlphaFactor, + SDL_BlendFactor dstAlphaFactor, + SDL_BlendOperation alphaOperation); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_blendmode_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_clipboard.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_clipboard.h new file mode 100644 index 00000000..7c351fbb --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_clipboard.h @@ -0,0 +1,141 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_clipboard.h + * + * Include file for SDL clipboard handling + */ + +#ifndef SDL_clipboard_h_ +#define SDL_clipboard_h_ + +#include "SDL_stdinc.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* Function prototypes */ + +/** + * Put UTF-8 text into the clipboard. + * + * \param text the text to store in the clipboard + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetClipboardText + * \sa SDL_HasClipboardText + */ +extern DECLSPEC int SDLCALL SDL_SetClipboardText(const char *text); + +/** + * Get UTF-8 text from the clipboard, which must be freed with SDL_free(). + * + * This functions returns empty string if there was not enough memory left for + * a copy of the clipboard's content. + * + * \returns the clipboard text on success or an empty string on failure; call + * SDL_GetError() for more information. Caller must call SDL_free() + * on the returned pointer when done with it (even if there was an + * error). + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HasClipboardText + * \sa SDL_SetClipboardText + */ +extern DECLSPEC char * SDLCALL SDL_GetClipboardText(void); + +/** + * Query whether the clipboard exists and contains a non-empty text string. + * + * \returns SDL_TRUE if the clipboard has text, or SDL_FALSE if it does not. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetClipboardText + * \sa SDL_SetClipboardText + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasClipboardText(void); + +/** + * Put UTF-8 text into the primary selection. + * + * \param text the text to store in the primary selection + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.26.0. + * + * \sa SDL_GetPrimarySelectionText + * \sa SDL_HasPrimarySelectionText + */ +extern DECLSPEC int SDLCALL SDL_SetPrimarySelectionText(const char *text); + +/** + * Get UTF-8 text from the primary selection, which must be freed with + * SDL_free(). + * + * This functions returns empty string if there was not enough memory left for + * a copy of the primary selection's content. + * + * \returns the primary selection text on success or an empty string on + * failure; call SDL_GetError() for more information. Caller must + * call SDL_free() on the returned pointer when done with it (even if + * there was an error). + * + * \since This function is available since SDL 2.26.0. + * + * \sa SDL_HasPrimarySelectionText + * \sa SDL_SetPrimarySelectionText + */ +extern DECLSPEC char * SDLCALL SDL_GetPrimarySelectionText(void); + +/** + * Query whether the primary selection exists and contains a non-empty text + * string. + * + * \returns SDL_TRUE if the primary selection has text, or SDL_FALSE if it + * does not. + * + * \since This function is available since SDL 2.26.0. + * + * \sa SDL_GetPrimarySelectionText + * \sa SDL_SetPrimarySelectionText + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasPrimarySelectionText(void); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_clipboard_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_config.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_config.h new file mode 100644 index 00000000..75116154 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_config.h @@ -0,0 +1,498 @@ +/* include/SDL_config.h. Generated from SDL_config.h.in by configure. */ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +#ifndef SDL_config_h_ +#define SDL_config_h_ + +/** + * \file SDL_config.h.in + * + * This is a set of defines to configure the SDL features + */ + +/* General platform specific identifiers */ +#include "SDL_platform.h" + +/* Make sure that this isn't included by Visual C++ */ +#ifdef _MSC_VER +#error You should run git checkout -f include/SDL_config.h +#endif + +/* C language features */ +/* #undef const */ +/* #undef inline */ +/* #undef volatile */ + +/* C datatypes */ +#if defined(__LP64__) || defined(_LP64) || defined(_WIN64) +#define SIZEOF_VOIDP 8 +#else +#define SIZEOF_VOIDP 4 +#endif + +#define HAVE_GCC_ATOMICS 1 +/* #undef HAVE_GCC_SYNC_LOCK_TEST_AND_SET */ + +/* Comment this if you want to build without any C library requirements */ +#define HAVE_LIBC 1 +#if HAVE_LIBC + +/* Useful headers */ +#define STDC_HEADERS 1 +#define HAVE_ALLOCA_H 1 +#define HAVE_CTYPE_H 1 +#define HAVE_FLOAT_H 1 +#define HAVE_ICONV_H 1 +#define HAVE_INTTYPES_H 1 +#define HAVE_LIMITS_H 1 +/* #undef HAVE_MALLOC_H */ +#define HAVE_MATH_H 1 +#define HAVE_MEMORY_H 1 +#define HAVE_SIGNAL_H 1 +#define HAVE_STDARG_H 1 +#define HAVE_STDINT_H 1 +#define HAVE_STDIO_H 1 +#define HAVE_STDLIB_H 1 +#define HAVE_STRINGS_H 1 +#define HAVE_STRING_H 1 +#define HAVE_SYS_TYPES_H 1 +#define HAVE_WCHAR_H 1 +/* #undef HAVE_LINUX_INPUT_H */ +/* #undef HAVE_PTHREAD_NP_H */ +#define HAVE_LIBUNWIND_H 1 + +/* C library functions */ +#define HAVE_DLOPEN 1 +#define HAVE_MALLOC 1 +#define HAVE_CALLOC 1 +#define HAVE_REALLOC 1 +#define HAVE_FREE 1 +#define HAVE_ALLOCA 1 +#ifndef __WIN32__ /* Don't use C runtime versions of these on Windows */ +#define HAVE_GETENV 1 +#define HAVE_SETENV 1 +#define HAVE_PUTENV 1 +#define HAVE_UNSETENV 1 +#endif +#define HAVE_QSORT 1 +#define HAVE_BSEARCH 1 +#define HAVE_ABS 1 +#define HAVE_BCOPY 1 +#define HAVE_MEMSET 1 +#define HAVE_MEMCPY 1 +#define HAVE_MEMMOVE 1 +#define HAVE_MEMCMP 1 +#define HAVE_WCSLEN 1 +#define HAVE_WCSLCPY 1 +#define HAVE_WCSLCAT 1 +/* #undef HAVE__WCSDUP */ +#define HAVE_WCSDUP 1 +#define HAVE_WCSSTR 1 +#define HAVE_WCSCMP 1 +#define HAVE_WCSNCMP 1 +#define HAVE_WCSCASECMP 1 +/* #undef HAVE__WCSICMP */ +#define HAVE_WCSNCASECMP 1 +/* #undef HAVE__WCSNICMP */ +#define HAVE_STRLEN 1 +#define HAVE_STRLCPY 1 +#define HAVE_STRLCAT 1 +/* #undef HAVE__STRREV */ +/* #undef HAVE__STRUPR */ +/* #undef HAVE__STRLWR */ +#define HAVE_INDEX 1 +#define HAVE_RINDEX 1 +#define HAVE_STRCHR 1 +#define HAVE_STRRCHR 1 +#define HAVE_STRSTR 1 +#define HAVE_STRTOK_R 1 +/* #undef HAVE_ITOA */ +/* #undef HAVE__LTOA */ +/* #undef HAVE__UITOA */ +/* #undef HAVE__ULTOA */ +#define HAVE_STRTOL 1 +#define HAVE_STRTOUL 1 +/* #undef HAVE__I64TOA */ +/* #undef HAVE__UI64TOA */ +#define HAVE_STRTOLL 1 +#define HAVE_STRTOULL 1 +#define HAVE_STRTOD 1 +#define HAVE_ATOI 1 +#define HAVE_ATOF 1 +#define HAVE_STRCMP 1 +#define HAVE_STRNCMP 1 +/* #undef HAVE__STRICMP */ +#define HAVE_STRCASECMP 1 +/* #undef HAVE__STRNICMP */ +#define HAVE_STRNCASECMP 1 +#define HAVE_STRCASESTR 1 +/* #undef HAVE_SSCANF */ +#define HAVE_VSSCANF 1 +/* #undef HAVE_SNPRINTF */ +#define HAVE_VSNPRINTF 1 +#define HAVE_M_PI /**/ +#define HAVE_ACOS 1 +#define HAVE_ACOSF 1 +#define HAVE_ASIN 1 +#define HAVE_ASINF 1 +#define HAVE_ATAN 1 +#define HAVE_ATANF 1 +#define HAVE_ATAN2 1 +#define HAVE_ATAN2F 1 +#define HAVE_CEIL 1 +#define HAVE_CEILF 1 +#define HAVE_COPYSIGN 1 +#define HAVE_COPYSIGNF 1 +#define HAVE_COS 1 +#define HAVE_COSF 1 +#define HAVE_EXP 1 +#define HAVE_EXPF 1 +#define HAVE_FABS 1 +#define HAVE_FABSF 1 +#define HAVE_FLOOR 1 +#define HAVE_FLOORF 1 +#define HAVE_FMOD 1 +#define HAVE_FMODF 1 +#define HAVE_LOG 1 +#define HAVE_LOGF 1 +#define HAVE_LOG10 1 +#define HAVE_LOG10F 1 +#define HAVE_LROUND 1 +#define HAVE_LROUNDF 1 +#define HAVE_POW 1 +#define HAVE_POWF 1 +#define HAVE_ROUND 1 +#define HAVE_ROUNDF 1 +#define HAVE_SCALBN 1 +#define HAVE_SCALBNF 1 +#define HAVE_SIN 1 +#define HAVE_SINF 1 +#define HAVE_SQRT 1 +#define HAVE_SQRTF 1 +#define HAVE_TAN 1 +#define HAVE_TANF 1 +#define HAVE_TRUNC 1 +#define HAVE_TRUNCF 1 +/* #undef HAVE_FOPEN64 */ +#define HAVE_FSEEKO 1 +/* #undef HAVE_FSEEKO64 */ +#define HAVE_SIGACTION 1 +#define HAVE_SA_SIGACTION 1 +#define HAVE_SETJMP 1 +#define HAVE_NANOSLEEP 1 +#define HAVE_SYSCONF 1 +#define HAVE_SYSCTLBYNAME 1 +/* #undef HAVE_CLOCK_GETTIME */ +/* #undef HAVE_GETPAGESIZE */ +#define HAVE_MPROTECT 1 +#define HAVE_ICONV 1 +#define HAVE_PTHREAD_SETNAME_NP 1 +/* #undef HAVE_PTHREAD_SET_NAME_NP */ +/* #undef HAVE_SEM_TIMEDWAIT */ +/* #undef HAVE_GETAUXVAL */ +/* #undef HAVE_ELF_AUX_INFO */ +#define HAVE_POLL 1 +#define HAVE__EXIT 1 + +#else +#define HAVE_STDARG_H 1 +#define HAVE_STDDEF_H 1 +#define HAVE_STDINT_H 1 +#endif /* HAVE_LIBC */ + +#define HAVE_O_CLOEXEC 1 +/* #undef HAVE_ALTIVEC_H */ +/* #undef HAVE_DBUS_DBUS_H */ +/* #undef HAVE_FCITX */ +/* #undef HAVE_SYS_INOTIFY_H */ +/* #undef HAVE_INOTIFY_INIT */ +/* #undef HAVE_INOTIFY_INIT1 */ +/* #undef HAVE_INOTIFY */ +/* #undef HAVE_IBUS_IBUS_H */ +/* #undef HAVE_IMMINTRIN_H */ +/* #undef HAVE_LIBUDEV_H */ +/* #undef HAVE_LIBUSB */ +/* #undef HAVE_LIBSAMPLERATE_H */ +/* #undef HAVE_LIBDECOR_H */ +/* #undef HAVE_LSXINTRIN_H */ +/* #undef HAVE_LASXINTRIN_H */ + +/* #undef HAVE_DDRAW_H */ +/* #undef HAVE_DINPUT_H */ +/* #undef HAVE_DSOUND_H */ +/* #undef HAVE_DXGI_H */ +/* #undef HAVE_WINDOWS_GAMING_INPUT_H */ +/* #undef HAVE_XINPUT_H */ +/* #undef HAVE_XINPUT_GAMEPAD_EX */ +/* #undef HAVE_XINPUT_STATE_EX */ + +/* #undef HAVE_MMDEVICEAPI_H */ +/* #undef HAVE_AUDIOCLIENT_H */ +/* #undef HAVE_TPCSHRD_H */ +/* #undef HAVE_SENSORSAPI_H */ +/* #undef HAVE_ROAPI_H */ +/* #undef HAVE_SHELLSCALINGAPI_H */ + +/* SDL internal assertion support */ +/* #undef SDL_DEFAULT_ASSERT_LEVEL */ + +/* Allow disabling of core subsystems */ +/* #undef SDL_ATOMIC_DISABLED */ +/* #undef SDL_AUDIO_DISABLED */ +/* #undef SDL_CPUINFO_DISABLED */ +/* #undef SDL_EVENTS_DISABLED */ +/* #undef SDL_FILE_DISABLED */ +/* #undef SDL_JOYSTICK_DISABLED */ +/* #undef SDL_HAPTIC_DISABLED */ +/* #undef SDL_HIDAPI_DISABLED */ +/* #undef SDL_SENSOR_DISABLED */ +/* #undef SDL_LOADSO_DISABLED */ +/* #undef SDL_RENDER_DISABLED */ +/* #undef SDL_THREADS_DISABLED */ +/* #undef SDL_TIMERS_DISABLED */ +/* #undef SDL_VIDEO_DISABLED */ +/* #undef SDL_POWER_DISABLED */ +/* #undef SDL_FILESYSTEM_DISABLED */ +/* #undef SDL_LOCALE_DISABLED */ +/* #undef SDL_MISC_DISABLED */ + +/* Enable various audio drivers */ +/* #undef SDL_AUDIO_DRIVER_AAUDIO */ +/* #undef SDL_AUDIO_DRIVER_ALSA */ +/* #undef SDL_AUDIO_DRIVER_ALSA_DYNAMIC */ +/* #undef SDL_AUDIO_DRIVER_ANDROID */ +/* #undef SDL_AUDIO_DRIVER_ARTS */ +/* #undef SDL_AUDIO_DRIVER_ARTS_DYNAMIC */ +#define SDL_AUDIO_DRIVER_COREAUDIO 1 +#define SDL_AUDIO_DRIVER_DISK 1 +/* #undef SDL_AUDIO_DRIVER_DSOUND */ +#define SDL_AUDIO_DRIVER_DUMMY 1 +/* #undef SDL_AUDIO_DRIVER_EMSCRIPTEN */ +/* #undef SDL_AUDIO_DRIVER_ESD */ +/* #undef SDL_AUDIO_DRIVER_ESD_DYNAMIC */ +/* #undef SDL_AUDIO_DRIVER_FUSIONSOUND */ +/* #undef SDL_AUDIO_DRIVER_FUSIONSOUND_DYNAMIC */ +/* #undef SDL_AUDIO_DRIVER_HAIKU */ +/* #undef SDL_AUDIO_DRIVER_JACK */ +/* #undef SDL_AUDIO_DRIVER_JACK_DYNAMIC */ +/* #undef SDL_AUDIO_DRIVER_NACL */ +/* #undef SDL_AUDIO_DRIVER_NAS */ +/* #undef SDL_AUDIO_DRIVER_NAS_DYNAMIC */ +/* #undef SDL_AUDIO_DRIVER_NETBSD */ +/* #undef SDL_AUDIO_DRIVER_OPENSLES */ +/* #undef SDL_AUDIO_DRIVER_OSS */ +/* #undef SDL_AUDIO_DRIVER_PAUDIO */ +/* #undef SDL_AUDIO_DRIVER_PIPEWIRE */ +/* #undef SDL_AUDIO_DRIVER_PIPEWIRE_DYNAMIC */ +/* #undef SDL_AUDIO_DRIVER_PULSEAUDIO */ +/* #undef SDL_AUDIO_DRIVER_PULSEAUDIO_DYNAMIC */ +/* #undef SDL_AUDIO_DRIVER_QSA */ +/* #undef SDL_AUDIO_DRIVER_SNDIO */ +/* #undef SDL_AUDIO_DRIVER_SNDIO_DYNAMIC */ +/* #undef SDL_AUDIO_DRIVER_SUNAUDIO */ +/* #undef SDL_AUDIO_DRIVER_WASAPI */ +/* #undef SDL_AUDIO_DRIVER_WINMM */ +/* #undef SDL_AUDIO_DRIVER_OS2 */ + +/* Enable various input drivers */ +/* #undef SDL_INPUT_LINUXEV */ +/* #undef SDL_INPUT_FBSDKBIO */ +/* #undef SDL_INPUT_LINUXKD */ +/* #undef SDL_INPUT_WSCONS */ +/* #undef SDL_JOYSTICK_HAIKU */ +/* #undef SDL_JOYSTICK_DINPUT */ +/* #undef SDL_JOYSTICK_WGI */ +/* #undef SDL_JOYSTICK_XINPUT */ +/* #undef SDL_JOYSTICK_DUMMY */ +#define SDL_JOYSTICK_IOKIT 1 +#define SDL_JOYSTICK_MFI 1 +/* #undef SDL_JOYSTICK_LINUX */ +/* #undef SDL_JOYSTICK_ANDROID */ +/* #undef SDL_JOYSTICK_OS2 */ +/* #undef SDL_JOYSTICK_USBHID */ +/* #undef SDL_HAVE_MACHINE_JOYSTICK_H */ +#define SDL_JOYSTICK_HIDAPI 1 +/* #undef SDL_JOYSTICK_RAWINPUT */ +/* #undef SDL_JOYSTICK_EMSCRIPTEN */ +#define SDL_JOYSTICK_VIRTUAL 1 +/* #undef SDL_HAPTIC_DUMMY */ +/* #undef SDL_HAPTIC_ANDROID */ +/* #undef SDL_HAPTIC_LINUX */ +#define SDL_HAPTIC_IOKIT 1 +/* #undef SDL_HAPTIC_DINPUT */ +/* #undef SDL_HAPTIC_XINPUT */ + +/* Enable various sensor drivers */ +/* #undef SDL_SENSOR_ANDROID */ +/* #undef SDL_SENSOR_COREMOTION */ +/* #undef SDL_SENSOR_WINDOWS */ +#define SDL_SENSOR_DUMMY 1 + +/* Enable various shared object loading systems */ +#define SDL_LOADSO_DLOPEN 1 +/* #undef SDL_LOADSO_DUMMY */ +/* #undef SDL_LOADSO_LDG */ +/* #undef SDL_LOADSO_WINDOWS */ +/* #undef SDL_LOADSO_OS2 */ + +/* Enable various threading systems */ +/* #undef SDL_THREAD_GENERIC_COND_SUFFIX */ +#define SDL_THREAD_PTHREAD 1 +#define SDL_THREAD_PTHREAD_RECURSIVE_MUTEX 1 +/* #undef SDL_THREAD_PTHREAD_RECURSIVE_MUTEX_NP */ +/* #undef SDL_THREAD_WINDOWS */ +/* #undef SDL_THREAD_OS2 */ + +/* Enable various timer systems */ +/* #undef SDL_TIMER_HAIKU */ +/* #undef SDL_TIMER_DUMMY */ +#define SDL_TIMER_UNIX 1 +/* #undef SDL_TIMER_WINDOWS */ +/* #undef SDL_TIMER_OS2 */ + +/* Enable various video drivers */ +/* #undef SDL_VIDEO_DRIVER_HAIKU */ +#define SDL_VIDEO_DRIVER_COCOA 1 +/* #undef SDL_VIDEO_DRIVER_DIRECTFB */ +/* #undef SDL_VIDEO_DRIVER_DIRECTFB_DYNAMIC */ +#define SDL_VIDEO_DRIVER_DUMMY 1 +/* #undef SDL_VIDEO_DRIVER_WINDOWS */ +/* #undef SDL_VIDEO_DRIVER_WAYLAND */ +/* #undef SDL_VIDEO_DRIVER_WAYLAND_QT_TOUCH */ +/* #undef SDL_VIDEO_DRIVER_WAYLAND_DYNAMIC */ +/* #undef SDL_VIDEO_DRIVER_WAYLAND_DYNAMIC_EGL */ +/* #undef SDL_VIDEO_DRIVER_WAYLAND_DYNAMIC_CURSOR */ +/* #undef SDL_VIDEO_DRIVER_WAYLAND_DYNAMIC_XKBCOMMON */ +/* #undef SDL_VIDEO_DRIVER_WAYLAND_DYNAMIC_LIBDECOR */ +/* #undef SDL_VIDEO_DRIVER_X11 */ +/* #undef SDL_VIDEO_DRIVER_RPI */ +/* #undef SDL_VIDEO_DRIVER_KMSDRM */ +/* #undef SDL_VIDEO_DRIVER_KMSDRM_DYNAMIC */ +/* #undef SDL_VIDEO_DRIVER_KMSDRM_DYNAMIC_GBM */ +/* #undef SDL_VIDEO_DRIVER_ANDROID */ +/* #undef SDL_VIDEO_DRIVER_EMSCRIPTEN */ +#define SDL_VIDEO_DRIVER_OFFSCREEN 1 +/* #undef SDL_VIDEO_DRIVER_X11_DYNAMIC */ +/* #undef SDL_VIDEO_DRIVER_X11_DYNAMIC_XEXT */ +/* #undef SDL_VIDEO_DRIVER_X11_DYNAMIC_XCURSOR */ +/* #undef SDL_VIDEO_DRIVER_X11_DYNAMIC_XINPUT2 */ +/* #undef SDL_VIDEO_DRIVER_X11_DYNAMIC_XFIXES */ +/* #undef SDL_VIDEO_DRIVER_X11_DYNAMIC_XRANDR */ +/* #undef SDL_VIDEO_DRIVER_X11_DYNAMIC_XSS */ +/* #undef SDL_VIDEO_DRIVER_X11_XCURSOR */ +/* #undef SDL_VIDEO_DRIVER_X11_XDBE */ +/* #undef SDL_VIDEO_DRIVER_X11_XINPUT2 */ +/* #undef SDL_VIDEO_DRIVER_X11_XINPUT2_SUPPORTS_MULTITOUCH */ +/* #undef SDL_VIDEO_DRIVER_X11_XFIXES */ +/* #undef SDL_VIDEO_DRIVER_X11_XRANDR */ +/* #undef SDL_VIDEO_DRIVER_X11_XSCRNSAVER */ +/* #undef SDL_VIDEO_DRIVER_X11_XSHAPE */ +/* #undef SDL_VIDEO_DRIVER_X11_SUPPORTS_GENERIC_EVENTS */ +/* #undef SDL_VIDEO_DRIVER_X11_HAS_XKBKEYCODETOKEYSYM */ +/* #undef SDL_VIDEO_DRIVER_NACL */ +/* #undef SDL_VIDEO_DRIVER_VIVANTE */ +/* #undef SDL_VIDEO_DRIVER_VIVANTE_VDK */ +/* #undef SDL_VIDEO_DRIVER_OS2 */ +/* #undef SDL_VIDEO_DRIVER_QNX */ +/* #undef SDL_VIDEO_DRIVER_RISCOS */ + +/* #undef SDL_VIDEO_RENDER_D3D */ +/* #undef SDL_VIDEO_RENDER_D3D11 */ +/* #undef SDL_VIDEO_RENDER_D3D12 */ +#define SDL_VIDEO_RENDER_OGL 1 +/* #undef SDL_VIDEO_RENDER_OGL_ES */ +#define SDL_VIDEO_RENDER_OGL_ES2 1 +/* #undef SDL_VIDEO_RENDER_DIRECTFB */ +#define SDL_VIDEO_RENDER_METAL 1 + +/* Enable OpenGL support */ +#define SDL_VIDEO_OPENGL 1 +/* #undef SDL_VIDEO_OPENGL_ES */ +#define SDL_VIDEO_OPENGL_ES2 1 +/* #undef SDL_VIDEO_OPENGL_BGL */ +#define SDL_VIDEO_OPENGL_CGL 1 +#define SDL_VIDEO_OPENGL_EGL 1 +/* #undef SDL_VIDEO_OPENGL_GLX */ +/* #undef SDL_VIDEO_OPENGL_WGL */ +/* #undef SDL_VIDEO_OPENGL_OSMESA */ +/* #undef SDL_VIDEO_OPENGL_OSMESA_DYNAMIC */ + +/* Enable Vulkan support */ +#define SDL_VIDEO_VULKAN 1 + +/* Enable Metal support */ +#define SDL_VIDEO_METAL 1 + +/* Enable system power support */ +/* #undef SDL_POWER_LINUX */ +/* #undef SDL_POWER_WINDOWS */ +#define SDL_POWER_MACOSX 1 +/* #undef SDL_POWER_HAIKU */ +/* #undef SDL_POWER_ANDROID */ +/* #undef SDL_POWER_EMSCRIPTEN */ +/* #undef SDL_POWER_HARDWIRED */ + +/* Enable system filesystem support */ +/* #undef SDL_FILESYSTEM_ANDROID */ +/* #undef SDL_FILESYSTEM_HAIKU */ +#define SDL_FILESYSTEM_COCOA 1 +/* #undef SDL_FILESYSTEM_DUMMY */ +/* #undef SDL_FILESYSTEM_RISCOS */ +/* #undef SDL_FILESYSTEM_UNIX */ +/* #undef SDL_FILESYSTEM_WINDOWS */ +/* #undef SDL_FILESYSTEM_NACL */ +/* #undef SDL_FILESYSTEM_EMSCRIPTEN */ +/* #undef SDL_FILESYSTEM_OS2 */ +/* #undef SDL_FILESYSTEM_VITA */ +/* #undef SDL_FILESYSTEM_PSP */ +/* #undef SDL_FILESYSTEM_PS2 */ + +/* Enable misc subsystem */ +/* #undef SDL_MISC_DUMMY */ + +/* Enable locale subsystem */ +/* #undef SDL_LOCALE_DUMMY */ + +/* Enable assembly routines */ +/* #undef SDL_ALTIVEC_BLITTERS */ +/* #undef SDL_ARM_SIMD_BLITTERS */ +/* #undef SDL_ARM_NEON_BLITTERS */ + +/* Whether SDL_DYNAMIC_API needs dlopen() */ +#define DYNAPI_NEEDS_DLOPEN 1 + +/* Enable ime support */ +/* #undef SDL_USE_IME */ + +/* Enable dynamic udev support */ +/* #undef SDL_UDEV_DYNAMIC */ + +/* Enable dynamic libusb support */ +/* #undef SDL_LIBUSB_DYNAMIC */ + +/* Enable dynamic libsamplerate support */ +/* #undef SDL_LIBSAMPLERATE_DYNAMIC */ + +/* Libdecor get min/max content size functions */ +/* #undef SDL_HAVE_LIBDECOR_GET_MIN_MAX */ + +#endif /* SDL_config_h_ */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_cpuinfo.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_cpuinfo.h new file mode 100644 index 00000000..ed5e9791 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_cpuinfo.h @@ -0,0 +1,594 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_cpuinfo.h + * + * CPU feature detection for SDL. + */ + +#ifndef SDL_cpuinfo_h_ +#define SDL_cpuinfo_h_ + +#include "SDL_stdinc.h" + +/* Need to do this here because intrin.h has C++ code in it */ +/* Visual Studio 2005 has a bug where intrin.h conflicts with winnt.h */ +#if defined(_MSC_VER) && (_MSC_VER >= 1500) && (defined(_M_IX86) || defined(_M_X64)) +#ifdef __clang__ +/* As of Clang 11, '_m_prefetchw' is conflicting with the winnt.h's version, + so we define the needed '_m_prefetch' here as a pseudo-header, until the issue is fixed. */ + +#ifndef __PRFCHWINTRIN_H +#define __PRFCHWINTRIN_H + +static __inline__ void __attribute__((__always_inline__, __nodebug__)) +_m_prefetch(void *__P) +{ + __builtin_prefetch (__P, 0, 3 /* _MM_HINT_T0 */); +} + +#endif /* __PRFCHWINTRIN_H */ +#endif /* __clang__ */ +#include +#ifndef _WIN64 +#ifndef __MMX__ +#define __MMX__ +#endif +#ifndef __3dNOW__ +#define __3dNOW__ +#endif +#endif +#ifndef __SSE__ +#define __SSE__ +#endif +#ifndef __SSE2__ +#define __SSE2__ +#endif +#ifndef __SSE3__ +#define __SSE3__ +#endif +#elif defined(__MINGW64_VERSION_MAJOR) +#include +#if !defined(SDL_DISABLE_ARM_NEON_H) && defined(__ARM_NEON) +# include +#endif +#else +/* altivec.h redefining bool causes a number of problems, see bugs 3993 and 4392, so you need to explicitly define SDL_ENABLE_ALTIVEC_H to have it included. */ +#if defined(HAVE_ALTIVEC_H) && defined(__ALTIVEC__) && !defined(__APPLE_ALTIVEC__) && defined(SDL_ENABLE_ALTIVEC_H) +#include +#endif +#if !defined(SDL_DISABLE_ARM_NEON_H) +# if defined(__ARM_NEON) +# include +# elif defined(__WINDOWS__) || defined(__WINRT__) || defined(__GDK__) +/* Visual Studio doesn't define __ARM_ARCH, but _M_ARM (if set, always 7), and _M_ARM64 (if set, always 1). */ +# if defined(_M_ARM) +# include +# include +# define __ARM_NEON 1 /* Set __ARM_NEON so that it can be used elsewhere, at compile time */ +# endif +# if defined (_M_ARM64) +# include +# include +# define __ARM_NEON 1 /* Set __ARM_NEON so that it can be used elsewhere, at compile time */ +# define __ARM_ARCH 8 +# endif +# endif +#endif +#endif /* compiler version */ + +#if defined(__3dNOW__) && !defined(SDL_DISABLE_MM3DNOW_H) +#include +#endif +#if defined(__loongarch_sx) && !defined(SDL_DISABLE_LSX_H) +#include +#define __LSX__ +#endif +#if defined(__loongarch_asx) && !defined(SDL_DISABLE_LASX_H) +#include +#define __LASX__ +#endif +#if defined(HAVE_IMMINTRIN_H) && !defined(SDL_DISABLE_IMMINTRIN_H) +#include +#else +#if defined(__MMX__) && !defined(SDL_DISABLE_MMINTRIN_H) +#include +#endif +#if defined(__SSE__) && !defined(SDL_DISABLE_XMMINTRIN_H) +#include +#endif +#if defined(__SSE2__) && !defined(SDL_DISABLE_EMMINTRIN_H) +#include +#endif +#if defined(__SSE3__) && !defined(SDL_DISABLE_PMMINTRIN_H) +#include +#endif +#endif /* HAVE_IMMINTRIN_H */ + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* This is a guess for the cacheline size used for padding. + * Most x86 processors have a 64 byte cache line. + * The 64-bit PowerPC processors have a 128 byte cache line. + * We'll use the larger value to be generally safe. + */ +#define SDL_CACHELINE_SIZE 128 + +/** + * Get the number of CPU cores available. + * + * \returns the total number of logical CPU cores. On CPUs that include + * technologies such as hyperthreading, the number of logical cores + * may be more than the number of physical cores. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC int SDLCALL SDL_GetCPUCount(void); + +/** + * Determine the L1 cache line size of the CPU. + * + * This is useful for determining multi-threaded structure padding or SIMD + * prefetch sizes. + * + * \returns the L1 cache line size of the CPU, in bytes. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC int SDLCALL SDL_GetCPUCacheLineSize(void); + +/** + * Determine whether the CPU has the RDTSC instruction. + * + * This always returns false on CPUs that aren't using Intel instruction sets. + * + * \returns SDL_TRUE if the CPU has the RDTSC instruction or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Has3DNow + * \sa SDL_HasAltiVec + * \sa SDL_HasAVX + * \sa SDL_HasAVX2 + * \sa SDL_HasMMX + * \sa SDL_HasSSE + * \sa SDL_HasSSE2 + * \sa SDL_HasSSE3 + * \sa SDL_HasSSE41 + * \sa SDL_HasSSE42 + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasRDTSC(void); + +/** + * Determine whether the CPU has AltiVec features. + * + * This always returns false on CPUs that aren't using PowerPC instruction + * sets. + * + * \returns SDL_TRUE if the CPU has AltiVec features or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Has3DNow + * \sa SDL_HasAVX + * \sa SDL_HasAVX2 + * \sa SDL_HasMMX + * \sa SDL_HasRDTSC + * \sa SDL_HasSSE + * \sa SDL_HasSSE2 + * \sa SDL_HasSSE3 + * \sa SDL_HasSSE41 + * \sa SDL_HasSSE42 + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasAltiVec(void); + +/** + * Determine whether the CPU has MMX features. + * + * This always returns false on CPUs that aren't using Intel instruction sets. + * + * \returns SDL_TRUE if the CPU has MMX features or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Has3DNow + * \sa SDL_HasAltiVec + * \sa SDL_HasAVX + * \sa SDL_HasAVX2 + * \sa SDL_HasRDTSC + * \sa SDL_HasSSE + * \sa SDL_HasSSE2 + * \sa SDL_HasSSE3 + * \sa SDL_HasSSE41 + * \sa SDL_HasSSE42 + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasMMX(void); + +/** + * Determine whether the CPU has 3DNow! features. + * + * This always returns false on CPUs that aren't using AMD instruction sets. + * + * \returns SDL_TRUE if the CPU has 3DNow! features or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HasAltiVec + * \sa SDL_HasAVX + * \sa SDL_HasAVX2 + * \sa SDL_HasMMX + * \sa SDL_HasRDTSC + * \sa SDL_HasSSE + * \sa SDL_HasSSE2 + * \sa SDL_HasSSE3 + * \sa SDL_HasSSE41 + * \sa SDL_HasSSE42 + */ +extern DECLSPEC SDL_bool SDLCALL SDL_Has3DNow(void); + +/** + * Determine whether the CPU has SSE features. + * + * This always returns false on CPUs that aren't using Intel instruction sets. + * + * \returns SDL_TRUE if the CPU has SSE features or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Has3DNow + * \sa SDL_HasAltiVec + * \sa SDL_HasAVX + * \sa SDL_HasAVX2 + * \sa SDL_HasMMX + * \sa SDL_HasRDTSC + * \sa SDL_HasSSE2 + * \sa SDL_HasSSE3 + * \sa SDL_HasSSE41 + * \sa SDL_HasSSE42 + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasSSE(void); + +/** + * Determine whether the CPU has SSE2 features. + * + * This always returns false on CPUs that aren't using Intel instruction sets. + * + * \returns SDL_TRUE if the CPU has SSE2 features or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Has3DNow + * \sa SDL_HasAltiVec + * \sa SDL_HasAVX + * \sa SDL_HasAVX2 + * \sa SDL_HasMMX + * \sa SDL_HasRDTSC + * \sa SDL_HasSSE + * \sa SDL_HasSSE3 + * \sa SDL_HasSSE41 + * \sa SDL_HasSSE42 + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasSSE2(void); + +/** + * Determine whether the CPU has SSE3 features. + * + * This always returns false on CPUs that aren't using Intel instruction sets. + * + * \returns SDL_TRUE if the CPU has SSE3 features or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Has3DNow + * \sa SDL_HasAltiVec + * \sa SDL_HasAVX + * \sa SDL_HasAVX2 + * \sa SDL_HasMMX + * \sa SDL_HasRDTSC + * \sa SDL_HasSSE + * \sa SDL_HasSSE2 + * \sa SDL_HasSSE41 + * \sa SDL_HasSSE42 + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasSSE3(void); + +/** + * Determine whether the CPU has SSE4.1 features. + * + * This always returns false on CPUs that aren't using Intel instruction sets. + * + * \returns SDL_TRUE if the CPU has SSE4.1 features or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Has3DNow + * \sa SDL_HasAltiVec + * \sa SDL_HasAVX + * \sa SDL_HasAVX2 + * \sa SDL_HasMMX + * \sa SDL_HasRDTSC + * \sa SDL_HasSSE + * \sa SDL_HasSSE2 + * \sa SDL_HasSSE3 + * \sa SDL_HasSSE42 + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasSSE41(void); + +/** + * Determine whether the CPU has SSE4.2 features. + * + * This always returns false on CPUs that aren't using Intel instruction sets. + * + * \returns SDL_TRUE if the CPU has SSE4.2 features or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Has3DNow + * \sa SDL_HasAltiVec + * \sa SDL_HasAVX + * \sa SDL_HasAVX2 + * \sa SDL_HasMMX + * \sa SDL_HasRDTSC + * \sa SDL_HasSSE + * \sa SDL_HasSSE2 + * \sa SDL_HasSSE3 + * \sa SDL_HasSSE41 + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasSSE42(void); + +/** + * Determine whether the CPU has AVX features. + * + * This always returns false on CPUs that aren't using Intel instruction sets. + * + * \returns SDL_TRUE if the CPU has AVX features or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.2. + * + * \sa SDL_Has3DNow + * \sa SDL_HasAltiVec + * \sa SDL_HasAVX2 + * \sa SDL_HasMMX + * \sa SDL_HasRDTSC + * \sa SDL_HasSSE + * \sa SDL_HasSSE2 + * \sa SDL_HasSSE3 + * \sa SDL_HasSSE41 + * \sa SDL_HasSSE42 + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasAVX(void); + +/** + * Determine whether the CPU has AVX2 features. + * + * This always returns false on CPUs that aren't using Intel instruction sets. + * + * \returns SDL_TRUE if the CPU has AVX2 features or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.4. + * + * \sa SDL_Has3DNow + * \sa SDL_HasAltiVec + * \sa SDL_HasAVX + * \sa SDL_HasMMX + * \sa SDL_HasRDTSC + * \sa SDL_HasSSE + * \sa SDL_HasSSE2 + * \sa SDL_HasSSE3 + * \sa SDL_HasSSE41 + * \sa SDL_HasSSE42 + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasAVX2(void); + +/** + * Determine whether the CPU has AVX-512F (foundation) features. + * + * This always returns false on CPUs that aren't using Intel instruction sets. + * + * \returns SDL_TRUE if the CPU has AVX-512F features or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.9. + * + * \sa SDL_HasAVX + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasAVX512F(void); + +/** + * Determine whether the CPU has ARM SIMD (ARMv6) features. + * + * This is different from ARM NEON, which is a different instruction set. + * + * This always returns false on CPUs that aren't using ARM instruction sets. + * + * \returns SDL_TRUE if the CPU has ARM SIMD features or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.12. + * + * \sa SDL_HasNEON + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasARMSIMD(void); + +/** + * Determine whether the CPU has NEON (ARM SIMD) features. + * + * This always returns false on CPUs that aren't using ARM instruction sets. + * + * \returns SDL_TRUE if the CPU has ARM NEON features or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasNEON(void); + +/** + * Determine whether the CPU has LSX (LOONGARCH SIMD) features. + * + * This always returns false on CPUs that aren't using LOONGARCH instruction + * sets. + * + * \returns SDL_TRUE if the CPU has LOONGARCH LSX features or SDL_FALSE if + * not. + * + * \since This function is available since SDL 2.24.0. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasLSX(void); + +/** + * Determine whether the CPU has LASX (LOONGARCH SIMD) features. + * + * This always returns false on CPUs that aren't using LOONGARCH instruction + * sets. + * + * \returns SDL_TRUE if the CPU has LOONGARCH LASX features or SDL_FALSE if + * not. + * + * \since This function is available since SDL 2.24.0. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasLASX(void); + +/** + * Get the amount of RAM configured in the system. + * + * \returns the amount of RAM configured in the system in MiB. + * + * \since This function is available since SDL 2.0.1. + */ +extern DECLSPEC int SDLCALL SDL_GetSystemRAM(void); + +/** + * Report the alignment this system needs for SIMD allocations. + * + * This will return the minimum number of bytes to which a pointer must be + * aligned to be compatible with SIMD instructions on the current machine. For + * example, if the machine supports SSE only, it will return 16, but if it + * supports AVX-512F, it'll return 64 (etc). This only reports values for + * instruction sets SDL knows about, so if your SDL build doesn't have + * SDL_HasAVX512F(), then it might return 16 for the SSE support it sees and + * not 64 for the AVX-512 instructions that exist but SDL doesn't know about. + * Plan accordingly. + * + * \returns the alignment in bytes needed for available, known SIMD + * instructions. + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC size_t SDLCALL SDL_SIMDGetAlignment(void); + +/** + * Allocate memory in a SIMD-friendly way. + * + * This will allocate a block of memory that is suitable for use with SIMD + * instructions. Specifically, it will be properly aligned and padded for the + * system's supported vector instructions. + * + * The memory returned will be padded such that it is safe to read or write an + * incomplete vector at the end of the memory block. This can be useful so you + * don't have to drop back to a scalar fallback at the end of your SIMD + * processing loop to deal with the final elements without overflowing the + * allocated buffer. + * + * You must free this memory with SDL_FreeSIMD(), not free() or SDL_free() or + * delete[], etc. + * + * Note that SDL will only deal with SIMD instruction sets it is aware of; for + * example, SDL 2.0.8 knows that SSE wants 16-byte vectors (SDL_HasSSE()), and + * AVX2 wants 32 bytes (SDL_HasAVX2()), but doesn't know that AVX-512 wants + * 64. To be clear: if you can't decide to use an instruction set with an + * SDL_Has*() function, don't use that instruction set with memory allocated + * through here. + * + * SDL_AllocSIMD(0) will return a non-NULL pointer, assuming the system isn't + * out of memory, but you are not allowed to dereference it (because you only + * own zero bytes of that buffer). + * + * \param len The length, in bytes, of the block to allocate. The actual + * allocated block might be larger due to padding, etc. + * \returns a pointer to the newly-allocated block, NULL if out of memory. + * + * \since This function is available since SDL 2.0.10. + * + * \sa SDL_SIMDGetAlignment + * \sa SDL_SIMDRealloc + * \sa SDL_SIMDFree + */ +extern DECLSPEC void * SDLCALL SDL_SIMDAlloc(const size_t len); + +/** + * Reallocate memory obtained from SDL_SIMDAlloc + * + * It is not valid to use this function on a pointer from anything but + * SDL_SIMDAlloc(). It can't be used on pointers from malloc, realloc, + * SDL_malloc, memalign, new[], etc. + * + * \param mem The pointer obtained from SDL_SIMDAlloc. This function also + * accepts NULL, at which point this function is the same as + * calling SDL_SIMDAlloc with a NULL pointer. + * \param len The length, in bytes, of the block to allocated. The actual + * allocated block might be larger due to padding, etc. Passing 0 + * will return a non-NULL pointer, assuming the system isn't out of + * memory. + * \returns a pointer to the newly-reallocated block, NULL if out of memory. + * + * \since This function is available since SDL 2.0.14. + * + * \sa SDL_SIMDGetAlignment + * \sa SDL_SIMDAlloc + * \sa SDL_SIMDFree + */ +extern DECLSPEC void * SDLCALL SDL_SIMDRealloc(void *mem, const size_t len); + +/** + * Deallocate memory obtained from SDL_SIMDAlloc + * + * It is not valid to use this function on a pointer from anything but + * SDL_SIMDAlloc() or SDL_SIMDRealloc(). It can't be used on pointers from + * malloc, realloc, SDL_malloc, memalign, new[], etc. + * + * However, SDL_SIMDFree(NULL) is a legal no-op. + * + * The memory pointed to by `ptr` is no longer valid for access upon return, + * and may be returned to the system or reused by a future allocation. The + * pointer passed to this function is no longer safe to dereference once this + * function returns, and should be discarded. + * + * \param ptr The pointer, returned from SDL_SIMDAlloc or SDL_SIMDRealloc, to + * deallocate. NULL is a legal no-op. + * + * \since This function is available since SDL 2.0.10. + * + * \sa SDL_SIMDAlloc + * \sa SDL_SIMDRealloc + */ +extern DECLSPEC void SDLCALL SDL_SIMDFree(void *ptr); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_cpuinfo_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_egl.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_egl.h new file mode 100644 index 00000000..6f51c083 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_egl.h @@ -0,0 +1,2352 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_egl.h + * + * This is a simple file to encapsulate the EGL API headers. + */ +#if !defined(_MSC_VER) && !defined(__ANDROID__) && !defined(SDL_USE_BUILTIN_OPENGL_DEFINITIONS) + +#if defined(__vita__) || defined(__psp2__) +#include +#endif + +#include +#include + +#else /* _MSC_VER */ + +/* EGL headers for Visual Studio */ + +#ifndef __khrplatform_h_ +#define __khrplatform_h_ + +/* +** Copyright (c) 2008-2018 The Khronos Group Inc. +** +** Permission is hereby granted, free of charge, to any person obtaining a +** copy of this software and/or associated documentation files (the +** "Materials"), to deal in the Materials without restriction, including +** without limitation the rights to use, copy, modify, merge, publish, +** distribute, sublicense, and/or sell copies of the Materials, and to +** permit persons to whom the Materials are furnished to do so, subject to +** the following conditions: +** +** The above copyright notice and this permission notice shall be included +** in all copies or substantial portions of the Materials. +** +** THE MATERIALS ARE PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. +** IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY +** CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, +** TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE +** MATERIALS OR THE USE OR OTHER DEALINGS IN THE MATERIALS. +*/ + +/* Khronos platform-specific types and definitions. + * + * The master copy of khrplatform.h is maintained in the Khronos EGL + * Registry repository at https://github.com/KhronosGroup/EGL-Registry + * The last semantic modification to khrplatform.h was at commit ID: + * 67a3e0864c2d75ea5287b9f3d2eb74a745936692 + * + * Adopters may modify this file to suit their platform. Adopters are + * encouraged to submit platform specific modifications to the Khronos + * group so that they can be included in future versions of this file. + * Please submit changes by filing pull requests or issues on + * the EGL Registry repository linked above. + * + * + * See the Implementer's Guidelines for information about where this file + * should be located on your system and for more details of its use: + * http://www.khronos.org/registry/implementers_guide.pdf + * + * This file should be included as + * #include + * by Khronos client API header files that use its types and defines. + * + * The types in khrplatform.h should only be used to define API-specific types. + * + * Types defined in khrplatform.h: + * khronos_int8_t signed 8 bit + * khronos_uint8_t unsigned 8 bit + * khronos_int16_t signed 16 bit + * khronos_uint16_t unsigned 16 bit + * khronos_int32_t signed 32 bit + * khronos_uint32_t unsigned 32 bit + * khronos_int64_t signed 64 bit + * khronos_uint64_t unsigned 64 bit + * khronos_intptr_t signed same number of bits as a pointer + * khronos_uintptr_t unsigned same number of bits as a pointer + * khronos_ssize_t signed size + * khronos_usize_t unsigned size + * khronos_float_t signed 32 bit floating point + * khronos_time_ns_t unsigned 64 bit time in nanoseconds + * khronos_utime_nanoseconds_t unsigned time interval or absolute time in + * nanoseconds + * khronos_stime_nanoseconds_t signed time interval in nanoseconds + * khronos_boolean_enum_t enumerated boolean type. This should + * only be used as a base type when a client API's boolean type is + * an enum. Client APIs which use an integer or other type for + * booleans cannot use this as the base type for their boolean. + * + * Tokens defined in khrplatform.h: + * + * KHRONOS_FALSE, KHRONOS_TRUE Enumerated boolean false/true values. + * + * KHRONOS_SUPPORT_INT64 is 1 if 64 bit integers are supported; otherwise 0. + * KHRONOS_SUPPORT_FLOAT is 1 if floats are supported; otherwise 0. + * + * Calling convention macros defined in this file: + * KHRONOS_APICALL + * KHRONOS_APIENTRY + * KHRONOS_APIATTRIBUTES + * + * These may be used in function prototypes as: + * + * KHRONOS_APICALL void KHRONOS_APIENTRY funcname( + * int arg1, + * int arg2) KHRONOS_APIATTRIBUTES; + */ + +#if defined(__SCITECH_SNAP__) && !defined(KHRONOS_STATIC) +# define KHRONOS_STATIC 1 +#endif + +/*------------------------------------------------------------------------- + * Definition of KHRONOS_APICALL + *------------------------------------------------------------------------- + * This precedes the return type of the function in the function prototype. + */ +#if defined(KHRONOS_STATIC) + /* If the preprocessor constant KHRONOS_STATIC is defined, make the + * header compatible with static linking. */ +# define KHRONOS_APICALL +#elif defined(_WIN32) +# define KHRONOS_APICALL __declspec(dllimport) +#elif defined (__SYMBIAN32__) +# define KHRONOS_APICALL IMPORT_C +#elif defined(__ANDROID__) +# define KHRONOS_APICALL __attribute__((visibility("default"))) +#else +# define KHRONOS_APICALL +#endif + +/*------------------------------------------------------------------------- + * Definition of KHRONOS_APIENTRY + *------------------------------------------------------------------------- + * This follows the return type of the function and precedes the function + * name in the function prototype. + */ +#if defined(_WIN32) && !defined(_WIN32_WCE) && !defined(__SCITECH_SNAP__) + /* Win32 but not WinCE */ +# define KHRONOS_APIENTRY __stdcall +#else +# define KHRONOS_APIENTRY +#endif + +/*------------------------------------------------------------------------- + * Definition of KHRONOS_APIATTRIBUTES + *------------------------------------------------------------------------- + * This follows the closing parenthesis of the function prototype arguments. + */ +#if defined (__ARMCC_2__) +#define KHRONOS_APIATTRIBUTES __softfp +#else +#define KHRONOS_APIATTRIBUTES +#endif + +/*------------------------------------------------------------------------- + * basic type definitions + *-----------------------------------------------------------------------*/ +#if (defined(__STDC_VERSION__) && __STDC_VERSION__ >= 199901L) || defined(__GNUC__) || defined(__SCO__) || defined(__USLC__) + + +/* + * Using + */ +#include +typedef int32_t khronos_int32_t; +typedef uint32_t khronos_uint32_t; +typedef int64_t khronos_int64_t; +typedef uint64_t khronos_uint64_t; +#define KHRONOS_SUPPORT_INT64 1 +#define KHRONOS_SUPPORT_FLOAT 1 +/* + * To support platform where unsigned long cannot be used interchangeably with + * inptr_t (e.g. CHERI-extended ISAs), we can use the stdint.h intptr_t. + * Ideally, we could just use (u)intptr_t everywhere, but this could result in + * ABI breakage if khronos_uintptr_t is changed from unsigned long to + * unsigned long long or similar (this results in different C++ name mangling). + * To avoid changes for existing platforms, we restrict usage of intptr_t to + * platforms where the size of a pointer is larger than the size of long. + */ +#if defined(__SIZEOF_LONG__) && defined(__SIZEOF_POINTER__) +#if __SIZEOF_POINTER__ > __SIZEOF_LONG__ +#define KHRONOS_USE_INTPTR_T +#endif +#endif + +#elif defined(__VMS ) || defined(__sgi) + +/* + * Using + */ +#include +typedef int32_t khronos_int32_t; +typedef uint32_t khronos_uint32_t; +typedef int64_t khronos_int64_t; +typedef uint64_t khronos_uint64_t; +#define KHRONOS_SUPPORT_INT64 1 +#define KHRONOS_SUPPORT_FLOAT 1 + +#elif defined(_WIN32) && !defined(__SCITECH_SNAP__) + +/* + * Win32 + */ +typedef __int32 khronos_int32_t; +typedef unsigned __int32 khronos_uint32_t; +typedef __int64 khronos_int64_t; +typedef unsigned __int64 khronos_uint64_t; +#define KHRONOS_SUPPORT_INT64 1 +#define KHRONOS_SUPPORT_FLOAT 1 + +#elif defined(__sun__) || defined(__digital__) + +/* + * Sun or Digital + */ +typedef int khronos_int32_t; +typedef unsigned int khronos_uint32_t; +#if defined(__arch64__) || defined(_LP64) +typedef long int khronos_int64_t; +typedef unsigned long int khronos_uint64_t; +#else +typedef long long int khronos_int64_t; +typedef unsigned long long int khronos_uint64_t; +#endif /* __arch64__ */ +#define KHRONOS_SUPPORT_INT64 1 +#define KHRONOS_SUPPORT_FLOAT 1 + +#elif 0 + +/* + * Hypothetical platform with no float or int64 support + */ +typedef int khronos_int32_t; +typedef unsigned int khronos_uint32_t; +#define KHRONOS_SUPPORT_INT64 0 +#define KHRONOS_SUPPORT_FLOAT 0 + +#else + +/* + * Generic fallback + */ +#include +typedef int32_t khronos_int32_t; +typedef uint32_t khronos_uint32_t; +typedef int64_t khronos_int64_t; +typedef uint64_t khronos_uint64_t; +#define KHRONOS_SUPPORT_INT64 1 +#define KHRONOS_SUPPORT_FLOAT 1 + +#endif + + +/* + * Types that are (so far) the same on all platforms + */ +typedef signed char khronos_int8_t; +typedef unsigned char khronos_uint8_t; +typedef signed short int khronos_int16_t; +typedef unsigned short int khronos_uint16_t; + +/* + * Types that differ between LLP64 and LP64 architectures - in LLP64, + * pointers are 64 bits, but 'long' is still 32 bits. Win64 appears + * to be the only LLP64 architecture in current use. + */ +#ifdef KHRONOS_USE_INTPTR_T +typedef intptr_t khronos_intptr_t; +typedef uintptr_t khronos_uintptr_t; +#elif defined(_WIN64) +typedef signed long long int khronos_intptr_t; +typedef unsigned long long int khronos_uintptr_t; +#else +typedef signed long int khronos_intptr_t; +typedef unsigned long int khronos_uintptr_t; +#endif + +#if defined(_WIN64) +typedef signed long long int khronos_ssize_t; +typedef unsigned long long int khronos_usize_t; +#else +typedef signed long int khronos_ssize_t; +typedef unsigned long int khronos_usize_t; +#endif + +#if KHRONOS_SUPPORT_FLOAT +/* + * Float type + */ +typedef float khronos_float_t; +#endif + +#if KHRONOS_SUPPORT_INT64 +/* Time types + * + * These types can be used to represent a time interval in nanoseconds or + * an absolute Unadjusted System Time. Unadjusted System Time is the number + * of nanoseconds since some arbitrary system event (e.g. since the last + * time the system booted). The Unadjusted System Time is an unsigned + * 64 bit value that wraps back to 0 every 584 years. Time intervals + * may be either signed or unsigned. + */ +typedef khronos_uint64_t khronos_utime_nanoseconds_t; +typedef khronos_int64_t khronos_stime_nanoseconds_t; +#endif + +/* + * Dummy value used to pad enum types to 32 bits. + */ +#ifndef KHRONOS_MAX_ENUM +#define KHRONOS_MAX_ENUM 0x7FFFFFFF +#endif + +/* + * Enumerated boolean type + * + * Values other than zero should be considered to be true. Therefore + * comparisons should not be made against KHRONOS_TRUE. + */ +typedef enum { + KHRONOS_FALSE = 0, + KHRONOS_TRUE = 1, + KHRONOS_BOOLEAN_ENUM_FORCE_SIZE = KHRONOS_MAX_ENUM +} khronos_boolean_enum_t; + +#endif /* __khrplatform_h_ */ + + +#ifndef __eglplatform_h_ +#define __eglplatform_h_ + +/* +** Copyright 2007-2020 The Khronos Group Inc. +** SPDX-License-Identifier: Apache-2.0 +*/ + +/* Platform-specific types and definitions for egl.h + * + * Adopters may modify khrplatform.h and this file to suit their platform. + * You are encouraged to submit all modifications to the Khronos group so that + * they can be included in future versions of this file. Please submit changes + * by filing an issue or pull request on the public Khronos EGL Registry, at + * https://www.github.com/KhronosGroup/EGL-Registry/ + */ + +/*#include */ + +/* Macros used in EGL function prototype declarations. + * + * EGL functions should be prototyped as: + * + * EGLAPI return-type EGLAPIENTRY eglFunction(arguments); + * typedef return-type (EXPAPIENTRYP PFNEGLFUNCTIONPROC) (arguments); + * + * KHRONOS_APICALL and KHRONOS_APIENTRY are defined in KHR/khrplatform.h + */ + +#ifndef EGLAPI +#define EGLAPI KHRONOS_APICALL +#endif + +#ifndef EGLAPIENTRY +#define EGLAPIENTRY KHRONOS_APIENTRY +#endif +#define EGLAPIENTRYP EGLAPIENTRY* + +/* The types NativeDisplayType, NativeWindowType, and NativePixmapType + * are aliases of window-system-dependent types, such as X Display * or + * Windows Device Context. They must be defined in platform-specific + * code below. The EGL-prefixed versions of Native*Type are the same + * types, renamed in EGL 1.3 so all types in the API start with "EGL". + * + * Khronos STRONGLY RECOMMENDS that you use the default definitions + * provided below, since these changes affect both binary and source + * portability of applications using EGL running on different EGL + * implementations. + */ + +#if defined(EGL_NO_PLATFORM_SPECIFIC_TYPES) + +typedef void *EGLNativeDisplayType; +typedef void *EGLNativePixmapType; +typedef void *EGLNativeWindowType; + +#elif defined(_WIN32) || defined(__VC32__) && !defined(__CYGWIN__) && !defined(__SCITECH_SNAP__) /* Win32 and WinCE */ +#ifndef WIN32_LEAN_AND_MEAN +#define WIN32_LEAN_AND_MEAN 1 +#endif +#include + +typedef HDC EGLNativeDisplayType; +typedef HBITMAP EGLNativePixmapType; +typedef HWND EGLNativeWindowType; + +#elif defined(__EMSCRIPTEN__) + +typedef int EGLNativeDisplayType; +typedef int EGLNativePixmapType; +typedef int EGLNativeWindowType; + +#elif defined(__WINSCW__) || defined(__SYMBIAN32__) /* Symbian */ + +typedef int EGLNativeDisplayType; +typedef void *EGLNativePixmapType; +typedef void *EGLNativeWindowType; + +#elif defined(WL_EGL_PLATFORM) + +typedef struct wl_display *EGLNativeDisplayType; +typedef struct wl_egl_pixmap *EGLNativePixmapType; +typedef struct wl_egl_window *EGLNativeWindowType; + +#elif defined(__GBM__) + +typedef struct gbm_device *EGLNativeDisplayType; +typedef struct gbm_bo *EGLNativePixmapType; +typedef void *EGLNativeWindowType; + +#elif defined(__ANDROID__) || defined(ANDROID) + +struct ANativeWindow; +struct egl_native_pixmap_t; + +typedef void* EGLNativeDisplayType; +typedef struct egl_native_pixmap_t* EGLNativePixmapType; +typedef struct ANativeWindow* EGLNativeWindowType; + +#elif defined(USE_OZONE) + +typedef intptr_t EGLNativeDisplayType; +typedef intptr_t EGLNativePixmapType; +typedef intptr_t EGLNativeWindowType; + +#elif defined(USE_X11) + +/* X11 (tentative) */ +#include +#include + +typedef Display *EGLNativeDisplayType; +typedef Pixmap EGLNativePixmapType; +typedef Window EGLNativeWindowType; + +#elif defined(__unix__) + +typedef void *EGLNativeDisplayType; +typedef khronos_uintptr_t EGLNativePixmapType; +typedef khronos_uintptr_t EGLNativeWindowType; + +#elif defined(__APPLE__) + +typedef int EGLNativeDisplayType; +typedef void *EGLNativePixmapType; +typedef void *EGLNativeWindowType; + +#elif defined(__HAIKU__) + +#include + +typedef void *EGLNativeDisplayType; +typedef khronos_uintptr_t EGLNativePixmapType; +typedef khronos_uintptr_t EGLNativeWindowType; + +#elif defined(__Fuchsia__) + +typedef void *EGLNativeDisplayType; +typedef khronos_uintptr_t EGLNativePixmapType; +typedef khronos_uintptr_t EGLNativeWindowType; + +#else +#error "Platform not recognized" +#endif + +/* EGL 1.2 types, renamed for consistency in EGL 1.3 */ +typedef EGLNativeDisplayType NativeDisplayType; +typedef EGLNativePixmapType NativePixmapType; +typedef EGLNativeWindowType NativeWindowType; + + +/* Define EGLint. This must be a signed integral type large enough to contain + * all legal attribute names and values passed into and out of EGL, whether + * their type is boolean, bitmask, enumerant (symbolic constant), integer, + * handle, or other. While in general a 32-bit integer will suffice, if + * handles are 64 bit types, then EGLint should be defined as a signed 64-bit + * integer type. + */ +typedef khronos_int32_t EGLint; + + +/* C++ / C typecast macros for special EGL handle values */ +#if defined(__cplusplus) +#define EGL_CAST(type, value) (static_cast(value)) +#else +#define EGL_CAST(type, value) ((type) (value)) +#endif + +#endif /* __eglplatform_h */ + + +#ifndef __egl_h_ +#define __egl_h_ 1 + +#ifdef __cplusplus +extern "C" { +#endif + +/* +** Copyright 2013-2020 The Khronos Group Inc. +** SPDX-License-Identifier: Apache-2.0 +** +** This header is generated from the Khronos EGL XML API Registry. +** The current version of the Registry, generator scripts +** used to make the header, and the header can be found at +** http://www.khronos.org/registry/egl +** +** Khronos $Git commit SHA1: 6fb1daea15 $ on $Git commit date: 2022-05-25 09:41:13 -0600 $ +*/ + +/*#include */ + +#ifndef EGL_EGL_PROTOTYPES +#define EGL_EGL_PROTOTYPES 1 +#endif + +/* Generated on date 20220525 */ + +/* Generated C header for: + * API: egl + * Versions considered: .* + * Versions emitted: .* + * Default extensions included: None + * Additional extensions included: _nomatch_^ + * Extensions removed: _nomatch_^ + */ + +#ifndef EGL_VERSION_1_0 +#define EGL_VERSION_1_0 1 +typedef unsigned int EGLBoolean; +typedef void *EGLDisplay; +/*#include */ +/*#include */ +typedef void *EGLConfig; +typedef void *EGLSurface; +typedef void *EGLContext; +typedef void (*__eglMustCastToProperFunctionPointerType)(void); +#define EGL_ALPHA_SIZE 0x3021 +#define EGL_BAD_ACCESS 0x3002 +#define EGL_BAD_ALLOC 0x3003 +#define EGL_BAD_ATTRIBUTE 0x3004 +#define EGL_BAD_CONFIG 0x3005 +#define EGL_BAD_CONTEXT 0x3006 +#define EGL_BAD_CURRENT_SURFACE 0x3007 +#define EGL_BAD_DISPLAY 0x3008 +#define EGL_BAD_MATCH 0x3009 +#define EGL_BAD_NATIVE_PIXMAP 0x300A +#define EGL_BAD_NATIVE_WINDOW 0x300B +#define EGL_BAD_PARAMETER 0x300C +#define EGL_BAD_SURFACE 0x300D +#define EGL_BLUE_SIZE 0x3022 +#define EGL_BUFFER_SIZE 0x3020 +#define EGL_CONFIG_CAVEAT 0x3027 +#define EGL_CONFIG_ID 0x3028 +#define EGL_CORE_NATIVE_ENGINE 0x305B +#define EGL_DEPTH_SIZE 0x3025 +#define EGL_DONT_CARE EGL_CAST(EGLint,-1) +#define EGL_DRAW 0x3059 +#define EGL_EXTENSIONS 0x3055 +#define EGL_FALSE 0 +#define EGL_GREEN_SIZE 0x3023 +#define EGL_HEIGHT 0x3056 +#define EGL_LARGEST_PBUFFER 0x3058 +#define EGL_LEVEL 0x3029 +#define EGL_MAX_PBUFFER_HEIGHT 0x302A +#define EGL_MAX_PBUFFER_PIXELS 0x302B +#define EGL_MAX_PBUFFER_WIDTH 0x302C +#define EGL_NATIVE_RENDERABLE 0x302D +#define EGL_NATIVE_VISUAL_ID 0x302E +#define EGL_NATIVE_VISUAL_TYPE 0x302F +#define EGL_NONE 0x3038 +#define EGL_NON_CONFORMANT_CONFIG 0x3051 +#define EGL_NOT_INITIALIZED 0x3001 +#define EGL_NO_CONTEXT EGL_CAST(EGLContext,0) +#define EGL_NO_DISPLAY EGL_CAST(EGLDisplay,0) +#define EGL_NO_SURFACE EGL_CAST(EGLSurface,0) +#define EGL_PBUFFER_BIT 0x0001 +#define EGL_PIXMAP_BIT 0x0002 +#define EGL_READ 0x305A +#define EGL_RED_SIZE 0x3024 +#define EGL_SAMPLES 0x3031 +#define EGL_SAMPLE_BUFFERS 0x3032 +#define EGL_SLOW_CONFIG 0x3050 +#define EGL_STENCIL_SIZE 0x3026 +#define EGL_SUCCESS 0x3000 +#define EGL_SURFACE_TYPE 0x3033 +#define EGL_TRANSPARENT_BLUE_VALUE 0x3035 +#define EGL_TRANSPARENT_GREEN_VALUE 0x3036 +#define EGL_TRANSPARENT_RED_VALUE 0x3037 +#define EGL_TRANSPARENT_RGB 0x3052 +#define EGL_TRANSPARENT_TYPE 0x3034 +#define EGL_TRUE 1 +#define EGL_VENDOR 0x3053 +#define EGL_VERSION 0x3054 +#define EGL_WIDTH 0x3057 +#define EGL_WINDOW_BIT 0x0004 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLCHOOSECONFIGPROC) (EGLDisplay dpy, const EGLint *attrib_list, EGLConfig *configs, EGLint config_size, EGLint *num_config); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLCOPYBUFFERSPROC) (EGLDisplay dpy, EGLSurface surface, EGLNativePixmapType target); +typedef EGLContext (EGLAPIENTRYP PFNEGLCREATECONTEXTPROC) (EGLDisplay dpy, EGLConfig config, EGLContext share_context, const EGLint *attrib_list); +typedef EGLSurface (EGLAPIENTRYP PFNEGLCREATEPBUFFERSURFACEPROC) (EGLDisplay dpy, EGLConfig config, const EGLint *attrib_list); +typedef EGLSurface (EGLAPIENTRYP PFNEGLCREATEPIXMAPSURFACEPROC) (EGLDisplay dpy, EGLConfig config, EGLNativePixmapType pixmap, const EGLint *attrib_list); +typedef EGLSurface (EGLAPIENTRYP PFNEGLCREATEWINDOWSURFACEPROC) (EGLDisplay dpy, EGLConfig config, EGLNativeWindowType win, const EGLint *attrib_list); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLDESTROYCONTEXTPROC) (EGLDisplay dpy, EGLContext ctx); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLDESTROYSURFACEPROC) (EGLDisplay dpy, EGLSurface surface); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLGETCONFIGATTRIBPROC) (EGLDisplay dpy, EGLConfig config, EGLint attribute, EGLint *value); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLGETCONFIGSPROC) (EGLDisplay dpy, EGLConfig *configs, EGLint config_size, EGLint *num_config); +typedef EGLDisplay (EGLAPIENTRYP PFNEGLGETCURRENTDISPLAYPROC) (void); +typedef EGLSurface (EGLAPIENTRYP PFNEGLGETCURRENTSURFACEPROC) (EGLint readdraw); +typedef EGLDisplay (EGLAPIENTRYP PFNEGLGETDISPLAYPROC) (EGLNativeDisplayType display_id); +typedef EGLint (EGLAPIENTRYP PFNEGLGETERRORPROC) (void); +typedef __eglMustCastToProperFunctionPointerType (EGLAPIENTRYP PFNEGLGETPROCADDRESSPROC) (const char *procname); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLINITIALIZEPROC) (EGLDisplay dpy, EGLint *major, EGLint *minor); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLMAKECURRENTPROC) (EGLDisplay dpy, EGLSurface draw, EGLSurface read, EGLContext ctx); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYCONTEXTPROC) (EGLDisplay dpy, EGLContext ctx, EGLint attribute, EGLint *value); +typedef const char *(EGLAPIENTRYP PFNEGLQUERYSTRINGPROC) (EGLDisplay dpy, EGLint name); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYSURFACEPROC) (EGLDisplay dpy, EGLSurface surface, EGLint attribute, EGLint *value); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSWAPBUFFERSPROC) (EGLDisplay dpy, EGLSurface surface); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLTERMINATEPROC) (EGLDisplay dpy); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLWAITGLPROC) (void); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLWAITNATIVEPROC) (EGLint engine); +#if EGL_EGL_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglChooseConfig (EGLDisplay dpy, const EGLint *attrib_list, EGLConfig *configs, EGLint config_size, EGLint *num_config); +EGLAPI EGLBoolean EGLAPIENTRY eglCopyBuffers (EGLDisplay dpy, EGLSurface surface, EGLNativePixmapType target); +EGLAPI EGLContext EGLAPIENTRY eglCreateContext (EGLDisplay dpy, EGLConfig config, EGLContext share_context, const EGLint *attrib_list); +EGLAPI EGLSurface EGLAPIENTRY eglCreatePbufferSurface (EGLDisplay dpy, EGLConfig config, const EGLint *attrib_list); +EGLAPI EGLSurface EGLAPIENTRY eglCreatePixmapSurface (EGLDisplay dpy, EGLConfig config, EGLNativePixmapType pixmap, const EGLint *attrib_list); +EGLAPI EGLSurface EGLAPIENTRY eglCreateWindowSurface (EGLDisplay dpy, EGLConfig config, EGLNativeWindowType win, const EGLint *attrib_list); +EGLAPI EGLBoolean EGLAPIENTRY eglDestroyContext (EGLDisplay dpy, EGLContext ctx); +EGLAPI EGLBoolean EGLAPIENTRY eglDestroySurface (EGLDisplay dpy, EGLSurface surface); +EGLAPI EGLBoolean EGLAPIENTRY eglGetConfigAttrib (EGLDisplay dpy, EGLConfig config, EGLint attribute, EGLint *value); +EGLAPI EGLBoolean EGLAPIENTRY eglGetConfigs (EGLDisplay dpy, EGLConfig *configs, EGLint config_size, EGLint *num_config); +EGLAPI EGLDisplay EGLAPIENTRY eglGetCurrentDisplay (void); +EGLAPI EGLSurface EGLAPIENTRY eglGetCurrentSurface (EGLint readdraw); +EGLAPI EGLDisplay EGLAPIENTRY eglGetDisplay (EGLNativeDisplayType display_id); +EGLAPI EGLint EGLAPIENTRY eglGetError (void); +EGLAPI __eglMustCastToProperFunctionPointerType EGLAPIENTRY eglGetProcAddress (const char *procname); +EGLAPI EGLBoolean EGLAPIENTRY eglInitialize (EGLDisplay dpy, EGLint *major, EGLint *minor); +EGLAPI EGLBoolean EGLAPIENTRY eglMakeCurrent (EGLDisplay dpy, EGLSurface draw, EGLSurface read, EGLContext ctx); +EGLAPI EGLBoolean EGLAPIENTRY eglQueryContext (EGLDisplay dpy, EGLContext ctx, EGLint attribute, EGLint *value); +EGLAPI const char *EGLAPIENTRY eglQueryString (EGLDisplay dpy, EGLint name); +EGLAPI EGLBoolean EGLAPIENTRY eglQuerySurface (EGLDisplay dpy, EGLSurface surface, EGLint attribute, EGLint *value); +EGLAPI EGLBoolean EGLAPIENTRY eglSwapBuffers (EGLDisplay dpy, EGLSurface surface); +EGLAPI EGLBoolean EGLAPIENTRY eglTerminate (EGLDisplay dpy); +EGLAPI EGLBoolean EGLAPIENTRY eglWaitGL (void); +EGLAPI EGLBoolean EGLAPIENTRY eglWaitNative (EGLint engine); +#endif +#endif /* EGL_VERSION_1_0 */ + +#ifndef EGL_VERSION_1_1 +#define EGL_VERSION_1_1 1 +#define EGL_BACK_BUFFER 0x3084 +#define EGL_BIND_TO_TEXTURE_RGB 0x3039 +#define EGL_BIND_TO_TEXTURE_RGBA 0x303A +#define EGL_CONTEXT_LOST 0x300E +#define EGL_MIN_SWAP_INTERVAL 0x303B +#define EGL_MAX_SWAP_INTERVAL 0x303C +#define EGL_MIPMAP_TEXTURE 0x3082 +#define EGL_MIPMAP_LEVEL 0x3083 +#define EGL_NO_TEXTURE 0x305C +#define EGL_TEXTURE_2D 0x305F +#define EGL_TEXTURE_FORMAT 0x3080 +#define EGL_TEXTURE_RGB 0x305D +#define EGL_TEXTURE_RGBA 0x305E +#define EGL_TEXTURE_TARGET 0x3081 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLBINDTEXIMAGEPROC) (EGLDisplay dpy, EGLSurface surface, EGLint buffer); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLRELEASETEXIMAGEPROC) (EGLDisplay dpy, EGLSurface surface, EGLint buffer); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSURFACEATTRIBPROC) (EGLDisplay dpy, EGLSurface surface, EGLint attribute, EGLint value); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSWAPINTERVALPROC) (EGLDisplay dpy, EGLint interval); +#if EGL_EGL_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglBindTexImage (EGLDisplay dpy, EGLSurface surface, EGLint buffer); +EGLAPI EGLBoolean EGLAPIENTRY eglReleaseTexImage (EGLDisplay dpy, EGLSurface surface, EGLint buffer); +EGLAPI EGLBoolean EGLAPIENTRY eglSurfaceAttrib (EGLDisplay dpy, EGLSurface surface, EGLint attribute, EGLint value); +EGLAPI EGLBoolean EGLAPIENTRY eglSwapInterval (EGLDisplay dpy, EGLint interval); +#endif +#endif /* EGL_VERSION_1_1 */ + +#ifndef EGL_VERSION_1_2 +#define EGL_VERSION_1_2 1 +typedef unsigned int EGLenum; +typedef void *EGLClientBuffer; +#define EGL_ALPHA_FORMAT 0x3088 +#define EGL_ALPHA_FORMAT_NONPRE 0x308B +#define EGL_ALPHA_FORMAT_PRE 0x308C +#define EGL_ALPHA_MASK_SIZE 0x303E +#define EGL_BUFFER_PRESERVED 0x3094 +#define EGL_BUFFER_DESTROYED 0x3095 +#define EGL_CLIENT_APIS 0x308D +#define EGL_COLORSPACE 0x3087 +#define EGL_COLORSPACE_sRGB 0x3089 +#define EGL_COLORSPACE_LINEAR 0x308A +#define EGL_COLOR_BUFFER_TYPE 0x303F +#define EGL_CONTEXT_CLIENT_TYPE 0x3097 +#define EGL_DISPLAY_SCALING 10000 +#define EGL_HORIZONTAL_RESOLUTION 0x3090 +#define EGL_LUMINANCE_BUFFER 0x308F +#define EGL_LUMINANCE_SIZE 0x303D +#define EGL_OPENGL_ES_BIT 0x0001 +#define EGL_OPENVG_BIT 0x0002 +#define EGL_OPENGL_ES_API 0x30A0 +#define EGL_OPENVG_API 0x30A1 +#define EGL_OPENVG_IMAGE 0x3096 +#define EGL_PIXEL_ASPECT_RATIO 0x3092 +#define EGL_RENDERABLE_TYPE 0x3040 +#define EGL_RENDER_BUFFER 0x3086 +#define EGL_RGB_BUFFER 0x308E +#define EGL_SINGLE_BUFFER 0x3085 +#define EGL_SWAP_BEHAVIOR 0x3093 +#define EGL_UNKNOWN EGL_CAST(EGLint,-1) +#define EGL_VERTICAL_RESOLUTION 0x3091 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLBINDAPIPROC) (EGLenum api); +typedef EGLenum (EGLAPIENTRYP PFNEGLQUERYAPIPROC) (void); +typedef EGLSurface (EGLAPIENTRYP PFNEGLCREATEPBUFFERFROMCLIENTBUFFERPROC) (EGLDisplay dpy, EGLenum buftype, EGLClientBuffer buffer, EGLConfig config, const EGLint *attrib_list); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLRELEASETHREADPROC) (void); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLWAITCLIENTPROC) (void); +#if EGL_EGL_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglBindAPI (EGLenum api); +EGLAPI EGLenum EGLAPIENTRY eglQueryAPI (void); +EGLAPI EGLSurface EGLAPIENTRY eglCreatePbufferFromClientBuffer (EGLDisplay dpy, EGLenum buftype, EGLClientBuffer buffer, EGLConfig config, const EGLint *attrib_list); +EGLAPI EGLBoolean EGLAPIENTRY eglReleaseThread (void); +EGLAPI EGLBoolean EGLAPIENTRY eglWaitClient (void); +#endif +#endif /* EGL_VERSION_1_2 */ + +#ifndef EGL_VERSION_1_3 +#define EGL_VERSION_1_3 1 +#define EGL_CONFORMANT 0x3042 +#define EGL_CONTEXT_CLIENT_VERSION 0x3098 +#define EGL_MATCH_NATIVE_PIXMAP 0x3041 +#define EGL_OPENGL_ES2_BIT 0x0004 +#define EGL_VG_ALPHA_FORMAT 0x3088 +#define EGL_VG_ALPHA_FORMAT_NONPRE 0x308B +#define EGL_VG_ALPHA_FORMAT_PRE 0x308C +#define EGL_VG_ALPHA_FORMAT_PRE_BIT 0x0040 +#define EGL_VG_COLORSPACE 0x3087 +#define EGL_VG_COLORSPACE_sRGB 0x3089 +#define EGL_VG_COLORSPACE_LINEAR 0x308A +#define EGL_VG_COLORSPACE_LINEAR_BIT 0x0020 +#endif /* EGL_VERSION_1_3 */ + +#ifndef EGL_VERSION_1_4 +#define EGL_VERSION_1_4 1 +#define EGL_DEFAULT_DISPLAY EGL_CAST(EGLNativeDisplayType,0) +#define EGL_MULTISAMPLE_RESOLVE_BOX_BIT 0x0200 +#define EGL_MULTISAMPLE_RESOLVE 0x3099 +#define EGL_MULTISAMPLE_RESOLVE_DEFAULT 0x309A +#define EGL_MULTISAMPLE_RESOLVE_BOX 0x309B +#define EGL_OPENGL_API 0x30A2 +#define EGL_OPENGL_BIT 0x0008 +#define EGL_SWAP_BEHAVIOR_PRESERVED_BIT 0x0400 +typedef EGLContext (EGLAPIENTRYP PFNEGLGETCURRENTCONTEXTPROC) (void); +#if EGL_EGL_PROTOTYPES +EGLAPI EGLContext EGLAPIENTRY eglGetCurrentContext (void); +#endif +#endif /* EGL_VERSION_1_4 */ + +#ifndef EGL_VERSION_1_5 +#define EGL_VERSION_1_5 1 +typedef void *EGLSync; +typedef intptr_t EGLAttrib; +typedef khronos_utime_nanoseconds_t EGLTime; +typedef void *EGLImage; +#define EGL_CONTEXT_MAJOR_VERSION 0x3098 +#define EGL_CONTEXT_MINOR_VERSION 0x30FB +#define EGL_CONTEXT_OPENGL_PROFILE_MASK 0x30FD +#define EGL_CONTEXT_OPENGL_RESET_NOTIFICATION_STRATEGY 0x31BD +#define EGL_NO_RESET_NOTIFICATION 0x31BE +#define EGL_LOSE_CONTEXT_ON_RESET 0x31BF +#define EGL_CONTEXT_OPENGL_CORE_PROFILE_BIT 0x00000001 +#define EGL_CONTEXT_OPENGL_COMPATIBILITY_PROFILE_BIT 0x00000002 +#define EGL_CONTEXT_OPENGL_DEBUG 0x31B0 +#define EGL_CONTEXT_OPENGL_FORWARD_COMPATIBLE 0x31B1 +#define EGL_CONTEXT_OPENGL_ROBUST_ACCESS 0x31B2 +#define EGL_OPENGL_ES3_BIT 0x00000040 +#define EGL_CL_EVENT_HANDLE 0x309C +#define EGL_SYNC_CL_EVENT 0x30FE +#define EGL_SYNC_CL_EVENT_COMPLETE 0x30FF +#define EGL_SYNC_PRIOR_COMMANDS_COMPLETE 0x30F0 +#define EGL_SYNC_TYPE 0x30F7 +#define EGL_SYNC_STATUS 0x30F1 +#define EGL_SYNC_CONDITION 0x30F8 +#define EGL_SIGNALED 0x30F2 +#define EGL_UNSIGNALED 0x30F3 +#define EGL_SYNC_FLUSH_COMMANDS_BIT 0x0001 +#define EGL_FOREVER 0xFFFFFFFFFFFFFFFFull +#define EGL_TIMEOUT_EXPIRED 0x30F5 +#define EGL_CONDITION_SATISFIED 0x30F6 +#define EGL_NO_SYNC EGL_CAST(EGLSync,0) +#define EGL_SYNC_FENCE 0x30F9 +#define EGL_GL_COLORSPACE 0x309D +#define EGL_GL_COLORSPACE_SRGB 0x3089 +#define EGL_GL_COLORSPACE_LINEAR 0x308A +#define EGL_GL_RENDERBUFFER 0x30B9 +#define EGL_GL_TEXTURE_2D 0x30B1 +#define EGL_GL_TEXTURE_LEVEL 0x30BC +#define EGL_GL_TEXTURE_3D 0x30B2 +#define EGL_GL_TEXTURE_ZOFFSET 0x30BD +#define EGL_GL_TEXTURE_CUBE_MAP_POSITIVE_X 0x30B3 +#define EGL_GL_TEXTURE_CUBE_MAP_NEGATIVE_X 0x30B4 +#define EGL_GL_TEXTURE_CUBE_MAP_POSITIVE_Y 0x30B5 +#define EGL_GL_TEXTURE_CUBE_MAP_NEGATIVE_Y 0x30B6 +#define EGL_GL_TEXTURE_CUBE_MAP_POSITIVE_Z 0x30B7 +#define EGL_GL_TEXTURE_CUBE_MAP_NEGATIVE_Z 0x30B8 +#define EGL_IMAGE_PRESERVED 0x30D2 +#define EGL_NO_IMAGE EGL_CAST(EGLImage,0) +typedef EGLSync (EGLAPIENTRYP PFNEGLCREATESYNCPROC) (EGLDisplay dpy, EGLenum type, const EGLAttrib *attrib_list); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLDESTROYSYNCPROC) (EGLDisplay dpy, EGLSync sync); +typedef EGLint (EGLAPIENTRYP PFNEGLCLIENTWAITSYNCPROC) (EGLDisplay dpy, EGLSync sync, EGLint flags, EGLTime timeout); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLGETSYNCATTRIBPROC) (EGLDisplay dpy, EGLSync sync, EGLint attribute, EGLAttrib *value); +typedef EGLImage (EGLAPIENTRYP PFNEGLCREATEIMAGEPROC) (EGLDisplay dpy, EGLContext ctx, EGLenum target, EGLClientBuffer buffer, const EGLAttrib *attrib_list); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLDESTROYIMAGEPROC) (EGLDisplay dpy, EGLImage image); +typedef EGLDisplay (EGLAPIENTRYP PFNEGLGETPLATFORMDISPLAYPROC) (EGLenum platform, void *native_display, const EGLAttrib *attrib_list); +typedef EGLSurface (EGLAPIENTRYP PFNEGLCREATEPLATFORMWINDOWSURFACEPROC) (EGLDisplay dpy, EGLConfig config, void *native_window, const EGLAttrib *attrib_list); +typedef EGLSurface (EGLAPIENTRYP PFNEGLCREATEPLATFORMPIXMAPSURFACEPROC) (EGLDisplay dpy, EGLConfig config, void *native_pixmap, const EGLAttrib *attrib_list); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLWAITSYNCPROC) (EGLDisplay dpy, EGLSync sync, EGLint flags); +#if EGL_EGL_PROTOTYPES +EGLAPI EGLSync EGLAPIENTRY eglCreateSync (EGLDisplay dpy, EGLenum type, const EGLAttrib *attrib_list); +EGLAPI EGLBoolean EGLAPIENTRY eglDestroySync (EGLDisplay dpy, EGLSync sync); +EGLAPI EGLint EGLAPIENTRY eglClientWaitSync (EGLDisplay dpy, EGLSync sync, EGLint flags, EGLTime timeout); +EGLAPI EGLBoolean EGLAPIENTRY eglGetSyncAttrib (EGLDisplay dpy, EGLSync sync, EGLint attribute, EGLAttrib *value); +EGLAPI EGLImage EGLAPIENTRY eglCreateImage (EGLDisplay dpy, EGLContext ctx, EGLenum target, EGLClientBuffer buffer, const EGLAttrib *attrib_list); +EGLAPI EGLBoolean EGLAPIENTRY eglDestroyImage (EGLDisplay dpy, EGLImage image); +EGLAPI EGLDisplay EGLAPIENTRY eglGetPlatformDisplay (EGLenum platform, void *native_display, const EGLAttrib *attrib_list); +EGLAPI EGLSurface EGLAPIENTRY eglCreatePlatformWindowSurface (EGLDisplay dpy, EGLConfig config, void *native_window, const EGLAttrib *attrib_list); +EGLAPI EGLSurface EGLAPIENTRY eglCreatePlatformPixmapSurface (EGLDisplay dpy, EGLConfig config, void *native_pixmap, const EGLAttrib *attrib_list); +EGLAPI EGLBoolean EGLAPIENTRY eglWaitSync (EGLDisplay dpy, EGLSync sync, EGLint flags); +#endif +#endif /* EGL_VERSION_1_5 */ + +#ifdef __cplusplus +} +#endif + +#endif /* __egl_h_ */ + + +#ifndef __eglext_h_ +#define __eglext_h_ 1 + +#ifdef __cplusplus +extern "C" { +#endif + +/* +** Copyright 2013-2020 The Khronos Group Inc. +** SPDX-License-Identifier: Apache-2.0 +** +** This header is generated from the Khronos EGL XML API Registry. +** The current version of the Registry, generator scripts +** used to make the header, and the header can be found at +** http://www.khronos.org/registry/egl +** +** Khronos $Git commit SHA1: 6fb1daea15 $ on $Git commit date: 2022-05-25 09:41:13 -0600 $ +*/ + +/*#include */ + +#define EGL_EGLEXT_VERSION 20220525 + +/* Generated C header for: + * API: egl + * Versions considered: .* + * Versions emitted: _nomatch_^ + * Default extensions included: egl + * Additional extensions included: _nomatch_^ + * Extensions removed: _nomatch_^ + */ + +#ifndef EGL_KHR_cl_event +#define EGL_KHR_cl_event 1 +#define EGL_CL_EVENT_HANDLE_KHR 0x309C +#define EGL_SYNC_CL_EVENT_KHR 0x30FE +#define EGL_SYNC_CL_EVENT_COMPLETE_KHR 0x30FF +#endif /* EGL_KHR_cl_event */ + +#ifndef EGL_KHR_cl_event2 +#define EGL_KHR_cl_event2 1 +typedef void *EGLSyncKHR; +typedef intptr_t EGLAttribKHR; +typedef EGLSyncKHR (EGLAPIENTRYP PFNEGLCREATESYNC64KHRPROC) (EGLDisplay dpy, EGLenum type, const EGLAttribKHR *attrib_list); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLSyncKHR EGLAPIENTRY eglCreateSync64KHR (EGLDisplay dpy, EGLenum type, const EGLAttribKHR *attrib_list); +#endif +#endif /* EGL_KHR_cl_event2 */ + +#ifndef EGL_KHR_client_get_all_proc_addresses +#define EGL_KHR_client_get_all_proc_addresses 1 +#endif /* EGL_KHR_client_get_all_proc_addresses */ + +#ifndef EGL_KHR_config_attribs +#define EGL_KHR_config_attribs 1 +#define EGL_CONFORMANT_KHR 0x3042 +#define EGL_VG_COLORSPACE_LINEAR_BIT_KHR 0x0020 +#define EGL_VG_ALPHA_FORMAT_PRE_BIT_KHR 0x0040 +#endif /* EGL_KHR_config_attribs */ + +#ifndef EGL_KHR_context_flush_control +#define EGL_KHR_context_flush_control 1 +#define EGL_CONTEXT_RELEASE_BEHAVIOR_NONE_KHR 0 +#define EGL_CONTEXT_RELEASE_BEHAVIOR_KHR 0x2097 +#define EGL_CONTEXT_RELEASE_BEHAVIOR_FLUSH_KHR 0x2098 +#endif /* EGL_KHR_context_flush_control */ + +#ifndef EGL_KHR_create_context +#define EGL_KHR_create_context 1 +#define EGL_CONTEXT_MAJOR_VERSION_KHR 0x3098 +#define EGL_CONTEXT_MINOR_VERSION_KHR 0x30FB +#define EGL_CONTEXT_FLAGS_KHR 0x30FC +#define EGL_CONTEXT_OPENGL_PROFILE_MASK_KHR 0x30FD +#define EGL_CONTEXT_OPENGL_RESET_NOTIFICATION_STRATEGY_KHR 0x31BD +#define EGL_NO_RESET_NOTIFICATION_KHR 0x31BE +#define EGL_LOSE_CONTEXT_ON_RESET_KHR 0x31BF +#define EGL_CONTEXT_OPENGL_DEBUG_BIT_KHR 0x00000001 +#define EGL_CONTEXT_OPENGL_FORWARD_COMPATIBLE_BIT_KHR 0x00000002 +#define EGL_CONTEXT_OPENGL_ROBUST_ACCESS_BIT_KHR 0x00000004 +#define EGL_CONTEXT_OPENGL_CORE_PROFILE_BIT_KHR 0x00000001 +#define EGL_CONTEXT_OPENGL_COMPATIBILITY_PROFILE_BIT_KHR 0x00000002 +#define EGL_OPENGL_ES3_BIT_KHR 0x00000040 +#endif /* EGL_KHR_create_context */ + +#ifndef EGL_KHR_create_context_no_error +#define EGL_KHR_create_context_no_error 1 +#define EGL_CONTEXT_OPENGL_NO_ERROR_KHR 0x31B3 +#endif /* EGL_KHR_create_context_no_error */ + +#ifndef EGL_KHR_debug +#define EGL_KHR_debug 1 +typedef void *EGLLabelKHR; +typedef void *EGLObjectKHR; +typedef void (EGLAPIENTRY *EGLDEBUGPROCKHR)(EGLenum error,const char *command,EGLint messageType,EGLLabelKHR threadLabel,EGLLabelKHR objectLabel,const char* message); +#define EGL_OBJECT_THREAD_KHR 0x33B0 +#define EGL_OBJECT_DISPLAY_KHR 0x33B1 +#define EGL_OBJECT_CONTEXT_KHR 0x33B2 +#define EGL_OBJECT_SURFACE_KHR 0x33B3 +#define EGL_OBJECT_IMAGE_KHR 0x33B4 +#define EGL_OBJECT_SYNC_KHR 0x33B5 +#define EGL_OBJECT_STREAM_KHR 0x33B6 +#define EGL_DEBUG_MSG_CRITICAL_KHR 0x33B9 +#define EGL_DEBUG_MSG_ERROR_KHR 0x33BA +#define EGL_DEBUG_MSG_WARN_KHR 0x33BB +#define EGL_DEBUG_MSG_INFO_KHR 0x33BC +#define EGL_DEBUG_CALLBACK_KHR 0x33B8 +typedef EGLint (EGLAPIENTRYP PFNEGLDEBUGMESSAGECONTROLKHRPROC) (EGLDEBUGPROCKHR callback, const EGLAttrib *attrib_list); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYDEBUGKHRPROC) (EGLint attribute, EGLAttrib *value); +typedef EGLint (EGLAPIENTRYP PFNEGLLABELOBJECTKHRPROC) (EGLDisplay display, EGLenum objectType, EGLObjectKHR object, EGLLabelKHR label); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLint EGLAPIENTRY eglDebugMessageControlKHR (EGLDEBUGPROCKHR callback, const EGLAttrib *attrib_list); +EGLAPI EGLBoolean EGLAPIENTRY eglQueryDebugKHR (EGLint attribute, EGLAttrib *value); +EGLAPI EGLint EGLAPIENTRY eglLabelObjectKHR (EGLDisplay display, EGLenum objectType, EGLObjectKHR object, EGLLabelKHR label); +#endif +#endif /* EGL_KHR_debug */ + +#ifndef EGL_KHR_display_reference +#define EGL_KHR_display_reference 1 +#define EGL_TRACK_REFERENCES_KHR 0x3352 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYDISPLAYATTRIBKHRPROC) (EGLDisplay dpy, EGLint name, EGLAttrib *value); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglQueryDisplayAttribKHR (EGLDisplay dpy, EGLint name, EGLAttrib *value); +#endif +#endif /* EGL_KHR_display_reference */ + +#ifndef EGL_KHR_fence_sync +#define EGL_KHR_fence_sync 1 +typedef khronos_utime_nanoseconds_t EGLTimeKHR; +#ifdef KHRONOS_SUPPORT_INT64 +#define EGL_SYNC_PRIOR_COMMANDS_COMPLETE_KHR 0x30F0 +#define EGL_SYNC_CONDITION_KHR 0x30F8 +#define EGL_SYNC_FENCE_KHR 0x30F9 +typedef EGLSyncKHR (EGLAPIENTRYP PFNEGLCREATESYNCKHRPROC) (EGLDisplay dpy, EGLenum type, const EGLint *attrib_list); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLDESTROYSYNCKHRPROC) (EGLDisplay dpy, EGLSyncKHR sync); +typedef EGLint (EGLAPIENTRYP PFNEGLCLIENTWAITSYNCKHRPROC) (EGLDisplay dpy, EGLSyncKHR sync, EGLint flags, EGLTimeKHR timeout); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLGETSYNCATTRIBKHRPROC) (EGLDisplay dpy, EGLSyncKHR sync, EGLint attribute, EGLint *value); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLSyncKHR EGLAPIENTRY eglCreateSyncKHR (EGLDisplay dpy, EGLenum type, const EGLint *attrib_list); +EGLAPI EGLBoolean EGLAPIENTRY eglDestroySyncKHR (EGLDisplay dpy, EGLSyncKHR sync); +EGLAPI EGLint EGLAPIENTRY eglClientWaitSyncKHR (EGLDisplay dpy, EGLSyncKHR sync, EGLint flags, EGLTimeKHR timeout); +EGLAPI EGLBoolean EGLAPIENTRY eglGetSyncAttribKHR (EGLDisplay dpy, EGLSyncKHR sync, EGLint attribute, EGLint *value); +#endif +#endif /* KHRONOS_SUPPORT_INT64 */ +#endif /* EGL_KHR_fence_sync */ + +#ifndef EGL_KHR_get_all_proc_addresses +#define EGL_KHR_get_all_proc_addresses 1 +#endif /* EGL_KHR_get_all_proc_addresses */ + +#ifndef EGL_KHR_gl_colorspace +#define EGL_KHR_gl_colorspace 1 +#define EGL_GL_COLORSPACE_KHR 0x309D +#define EGL_GL_COLORSPACE_SRGB_KHR 0x3089 +#define EGL_GL_COLORSPACE_LINEAR_KHR 0x308A +#endif /* EGL_KHR_gl_colorspace */ + +#ifndef EGL_KHR_gl_renderbuffer_image +#define EGL_KHR_gl_renderbuffer_image 1 +#define EGL_GL_RENDERBUFFER_KHR 0x30B9 +#endif /* EGL_KHR_gl_renderbuffer_image */ + +#ifndef EGL_KHR_gl_texture_2D_image +#define EGL_KHR_gl_texture_2D_image 1 +#define EGL_GL_TEXTURE_2D_KHR 0x30B1 +#define EGL_GL_TEXTURE_LEVEL_KHR 0x30BC +#endif /* EGL_KHR_gl_texture_2D_image */ + +#ifndef EGL_KHR_gl_texture_3D_image +#define EGL_KHR_gl_texture_3D_image 1 +#define EGL_GL_TEXTURE_3D_KHR 0x30B2 +#define EGL_GL_TEXTURE_ZOFFSET_KHR 0x30BD +#endif /* EGL_KHR_gl_texture_3D_image */ + +#ifndef EGL_KHR_gl_texture_cubemap_image +#define EGL_KHR_gl_texture_cubemap_image 1 +#define EGL_GL_TEXTURE_CUBE_MAP_POSITIVE_X_KHR 0x30B3 +#define EGL_GL_TEXTURE_CUBE_MAP_NEGATIVE_X_KHR 0x30B4 +#define EGL_GL_TEXTURE_CUBE_MAP_POSITIVE_Y_KHR 0x30B5 +#define EGL_GL_TEXTURE_CUBE_MAP_NEGATIVE_Y_KHR 0x30B6 +#define EGL_GL_TEXTURE_CUBE_MAP_POSITIVE_Z_KHR 0x30B7 +#define EGL_GL_TEXTURE_CUBE_MAP_NEGATIVE_Z_KHR 0x30B8 +#endif /* EGL_KHR_gl_texture_cubemap_image */ + +#ifndef EGL_KHR_image +#define EGL_KHR_image 1 +typedef void *EGLImageKHR; +#define EGL_NATIVE_PIXMAP_KHR 0x30B0 +#define EGL_NO_IMAGE_KHR EGL_CAST(EGLImageKHR,0) +typedef EGLImageKHR (EGLAPIENTRYP PFNEGLCREATEIMAGEKHRPROC) (EGLDisplay dpy, EGLContext ctx, EGLenum target, EGLClientBuffer buffer, const EGLint *attrib_list); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLDESTROYIMAGEKHRPROC) (EGLDisplay dpy, EGLImageKHR image); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLImageKHR EGLAPIENTRY eglCreateImageKHR (EGLDisplay dpy, EGLContext ctx, EGLenum target, EGLClientBuffer buffer, const EGLint *attrib_list); +EGLAPI EGLBoolean EGLAPIENTRY eglDestroyImageKHR (EGLDisplay dpy, EGLImageKHR image); +#endif +#endif /* EGL_KHR_image */ + +#ifndef EGL_KHR_image_base +#define EGL_KHR_image_base 1 +#define EGL_IMAGE_PRESERVED_KHR 0x30D2 +#endif /* EGL_KHR_image_base */ + +#ifndef EGL_KHR_image_pixmap +#define EGL_KHR_image_pixmap 1 +#endif /* EGL_KHR_image_pixmap */ + +#ifndef EGL_KHR_lock_surface +#define EGL_KHR_lock_surface 1 +#define EGL_READ_SURFACE_BIT_KHR 0x0001 +#define EGL_WRITE_SURFACE_BIT_KHR 0x0002 +#define EGL_LOCK_SURFACE_BIT_KHR 0x0080 +#define EGL_OPTIMAL_FORMAT_BIT_KHR 0x0100 +#define EGL_MATCH_FORMAT_KHR 0x3043 +#define EGL_FORMAT_RGB_565_EXACT_KHR 0x30C0 +#define EGL_FORMAT_RGB_565_KHR 0x30C1 +#define EGL_FORMAT_RGBA_8888_EXACT_KHR 0x30C2 +#define EGL_FORMAT_RGBA_8888_KHR 0x30C3 +#define EGL_MAP_PRESERVE_PIXELS_KHR 0x30C4 +#define EGL_LOCK_USAGE_HINT_KHR 0x30C5 +#define EGL_BITMAP_POINTER_KHR 0x30C6 +#define EGL_BITMAP_PITCH_KHR 0x30C7 +#define EGL_BITMAP_ORIGIN_KHR 0x30C8 +#define EGL_BITMAP_PIXEL_RED_OFFSET_KHR 0x30C9 +#define EGL_BITMAP_PIXEL_GREEN_OFFSET_KHR 0x30CA +#define EGL_BITMAP_PIXEL_BLUE_OFFSET_KHR 0x30CB +#define EGL_BITMAP_PIXEL_ALPHA_OFFSET_KHR 0x30CC +#define EGL_BITMAP_PIXEL_LUMINANCE_OFFSET_KHR 0x30CD +#define EGL_LOWER_LEFT_KHR 0x30CE +#define EGL_UPPER_LEFT_KHR 0x30CF +typedef EGLBoolean (EGLAPIENTRYP PFNEGLLOCKSURFACEKHRPROC) (EGLDisplay dpy, EGLSurface surface, const EGLint *attrib_list); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLUNLOCKSURFACEKHRPROC) (EGLDisplay dpy, EGLSurface surface); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglLockSurfaceKHR (EGLDisplay dpy, EGLSurface surface, const EGLint *attrib_list); +EGLAPI EGLBoolean EGLAPIENTRY eglUnlockSurfaceKHR (EGLDisplay dpy, EGLSurface surface); +#endif +#endif /* EGL_KHR_lock_surface */ + +#ifndef EGL_KHR_lock_surface2 +#define EGL_KHR_lock_surface2 1 +#define EGL_BITMAP_PIXEL_SIZE_KHR 0x3110 +#endif /* EGL_KHR_lock_surface2 */ + +#ifndef EGL_KHR_lock_surface3 +#define EGL_KHR_lock_surface3 1 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYSURFACE64KHRPROC) (EGLDisplay dpy, EGLSurface surface, EGLint attribute, EGLAttribKHR *value); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglQuerySurface64KHR (EGLDisplay dpy, EGLSurface surface, EGLint attribute, EGLAttribKHR *value); +#endif +#endif /* EGL_KHR_lock_surface3 */ + +#ifndef EGL_KHR_mutable_render_buffer +#define EGL_KHR_mutable_render_buffer 1 +#define EGL_MUTABLE_RENDER_BUFFER_BIT_KHR 0x1000 +#endif /* EGL_KHR_mutable_render_buffer */ + +#ifndef EGL_KHR_no_config_context +#define EGL_KHR_no_config_context 1 +#define EGL_NO_CONFIG_KHR EGL_CAST(EGLConfig,0) +#endif /* EGL_KHR_no_config_context */ + +#ifndef EGL_KHR_partial_update +#define EGL_KHR_partial_update 1 +#define EGL_BUFFER_AGE_KHR 0x313D +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSETDAMAGEREGIONKHRPROC) (EGLDisplay dpy, EGLSurface surface, EGLint *rects, EGLint n_rects); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglSetDamageRegionKHR (EGLDisplay dpy, EGLSurface surface, EGLint *rects, EGLint n_rects); +#endif +#endif /* EGL_KHR_partial_update */ + +#ifndef EGL_KHR_platform_android +#define EGL_KHR_platform_android 1 +#define EGL_PLATFORM_ANDROID_KHR 0x3141 +#endif /* EGL_KHR_platform_android */ + +#ifndef EGL_KHR_platform_gbm +#define EGL_KHR_platform_gbm 1 +#define EGL_PLATFORM_GBM_KHR 0x31D7 +#endif /* EGL_KHR_platform_gbm */ + +#ifndef EGL_KHR_platform_wayland +#define EGL_KHR_platform_wayland 1 +#define EGL_PLATFORM_WAYLAND_KHR 0x31D8 +#endif /* EGL_KHR_platform_wayland */ + +#ifndef EGL_KHR_platform_x11 +#define EGL_KHR_platform_x11 1 +#define EGL_PLATFORM_X11_KHR 0x31D5 +#define EGL_PLATFORM_X11_SCREEN_KHR 0x31D6 +#endif /* EGL_KHR_platform_x11 */ + +#ifndef EGL_KHR_reusable_sync +#define EGL_KHR_reusable_sync 1 +#ifdef KHRONOS_SUPPORT_INT64 +#define EGL_SYNC_STATUS_KHR 0x30F1 +#define EGL_SIGNALED_KHR 0x30F2 +#define EGL_UNSIGNALED_KHR 0x30F3 +#define EGL_TIMEOUT_EXPIRED_KHR 0x30F5 +#define EGL_CONDITION_SATISFIED_KHR 0x30F6 +#define EGL_SYNC_TYPE_KHR 0x30F7 +#define EGL_SYNC_REUSABLE_KHR 0x30FA +#define EGL_SYNC_FLUSH_COMMANDS_BIT_KHR 0x0001 +#define EGL_FOREVER_KHR 0xFFFFFFFFFFFFFFFFull +#define EGL_NO_SYNC_KHR EGL_CAST(EGLSyncKHR,0) +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSIGNALSYNCKHRPROC) (EGLDisplay dpy, EGLSyncKHR sync, EGLenum mode); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglSignalSyncKHR (EGLDisplay dpy, EGLSyncKHR sync, EGLenum mode); +#endif +#endif /* KHRONOS_SUPPORT_INT64 */ +#endif /* EGL_KHR_reusable_sync */ + +#ifndef EGL_KHR_stream +#define EGL_KHR_stream 1 +typedef void *EGLStreamKHR; +typedef khronos_uint64_t EGLuint64KHR; +#ifdef KHRONOS_SUPPORT_INT64 +#define EGL_NO_STREAM_KHR EGL_CAST(EGLStreamKHR,0) +#define EGL_CONSUMER_LATENCY_USEC_KHR 0x3210 +#define EGL_PRODUCER_FRAME_KHR 0x3212 +#define EGL_CONSUMER_FRAME_KHR 0x3213 +#define EGL_STREAM_STATE_KHR 0x3214 +#define EGL_STREAM_STATE_CREATED_KHR 0x3215 +#define EGL_STREAM_STATE_CONNECTING_KHR 0x3216 +#define EGL_STREAM_STATE_EMPTY_KHR 0x3217 +#define EGL_STREAM_STATE_NEW_FRAME_AVAILABLE_KHR 0x3218 +#define EGL_STREAM_STATE_OLD_FRAME_AVAILABLE_KHR 0x3219 +#define EGL_STREAM_STATE_DISCONNECTED_KHR 0x321A +#define EGL_BAD_STREAM_KHR 0x321B +#define EGL_BAD_STATE_KHR 0x321C +typedef EGLStreamKHR (EGLAPIENTRYP PFNEGLCREATESTREAMKHRPROC) (EGLDisplay dpy, const EGLint *attrib_list); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLDESTROYSTREAMKHRPROC) (EGLDisplay dpy, EGLStreamKHR stream); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSTREAMATTRIBKHRPROC) (EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, EGLint value); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYSTREAMKHRPROC) (EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, EGLint *value); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYSTREAMU64KHRPROC) (EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, EGLuint64KHR *value); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLStreamKHR EGLAPIENTRY eglCreateStreamKHR (EGLDisplay dpy, const EGLint *attrib_list); +EGLAPI EGLBoolean EGLAPIENTRY eglDestroyStreamKHR (EGLDisplay dpy, EGLStreamKHR stream); +EGLAPI EGLBoolean EGLAPIENTRY eglStreamAttribKHR (EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, EGLint value); +EGLAPI EGLBoolean EGLAPIENTRY eglQueryStreamKHR (EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, EGLint *value); +EGLAPI EGLBoolean EGLAPIENTRY eglQueryStreamu64KHR (EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, EGLuint64KHR *value); +#endif +#endif /* KHRONOS_SUPPORT_INT64 */ +#endif /* EGL_KHR_stream */ + +#ifndef EGL_KHR_stream_attrib +#define EGL_KHR_stream_attrib 1 +#ifdef KHRONOS_SUPPORT_INT64 +typedef EGLStreamKHR (EGLAPIENTRYP PFNEGLCREATESTREAMATTRIBKHRPROC) (EGLDisplay dpy, const EGLAttrib *attrib_list); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSETSTREAMATTRIBKHRPROC) (EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, EGLAttrib value); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYSTREAMATTRIBKHRPROC) (EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, EGLAttrib *value); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSTREAMCONSUMERACQUIREATTRIBKHRPROC) (EGLDisplay dpy, EGLStreamKHR stream, const EGLAttrib *attrib_list); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSTREAMCONSUMERRELEASEATTRIBKHRPROC) (EGLDisplay dpy, EGLStreamKHR stream, const EGLAttrib *attrib_list); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLStreamKHR EGLAPIENTRY eglCreateStreamAttribKHR (EGLDisplay dpy, const EGLAttrib *attrib_list); +EGLAPI EGLBoolean EGLAPIENTRY eglSetStreamAttribKHR (EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, EGLAttrib value); +EGLAPI EGLBoolean EGLAPIENTRY eglQueryStreamAttribKHR (EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, EGLAttrib *value); +EGLAPI EGLBoolean EGLAPIENTRY eglStreamConsumerAcquireAttribKHR (EGLDisplay dpy, EGLStreamKHR stream, const EGLAttrib *attrib_list); +EGLAPI EGLBoolean EGLAPIENTRY eglStreamConsumerReleaseAttribKHR (EGLDisplay dpy, EGLStreamKHR stream, const EGLAttrib *attrib_list); +#endif +#endif /* KHRONOS_SUPPORT_INT64 */ +#endif /* EGL_KHR_stream_attrib */ + +#ifndef EGL_KHR_stream_consumer_gltexture +#define EGL_KHR_stream_consumer_gltexture 1 +#ifdef EGL_KHR_stream +#define EGL_CONSUMER_ACQUIRE_TIMEOUT_USEC_KHR 0x321E +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSTREAMCONSUMERGLTEXTUREEXTERNALKHRPROC) (EGLDisplay dpy, EGLStreamKHR stream); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSTREAMCONSUMERACQUIREKHRPROC) (EGLDisplay dpy, EGLStreamKHR stream); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSTREAMCONSUMERRELEASEKHRPROC) (EGLDisplay dpy, EGLStreamKHR stream); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglStreamConsumerGLTextureExternalKHR (EGLDisplay dpy, EGLStreamKHR stream); +EGLAPI EGLBoolean EGLAPIENTRY eglStreamConsumerAcquireKHR (EGLDisplay dpy, EGLStreamKHR stream); +EGLAPI EGLBoolean EGLAPIENTRY eglStreamConsumerReleaseKHR (EGLDisplay dpy, EGLStreamKHR stream); +#endif +#endif /* EGL_KHR_stream */ +#endif /* EGL_KHR_stream_consumer_gltexture */ + +#ifndef EGL_KHR_stream_cross_process_fd +#define EGL_KHR_stream_cross_process_fd 1 +typedef int EGLNativeFileDescriptorKHR; +#ifdef EGL_KHR_stream +#define EGL_NO_FILE_DESCRIPTOR_KHR EGL_CAST(EGLNativeFileDescriptorKHR,-1) +typedef EGLNativeFileDescriptorKHR (EGLAPIENTRYP PFNEGLGETSTREAMFILEDESCRIPTORKHRPROC) (EGLDisplay dpy, EGLStreamKHR stream); +typedef EGLStreamKHR (EGLAPIENTRYP PFNEGLCREATESTREAMFROMFILEDESCRIPTORKHRPROC) (EGLDisplay dpy, EGLNativeFileDescriptorKHR file_descriptor); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLNativeFileDescriptorKHR EGLAPIENTRY eglGetStreamFileDescriptorKHR (EGLDisplay dpy, EGLStreamKHR stream); +EGLAPI EGLStreamKHR EGLAPIENTRY eglCreateStreamFromFileDescriptorKHR (EGLDisplay dpy, EGLNativeFileDescriptorKHR file_descriptor); +#endif +#endif /* EGL_KHR_stream */ +#endif /* EGL_KHR_stream_cross_process_fd */ + +#ifndef EGL_KHR_stream_fifo +#define EGL_KHR_stream_fifo 1 +#ifdef EGL_KHR_stream +#define EGL_STREAM_FIFO_LENGTH_KHR 0x31FC +#define EGL_STREAM_TIME_NOW_KHR 0x31FD +#define EGL_STREAM_TIME_CONSUMER_KHR 0x31FE +#define EGL_STREAM_TIME_PRODUCER_KHR 0x31FF +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYSTREAMTIMEKHRPROC) (EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, EGLTimeKHR *value); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglQueryStreamTimeKHR (EGLDisplay dpy, EGLStreamKHR stream, EGLenum attribute, EGLTimeKHR *value); +#endif +#endif /* EGL_KHR_stream */ +#endif /* EGL_KHR_stream_fifo */ + +#ifndef EGL_KHR_stream_producer_aldatalocator +#define EGL_KHR_stream_producer_aldatalocator 1 +#ifdef EGL_KHR_stream +#endif /* EGL_KHR_stream */ +#endif /* EGL_KHR_stream_producer_aldatalocator */ + +#ifndef EGL_KHR_stream_producer_eglsurface +#define EGL_KHR_stream_producer_eglsurface 1 +#ifdef EGL_KHR_stream +#define EGL_STREAM_BIT_KHR 0x0800 +typedef EGLSurface (EGLAPIENTRYP PFNEGLCREATESTREAMPRODUCERSURFACEKHRPROC) (EGLDisplay dpy, EGLConfig config, EGLStreamKHR stream, const EGLint *attrib_list); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLSurface EGLAPIENTRY eglCreateStreamProducerSurfaceKHR (EGLDisplay dpy, EGLConfig config, EGLStreamKHR stream, const EGLint *attrib_list); +#endif +#endif /* EGL_KHR_stream */ +#endif /* EGL_KHR_stream_producer_eglsurface */ + +#ifndef EGL_KHR_surfaceless_context +#define EGL_KHR_surfaceless_context 1 +#endif /* EGL_KHR_surfaceless_context */ + +#ifndef EGL_KHR_swap_buffers_with_damage +#define EGL_KHR_swap_buffers_with_damage 1 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSWAPBUFFERSWITHDAMAGEKHRPROC) (EGLDisplay dpy, EGLSurface surface, const EGLint *rects, EGLint n_rects); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglSwapBuffersWithDamageKHR (EGLDisplay dpy, EGLSurface surface, const EGLint *rects, EGLint n_rects); +#endif +#endif /* EGL_KHR_swap_buffers_with_damage */ + +#ifndef EGL_KHR_vg_parent_image +#define EGL_KHR_vg_parent_image 1 +#define EGL_VG_PARENT_IMAGE_KHR 0x30BA +#endif /* EGL_KHR_vg_parent_image */ + +#ifndef EGL_KHR_wait_sync +#define EGL_KHR_wait_sync 1 +typedef EGLint (EGLAPIENTRYP PFNEGLWAITSYNCKHRPROC) (EGLDisplay dpy, EGLSyncKHR sync, EGLint flags); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLint EGLAPIENTRY eglWaitSyncKHR (EGLDisplay dpy, EGLSyncKHR sync, EGLint flags); +#endif +#endif /* EGL_KHR_wait_sync */ + +#ifndef EGL_ANDROID_GLES_layers +#define EGL_ANDROID_GLES_layers 1 +#endif /* EGL_ANDROID_GLES_layers */ + +#ifndef EGL_ANDROID_blob_cache +#define EGL_ANDROID_blob_cache 1 +typedef khronos_ssize_t EGLsizeiANDROID; +typedef void (*EGLSetBlobFuncANDROID) (const void *key, EGLsizeiANDROID keySize, const void *value, EGLsizeiANDROID valueSize); +typedef EGLsizeiANDROID (*EGLGetBlobFuncANDROID) (const void *key, EGLsizeiANDROID keySize, void *value, EGLsizeiANDROID valueSize); +typedef void (EGLAPIENTRYP PFNEGLSETBLOBCACHEFUNCSANDROIDPROC) (EGLDisplay dpy, EGLSetBlobFuncANDROID set, EGLGetBlobFuncANDROID get); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI void EGLAPIENTRY eglSetBlobCacheFuncsANDROID (EGLDisplay dpy, EGLSetBlobFuncANDROID set, EGLGetBlobFuncANDROID get); +#endif +#endif /* EGL_ANDROID_blob_cache */ + +#ifndef EGL_ANDROID_create_native_client_buffer +#define EGL_ANDROID_create_native_client_buffer 1 +#define EGL_NATIVE_BUFFER_USAGE_ANDROID 0x3143 +#define EGL_NATIVE_BUFFER_USAGE_PROTECTED_BIT_ANDROID 0x00000001 +#define EGL_NATIVE_BUFFER_USAGE_RENDERBUFFER_BIT_ANDROID 0x00000002 +#define EGL_NATIVE_BUFFER_USAGE_TEXTURE_BIT_ANDROID 0x00000004 +typedef EGLClientBuffer (EGLAPIENTRYP PFNEGLCREATENATIVECLIENTBUFFERANDROIDPROC) (const EGLint *attrib_list); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLClientBuffer EGLAPIENTRY eglCreateNativeClientBufferANDROID (const EGLint *attrib_list); +#endif +#endif /* EGL_ANDROID_create_native_client_buffer */ + +#ifndef EGL_ANDROID_framebuffer_target +#define EGL_ANDROID_framebuffer_target 1 +#define EGL_FRAMEBUFFER_TARGET_ANDROID 0x3147 +#endif /* EGL_ANDROID_framebuffer_target */ + +#ifndef EGL_ANDROID_front_buffer_auto_refresh +#define EGL_ANDROID_front_buffer_auto_refresh 1 +#define EGL_FRONT_BUFFER_AUTO_REFRESH_ANDROID 0x314C +#endif /* EGL_ANDROID_front_buffer_auto_refresh */ + +#ifndef EGL_ANDROID_get_frame_timestamps +#define EGL_ANDROID_get_frame_timestamps 1 +typedef khronos_stime_nanoseconds_t EGLnsecsANDROID; +#define EGL_TIMESTAMP_PENDING_ANDROID EGL_CAST(EGLnsecsANDROID,-2) +#define EGL_TIMESTAMP_INVALID_ANDROID EGL_CAST(EGLnsecsANDROID,-1) +#define EGL_TIMESTAMPS_ANDROID 0x3430 +#define EGL_COMPOSITE_DEADLINE_ANDROID 0x3431 +#define EGL_COMPOSITE_INTERVAL_ANDROID 0x3432 +#define EGL_COMPOSITE_TO_PRESENT_LATENCY_ANDROID 0x3433 +#define EGL_REQUESTED_PRESENT_TIME_ANDROID 0x3434 +#define EGL_RENDERING_COMPLETE_TIME_ANDROID 0x3435 +#define EGL_COMPOSITION_LATCH_TIME_ANDROID 0x3436 +#define EGL_FIRST_COMPOSITION_START_TIME_ANDROID 0x3437 +#define EGL_LAST_COMPOSITION_START_TIME_ANDROID 0x3438 +#define EGL_FIRST_COMPOSITION_GPU_FINISHED_TIME_ANDROID 0x3439 +#define EGL_DISPLAY_PRESENT_TIME_ANDROID 0x343A +#define EGL_DEQUEUE_READY_TIME_ANDROID 0x343B +#define EGL_READS_DONE_TIME_ANDROID 0x343C +typedef EGLBoolean (EGLAPIENTRYP PFNEGLGETCOMPOSITORTIMINGSUPPORTEDANDROIDPROC) (EGLDisplay dpy, EGLSurface surface, EGLint name); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLGETCOMPOSITORTIMINGANDROIDPROC) (EGLDisplay dpy, EGLSurface surface, EGLint numTimestamps, const EGLint *names, EGLnsecsANDROID *values); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLGETNEXTFRAMEIDANDROIDPROC) (EGLDisplay dpy, EGLSurface surface, EGLuint64KHR *frameId); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLGETFRAMETIMESTAMPSUPPORTEDANDROIDPROC) (EGLDisplay dpy, EGLSurface surface, EGLint timestamp); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLGETFRAMETIMESTAMPSANDROIDPROC) (EGLDisplay dpy, EGLSurface surface, EGLuint64KHR frameId, EGLint numTimestamps, const EGLint *timestamps, EGLnsecsANDROID *values); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglGetCompositorTimingSupportedANDROID (EGLDisplay dpy, EGLSurface surface, EGLint name); +EGLAPI EGLBoolean EGLAPIENTRY eglGetCompositorTimingANDROID (EGLDisplay dpy, EGLSurface surface, EGLint numTimestamps, const EGLint *names, EGLnsecsANDROID *values); +EGLAPI EGLBoolean EGLAPIENTRY eglGetNextFrameIdANDROID (EGLDisplay dpy, EGLSurface surface, EGLuint64KHR *frameId); +EGLAPI EGLBoolean EGLAPIENTRY eglGetFrameTimestampSupportedANDROID (EGLDisplay dpy, EGLSurface surface, EGLint timestamp); +EGLAPI EGLBoolean EGLAPIENTRY eglGetFrameTimestampsANDROID (EGLDisplay dpy, EGLSurface surface, EGLuint64KHR frameId, EGLint numTimestamps, const EGLint *timestamps, EGLnsecsANDROID *values); +#endif +#endif /* EGL_ANDROID_get_frame_timestamps */ + +#ifndef EGL_ANDROID_get_native_client_buffer +#define EGL_ANDROID_get_native_client_buffer 1 +struct AHardwareBuffer; +typedef EGLClientBuffer (EGLAPIENTRYP PFNEGLGETNATIVECLIENTBUFFERANDROIDPROC) (const struct AHardwareBuffer *buffer); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLClientBuffer EGLAPIENTRY eglGetNativeClientBufferANDROID (const struct AHardwareBuffer *buffer); +#endif +#endif /* EGL_ANDROID_get_native_client_buffer */ + +#ifndef EGL_ANDROID_image_native_buffer +#define EGL_ANDROID_image_native_buffer 1 +#define EGL_NATIVE_BUFFER_ANDROID 0x3140 +#endif /* EGL_ANDROID_image_native_buffer */ + +#ifndef EGL_ANDROID_native_fence_sync +#define EGL_ANDROID_native_fence_sync 1 +#define EGL_SYNC_NATIVE_FENCE_ANDROID 0x3144 +#define EGL_SYNC_NATIVE_FENCE_FD_ANDROID 0x3145 +#define EGL_SYNC_NATIVE_FENCE_SIGNALED_ANDROID 0x3146 +#define EGL_NO_NATIVE_FENCE_FD_ANDROID -1 +typedef EGLint (EGLAPIENTRYP PFNEGLDUPNATIVEFENCEFDANDROIDPROC) (EGLDisplay dpy, EGLSyncKHR sync); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLint EGLAPIENTRY eglDupNativeFenceFDANDROID (EGLDisplay dpy, EGLSyncKHR sync); +#endif +#endif /* EGL_ANDROID_native_fence_sync */ + +#ifndef EGL_ANDROID_presentation_time +#define EGL_ANDROID_presentation_time 1 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLPRESENTATIONTIMEANDROIDPROC) (EGLDisplay dpy, EGLSurface surface, EGLnsecsANDROID time); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglPresentationTimeANDROID (EGLDisplay dpy, EGLSurface surface, EGLnsecsANDROID time); +#endif +#endif /* EGL_ANDROID_presentation_time */ + +#ifndef EGL_ANDROID_recordable +#define EGL_ANDROID_recordable 1 +#define EGL_RECORDABLE_ANDROID 0x3142 +#endif /* EGL_ANDROID_recordable */ + +#ifndef EGL_ANGLE_d3d_share_handle_client_buffer +#define EGL_ANGLE_d3d_share_handle_client_buffer 1 +#define EGL_D3D_TEXTURE_2D_SHARE_HANDLE_ANGLE 0x3200 +#endif /* EGL_ANGLE_d3d_share_handle_client_buffer */ + +#ifndef EGL_ANGLE_device_d3d +#define EGL_ANGLE_device_d3d 1 +#define EGL_D3D9_DEVICE_ANGLE 0x33A0 +#define EGL_D3D11_DEVICE_ANGLE 0x33A1 +#endif /* EGL_ANGLE_device_d3d */ + +#ifndef EGL_ANGLE_query_surface_pointer +#define EGL_ANGLE_query_surface_pointer 1 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYSURFACEPOINTERANGLEPROC) (EGLDisplay dpy, EGLSurface surface, EGLint attribute, void **value); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglQuerySurfacePointerANGLE (EGLDisplay dpy, EGLSurface surface, EGLint attribute, void **value); +#endif +#endif /* EGL_ANGLE_query_surface_pointer */ + +#ifndef EGL_ANGLE_surface_d3d_texture_2d_share_handle +#define EGL_ANGLE_surface_d3d_texture_2d_share_handle 1 +#endif /* EGL_ANGLE_surface_d3d_texture_2d_share_handle */ + +#ifndef EGL_ANGLE_sync_control_rate +#define EGL_ANGLE_sync_control_rate 1 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLGETMSCRATEANGLEPROC) (EGLDisplay dpy, EGLSurface surface, EGLint *numerator, EGLint *denominator); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglGetMscRateANGLE (EGLDisplay dpy, EGLSurface surface, EGLint *numerator, EGLint *denominator); +#endif +#endif /* EGL_ANGLE_sync_control_rate */ + +#ifndef EGL_ANGLE_window_fixed_size +#define EGL_ANGLE_window_fixed_size 1 +#define EGL_FIXED_SIZE_ANGLE 0x3201 +#endif /* EGL_ANGLE_window_fixed_size */ + +#ifndef EGL_ARM_image_format +#define EGL_ARM_image_format 1 +#define EGL_COLOR_COMPONENT_TYPE_UNSIGNED_INTEGER_ARM 0x3287 +#define EGL_COLOR_COMPONENT_TYPE_INTEGER_ARM 0x3288 +#endif /* EGL_ARM_image_format */ + +#ifndef EGL_ARM_implicit_external_sync +#define EGL_ARM_implicit_external_sync 1 +#define EGL_SYNC_PRIOR_COMMANDS_IMPLICIT_EXTERNAL_ARM 0x328A +#endif /* EGL_ARM_implicit_external_sync */ + +#ifndef EGL_ARM_pixmap_multisample_discard +#define EGL_ARM_pixmap_multisample_discard 1 +#define EGL_DISCARD_SAMPLES_ARM 0x3286 +#endif /* EGL_ARM_pixmap_multisample_discard */ + +#ifndef EGL_EXT_bind_to_front +#define EGL_EXT_bind_to_front 1 +#define EGL_FRONT_BUFFER_EXT 0x3464 +#endif /* EGL_EXT_bind_to_front */ + +#ifndef EGL_EXT_buffer_age +#define EGL_EXT_buffer_age 1 +#define EGL_BUFFER_AGE_EXT 0x313D +#endif /* EGL_EXT_buffer_age */ + +#ifndef EGL_EXT_client_extensions +#define EGL_EXT_client_extensions 1 +#endif /* EGL_EXT_client_extensions */ + +#ifndef EGL_EXT_client_sync +#define EGL_EXT_client_sync 1 +#define EGL_SYNC_CLIENT_EXT 0x3364 +#define EGL_SYNC_CLIENT_SIGNAL_EXT 0x3365 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLCLIENTSIGNALSYNCEXTPROC) (EGLDisplay dpy, EGLSync sync, const EGLAttrib *attrib_list); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglClientSignalSyncEXT (EGLDisplay dpy, EGLSync sync, const EGLAttrib *attrib_list); +#endif +#endif /* EGL_EXT_client_sync */ + +#ifndef EGL_EXT_compositor +#define EGL_EXT_compositor 1 +#define EGL_PRIMARY_COMPOSITOR_CONTEXT_EXT 0x3460 +#define EGL_EXTERNAL_REF_ID_EXT 0x3461 +#define EGL_COMPOSITOR_DROP_NEWEST_FRAME_EXT 0x3462 +#define EGL_COMPOSITOR_KEEP_NEWEST_FRAME_EXT 0x3463 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLCOMPOSITORSETCONTEXTLISTEXTPROC) (const EGLint *external_ref_ids, EGLint num_entries); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLCOMPOSITORSETCONTEXTATTRIBUTESEXTPROC) (EGLint external_ref_id, const EGLint *context_attributes, EGLint num_entries); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLCOMPOSITORSETWINDOWLISTEXTPROC) (EGLint external_ref_id, const EGLint *external_win_ids, EGLint num_entries); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLCOMPOSITORSETWINDOWATTRIBUTESEXTPROC) (EGLint external_win_id, const EGLint *window_attributes, EGLint num_entries); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLCOMPOSITORBINDTEXWINDOWEXTPROC) (EGLint external_win_id); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLCOMPOSITORSETSIZEEXTPROC) (EGLint external_win_id, EGLint width, EGLint height); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLCOMPOSITORSWAPPOLICYEXTPROC) (EGLint external_win_id, EGLint policy); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglCompositorSetContextListEXT (const EGLint *external_ref_ids, EGLint num_entries); +EGLAPI EGLBoolean EGLAPIENTRY eglCompositorSetContextAttributesEXT (EGLint external_ref_id, const EGLint *context_attributes, EGLint num_entries); +EGLAPI EGLBoolean EGLAPIENTRY eglCompositorSetWindowListEXT (EGLint external_ref_id, const EGLint *external_win_ids, EGLint num_entries); +EGLAPI EGLBoolean EGLAPIENTRY eglCompositorSetWindowAttributesEXT (EGLint external_win_id, const EGLint *window_attributes, EGLint num_entries); +EGLAPI EGLBoolean EGLAPIENTRY eglCompositorBindTexWindowEXT (EGLint external_win_id); +EGLAPI EGLBoolean EGLAPIENTRY eglCompositorSetSizeEXT (EGLint external_win_id, EGLint width, EGLint height); +EGLAPI EGLBoolean EGLAPIENTRY eglCompositorSwapPolicyEXT (EGLint external_win_id, EGLint policy); +#endif +#endif /* EGL_EXT_compositor */ + +#ifndef EGL_EXT_config_select_group +#define EGL_EXT_config_select_group 1 +#define EGL_CONFIG_SELECT_GROUP_EXT 0x34C0 +#endif /* EGL_EXT_config_select_group */ + +#ifndef EGL_EXT_create_context_robustness +#define EGL_EXT_create_context_robustness 1 +#define EGL_CONTEXT_OPENGL_ROBUST_ACCESS_EXT 0x30BF +#define EGL_CONTEXT_OPENGL_RESET_NOTIFICATION_STRATEGY_EXT 0x3138 +#define EGL_NO_RESET_NOTIFICATION_EXT 0x31BE +#define EGL_LOSE_CONTEXT_ON_RESET_EXT 0x31BF +#endif /* EGL_EXT_create_context_robustness */ + +#ifndef EGL_EXT_device_base +#define EGL_EXT_device_base 1 +typedef void *EGLDeviceEXT; +#define EGL_NO_DEVICE_EXT EGL_CAST(EGLDeviceEXT,0) +#define EGL_BAD_DEVICE_EXT 0x322B +#define EGL_DEVICE_EXT 0x322C +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYDEVICEATTRIBEXTPROC) (EGLDeviceEXT device, EGLint attribute, EGLAttrib *value); +typedef const char *(EGLAPIENTRYP PFNEGLQUERYDEVICESTRINGEXTPROC) (EGLDeviceEXT device, EGLint name); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYDEVICESEXTPROC) (EGLint max_devices, EGLDeviceEXT *devices, EGLint *num_devices); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYDISPLAYATTRIBEXTPROC) (EGLDisplay dpy, EGLint attribute, EGLAttrib *value); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglQueryDeviceAttribEXT (EGLDeviceEXT device, EGLint attribute, EGLAttrib *value); +EGLAPI const char *EGLAPIENTRY eglQueryDeviceStringEXT (EGLDeviceEXT device, EGLint name); +EGLAPI EGLBoolean EGLAPIENTRY eglQueryDevicesEXT (EGLint max_devices, EGLDeviceEXT *devices, EGLint *num_devices); +EGLAPI EGLBoolean EGLAPIENTRY eglQueryDisplayAttribEXT (EGLDisplay dpy, EGLint attribute, EGLAttrib *value); +#endif +#endif /* EGL_EXT_device_base */ + +#ifndef EGL_EXT_device_drm +#define EGL_EXT_device_drm 1 +#define EGL_DRM_DEVICE_FILE_EXT 0x3233 +#define EGL_DRM_MASTER_FD_EXT 0x333C +#endif /* EGL_EXT_device_drm */ + +#ifndef EGL_EXT_device_drm_render_node +#define EGL_EXT_device_drm_render_node 1 +#define EGL_DRM_RENDER_NODE_FILE_EXT 0x3377 +#endif /* EGL_EXT_device_drm_render_node */ + +#ifndef EGL_EXT_device_enumeration +#define EGL_EXT_device_enumeration 1 +#endif /* EGL_EXT_device_enumeration */ + +#ifndef EGL_EXT_device_openwf +#define EGL_EXT_device_openwf 1 +#define EGL_OPENWF_DEVICE_ID_EXT 0x3237 +#define EGL_OPENWF_DEVICE_EXT 0x333D +#endif /* EGL_EXT_device_openwf */ + +#ifndef EGL_EXT_device_persistent_id +#define EGL_EXT_device_persistent_id 1 +#define EGL_DEVICE_UUID_EXT 0x335C +#define EGL_DRIVER_UUID_EXT 0x335D +#define EGL_DRIVER_NAME_EXT 0x335E +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYDEVICEBINARYEXTPROC) (EGLDeviceEXT device, EGLint name, EGLint max_size, void *value, EGLint *size); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglQueryDeviceBinaryEXT (EGLDeviceEXT device, EGLint name, EGLint max_size, void *value, EGLint *size); +#endif +#endif /* EGL_EXT_device_persistent_id */ + +#ifndef EGL_EXT_device_query +#define EGL_EXT_device_query 1 +#endif /* EGL_EXT_device_query */ + +#ifndef EGL_EXT_device_query_name +#define EGL_EXT_device_query_name 1 +#define EGL_RENDERER_EXT 0x335F +#endif /* EGL_EXT_device_query_name */ + +#ifndef EGL_EXT_explicit_device +#define EGL_EXT_explicit_device 1 +#endif /* EGL_EXT_explicit_device */ + +#ifndef EGL_EXT_gl_colorspace_bt2020_linear +#define EGL_EXT_gl_colorspace_bt2020_linear 1 +#define EGL_GL_COLORSPACE_BT2020_LINEAR_EXT 0x333F +#endif /* EGL_EXT_gl_colorspace_bt2020_linear */ + +#ifndef EGL_EXT_gl_colorspace_bt2020_pq +#define EGL_EXT_gl_colorspace_bt2020_pq 1 +#define EGL_GL_COLORSPACE_BT2020_PQ_EXT 0x3340 +#endif /* EGL_EXT_gl_colorspace_bt2020_pq */ + +#ifndef EGL_EXT_gl_colorspace_display_p3 +#define EGL_EXT_gl_colorspace_display_p3 1 +#define EGL_GL_COLORSPACE_DISPLAY_P3_EXT 0x3363 +#endif /* EGL_EXT_gl_colorspace_display_p3 */ + +#ifndef EGL_EXT_gl_colorspace_display_p3_linear +#define EGL_EXT_gl_colorspace_display_p3_linear 1 +#define EGL_GL_COLORSPACE_DISPLAY_P3_LINEAR_EXT 0x3362 +#endif /* EGL_EXT_gl_colorspace_display_p3_linear */ + +#ifndef EGL_EXT_gl_colorspace_display_p3_passthrough +#define EGL_EXT_gl_colorspace_display_p3_passthrough 1 +#define EGL_GL_COLORSPACE_DISPLAY_P3_PASSTHROUGH_EXT 0x3490 +#endif /* EGL_EXT_gl_colorspace_display_p3_passthrough */ + +#ifndef EGL_EXT_gl_colorspace_scrgb +#define EGL_EXT_gl_colorspace_scrgb 1 +#define EGL_GL_COLORSPACE_SCRGB_EXT 0x3351 +#endif /* EGL_EXT_gl_colorspace_scrgb */ + +#ifndef EGL_EXT_gl_colorspace_scrgb_linear +#define EGL_EXT_gl_colorspace_scrgb_linear 1 +#define EGL_GL_COLORSPACE_SCRGB_LINEAR_EXT 0x3350 +#endif /* EGL_EXT_gl_colorspace_scrgb_linear */ + +#ifndef EGL_EXT_image_dma_buf_import +#define EGL_EXT_image_dma_buf_import 1 +#define EGL_LINUX_DMA_BUF_EXT 0x3270 +#define EGL_LINUX_DRM_FOURCC_EXT 0x3271 +#define EGL_DMA_BUF_PLANE0_FD_EXT 0x3272 +#define EGL_DMA_BUF_PLANE0_OFFSET_EXT 0x3273 +#define EGL_DMA_BUF_PLANE0_PITCH_EXT 0x3274 +#define EGL_DMA_BUF_PLANE1_FD_EXT 0x3275 +#define EGL_DMA_BUF_PLANE1_OFFSET_EXT 0x3276 +#define EGL_DMA_BUF_PLANE1_PITCH_EXT 0x3277 +#define EGL_DMA_BUF_PLANE2_FD_EXT 0x3278 +#define EGL_DMA_BUF_PLANE2_OFFSET_EXT 0x3279 +#define EGL_DMA_BUF_PLANE2_PITCH_EXT 0x327A +#define EGL_YUV_COLOR_SPACE_HINT_EXT 0x327B +#define EGL_SAMPLE_RANGE_HINT_EXT 0x327C +#define EGL_YUV_CHROMA_HORIZONTAL_SITING_HINT_EXT 0x327D +#define EGL_YUV_CHROMA_VERTICAL_SITING_HINT_EXT 0x327E +#define EGL_ITU_REC601_EXT 0x327F +#define EGL_ITU_REC709_EXT 0x3280 +#define EGL_ITU_REC2020_EXT 0x3281 +#define EGL_YUV_FULL_RANGE_EXT 0x3282 +#define EGL_YUV_NARROW_RANGE_EXT 0x3283 +#define EGL_YUV_CHROMA_SITING_0_EXT 0x3284 +#define EGL_YUV_CHROMA_SITING_0_5_EXT 0x3285 +#endif /* EGL_EXT_image_dma_buf_import */ + +#ifndef EGL_EXT_image_dma_buf_import_modifiers +#define EGL_EXT_image_dma_buf_import_modifiers 1 +#define EGL_DMA_BUF_PLANE3_FD_EXT 0x3440 +#define EGL_DMA_BUF_PLANE3_OFFSET_EXT 0x3441 +#define EGL_DMA_BUF_PLANE3_PITCH_EXT 0x3442 +#define EGL_DMA_BUF_PLANE0_MODIFIER_LO_EXT 0x3443 +#define EGL_DMA_BUF_PLANE0_MODIFIER_HI_EXT 0x3444 +#define EGL_DMA_BUF_PLANE1_MODIFIER_LO_EXT 0x3445 +#define EGL_DMA_BUF_PLANE1_MODIFIER_HI_EXT 0x3446 +#define EGL_DMA_BUF_PLANE2_MODIFIER_LO_EXT 0x3447 +#define EGL_DMA_BUF_PLANE2_MODIFIER_HI_EXT 0x3448 +#define EGL_DMA_BUF_PLANE3_MODIFIER_LO_EXT 0x3449 +#define EGL_DMA_BUF_PLANE3_MODIFIER_HI_EXT 0x344A +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYDMABUFFORMATSEXTPROC) (EGLDisplay dpy, EGLint max_formats, EGLint *formats, EGLint *num_formats); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYDMABUFMODIFIERSEXTPROC) (EGLDisplay dpy, EGLint format, EGLint max_modifiers, EGLuint64KHR *modifiers, EGLBoolean *external_only, EGLint *num_modifiers); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglQueryDmaBufFormatsEXT (EGLDisplay dpy, EGLint max_formats, EGLint *formats, EGLint *num_formats); +EGLAPI EGLBoolean EGLAPIENTRY eglQueryDmaBufModifiersEXT (EGLDisplay dpy, EGLint format, EGLint max_modifiers, EGLuint64KHR *modifiers, EGLBoolean *external_only, EGLint *num_modifiers); +#endif +#endif /* EGL_EXT_image_dma_buf_import_modifiers */ + +#ifndef EGL_EXT_image_gl_colorspace +#define EGL_EXT_image_gl_colorspace 1 +#define EGL_GL_COLORSPACE_DEFAULT_EXT 0x314D +#endif /* EGL_EXT_image_gl_colorspace */ + +#ifndef EGL_EXT_image_implicit_sync_control +#define EGL_EXT_image_implicit_sync_control 1 +#define EGL_IMPORT_SYNC_TYPE_EXT 0x3470 +#define EGL_IMPORT_IMPLICIT_SYNC_EXT 0x3471 +#define EGL_IMPORT_EXPLICIT_SYNC_EXT 0x3472 +#endif /* EGL_EXT_image_implicit_sync_control */ + +#ifndef EGL_EXT_multiview_window +#define EGL_EXT_multiview_window 1 +#define EGL_MULTIVIEW_VIEW_COUNT_EXT 0x3134 +#endif /* EGL_EXT_multiview_window */ + +#ifndef EGL_EXT_output_base +#define EGL_EXT_output_base 1 +typedef void *EGLOutputLayerEXT; +typedef void *EGLOutputPortEXT; +#define EGL_NO_OUTPUT_LAYER_EXT EGL_CAST(EGLOutputLayerEXT,0) +#define EGL_NO_OUTPUT_PORT_EXT EGL_CAST(EGLOutputPortEXT,0) +#define EGL_BAD_OUTPUT_LAYER_EXT 0x322D +#define EGL_BAD_OUTPUT_PORT_EXT 0x322E +#define EGL_SWAP_INTERVAL_EXT 0x322F +typedef EGLBoolean (EGLAPIENTRYP PFNEGLGETOUTPUTLAYERSEXTPROC) (EGLDisplay dpy, const EGLAttrib *attrib_list, EGLOutputLayerEXT *layers, EGLint max_layers, EGLint *num_layers); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLGETOUTPUTPORTSEXTPROC) (EGLDisplay dpy, const EGLAttrib *attrib_list, EGLOutputPortEXT *ports, EGLint max_ports, EGLint *num_ports); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLOUTPUTLAYERATTRIBEXTPROC) (EGLDisplay dpy, EGLOutputLayerEXT layer, EGLint attribute, EGLAttrib value); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYOUTPUTLAYERATTRIBEXTPROC) (EGLDisplay dpy, EGLOutputLayerEXT layer, EGLint attribute, EGLAttrib *value); +typedef const char *(EGLAPIENTRYP PFNEGLQUERYOUTPUTLAYERSTRINGEXTPROC) (EGLDisplay dpy, EGLOutputLayerEXT layer, EGLint name); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLOUTPUTPORTATTRIBEXTPROC) (EGLDisplay dpy, EGLOutputPortEXT port, EGLint attribute, EGLAttrib value); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYOUTPUTPORTATTRIBEXTPROC) (EGLDisplay dpy, EGLOutputPortEXT port, EGLint attribute, EGLAttrib *value); +typedef const char *(EGLAPIENTRYP PFNEGLQUERYOUTPUTPORTSTRINGEXTPROC) (EGLDisplay dpy, EGLOutputPortEXT port, EGLint name); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglGetOutputLayersEXT (EGLDisplay dpy, const EGLAttrib *attrib_list, EGLOutputLayerEXT *layers, EGLint max_layers, EGLint *num_layers); +EGLAPI EGLBoolean EGLAPIENTRY eglGetOutputPortsEXT (EGLDisplay dpy, const EGLAttrib *attrib_list, EGLOutputPortEXT *ports, EGLint max_ports, EGLint *num_ports); +EGLAPI EGLBoolean EGLAPIENTRY eglOutputLayerAttribEXT (EGLDisplay dpy, EGLOutputLayerEXT layer, EGLint attribute, EGLAttrib value); +EGLAPI EGLBoolean EGLAPIENTRY eglQueryOutputLayerAttribEXT (EGLDisplay dpy, EGLOutputLayerEXT layer, EGLint attribute, EGLAttrib *value); +EGLAPI const char *EGLAPIENTRY eglQueryOutputLayerStringEXT (EGLDisplay dpy, EGLOutputLayerEXT layer, EGLint name); +EGLAPI EGLBoolean EGLAPIENTRY eglOutputPortAttribEXT (EGLDisplay dpy, EGLOutputPortEXT port, EGLint attribute, EGLAttrib value); +EGLAPI EGLBoolean EGLAPIENTRY eglQueryOutputPortAttribEXT (EGLDisplay dpy, EGLOutputPortEXT port, EGLint attribute, EGLAttrib *value); +EGLAPI const char *EGLAPIENTRY eglQueryOutputPortStringEXT (EGLDisplay dpy, EGLOutputPortEXT port, EGLint name); +#endif +#endif /* EGL_EXT_output_base */ + +#ifndef EGL_EXT_output_drm +#define EGL_EXT_output_drm 1 +#define EGL_DRM_CRTC_EXT 0x3234 +#define EGL_DRM_PLANE_EXT 0x3235 +#define EGL_DRM_CONNECTOR_EXT 0x3236 +#endif /* EGL_EXT_output_drm */ + +#ifndef EGL_EXT_output_openwf +#define EGL_EXT_output_openwf 1 +#define EGL_OPENWF_PIPELINE_ID_EXT 0x3238 +#define EGL_OPENWF_PORT_ID_EXT 0x3239 +#endif /* EGL_EXT_output_openwf */ + +#ifndef EGL_EXT_pixel_format_float +#define EGL_EXT_pixel_format_float 1 +#define EGL_COLOR_COMPONENT_TYPE_EXT 0x3339 +#define EGL_COLOR_COMPONENT_TYPE_FIXED_EXT 0x333A +#define EGL_COLOR_COMPONENT_TYPE_FLOAT_EXT 0x333B +#endif /* EGL_EXT_pixel_format_float */ + +#ifndef EGL_EXT_platform_base +#define EGL_EXT_platform_base 1 +typedef EGLDisplay (EGLAPIENTRYP PFNEGLGETPLATFORMDISPLAYEXTPROC) (EGLenum platform, void *native_display, const EGLint *attrib_list); +typedef EGLSurface (EGLAPIENTRYP PFNEGLCREATEPLATFORMWINDOWSURFACEEXTPROC) (EGLDisplay dpy, EGLConfig config, void *native_window, const EGLint *attrib_list); +typedef EGLSurface (EGLAPIENTRYP PFNEGLCREATEPLATFORMPIXMAPSURFACEEXTPROC) (EGLDisplay dpy, EGLConfig config, void *native_pixmap, const EGLint *attrib_list); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLDisplay EGLAPIENTRY eglGetPlatformDisplayEXT (EGLenum platform, void *native_display, const EGLint *attrib_list); +EGLAPI EGLSurface EGLAPIENTRY eglCreatePlatformWindowSurfaceEXT (EGLDisplay dpy, EGLConfig config, void *native_window, const EGLint *attrib_list); +EGLAPI EGLSurface EGLAPIENTRY eglCreatePlatformPixmapSurfaceEXT (EGLDisplay dpy, EGLConfig config, void *native_pixmap, const EGLint *attrib_list); +#endif +#endif /* EGL_EXT_platform_base */ + +#ifndef EGL_EXT_platform_device +#define EGL_EXT_platform_device 1 +#define EGL_PLATFORM_DEVICE_EXT 0x313F +#endif /* EGL_EXT_platform_device */ + +#ifndef EGL_EXT_platform_wayland +#define EGL_EXT_platform_wayland 1 +#define EGL_PLATFORM_WAYLAND_EXT 0x31D8 +#endif /* EGL_EXT_platform_wayland */ + +#ifndef EGL_EXT_platform_x11 +#define EGL_EXT_platform_x11 1 +#define EGL_PLATFORM_X11_EXT 0x31D5 +#define EGL_PLATFORM_X11_SCREEN_EXT 0x31D6 +#endif /* EGL_EXT_platform_x11 */ + +#ifndef EGL_EXT_platform_xcb +#define EGL_EXT_platform_xcb 1 +#define EGL_PLATFORM_XCB_EXT 0x31DC +#define EGL_PLATFORM_XCB_SCREEN_EXT 0x31DE +#endif /* EGL_EXT_platform_xcb */ + +#ifndef EGL_EXT_present_opaque +#define EGL_EXT_present_opaque 1 +#define EGL_PRESENT_OPAQUE_EXT 0x31DF +#endif /* EGL_EXT_present_opaque */ + +#ifndef EGL_EXT_protected_content +#define EGL_EXT_protected_content 1 +#define EGL_PROTECTED_CONTENT_EXT 0x32C0 +#endif /* EGL_EXT_protected_content */ + +#ifndef EGL_EXT_protected_surface +#define EGL_EXT_protected_surface 1 +#endif /* EGL_EXT_protected_surface */ + +#ifndef EGL_EXT_stream_consumer_egloutput +#define EGL_EXT_stream_consumer_egloutput 1 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSTREAMCONSUMEROUTPUTEXTPROC) (EGLDisplay dpy, EGLStreamKHR stream, EGLOutputLayerEXT layer); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglStreamConsumerOutputEXT (EGLDisplay dpy, EGLStreamKHR stream, EGLOutputLayerEXT layer); +#endif +#endif /* EGL_EXT_stream_consumer_egloutput */ + +#ifndef EGL_EXT_surface_CTA861_3_metadata +#define EGL_EXT_surface_CTA861_3_metadata 1 +#define EGL_CTA861_3_MAX_CONTENT_LIGHT_LEVEL_EXT 0x3360 +#define EGL_CTA861_3_MAX_FRAME_AVERAGE_LEVEL_EXT 0x3361 +#endif /* EGL_EXT_surface_CTA861_3_metadata */ + +#ifndef EGL_EXT_surface_SMPTE2086_metadata +#define EGL_EXT_surface_SMPTE2086_metadata 1 +#define EGL_SMPTE2086_DISPLAY_PRIMARY_RX_EXT 0x3341 +#define EGL_SMPTE2086_DISPLAY_PRIMARY_RY_EXT 0x3342 +#define EGL_SMPTE2086_DISPLAY_PRIMARY_GX_EXT 0x3343 +#define EGL_SMPTE2086_DISPLAY_PRIMARY_GY_EXT 0x3344 +#define EGL_SMPTE2086_DISPLAY_PRIMARY_BX_EXT 0x3345 +#define EGL_SMPTE2086_DISPLAY_PRIMARY_BY_EXT 0x3346 +#define EGL_SMPTE2086_WHITE_POINT_X_EXT 0x3347 +#define EGL_SMPTE2086_WHITE_POINT_Y_EXT 0x3348 +#define EGL_SMPTE2086_MAX_LUMINANCE_EXT 0x3349 +#define EGL_SMPTE2086_MIN_LUMINANCE_EXT 0x334A +#define EGL_METADATA_SCALING_EXT 50000 +#endif /* EGL_EXT_surface_SMPTE2086_metadata */ + +#ifndef EGL_EXT_surface_compression +#define EGL_EXT_surface_compression 1 +#define EGL_SURFACE_COMPRESSION_EXT 0x34B0 +#define EGL_SURFACE_COMPRESSION_PLANE1_EXT 0x328E +#define EGL_SURFACE_COMPRESSION_PLANE2_EXT 0x328F +#define EGL_SURFACE_COMPRESSION_FIXED_RATE_NONE_EXT 0x34B1 +#define EGL_SURFACE_COMPRESSION_FIXED_RATE_DEFAULT_EXT 0x34B2 +#define EGL_SURFACE_COMPRESSION_FIXED_RATE_1BPC_EXT 0x34B4 +#define EGL_SURFACE_COMPRESSION_FIXED_RATE_2BPC_EXT 0x34B5 +#define EGL_SURFACE_COMPRESSION_FIXED_RATE_3BPC_EXT 0x34B6 +#define EGL_SURFACE_COMPRESSION_FIXED_RATE_4BPC_EXT 0x34B7 +#define EGL_SURFACE_COMPRESSION_FIXED_RATE_5BPC_EXT 0x34B8 +#define EGL_SURFACE_COMPRESSION_FIXED_RATE_6BPC_EXT 0x34B9 +#define EGL_SURFACE_COMPRESSION_FIXED_RATE_7BPC_EXT 0x34BA +#define EGL_SURFACE_COMPRESSION_FIXED_RATE_8BPC_EXT 0x34BB +#define EGL_SURFACE_COMPRESSION_FIXED_RATE_9BPC_EXT 0x34BC +#define EGL_SURFACE_COMPRESSION_FIXED_RATE_10BPC_EXT 0x34BD +#define EGL_SURFACE_COMPRESSION_FIXED_RATE_11BPC_EXT 0x34BE +#define EGL_SURFACE_COMPRESSION_FIXED_RATE_12BPC_EXT 0x34BF +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYSUPPORTEDCOMPRESSIONRATESEXTPROC) (EGLDisplay dpy, EGLConfig config, const EGLAttrib *attrib_list, EGLint *rates, EGLint rate_size, EGLint *num_rates); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglQuerySupportedCompressionRatesEXT (EGLDisplay dpy, EGLConfig config, const EGLAttrib *attrib_list, EGLint *rates, EGLint rate_size, EGLint *num_rates); +#endif +#endif /* EGL_EXT_surface_compression */ + +#ifndef EGL_EXT_swap_buffers_with_damage +#define EGL_EXT_swap_buffers_with_damage 1 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSWAPBUFFERSWITHDAMAGEEXTPROC) (EGLDisplay dpy, EGLSurface surface, const EGLint *rects, EGLint n_rects); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglSwapBuffersWithDamageEXT (EGLDisplay dpy, EGLSurface surface, const EGLint *rects, EGLint n_rects); +#endif +#endif /* EGL_EXT_swap_buffers_with_damage */ + +#ifndef EGL_EXT_sync_reuse +#define EGL_EXT_sync_reuse 1 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLUNSIGNALSYNCEXTPROC) (EGLDisplay dpy, EGLSync sync, const EGLAttrib *attrib_list); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglUnsignalSyncEXT (EGLDisplay dpy, EGLSync sync, const EGLAttrib *attrib_list); +#endif +#endif /* EGL_EXT_sync_reuse */ + +#ifndef EGL_EXT_yuv_surface +#define EGL_EXT_yuv_surface 1 +#define EGL_YUV_ORDER_EXT 0x3301 +#define EGL_YUV_NUMBER_OF_PLANES_EXT 0x3311 +#define EGL_YUV_SUBSAMPLE_EXT 0x3312 +#define EGL_YUV_DEPTH_RANGE_EXT 0x3317 +#define EGL_YUV_CSC_STANDARD_EXT 0x330A +#define EGL_YUV_PLANE_BPP_EXT 0x331A +#define EGL_YUV_BUFFER_EXT 0x3300 +#define EGL_YUV_ORDER_YUV_EXT 0x3302 +#define EGL_YUV_ORDER_YVU_EXT 0x3303 +#define EGL_YUV_ORDER_YUYV_EXT 0x3304 +#define EGL_YUV_ORDER_UYVY_EXT 0x3305 +#define EGL_YUV_ORDER_YVYU_EXT 0x3306 +#define EGL_YUV_ORDER_VYUY_EXT 0x3307 +#define EGL_YUV_ORDER_AYUV_EXT 0x3308 +#define EGL_YUV_SUBSAMPLE_4_2_0_EXT 0x3313 +#define EGL_YUV_SUBSAMPLE_4_2_2_EXT 0x3314 +#define EGL_YUV_SUBSAMPLE_4_4_4_EXT 0x3315 +#define EGL_YUV_DEPTH_RANGE_LIMITED_EXT 0x3318 +#define EGL_YUV_DEPTH_RANGE_FULL_EXT 0x3319 +#define EGL_YUV_CSC_STANDARD_601_EXT 0x330B +#define EGL_YUV_CSC_STANDARD_709_EXT 0x330C +#define EGL_YUV_CSC_STANDARD_2020_EXT 0x330D +#define EGL_YUV_PLANE_BPP_0_EXT 0x331B +#define EGL_YUV_PLANE_BPP_8_EXT 0x331C +#define EGL_YUV_PLANE_BPP_10_EXT 0x331D +#endif /* EGL_EXT_yuv_surface */ + +#ifndef EGL_HI_clientpixmap +#define EGL_HI_clientpixmap 1 +struct EGLClientPixmapHI { + void *pData; + EGLint iWidth; + EGLint iHeight; + EGLint iStride; +}; +#define EGL_CLIENT_PIXMAP_POINTER_HI 0x8F74 +typedef EGLSurface (EGLAPIENTRYP PFNEGLCREATEPIXMAPSURFACEHIPROC) (EGLDisplay dpy, EGLConfig config, struct EGLClientPixmapHI *pixmap); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLSurface EGLAPIENTRY eglCreatePixmapSurfaceHI (EGLDisplay dpy, EGLConfig config, struct EGLClientPixmapHI *pixmap); +#endif +#endif /* EGL_HI_clientpixmap */ + +#ifndef EGL_HI_colorformats +#define EGL_HI_colorformats 1 +#define EGL_COLOR_FORMAT_HI 0x8F70 +#define EGL_COLOR_RGB_HI 0x8F71 +#define EGL_COLOR_RGBA_HI 0x8F72 +#define EGL_COLOR_ARGB_HI 0x8F73 +#endif /* EGL_HI_colorformats */ + +#ifndef EGL_IMG_context_priority +#define EGL_IMG_context_priority 1 +#define EGL_CONTEXT_PRIORITY_LEVEL_IMG 0x3100 +#define EGL_CONTEXT_PRIORITY_HIGH_IMG 0x3101 +#define EGL_CONTEXT_PRIORITY_MEDIUM_IMG 0x3102 +#define EGL_CONTEXT_PRIORITY_LOW_IMG 0x3103 +#endif /* EGL_IMG_context_priority */ + +#ifndef EGL_IMG_image_plane_attribs +#define EGL_IMG_image_plane_attribs 1 +#define EGL_NATIVE_BUFFER_MULTIPLANE_SEPARATE_IMG 0x3105 +#define EGL_NATIVE_BUFFER_PLANE_OFFSET_IMG 0x3106 +#endif /* EGL_IMG_image_plane_attribs */ + +#ifndef EGL_MESA_drm_image +#define EGL_MESA_drm_image 1 +#define EGL_DRM_BUFFER_FORMAT_MESA 0x31D0 +#define EGL_DRM_BUFFER_USE_MESA 0x31D1 +#define EGL_DRM_BUFFER_FORMAT_ARGB32_MESA 0x31D2 +#define EGL_DRM_BUFFER_MESA 0x31D3 +#define EGL_DRM_BUFFER_STRIDE_MESA 0x31D4 +#define EGL_DRM_BUFFER_USE_SCANOUT_MESA 0x00000001 +#define EGL_DRM_BUFFER_USE_SHARE_MESA 0x00000002 +#define EGL_DRM_BUFFER_USE_CURSOR_MESA 0x00000004 +typedef EGLImageKHR (EGLAPIENTRYP PFNEGLCREATEDRMIMAGEMESAPROC) (EGLDisplay dpy, const EGLint *attrib_list); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLEXPORTDRMIMAGEMESAPROC) (EGLDisplay dpy, EGLImageKHR image, EGLint *name, EGLint *handle, EGLint *stride); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLImageKHR EGLAPIENTRY eglCreateDRMImageMESA (EGLDisplay dpy, const EGLint *attrib_list); +EGLAPI EGLBoolean EGLAPIENTRY eglExportDRMImageMESA (EGLDisplay dpy, EGLImageKHR image, EGLint *name, EGLint *handle, EGLint *stride); +#endif +#endif /* EGL_MESA_drm_image */ + +#ifndef EGL_MESA_image_dma_buf_export +#define EGL_MESA_image_dma_buf_export 1 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLEXPORTDMABUFIMAGEQUERYMESAPROC) (EGLDisplay dpy, EGLImageKHR image, int *fourcc, int *num_planes, EGLuint64KHR *modifiers); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLEXPORTDMABUFIMAGEMESAPROC) (EGLDisplay dpy, EGLImageKHR image, int *fds, EGLint *strides, EGLint *offsets); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglExportDMABUFImageQueryMESA (EGLDisplay dpy, EGLImageKHR image, int *fourcc, int *num_planes, EGLuint64KHR *modifiers); +EGLAPI EGLBoolean EGLAPIENTRY eglExportDMABUFImageMESA (EGLDisplay dpy, EGLImageKHR image, int *fds, EGLint *strides, EGLint *offsets); +#endif +#endif /* EGL_MESA_image_dma_buf_export */ + +#ifndef EGL_MESA_platform_gbm +#define EGL_MESA_platform_gbm 1 +#define EGL_PLATFORM_GBM_MESA 0x31D7 +#endif /* EGL_MESA_platform_gbm */ + +#ifndef EGL_MESA_platform_surfaceless +#define EGL_MESA_platform_surfaceless 1 +#define EGL_PLATFORM_SURFACELESS_MESA 0x31DD +#endif /* EGL_MESA_platform_surfaceless */ + +#ifndef EGL_MESA_query_driver +#define EGL_MESA_query_driver 1 +typedef char *(EGLAPIENTRYP PFNEGLGETDISPLAYDRIVERCONFIGPROC) (EGLDisplay dpy); +typedef const char *(EGLAPIENTRYP PFNEGLGETDISPLAYDRIVERNAMEPROC) (EGLDisplay dpy); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI char *EGLAPIENTRY eglGetDisplayDriverConfig (EGLDisplay dpy); +EGLAPI const char *EGLAPIENTRY eglGetDisplayDriverName (EGLDisplay dpy); +#endif +#endif /* EGL_MESA_query_driver */ + +#ifndef EGL_NOK_swap_region +#define EGL_NOK_swap_region 1 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSWAPBUFFERSREGIONNOKPROC) (EGLDisplay dpy, EGLSurface surface, EGLint numRects, const EGLint *rects); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglSwapBuffersRegionNOK (EGLDisplay dpy, EGLSurface surface, EGLint numRects, const EGLint *rects); +#endif +#endif /* EGL_NOK_swap_region */ + +#ifndef EGL_NOK_swap_region2 +#define EGL_NOK_swap_region2 1 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSWAPBUFFERSREGION2NOKPROC) (EGLDisplay dpy, EGLSurface surface, EGLint numRects, const EGLint *rects); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglSwapBuffersRegion2NOK (EGLDisplay dpy, EGLSurface surface, EGLint numRects, const EGLint *rects); +#endif +#endif /* EGL_NOK_swap_region2 */ + +#ifndef EGL_NOK_texture_from_pixmap +#define EGL_NOK_texture_from_pixmap 1 +#define EGL_Y_INVERTED_NOK 0x307F +#endif /* EGL_NOK_texture_from_pixmap */ + +#ifndef EGL_NV_3dvision_surface +#define EGL_NV_3dvision_surface 1 +#define EGL_AUTO_STEREO_NV 0x3136 +#endif /* EGL_NV_3dvision_surface */ + +#ifndef EGL_NV_context_priority_realtime +#define EGL_NV_context_priority_realtime 1 +#define EGL_CONTEXT_PRIORITY_REALTIME_NV 0x3357 +#endif /* EGL_NV_context_priority_realtime */ + +#ifndef EGL_NV_coverage_sample +#define EGL_NV_coverage_sample 1 +#define EGL_COVERAGE_BUFFERS_NV 0x30E0 +#define EGL_COVERAGE_SAMPLES_NV 0x30E1 +#endif /* EGL_NV_coverage_sample */ + +#ifndef EGL_NV_coverage_sample_resolve +#define EGL_NV_coverage_sample_resolve 1 +#define EGL_COVERAGE_SAMPLE_RESOLVE_NV 0x3131 +#define EGL_COVERAGE_SAMPLE_RESOLVE_DEFAULT_NV 0x3132 +#define EGL_COVERAGE_SAMPLE_RESOLVE_NONE_NV 0x3133 +#endif /* EGL_NV_coverage_sample_resolve */ + +#ifndef EGL_NV_cuda_event +#define EGL_NV_cuda_event 1 +#define EGL_CUDA_EVENT_HANDLE_NV 0x323B +#define EGL_SYNC_CUDA_EVENT_NV 0x323C +#define EGL_SYNC_CUDA_EVENT_COMPLETE_NV 0x323D +#endif /* EGL_NV_cuda_event */ + +#ifndef EGL_NV_depth_nonlinear +#define EGL_NV_depth_nonlinear 1 +#define EGL_DEPTH_ENCODING_NV 0x30E2 +#define EGL_DEPTH_ENCODING_NONE_NV 0 +#define EGL_DEPTH_ENCODING_NONLINEAR_NV 0x30E3 +#endif /* EGL_NV_depth_nonlinear */ + +#ifndef EGL_NV_device_cuda +#define EGL_NV_device_cuda 1 +#define EGL_CUDA_DEVICE_NV 0x323A +#endif /* EGL_NV_device_cuda */ + +#ifndef EGL_NV_native_query +#define EGL_NV_native_query 1 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYNATIVEDISPLAYNVPROC) (EGLDisplay dpy, EGLNativeDisplayType *display_id); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYNATIVEWINDOWNVPROC) (EGLDisplay dpy, EGLSurface surf, EGLNativeWindowType *window); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYNATIVEPIXMAPNVPROC) (EGLDisplay dpy, EGLSurface surf, EGLNativePixmapType *pixmap); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglQueryNativeDisplayNV (EGLDisplay dpy, EGLNativeDisplayType *display_id); +EGLAPI EGLBoolean EGLAPIENTRY eglQueryNativeWindowNV (EGLDisplay dpy, EGLSurface surf, EGLNativeWindowType *window); +EGLAPI EGLBoolean EGLAPIENTRY eglQueryNativePixmapNV (EGLDisplay dpy, EGLSurface surf, EGLNativePixmapType *pixmap); +#endif +#endif /* EGL_NV_native_query */ + +#ifndef EGL_NV_post_convert_rounding +#define EGL_NV_post_convert_rounding 1 +#endif /* EGL_NV_post_convert_rounding */ + +#ifndef EGL_NV_post_sub_buffer +#define EGL_NV_post_sub_buffer 1 +#define EGL_POST_SUB_BUFFER_SUPPORTED_NV 0x30BE +typedef EGLBoolean (EGLAPIENTRYP PFNEGLPOSTSUBBUFFERNVPROC) (EGLDisplay dpy, EGLSurface surface, EGLint x, EGLint y, EGLint width, EGLint height); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglPostSubBufferNV (EGLDisplay dpy, EGLSurface surface, EGLint x, EGLint y, EGLint width, EGLint height); +#endif +#endif /* EGL_NV_post_sub_buffer */ + +#ifndef EGL_NV_quadruple_buffer +#define EGL_NV_quadruple_buffer 1 +#define EGL_QUADRUPLE_BUFFER_NV 0x3231 +#endif /* EGL_NV_quadruple_buffer */ + +#ifndef EGL_NV_robustness_video_memory_purge +#define EGL_NV_robustness_video_memory_purge 1 +#define EGL_GENERATE_RESET_ON_VIDEO_MEMORY_PURGE_NV 0x334C +#endif /* EGL_NV_robustness_video_memory_purge */ + +#ifndef EGL_NV_stream_consumer_eglimage +#define EGL_NV_stream_consumer_eglimage 1 +#define EGL_STREAM_CONSUMER_IMAGE_NV 0x3373 +#define EGL_STREAM_IMAGE_ADD_NV 0x3374 +#define EGL_STREAM_IMAGE_REMOVE_NV 0x3375 +#define EGL_STREAM_IMAGE_AVAILABLE_NV 0x3376 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSTREAMIMAGECONSUMERCONNECTNVPROC) (EGLDisplay dpy, EGLStreamKHR stream, EGLint num_modifiers, const EGLuint64KHR *modifiers, const EGLAttrib *attrib_list); +typedef EGLint (EGLAPIENTRYP PFNEGLQUERYSTREAMCONSUMEREVENTNVPROC) (EGLDisplay dpy, EGLStreamKHR stream, EGLTime timeout, EGLenum *event, EGLAttrib *aux); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSTREAMACQUIREIMAGENVPROC) (EGLDisplay dpy, EGLStreamKHR stream, EGLImage *pImage, EGLSync sync); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSTREAMRELEASEIMAGENVPROC) (EGLDisplay dpy, EGLStreamKHR stream, EGLImage image, EGLSync sync); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglStreamImageConsumerConnectNV (EGLDisplay dpy, EGLStreamKHR stream, EGLint num_modifiers, const EGLuint64KHR *modifiers, const EGLAttrib *attrib_list); +EGLAPI EGLint EGLAPIENTRY eglQueryStreamConsumerEventNV (EGLDisplay dpy, EGLStreamKHR stream, EGLTime timeout, EGLenum *event, EGLAttrib *aux); +EGLAPI EGLBoolean EGLAPIENTRY eglStreamAcquireImageNV (EGLDisplay dpy, EGLStreamKHR stream, EGLImage *pImage, EGLSync sync); +EGLAPI EGLBoolean EGLAPIENTRY eglStreamReleaseImageNV (EGLDisplay dpy, EGLStreamKHR stream, EGLImage image, EGLSync sync); +#endif +#endif /* EGL_NV_stream_consumer_eglimage */ + +#ifndef EGL_NV_stream_consumer_gltexture_yuv +#define EGL_NV_stream_consumer_gltexture_yuv 1 +#define EGL_YUV_PLANE0_TEXTURE_UNIT_NV 0x332C +#define EGL_YUV_PLANE1_TEXTURE_UNIT_NV 0x332D +#define EGL_YUV_PLANE2_TEXTURE_UNIT_NV 0x332E +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSTREAMCONSUMERGLTEXTUREEXTERNALATTRIBSNVPROC) (EGLDisplay dpy, EGLStreamKHR stream, const EGLAttrib *attrib_list); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglStreamConsumerGLTextureExternalAttribsNV (EGLDisplay dpy, EGLStreamKHR stream, const EGLAttrib *attrib_list); +#endif +#endif /* EGL_NV_stream_consumer_gltexture_yuv */ + +#ifndef EGL_NV_stream_cross_display +#define EGL_NV_stream_cross_display 1 +#define EGL_STREAM_CROSS_DISPLAY_NV 0x334E +#endif /* EGL_NV_stream_cross_display */ + +#ifndef EGL_NV_stream_cross_object +#define EGL_NV_stream_cross_object 1 +#define EGL_STREAM_CROSS_OBJECT_NV 0x334D +#endif /* EGL_NV_stream_cross_object */ + +#ifndef EGL_NV_stream_cross_partition +#define EGL_NV_stream_cross_partition 1 +#define EGL_STREAM_CROSS_PARTITION_NV 0x323F +#endif /* EGL_NV_stream_cross_partition */ + +#ifndef EGL_NV_stream_cross_process +#define EGL_NV_stream_cross_process 1 +#define EGL_STREAM_CROSS_PROCESS_NV 0x3245 +#endif /* EGL_NV_stream_cross_process */ + +#ifndef EGL_NV_stream_cross_system +#define EGL_NV_stream_cross_system 1 +#define EGL_STREAM_CROSS_SYSTEM_NV 0x334F +#endif /* EGL_NV_stream_cross_system */ + +#ifndef EGL_NV_stream_dma +#define EGL_NV_stream_dma 1 +#define EGL_STREAM_DMA_NV 0x3371 +#define EGL_STREAM_DMA_SERVER_NV 0x3372 +#endif /* EGL_NV_stream_dma */ + +#ifndef EGL_NV_stream_fifo_next +#define EGL_NV_stream_fifo_next 1 +#define EGL_PENDING_FRAME_NV 0x3329 +#define EGL_STREAM_TIME_PENDING_NV 0x332A +#endif /* EGL_NV_stream_fifo_next */ + +#ifndef EGL_NV_stream_fifo_synchronous +#define EGL_NV_stream_fifo_synchronous 1 +#define EGL_STREAM_FIFO_SYNCHRONOUS_NV 0x3336 +#endif /* EGL_NV_stream_fifo_synchronous */ + +#ifndef EGL_NV_stream_flush +#define EGL_NV_stream_flush 1 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSTREAMFLUSHNVPROC) (EGLDisplay dpy, EGLStreamKHR stream); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglStreamFlushNV (EGLDisplay dpy, EGLStreamKHR stream); +#endif +#endif /* EGL_NV_stream_flush */ + +#ifndef EGL_NV_stream_frame_limits +#define EGL_NV_stream_frame_limits 1 +#define EGL_PRODUCER_MAX_FRAME_HINT_NV 0x3337 +#define EGL_CONSUMER_MAX_FRAME_HINT_NV 0x3338 +#endif /* EGL_NV_stream_frame_limits */ + +#ifndef EGL_NV_stream_metadata +#define EGL_NV_stream_metadata 1 +#define EGL_MAX_STREAM_METADATA_BLOCKS_NV 0x3250 +#define EGL_MAX_STREAM_METADATA_BLOCK_SIZE_NV 0x3251 +#define EGL_MAX_STREAM_METADATA_TOTAL_SIZE_NV 0x3252 +#define EGL_PRODUCER_METADATA_NV 0x3253 +#define EGL_CONSUMER_METADATA_NV 0x3254 +#define EGL_PENDING_METADATA_NV 0x3328 +#define EGL_METADATA0_SIZE_NV 0x3255 +#define EGL_METADATA1_SIZE_NV 0x3256 +#define EGL_METADATA2_SIZE_NV 0x3257 +#define EGL_METADATA3_SIZE_NV 0x3258 +#define EGL_METADATA0_TYPE_NV 0x3259 +#define EGL_METADATA1_TYPE_NV 0x325A +#define EGL_METADATA2_TYPE_NV 0x325B +#define EGL_METADATA3_TYPE_NV 0x325C +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYDISPLAYATTRIBNVPROC) (EGLDisplay dpy, EGLint attribute, EGLAttrib *value); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSETSTREAMMETADATANVPROC) (EGLDisplay dpy, EGLStreamKHR stream, EGLint n, EGLint offset, EGLint size, const void *data); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYSTREAMMETADATANVPROC) (EGLDisplay dpy, EGLStreamKHR stream, EGLenum name, EGLint n, EGLint offset, EGLint size, void *data); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglQueryDisplayAttribNV (EGLDisplay dpy, EGLint attribute, EGLAttrib *value); +EGLAPI EGLBoolean EGLAPIENTRY eglSetStreamMetadataNV (EGLDisplay dpy, EGLStreamKHR stream, EGLint n, EGLint offset, EGLint size, const void *data); +EGLAPI EGLBoolean EGLAPIENTRY eglQueryStreamMetadataNV (EGLDisplay dpy, EGLStreamKHR stream, EGLenum name, EGLint n, EGLint offset, EGLint size, void *data); +#endif +#endif /* EGL_NV_stream_metadata */ + +#ifndef EGL_NV_stream_origin +#define EGL_NV_stream_origin 1 +#define EGL_STREAM_FRAME_ORIGIN_X_NV 0x3366 +#define EGL_STREAM_FRAME_ORIGIN_Y_NV 0x3367 +#define EGL_STREAM_FRAME_MAJOR_AXIS_NV 0x3368 +#define EGL_CONSUMER_AUTO_ORIENTATION_NV 0x3369 +#define EGL_PRODUCER_AUTO_ORIENTATION_NV 0x336A +#define EGL_LEFT_NV 0x336B +#define EGL_RIGHT_NV 0x336C +#define EGL_TOP_NV 0x336D +#define EGL_BOTTOM_NV 0x336E +#define EGL_X_AXIS_NV 0x336F +#define EGL_Y_AXIS_NV 0x3370 +#endif /* EGL_NV_stream_origin */ + +#ifndef EGL_NV_stream_remote +#define EGL_NV_stream_remote 1 +#define EGL_STREAM_STATE_INITIALIZING_NV 0x3240 +#define EGL_STREAM_TYPE_NV 0x3241 +#define EGL_STREAM_PROTOCOL_NV 0x3242 +#define EGL_STREAM_ENDPOINT_NV 0x3243 +#define EGL_STREAM_LOCAL_NV 0x3244 +#define EGL_STREAM_PRODUCER_NV 0x3247 +#define EGL_STREAM_CONSUMER_NV 0x3248 +#define EGL_STREAM_PROTOCOL_FD_NV 0x3246 +#endif /* EGL_NV_stream_remote */ + +#ifndef EGL_NV_stream_reset +#define EGL_NV_stream_reset 1 +#define EGL_SUPPORT_RESET_NV 0x3334 +#define EGL_SUPPORT_REUSE_NV 0x3335 +typedef EGLBoolean (EGLAPIENTRYP PFNEGLRESETSTREAMNVPROC) (EGLDisplay dpy, EGLStreamKHR stream); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglResetStreamNV (EGLDisplay dpy, EGLStreamKHR stream); +#endif +#endif /* EGL_NV_stream_reset */ + +#ifndef EGL_NV_stream_socket +#define EGL_NV_stream_socket 1 +#define EGL_STREAM_PROTOCOL_SOCKET_NV 0x324B +#define EGL_SOCKET_HANDLE_NV 0x324C +#define EGL_SOCKET_TYPE_NV 0x324D +#endif /* EGL_NV_stream_socket */ + +#ifndef EGL_NV_stream_socket_inet +#define EGL_NV_stream_socket_inet 1 +#define EGL_SOCKET_TYPE_INET_NV 0x324F +#endif /* EGL_NV_stream_socket_inet */ + +#ifndef EGL_NV_stream_socket_unix +#define EGL_NV_stream_socket_unix 1 +#define EGL_SOCKET_TYPE_UNIX_NV 0x324E +#endif /* EGL_NV_stream_socket_unix */ + +#ifndef EGL_NV_stream_sync +#define EGL_NV_stream_sync 1 +#define EGL_SYNC_NEW_FRAME_NV 0x321F +typedef EGLSyncKHR (EGLAPIENTRYP PFNEGLCREATESTREAMSYNCNVPROC) (EGLDisplay dpy, EGLStreamKHR stream, EGLenum type, const EGLint *attrib_list); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLSyncKHR EGLAPIENTRY eglCreateStreamSyncNV (EGLDisplay dpy, EGLStreamKHR stream, EGLenum type, const EGLint *attrib_list); +#endif +#endif /* EGL_NV_stream_sync */ + +#ifndef EGL_NV_sync +#define EGL_NV_sync 1 +typedef void *EGLSyncNV; +typedef khronos_utime_nanoseconds_t EGLTimeNV; +#ifdef KHRONOS_SUPPORT_INT64 +#define EGL_SYNC_PRIOR_COMMANDS_COMPLETE_NV 0x30E6 +#define EGL_SYNC_STATUS_NV 0x30E7 +#define EGL_SIGNALED_NV 0x30E8 +#define EGL_UNSIGNALED_NV 0x30E9 +#define EGL_SYNC_FLUSH_COMMANDS_BIT_NV 0x0001 +#define EGL_FOREVER_NV 0xFFFFFFFFFFFFFFFFull +#define EGL_ALREADY_SIGNALED_NV 0x30EA +#define EGL_TIMEOUT_EXPIRED_NV 0x30EB +#define EGL_CONDITION_SATISFIED_NV 0x30EC +#define EGL_SYNC_TYPE_NV 0x30ED +#define EGL_SYNC_CONDITION_NV 0x30EE +#define EGL_SYNC_FENCE_NV 0x30EF +#define EGL_NO_SYNC_NV EGL_CAST(EGLSyncNV,0) +typedef EGLSyncNV (EGLAPIENTRYP PFNEGLCREATEFENCESYNCNVPROC) (EGLDisplay dpy, EGLenum condition, const EGLint *attrib_list); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLDESTROYSYNCNVPROC) (EGLSyncNV sync); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLFENCENVPROC) (EGLSyncNV sync); +typedef EGLint (EGLAPIENTRYP PFNEGLCLIENTWAITSYNCNVPROC) (EGLSyncNV sync, EGLint flags, EGLTimeNV timeout); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLSIGNALSYNCNVPROC) (EGLSyncNV sync, EGLenum mode); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLGETSYNCATTRIBNVPROC) (EGLSyncNV sync, EGLint attribute, EGLint *value); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLSyncNV EGLAPIENTRY eglCreateFenceSyncNV (EGLDisplay dpy, EGLenum condition, const EGLint *attrib_list); +EGLAPI EGLBoolean EGLAPIENTRY eglDestroySyncNV (EGLSyncNV sync); +EGLAPI EGLBoolean EGLAPIENTRY eglFenceNV (EGLSyncNV sync); +EGLAPI EGLint EGLAPIENTRY eglClientWaitSyncNV (EGLSyncNV sync, EGLint flags, EGLTimeNV timeout); +EGLAPI EGLBoolean EGLAPIENTRY eglSignalSyncNV (EGLSyncNV sync, EGLenum mode); +EGLAPI EGLBoolean EGLAPIENTRY eglGetSyncAttribNV (EGLSyncNV sync, EGLint attribute, EGLint *value); +#endif +#endif /* KHRONOS_SUPPORT_INT64 */ +#endif /* EGL_NV_sync */ + +#ifndef EGL_NV_system_time +#define EGL_NV_system_time 1 +typedef khronos_utime_nanoseconds_t EGLuint64NV; +#ifdef KHRONOS_SUPPORT_INT64 +typedef EGLuint64NV (EGLAPIENTRYP PFNEGLGETSYSTEMTIMEFREQUENCYNVPROC) (void); +typedef EGLuint64NV (EGLAPIENTRYP PFNEGLGETSYSTEMTIMENVPROC) (void); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLuint64NV EGLAPIENTRY eglGetSystemTimeFrequencyNV (void); +EGLAPI EGLuint64NV EGLAPIENTRY eglGetSystemTimeNV (void); +#endif +#endif /* KHRONOS_SUPPORT_INT64 */ +#endif /* EGL_NV_system_time */ + +#ifndef EGL_NV_triple_buffer +#define EGL_NV_triple_buffer 1 +#define EGL_TRIPLE_BUFFER_NV 0x3230 +#endif /* EGL_NV_triple_buffer */ + +#ifndef EGL_TIZEN_image_native_buffer +#define EGL_TIZEN_image_native_buffer 1 +#define EGL_NATIVE_BUFFER_TIZEN 0x32A0 +#endif /* EGL_TIZEN_image_native_buffer */ + +#ifndef EGL_TIZEN_image_native_surface +#define EGL_TIZEN_image_native_surface 1 +#define EGL_NATIVE_SURFACE_TIZEN 0x32A1 +#endif /* EGL_TIZEN_image_native_surface */ + +#ifndef EGL_WL_bind_wayland_display +#define EGL_WL_bind_wayland_display 1 +#define PFNEGLBINDWAYLANDDISPLAYWL PFNEGLBINDWAYLANDDISPLAYWLPROC +#define PFNEGLUNBINDWAYLANDDISPLAYWL PFNEGLUNBINDWAYLANDDISPLAYWLPROC +#define PFNEGLQUERYWAYLANDBUFFERWL PFNEGLQUERYWAYLANDBUFFERWLPROC +struct wl_display; +struct wl_resource; +#define EGL_WAYLAND_BUFFER_WL 0x31D5 +#define EGL_WAYLAND_PLANE_WL 0x31D6 +#define EGL_TEXTURE_Y_U_V_WL 0x31D7 +#define EGL_TEXTURE_Y_UV_WL 0x31D8 +#define EGL_TEXTURE_Y_XUXV_WL 0x31D9 +#define EGL_TEXTURE_EXTERNAL_WL 0x31DA +#define EGL_WAYLAND_Y_INVERTED_WL 0x31DB +typedef EGLBoolean (EGLAPIENTRYP PFNEGLBINDWAYLANDDISPLAYWLPROC) (EGLDisplay dpy, struct wl_display *display); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLUNBINDWAYLANDDISPLAYWLPROC) (EGLDisplay dpy, struct wl_display *display); +typedef EGLBoolean (EGLAPIENTRYP PFNEGLQUERYWAYLANDBUFFERWLPROC) (EGLDisplay dpy, struct wl_resource *buffer, EGLint attribute, EGLint *value); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI EGLBoolean EGLAPIENTRY eglBindWaylandDisplayWL (EGLDisplay dpy, struct wl_display *display); +EGLAPI EGLBoolean EGLAPIENTRY eglUnbindWaylandDisplayWL (EGLDisplay dpy, struct wl_display *display); +EGLAPI EGLBoolean EGLAPIENTRY eglQueryWaylandBufferWL (EGLDisplay dpy, struct wl_resource *buffer, EGLint attribute, EGLint *value); +#endif +#endif /* EGL_WL_bind_wayland_display */ + +#ifndef EGL_WL_create_wayland_buffer_from_image +#define EGL_WL_create_wayland_buffer_from_image 1 +#define PFNEGLCREATEWAYLANDBUFFERFROMIMAGEWL PFNEGLCREATEWAYLANDBUFFERFROMIMAGEWLPROC +struct wl_buffer; +typedef struct wl_buffer *(EGLAPIENTRYP PFNEGLCREATEWAYLANDBUFFERFROMIMAGEWLPROC) (EGLDisplay dpy, EGLImageKHR image); +#ifdef EGL_EGLEXT_PROTOTYPES +EGLAPI struct wl_buffer *EGLAPIENTRY eglCreateWaylandBufferFromImageWL (EGLDisplay dpy, EGLImageKHR image); +#endif +#endif /* EGL_WL_create_wayland_buffer_from_image */ + +#ifdef __cplusplus +} +#endif + +#endif /* __eglext_h_ */ + +#endif /* _MSC_VER */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_endian.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_endian.h new file mode 100644 index 00000000..71bc0672 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_endian.h @@ -0,0 +1,348 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_endian.h + * + * Functions for reading and writing endian-specific values + */ + +#ifndef SDL_endian_h_ +#define SDL_endian_h_ + +#include "SDL_stdinc.h" + +#if defined(_MSC_VER) && (_MSC_VER >= 1400) +/* As of Clang 11, '_m_prefetchw' is conflicting with the winnt.h's version, + so we define the needed '_m_prefetch' here as a pseudo-header, until the issue is fixed. */ +#ifdef __clang__ +#ifndef __PRFCHWINTRIN_H +#define __PRFCHWINTRIN_H +static __inline__ void __attribute__((__always_inline__, __nodebug__)) +_m_prefetch(void *__P) +{ + __builtin_prefetch(__P, 0, 3 /* _MM_HINT_T0 */); +} +#endif /* __PRFCHWINTRIN_H */ +#endif /* __clang__ */ + +#include +#endif + +/** + * \name The two types of endianness + */ +/* @{ */ +#define SDL_LIL_ENDIAN 1234 +#define SDL_BIG_ENDIAN 4321 +/* @} */ + +#ifndef SDL_BYTEORDER /* Not defined in SDL_config.h? */ +#ifdef __linux__ +#include +#define SDL_BYTEORDER __BYTE_ORDER +#elif defined(__OpenBSD__) || defined(__DragonFly__) +#include +#define SDL_BYTEORDER BYTE_ORDER +#elif defined(__FreeBSD__) || defined(__NetBSD__) +#include +#define SDL_BYTEORDER BYTE_ORDER +/* predefs from newer gcc and clang versions: */ +#elif defined(__ORDER_LITTLE_ENDIAN__) && defined(__ORDER_BIG_ENDIAN__) && defined(__BYTE_ORDER__) +#if (__BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__) +#define SDL_BYTEORDER SDL_LIL_ENDIAN +#elif (__BYTE_ORDER__ == __ORDER_BIG_ENDIAN__) +#define SDL_BYTEORDER SDL_BIG_ENDIAN +#else +#error Unsupported endianness +#endif /**/ +#else +#if defined(__hppa__) || \ + defined(__m68k__) || defined(mc68000) || defined(_M_M68K) || \ + (defined(__MIPS__) && defined(__MIPSEB__)) || \ + defined(__ppc__) || defined(__POWERPC__) || defined(__powerpc__) || defined(__PPC__) || \ + defined(__sparc__) +#define SDL_BYTEORDER SDL_BIG_ENDIAN +#else +#define SDL_BYTEORDER SDL_LIL_ENDIAN +#endif +#endif /* __linux__ */ +#endif /* !SDL_BYTEORDER */ + +#ifndef SDL_FLOATWORDORDER /* Not defined in SDL_config.h? */ +/* predefs from newer gcc versions: */ +#if defined(__ORDER_LITTLE_ENDIAN__) && defined(__ORDER_BIG_ENDIAN__) && defined(__FLOAT_WORD_ORDER__) +#if (__FLOAT_WORD_ORDER__ == __ORDER_LITTLE_ENDIAN__) +#define SDL_FLOATWORDORDER SDL_LIL_ENDIAN +#elif (__FLOAT_WORD_ORDER__ == __ORDER_BIG_ENDIAN__) +#define SDL_FLOATWORDORDER SDL_BIG_ENDIAN +#else +#error Unsupported endianness +#endif /**/ +#elif defined(__MAVERICK__) +/* For Maverick, float words are always little-endian. */ +#define SDL_FLOATWORDORDER SDL_LIL_ENDIAN +#elif (defined(__arm__) || defined(__thumb__)) && !defined(__VFP_FP__) && !defined(__ARM_EABI__) +/* For FPA, float words are always big-endian. */ +#define SDL_FLOATWORDORDER SDL_BIG_ENDIAN +#else +/* By default, assume that floats words follow the memory system mode. */ +#define SDL_FLOATWORDORDER SDL_BYTEORDER +#endif /* __FLOAT_WORD_ORDER__ */ +#endif /* !SDL_FLOATWORDORDER */ + + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \file SDL_endian.h + */ + +/* various modern compilers may have builtin swap */ +#if defined(__GNUC__) || defined(__clang__) +# define HAS_BUILTIN_BSWAP16 (_SDL_HAS_BUILTIN(__builtin_bswap16)) || \ + (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 8)) +# define HAS_BUILTIN_BSWAP32 (_SDL_HAS_BUILTIN(__builtin_bswap32)) || \ + (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)) +# define HAS_BUILTIN_BSWAP64 (_SDL_HAS_BUILTIN(__builtin_bswap64)) || \ + (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)) + + /* this one is broken */ +# define HAS_BROKEN_BSWAP (__GNUC__ == 2 && __GNUC_MINOR__ <= 95) +#else +# define HAS_BUILTIN_BSWAP16 0 +# define HAS_BUILTIN_BSWAP32 0 +# define HAS_BUILTIN_BSWAP64 0 +# define HAS_BROKEN_BSWAP 0 +#endif + +#if HAS_BUILTIN_BSWAP16 +#define SDL_Swap16(x) __builtin_bswap16(x) +#elif (defined(_MSC_VER) && (_MSC_VER >= 1400)) && !defined(__ICL) +#pragma intrinsic(_byteswap_ushort) +#define SDL_Swap16(x) _byteswap_ushort(x) +#elif defined(__i386__) && !HAS_BROKEN_BSWAP +SDL_FORCE_INLINE Uint16 +SDL_Swap16(Uint16 x) +{ + __asm__("xchgb %b0,%h0": "=q"(x):"0"(x)); + return x; +} +#elif defined(__x86_64__) +SDL_FORCE_INLINE Uint16 +SDL_Swap16(Uint16 x) +{ + __asm__("xchgb %b0,%h0": "=Q"(x):"0"(x)); + return x; +} +#elif (defined(__powerpc__) || defined(__ppc__)) +SDL_FORCE_INLINE Uint16 +SDL_Swap16(Uint16 x) +{ + int result; + + __asm__("rlwimi %0,%2,8,16,23": "=&r"(result):"0"(x >> 8), "r"(x)); + return (Uint16)result; +} +#elif (defined(__m68k__) && !defined(__mcoldfire__)) +SDL_FORCE_INLINE Uint16 +SDL_Swap16(Uint16 x) +{ + __asm__("rorw #8,%0": "=d"(x): "0"(x):"cc"); + return x; +} +#elif defined(__WATCOMC__) && defined(__386__) +extern __inline Uint16 SDL_Swap16(Uint16); +#pragma aux SDL_Swap16 = \ + "xchg al, ah" \ + parm [ax] \ + modify [ax]; +#else +SDL_FORCE_INLINE Uint16 +SDL_Swap16(Uint16 x) +{ + return SDL_static_cast(Uint16, ((x << 8) | (x >> 8))); +} +#endif + +#if HAS_BUILTIN_BSWAP32 +#define SDL_Swap32(x) __builtin_bswap32(x) +#elif (defined(_MSC_VER) && (_MSC_VER >= 1400)) && !defined(__ICL) +#pragma intrinsic(_byteswap_ulong) +#define SDL_Swap32(x) _byteswap_ulong(x) +#elif defined(__i386__) && !HAS_BROKEN_BSWAP +SDL_FORCE_INLINE Uint32 +SDL_Swap32(Uint32 x) +{ + __asm__("bswap %0": "=r"(x):"0"(x)); + return x; +} +#elif defined(__x86_64__) +SDL_FORCE_INLINE Uint32 +SDL_Swap32(Uint32 x) +{ + __asm__("bswapl %0": "=r"(x):"0"(x)); + return x; +} +#elif (defined(__powerpc__) || defined(__ppc__)) +SDL_FORCE_INLINE Uint32 +SDL_Swap32(Uint32 x) +{ + Uint32 result; + + __asm__("rlwimi %0,%2,24,16,23": "=&r"(result): "0" (x>>24), "r"(x)); + __asm__("rlwimi %0,%2,8,8,15" : "=&r"(result): "0" (result), "r"(x)); + __asm__("rlwimi %0,%2,24,0,7" : "=&r"(result): "0" (result), "r"(x)); + return result; +} +#elif (defined(__m68k__) && !defined(__mcoldfire__)) +SDL_FORCE_INLINE Uint32 +SDL_Swap32(Uint32 x) +{ + __asm__("rorw #8,%0\n\tswap %0\n\trorw #8,%0": "=d"(x): "0"(x):"cc"); + return x; +} +#elif defined(__WATCOMC__) && defined(__386__) +extern __inline Uint32 SDL_Swap32(Uint32); +#pragma aux SDL_Swap32 = \ + "bswap eax" \ + parm [eax] \ + modify [eax]; +#else +SDL_FORCE_INLINE Uint32 +SDL_Swap32(Uint32 x) +{ + return SDL_static_cast(Uint32, ((x << 24) | ((x << 8) & 0x00FF0000) | + ((x >> 8) & 0x0000FF00) | (x >> 24))); +} +#endif + +#if HAS_BUILTIN_BSWAP64 +#define SDL_Swap64(x) __builtin_bswap64(x) +#elif (defined(_MSC_VER) && (_MSC_VER >= 1400)) && !defined(__ICL) +#pragma intrinsic(_byteswap_uint64) +#define SDL_Swap64(x) _byteswap_uint64(x) +#elif defined(__i386__) && !HAS_BROKEN_BSWAP +SDL_FORCE_INLINE Uint64 +SDL_Swap64(Uint64 x) +{ + union { + struct { + Uint32 a, b; + } s; + Uint64 u; + } v; + v.u = x; + __asm__("bswapl %0 ; bswapl %1 ; xchgl %0,%1" + : "=r"(v.s.a), "=r"(v.s.b) + : "0" (v.s.a), "1"(v.s.b)); + return v.u; +} +#elif defined(__x86_64__) +SDL_FORCE_INLINE Uint64 +SDL_Swap64(Uint64 x) +{ + __asm__("bswapq %0": "=r"(x):"0"(x)); + return x; +} +#elif defined(__WATCOMC__) && defined(__386__) +extern __inline Uint64 SDL_Swap64(Uint64); +#pragma aux SDL_Swap64 = \ + "bswap eax" \ + "bswap edx" \ + "xchg eax,edx" \ + parm [eax edx] \ + modify [eax edx]; +#else +SDL_FORCE_INLINE Uint64 +SDL_Swap64(Uint64 x) +{ + Uint32 hi, lo; + + /* Separate into high and low 32-bit values and swap them */ + lo = SDL_static_cast(Uint32, x & 0xFFFFFFFF); + x >>= 32; + hi = SDL_static_cast(Uint32, x & 0xFFFFFFFF); + x = SDL_Swap32(lo); + x <<= 32; + x |= SDL_Swap32(hi); + return (x); +} +#endif + + +SDL_FORCE_INLINE float +SDL_SwapFloat(float x) +{ + union { + float f; + Uint32 ui32; + } swapper; + swapper.f = x; + swapper.ui32 = SDL_Swap32(swapper.ui32); + return swapper.f; +} + +/* remove extra macros */ +#undef HAS_BROKEN_BSWAP +#undef HAS_BUILTIN_BSWAP16 +#undef HAS_BUILTIN_BSWAP32 +#undef HAS_BUILTIN_BSWAP64 + +/** + * \name Swap to native + * Byteswap item from the specified endianness to the native endianness. + */ +/* @{ */ +#if SDL_BYTEORDER == SDL_LIL_ENDIAN +#define SDL_SwapLE16(X) (X) +#define SDL_SwapLE32(X) (X) +#define SDL_SwapLE64(X) (X) +#define SDL_SwapFloatLE(X) (X) +#define SDL_SwapBE16(X) SDL_Swap16(X) +#define SDL_SwapBE32(X) SDL_Swap32(X) +#define SDL_SwapBE64(X) SDL_Swap64(X) +#define SDL_SwapFloatBE(X) SDL_SwapFloat(X) +#else +#define SDL_SwapLE16(X) SDL_Swap16(X) +#define SDL_SwapLE32(X) SDL_Swap32(X) +#define SDL_SwapLE64(X) SDL_Swap64(X) +#define SDL_SwapFloatLE(X) SDL_SwapFloat(X) +#define SDL_SwapBE16(X) (X) +#define SDL_SwapBE32(X) (X) +#define SDL_SwapBE64(X) (X) +#define SDL_SwapFloatBE(X) (X) +#endif +/* @} *//* Swap to native */ + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_endian_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_error.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_error.h new file mode 100644 index 00000000..31c22616 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_error.h @@ -0,0 +1,163 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_error.h + * + * Simple error message routines for SDL. + */ + +#ifndef SDL_error_h_ +#define SDL_error_h_ + +#include "SDL_stdinc.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* Public functions */ + + +/** + * Set the SDL error message for the current thread. + * + * Calling this function will replace any previous error message that was set. + * + * This function always returns -1, since SDL frequently uses -1 to signify an + * failing result, leading to this idiom: + * + * ```c + * if (error_code) { + * return SDL_SetError("This operation has failed: %d", error_code); + * } + * ``` + * + * \param fmt a printf()-style message format string + * \param ... additional parameters matching % tokens in the `fmt` string, if + * any + * \returns always -1. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ClearError + * \sa SDL_GetError + */ +extern DECLSPEC int SDLCALL SDL_SetError(SDL_PRINTF_FORMAT_STRING const char *fmt, ...) SDL_PRINTF_VARARG_FUNC(1); + +/** + * Retrieve a message about the last error that occurred on the current + * thread. + * + * It is possible for multiple errors to occur before calling SDL_GetError(). + * Only the last error is returned. + * + * The message is only applicable when an SDL function has signaled an error. + * You must check the return values of SDL function calls to determine when to + * appropriately call SDL_GetError(). You should *not* use the results of + * SDL_GetError() to decide if an error has occurred! Sometimes SDL will set + * an error string even when reporting success. + * + * SDL will *not* clear the error string for successful API calls. You *must* + * check return values for failure cases before you can assume the error + * string applies. + * + * Error strings are set per-thread, so an error set in a different thread + * will not interfere with the current thread's operation. + * + * The returned string is internally allocated and must not be freed by the + * application. + * + * \returns a message with information about the specific error that occurred, + * or an empty string if there hasn't been an error message set since + * the last call to SDL_ClearError(). The message is only applicable + * when an SDL function has signaled an error. You must check the + * return values of SDL function calls to determine when to + * appropriately call SDL_GetError(). + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ClearError + * \sa SDL_SetError + */ +extern DECLSPEC const char *SDLCALL SDL_GetError(void); + +/** + * Get the last error message that was set for the current thread. + * + * This allows the caller to copy the error string into a provided buffer, but + * otherwise operates exactly the same as SDL_GetError(). + * + * \param errstr A buffer to fill with the last error message that was set for + * the current thread + * \param maxlen The size of the buffer pointed to by the errstr parameter + * \returns the pointer passed in as the `errstr` parameter. + * + * \since This function is available since SDL 2.0.14. + * + * \sa SDL_GetError + */ +extern DECLSPEC char * SDLCALL SDL_GetErrorMsg(char *errstr, int maxlen); + +/** + * Clear any previous error message for this thread. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetError + * \sa SDL_SetError + */ +extern DECLSPEC void SDLCALL SDL_ClearError(void); + +/** + * \name Internal error functions + * + * \internal + * Private error reporting function - used internally. + */ +/* @{ */ +#define SDL_OutOfMemory() SDL_Error(SDL_ENOMEM) +#define SDL_Unsupported() SDL_Error(SDL_UNSUPPORTED) +#define SDL_InvalidParamError(param) SDL_SetError("Parameter '%s' is invalid", (param)) +typedef enum +{ + SDL_ENOMEM, + SDL_EFREAD, + SDL_EFWRITE, + SDL_EFSEEK, + SDL_UNSUPPORTED, + SDL_LASTERROR +} SDL_errorcode; +/* SDL_Error() unconditionally returns -1. */ +extern DECLSPEC int SDLCALL SDL_Error(SDL_errorcode code); +/* @} *//* Internal error functions */ + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_error_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_events.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_events.h new file mode 100644 index 00000000..9d097031 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_events.h @@ -0,0 +1,1166 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_events.h + * + * Include file for SDL event handling. + */ + +#ifndef SDL_events_h_ +#define SDL_events_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_video.h" +#include "SDL_keyboard.h" +#include "SDL_mouse.h" +#include "SDL_joystick.h" +#include "SDL_gamecontroller.h" +#include "SDL_quit.h" +#include "SDL_gesture.h" +#include "SDL_touch.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* General keyboard/mouse state definitions */ +#define SDL_RELEASED 0 +#define SDL_PRESSED 1 + +/** + * The types of events that can be delivered. + */ +typedef enum +{ + SDL_FIRSTEVENT = 0, /**< Unused (do not remove) */ + + /* Application events */ + SDL_QUIT = 0x100, /**< User-requested quit */ + + /* These application events have special meaning on iOS, see README-ios.md for details */ + SDL_APP_TERMINATING, /**< The application is being terminated by the OS + Called on iOS in applicationWillTerminate() + Called on Android in onDestroy() + */ + SDL_APP_LOWMEMORY, /**< The application is low on memory, free memory if possible. + Called on iOS in applicationDidReceiveMemoryWarning() + Called on Android in onLowMemory() + */ + SDL_APP_WILLENTERBACKGROUND, /**< The application is about to enter the background + Called on iOS in applicationWillResignActive() + Called on Android in onPause() + */ + SDL_APP_DIDENTERBACKGROUND, /**< The application did enter the background and may not get CPU for some time + Called on iOS in applicationDidEnterBackground() + Called on Android in onPause() + */ + SDL_APP_WILLENTERFOREGROUND, /**< The application is about to enter the foreground + Called on iOS in applicationWillEnterForeground() + Called on Android in onResume() + */ + SDL_APP_DIDENTERFOREGROUND, /**< The application is now interactive + Called on iOS in applicationDidBecomeActive() + Called on Android in onResume() + */ + + SDL_LOCALECHANGED, /**< The user's locale preferences have changed. */ + + /* Display events */ + SDL_DISPLAYEVENT = 0x150, /**< Display state change */ + + /* Window events */ + SDL_WINDOWEVENT = 0x200, /**< Window state change */ + SDL_SYSWMEVENT, /**< System specific event */ + + /* Keyboard events */ + SDL_KEYDOWN = 0x300, /**< Key pressed */ + SDL_KEYUP, /**< Key released */ + SDL_TEXTEDITING, /**< Keyboard text editing (composition) */ + SDL_TEXTINPUT, /**< Keyboard text input */ + SDL_KEYMAPCHANGED, /**< Keymap changed due to a system event such as an + input language or keyboard layout change. + */ + SDL_TEXTEDITING_EXT, /**< Extended keyboard text editing (composition) */ + + /* Mouse events */ + SDL_MOUSEMOTION = 0x400, /**< Mouse moved */ + SDL_MOUSEBUTTONDOWN, /**< Mouse button pressed */ + SDL_MOUSEBUTTONUP, /**< Mouse button released */ + SDL_MOUSEWHEEL, /**< Mouse wheel motion */ + + /* Joystick events */ + SDL_JOYAXISMOTION = 0x600, /**< Joystick axis motion */ + SDL_JOYBALLMOTION, /**< Joystick trackball motion */ + SDL_JOYHATMOTION, /**< Joystick hat position change */ + SDL_JOYBUTTONDOWN, /**< Joystick button pressed */ + SDL_JOYBUTTONUP, /**< Joystick button released */ + SDL_JOYDEVICEADDED, /**< A new joystick has been inserted into the system */ + SDL_JOYDEVICEREMOVED, /**< An opened joystick has been removed */ + SDL_JOYBATTERYUPDATED, /**< Joystick battery level change */ + + /* Game controller events */ + SDL_CONTROLLERAXISMOTION = 0x650, /**< Game controller axis motion */ + SDL_CONTROLLERBUTTONDOWN, /**< Game controller button pressed */ + SDL_CONTROLLERBUTTONUP, /**< Game controller button released */ + SDL_CONTROLLERDEVICEADDED, /**< A new Game controller has been inserted into the system */ + SDL_CONTROLLERDEVICEREMOVED, /**< An opened Game controller has been removed */ + SDL_CONTROLLERDEVICEREMAPPED, /**< The controller mapping was updated */ + SDL_CONTROLLERTOUCHPADDOWN, /**< Game controller touchpad was touched */ + SDL_CONTROLLERTOUCHPADMOTION, /**< Game controller touchpad finger was moved */ + SDL_CONTROLLERTOUCHPADUP, /**< Game controller touchpad finger was lifted */ + SDL_CONTROLLERSENSORUPDATE, /**< Game controller sensor was updated */ + + /* Touch events */ + SDL_FINGERDOWN = 0x700, + SDL_FINGERUP, + SDL_FINGERMOTION, + + /* Gesture events */ + SDL_DOLLARGESTURE = 0x800, + SDL_DOLLARRECORD, + SDL_MULTIGESTURE, + + /* Clipboard events */ + SDL_CLIPBOARDUPDATE = 0x900, /**< The clipboard or primary selection changed */ + + /* Drag and drop events */ + SDL_DROPFILE = 0x1000, /**< The system requests a file open */ + SDL_DROPTEXT, /**< text/plain drag-and-drop event */ + SDL_DROPBEGIN, /**< A new set of drops is beginning (NULL filename) */ + SDL_DROPCOMPLETE, /**< Current set of drops is now complete (NULL filename) */ + + /* Audio hotplug events */ + SDL_AUDIODEVICEADDED = 0x1100, /**< A new audio device is available */ + SDL_AUDIODEVICEREMOVED, /**< An audio device has been removed. */ + + /* Sensor events */ + SDL_SENSORUPDATE = 0x1200, /**< A sensor was updated */ + + /* Render events */ + SDL_RENDER_TARGETS_RESET = 0x2000, /**< The render targets have been reset and their contents need to be updated */ + SDL_RENDER_DEVICE_RESET, /**< The device has been reset and all textures need to be recreated */ + + /* Internal events */ + SDL_POLLSENTINEL = 0x7F00, /**< Signals the end of an event poll cycle */ + + /** Events ::SDL_USEREVENT through ::SDL_LASTEVENT are for your use, + * and should be allocated with SDL_RegisterEvents() + */ + SDL_USEREVENT = 0x8000, + + /** + * This last event is only for bounding internal arrays + */ + SDL_LASTEVENT = 0xFFFF +} SDL_EventType; + +/** + * \brief Fields shared by every event + */ +typedef struct SDL_CommonEvent +{ + Uint32 type; + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ +} SDL_CommonEvent; + +/** + * \brief Display state change event data (event.display.*) + */ +typedef struct SDL_DisplayEvent +{ + Uint32 type; /**< ::SDL_DISPLAYEVENT */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + Uint32 display; /**< The associated display index */ + Uint8 event; /**< ::SDL_DisplayEventID */ + Uint8 padding1; + Uint8 padding2; + Uint8 padding3; + Sint32 data1; /**< event dependent data */ +} SDL_DisplayEvent; + +/** + * \brief Window state change event data (event.window.*) + */ +typedef struct SDL_WindowEvent +{ + Uint32 type; /**< ::SDL_WINDOWEVENT */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + Uint32 windowID; /**< The associated window */ + Uint8 event; /**< ::SDL_WindowEventID */ + Uint8 padding1; + Uint8 padding2; + Uint8 padding3; + Sint32 data1; /**< event dependent data */ + Sint32 data2; /**< event dependent data */ +} SDL_WindowEvent; + +/** + * \brief Keyboard button event structure (event.key.*) + */ +typedef struct SDL_KeyboardEvent +{ + Uint32 type; /**< ::SDL_KEYDOWN or ::SDL_KEYUP */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + Uint32 windowID; /**< The window with keyboard focus, if any */ + Uint8 state; /**< ::SDL_PRESSED or ::SDL_RELEASED */ + Uint8 repeat; /**< Non-zero if this is a key repeat */ + Uint8 padding2; + Uint8 padding3; + SDL_Keysym keysym; /**< The key that was pressed or released */ +} SDL_KeyboardEvent; + +#define SDL_TEXTEDITINGEVENT_TEXT_SIZE (32) +/** + * \brief Keyboard text editing event structure (event.edit.*) + */ +typedef struct SDL_TextEditingEvent +{ + Uint32 type; /**< ::SDL_TEXTEDITING */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + Uint32 windowID; /**< The window with keyboard focus, if any */ + char text[SDL_TEXTEDITINGEVENT_TEXT_SIZE]; /**< The editing text */ + Sint32 start; /**< The start cursor of selected editing text */ + Sint32 length; /**< The length of selected editing text */ +} SDL_TextEditingEvent; + +/** + * \brief Extended keyboard text editing event structure (event.editExt.*) when text would be + * truncated if stored in the text buffer SDL_TextEditingEvent + */ +typedef struct SDL_TextEditingExtEvent +{ + Uint32 type; /**< ::SDL_TEXTEDITING_EXT */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + Uint32 windowID; /**< The window with keyboard focus, if any */ + char* text; /**< The editing text, which should be freed with SDL_free(), and will not be NULL */ + Sint32 start; /**< The start cursor of selected editing text */ + Sint32 length; /**< The length of selected editing text */ +} SDL_TextEditingExtEvent; + +#define SDL_TEXTINPUTEVENT_TEXT_SIZE (32) +/** + * \brief Keyboard text input event structure (event.text.*) + */ +typedef struct SDL_TextInputEvent +{ + Uint32 type; /**< ::SDL_TEXTINPUT */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + Uint32 windowID; /**< The window with keyboard focus, if any */ + char text[SDL_TEXTINPUTEVENT_TEXT_SIZE]; /**< The input text */ +} SDL_TextInputEvent; + +/** + * \brief Mouse motion event structure (event.motion.*) + */ +typedef struct SDL_MouseMotionEvent +{ + Uint32 type; /**< ::SDL_MOUSEMOTION */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + Uint32 windowID; /**< The window with mouse focus, if any */ + Uint32 which; /**< The mouse instance id, or SDL_TOUCH_MOUSEID */ + Uint32 state; /**< The current button state */ + Sint32 x; /**< X coordinate, relative to window */ + Sint32 y; /**< Y coordinate, relative to window */ + Sint32 xrel; /**< The relative motion in the X direction */ + Sint32 yrel; /**< The relative motion in the Y direction */ +} SDL_MouseMotionEvent; + +/** + * \brief Mouse button event structure (event.button.*) + */ +typedef struct SDL_MouseButtonEvent +{ + Uint32 type; /**< ::SDL_MOUSEBUTTONDOWN or ::SDL_MOUSEBUTTONUP */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + Uint32 windowID; /**< The window with mouse focus, if any */ + Uint32 which; /**< The mouse instance id, or SDL_TOUCH_MOUSEID */ + Uint8 button; /**< The mouse button index */ + Uint8 state; /**< ::SDL_PRESSED or ::SDL_RELEASED */ + Uint8 clicks; /**< 1 for single-click, 2 for double-click, etc. */ + Uint8 padding1; + Sint32 x; /**< X coordinate, relative to window */ + Sint32 y; /**< Y coordinate, relative to window */ +} SDL_MouseButtonEvent; + +/** + * \brief Mouse wheel event structure (event.wheel.*) + */ +typedef struct SDL_MouseWheelEvent +{ + Uint32 type; /**< ::SDL_MOUSEWHEEL */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + Uint32 windowID; /**< The window with mouse focus, if any */ + Uint32 which; /**< The mouse instance id, or SDL_TOUCH_MOUSEID */ + Sint32 x; /**< The amount scrolled horizontally, positive to the right and negative to the left */ + Sint32 y; /**< The amount scrolled vertically, positive away from the user and negative toward the user */ + Uint32 direction; /**< Set to one of the SDL_MOUSEWHEEL_* defines. When FLIPPED the values in X and Y will be opposite. Multiply by -1 to change them back */ + float preciseX; /**< The amount scrolled horizontally, positive to the right and negative to the left, with float precision (added in 2.0.18) */ + float preciseY; /**< The amount scrolled vertically, positive away from the user and negative toward the user, with float precision (added in 2.0.18) */ + Sint32 mouseX; /**< X coordinate, relative to window (added in 2.26.0) */ + Sint32 mouseY; /**< Y coordinate, relative to window (added in 2.26.0) */ +} SDL_MouseWheelEvent; + +/** + * \brief Joystick axis motion event structure (event.jaxis.*) + */ +typedef struct SDL_JoyAxisEvent +{ + Uint32 type; /**< ::SDL_JOYAXISMOTION */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + SDL_JoystickID which; /**< The joystick instance id */ + Uint8 axis; /**< The joystick axis index */ + Uint8 padding1; + Uint8 padding2; + Uint8 padding3; + Sint16 value; /**< The axis value (range: -32768 to 32767) */ + Uint16 padding4; +} SDL_JoyAxisEvent; + +/** + * \brief Joystick trackball motion event structure (event.jball.*) + */ +typedef struct SDL_JoyBallEvent +{ + Uint32 type; /**< ::SDL_JOYBALLMOTION */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + SDL_JoystickID which; /**< The joystick instance id */ + Uint8 ball; /**< The joystick trackball index */ + Uint8 padding1; + Uint8 padding2; + Uint8 padding3; + Sint16 xrel; /**< The relative motion in the X direction */ + Sint16 yrel; /**< The relative motion in the Y direction */ +} SDL_JoyBallEvent; + +/** + * \brief Joystick hat position change event structure (event.jhat.*) + */ +typedef struct SDL_JoyHatEvent +{ + Uint32 type; /**< ::SDL_JOYHATMOTION */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + SDL_JoystickID which; /**< The joystick instance id */ + Uint8 hat; /**< The joystick hat index */ + Uint8 value; /**< The hat position value. + * \sa ::SDL_HAT_LEFTUP ::SDL_HAT_UP ::SDL_HAT_RIGHTUP + * \sa ::SDL_HAT_LEFT ::SDL_HAT_CENTERED ::SDL_HAT_RIGHT + * \sa ::SDL_HAT_LEFTDOWN ::SDL_HAT_DOWN ::SDL_HAT_RIGHTDOWN + * + * Note that zero means the POV is centered. + */ + Uint8 padding1; + Uint8 padding2; +} SDL_JoyHatEvent; + +/** + * \brief Joystick button event structure (event.jbutton.*) + */ +typedef struct SDL_JoyButtonEvent +{ + Uint32 type; /**< ::SDL_JOYBUTTONDOWN or ::SDL_JOYBUTTONUP */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + SDL_JoystickID which; /**< The joystick instance id */ + Uint8 button; /**< The joystick button index */ + Uint8 state; /**< ::SDL_PRESSED or ::SDL_RELEASED */ + Uint8 padding1; + Uint8 padding2; +} SDL_JoyButtonEvent; + +/** + * \brief Joystick device event structure (event.jdevice.*) + */ +typedef struct SDL_JoyDeviceEvent +{ + Uint32 type; /**< ::SDL_JOYDEVICEADDED or ::SDL_JOYDEVICEREMOVED */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + Sint32 which; /**< The joystick device index for the ADDED event, instance id for the REMOVED event */ +} SDL_JoyDeviceEvent; + +/** + * \brief Joysick battery level change event structure (event.jbattery.*) + */ +typedef struct SDL_JoyBatteryEvent +{ + Uint32 type; /**< ::SDL_JOYBATTERYUPDATED */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + SDL_JoystickID which; /**< The joystick instance id */ + SDL_JoystickPowerLevel level; /**< The joystick battery level */ +} SDL_JoyBatteryEvent; + +/** + * \brief Game controller axis motion event structure (event.caxis.*) + */ +typedef struct SDL_ControllerAxisEvent +{ + Uint32 type; /**< ::SDL_CONTROLLERAXISMOTION */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + SDL_JoystickID which; /**< The joystick instance id */ + Uint8 axis; /**< The controller axis (SDL_GameControllerAxis) */ + Uint8 padding1; + Uint8 padding2; + Uint8 padding3; + Sint16 value; /**< The axis value (range: -32768 to 32767) */ + Uint16 padding4; +} SDL_ControllerAxisEvent; + + +/** + * \brief Game controller button event structure (event.cbutton.*) + */ +typedef struct SDL_ControllerButtonEvent +{ + Uint32 type; /**< ::SDL_CONTROLLERBUTTONDOWN or ::SDL_CONTROLLERBUTTONUP */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + SDL_JoystickID which; /**< The joystick instance id */ + Uint8 button; /**< The controller button (SDL_GameControllerButton) */ + Uint8 state; /**< ::SDL_PRESSED or ::SDL_RELEASED */ + Uint8 padding1; + Uint8 padding2; +} SDL_ControllerButtonEvent; + + +/** + * \brief Controller device event structure (event.cdevice.*) + */ +typedef struct SDL_ControllerDeviceEvent +{ + Uint32 type; /**< ::SDL_CONTROLLERDEVICEADDED, ::SDL_CONTROLLERDEVICEREMOVED, or ::SDL_CONTROLLERDEVICEREMAPPED */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + Sint32 which; /**< The joystick device index for the ADDED event, instance id for the REMOVED or REMAPPED event */ +} SDL_ControllerDeviceEvent; + +/** + * \brief Game controller touchpad event structure (event.ctouchpad.*) + */ +typedef struct SDL_ControllerTouchpadEvent +{ + Uint32 type; /**< ::SDL_CONTROLLERTOUCHPADDOWN or ::SDL_CONTROLLERTOUCHPADMOTION or ::SDL_CONTROLLERTOUCHPADUP */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + SDL_JoystickID which; /**< The joystick instance id */ + Sint32 touchpad; /**< The index of the touchpad */ + Sint32 finger; /**< The index of the finger on the touchpad */ + float x; /**< Normalized in the range 0...1 with 0 being on the left */ + float y; /**< Normalized in the range 0...1 with 0 being at the top */ + float pressure; /**< Normalized in the range 0...1 */ +} SDL_ControllerTouchpadEvent; + +/** + * \brief Game controller sensor event structure (event.csensor.*) + */ +typedef struct SDL_ControllerSensorEvent +{ + Uint32 type; /**< ::SDL_CONTROLLERSENSORUPDATE */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + SDL_JoystickID which; /**< The joystick instance id */ + Sint32 sensor; /**< The type of the sensor, one of the values of ::SDL_SensorType */ + float data[3]; /**< Up to 3 values from the sensor, as defined in SDL_sensor.h */ + Uint64 timestamp_us; /**< The timestamp of the sensor reading in microseconds, if the hardware provides this information. */ +} SDL_ControllerSensorEvent; + +/** + * \brief Audio device event structure (event.adevice.*) + */ +typedef struct SDL_AudioDeviceEvent +{ + Uint32 type; /**< ::SDL_AUDIODEVICEADDED, or ::SDL_AUDIODEVICEREMOVED */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + Uint32 which; /**< The audio device index for the ADDED event (valid until next SDL_GetNumAudioDevices() call), SDL_AudioDeviceID for the REMOVED event */ + Uint8 iscapture; /**< zero if an output device, non-zero if a capture device. */ + Uint8 padding1; + Uint8 padding2; + Uint8 padding3; +} SDL_AudioDeviceEvent; + + +/** + * \brief Touch finger event structure (event.tfinger.*) + */ +typedef struct SDL_TouchFingerEvent +{ + Uint32 type; /**< ::SDL_FINGERMOTION or ::SDL_FINGERDOWN or ::SDL_FINGERUP */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + SDL_TouchID touchId; /**< The touch device id */ + SDL_FingerID fingerId; + float x; /**< Normalized in the range 0...1 */ + float y; /**< Normalized in the range 0...1 */ + float dx; /**< Normalized in the range -1...1 */ + float dy; /**< Normalized in the range -1...1 */ + float pressure; /**< Normalized in the range 0...1 */ + Uint32 windowID; /**< The window underneath the finger, if any */ +} SDL_TouchFingerEvent; + + +/** + * \brief Multiple Finger Gesture Event (event.mgesture.*) + */ +typedef struct SDL_MultiGestureEvent +{ + Uint32 type; /**< ::SDL_MULTIGESTURE */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + SDL_TouchID touchId; /**< The touch device id */ + float dTheta; + float dDist; + float x; + float y; + Uint16 numFingers; + Uint16 padding; +} SDL_MultiGestureEvent; + + +/** + * \brief Dollar Gesture Event (event.dgesture.*) + */ +typedef struct SDL_DollarGestureEvent +{ + Uint32 type; /**< ::SDL_DOLLARGESTURE or ::SDL_DOLLARRECORD */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + SDL_TouchID touchId; /**< The touch device id */ + SDL_GestureID gestureId; + Uint32 numFingers; + float error; + float x; /**< Normalized center of gesture */ + float y; /**< Normalized center of gesture */ +} SDL_DollarGestureEvent; + + +/** + * \brief An event used to request a file open by the system (event.drop.*) + * This event is enabled by default, you can disable it with SDL_EventState(). + * \note If this event is enabled, you must free the filename in the event. + */ +typedef struct SDL_DropEvent +{ + Uint32 type; /**< ::SDL_DROPBEGIN or ::SDL_DROPFILE or ::SDL_DROPTEXT or ::SDL_DROPCOMPLETE */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + char *file; /**< The file name, which should be freed with SDL_free(), is NULL on begin/complete */ + Uint32 windowID; /**< The window that was dropped on, if any */ +} SDL_DropEvent; + + +/** + * \brief Sensor event structure (event.sensor.*) + */ +typedef struct SDL_SensorEvent +{ + Uint32 type; /**< ::SDL_SENSORUPDATE */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + Sint32 which; /**< The instance ID of the sensor */ + float data[6]; /**< Up to 6 values from the sensor - additional values can be queried using SDL_SensorGetData() */ + Uint64 timestamp_us; /**< The timestamp of the sensor reading in microseconds, if the hardware provides this information. */ +} SDL_SensorEvent; + +/** + * \brief The "quit requested" event + */ +typedef struct SDL_QuitEvent +{ + Uint32 type; /**< ::SDL_QUIT */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ +} SDL_QuitEvent; + +/** + * \brief OS Specific event + */ +typedef struct SDL_OSEvent +{ + Uint32 type; /**< ::SDL_QUIT */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ +} SDL_OSEvent; + +/** + * \brief A user-defined event type (event.user.*) + */ +typedef struct SDL_UserEvent +{ + Uint32 type; /**< ::SDL_USEREVENT through ::SDL_LASTEVENT-1 */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + Uint32 windowID; /**< The associated window if any */ + Sint32 code; /**< User defined event code */ + void *data1; /**< User defined data pointer */ + void *data2; /**< User defined data pointer */ +} SDL_UserEvent; + + +struct SDL_SysWMmsg; +typedef struct SDL_SysWMmsg SDL_SysWMmsg; + +/** + * \brief A video driver dependent system event (event.syswm.*) + * This event is disabled by default, you can enable it with SDL_EventState() + * + * \note If you want to use this event, you should include SDL_syswm.h. + */ +typedef struct SDL_SysWMEvent +{ + Uint32 type; /**< ::SDL_SYSWMEVENT */ + Uint32 timestamp; /**< In milliseconds, populated using SDL_GetTicks() */ + SDL_SysWMmsg *msg; /**< driver dependent data, defined in SDL_syswm.h */ +} SDL_SysWMEvent; + +/** + * \brief General event structure + */ +typedef union SDL_Event +{ + Uint32 type; /**< Event type, shared with all events */ + SDL_CommonEvent common; /**< Common event data */ + SDL_DisplayEvent display; /**< Display event data */ + SDL_WindowEvent window; /**< Window event data */ + SDL_KeyboardEvent key; /**< Keyboard event data */ + SDL_TextEditingEvent edit; /**< Text editing event data */ + SDL_TextEditingExtEvent editExt; /**< Extended text editing event data */ + SDL_TextInputEvent text; /**< Text input event data */ + SDL_MouseMotionEvent motion; /**< Mouse motion event data */ + SDL_MouseButtonEvent button; /**< Mouse button event data */ + SDL_MouseWheelEvent wheel; /**< Mouse wheel event data */ + SDL_JoyAxisEvent jaxis; /**< Joystick axis event data */ + SDL_JoyBallEvent jball; /**< Joystick ball event data */ + SDL_JoyHatEvent jhat; /**< Joystick hat event data */ + SDL_JoyButtonEvent jbutton; /**< Joystick button event data */ + SDL_JoyDeviceEvent jdevice; /**< Joystick device change event data */ + SDL_JoyBatteryEvent jbattery; /**< Joystick battery event data */ + SDL_ControllerAxisEvent caxis; /**< Game Controller axis event data */ + SDL_ControllerButtonEvent cbutton; /**< Game Controller button event data */ + SDL_ControllerDeviceEvent cdevice; /**< Game Controller device event data */ + SDL_ControllerTouchpadEvent ctouchpad; /**< Game Controller touchpad event data */ + SDL_ControllerSensorEvent csensor; /**< Game Controller sensor event data */ + SDL_AudioDeviceEvent adevice; /**< Audio device event data */ + SDL_SensorEvent sensor; /**< Sensor event data */ + SDL_QuitEvent quit; /**< Quit request event data */ + SDL_UserEvent user; /**< Custom event data */ + SDL_SysWMEvent syswm; /**< System dependent window event data */ + SDL_TouchFingerEvent tfinger; /**< Touch finger event data */ + SDL_MultiGestureEvent mgesture; /**< Gesture event data */ + SDL_DollarGestureEvent dgesture; /**< Gesture event data */ + SDL_DropEvent drop; /**< Drag and drop event data */ + + /* This is necessary for ABI compatibility between Visual C++ and GCC. + Visual C++ will respect the push pack pragma and use 52 bytes (size of + SDL_TextEditingEvent, the largest structure for 32-bit and 64-bit + architectures) for this union, and GCC will use the alignment of the + largest datatype within the union, which is 8 bytes on 64-bit + architectures. + + So... we'll add padding to force the size to be 56 bytes for both. + + On architectures where pointers are 16 bytes, this needs rounding up to + the next multiple of 16, 64, and on architectures where pointers are + even larger the size of SDL_UserEvent will dominate as being 3 pointers. + */ + Uint8 padding[sizeof(void *) <= 8 ? 56 : sizeof(void *) == 16 ? 64 : 3 * sizeof(void *)]; +} SDL_Event; + +/* Make sure we haven't broken binary compatibility */ +SDL_COMPILE_TIME_ASSERT(SDL_Event, sizeof(SDL_Event) == sizeof(((SDL_Event *)NULL)->padding)); + + +/* Function prototypes */ + +/** + * Pump the event loop, gathering events from the input devices. + * + * This function updates the event queue and internal input device state. + * + * **WARNING**: This should only be run in the thread that initialized the + * video subsystem, and for extra safety, you should consider only doing those + * things on the main thread in any case. + * + * SDL_PumpEvents() gathers all the pending input information from devices and + * places it in the event queue. Without calls to SDL_PumpEvents() no events + * would ever be placed on the queue. Often the need for calls to + * SDL_PumpEvents() is hidden from the user since SDL_PollEvent() and + * SDL_WaitEvent() implicitly call SDL_PumpEvents(). However, if you are not + * polling or waiting for events (e.g. you are filtering them), then you must + * call SDL_PumpEvents() to force an event queue update. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_PollEvent + * \sa SDL_WaitEvent + */ +extern DECLSPEC void SDLCALL SDL_PumpEvents(void); + +/* @{ */ +typedef enum +{ + SDL_ADDEVENT, + SDL_PEEKEVENT, + SDL_GETEVENT +} SDL_eventaction; + +/** + * Check the event queue for messages and optionally return them. + * + * `action` may be any of the following: + * + * - `SDL_ADDEVENT`: up to `numevents` events will be added to the back of the + * event queue. + * - `SDL_PEEKEVENT`: `numevents` events at the front of the event queue, + * within the specified minimum and maximum type, will be returned to the + * caller and will _not_ be removed from the queue. + * - `SDL_GETEVENT`: up to `numevents` events at the front of the event queue, + * within the specified minimum and maximum type, will be returned to the + * caller and will be removed from the queue. + * + * You may have to call SDL_PumpEvents() before calling this function. + * Otherwise, the events may not be ready to be filtered when you call + * SDL_PeepEvents(). + * + * This function is thread-safe. + * + * \param events destination buffer for the retrieved events + * \param numevents if action is SDL_ADDEVENT, the number of events to add + * back to the event queue; if action is SDL_PEEKEVENT or + * SDL_GETEVENT, the maximum number of events to retrieve + * \param action action to take; see [[#action|Remarks]] for details + * \param minType minimum value of the event type to be considered; + * SDL_FIRSTEVENT is a safe choice + * \param maxType maximum value of the event type to be considered; + * SDL_LASTEVENT is a safe choice + * \returns the number of events actually stored or a negative error code on + * failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_PollEvent + * \sa SDL_PumpEvents + * \sa SDL_PushEvent + */ +extern DECLSPEC int SDLCALL SDL_PeepEvents(SDL_Event * events, int numevents, + SDL_eventaction action, + Uint32 minType, Uint32 maxType); +/* @} */ + +/** + * Check for the existence of a certain event type in the event queue. + * + * If you need to check for a range of event types, use SDL_HasEvents() + * instead. + * + * \param type the type of event to be queried; see SDL_EventType for details + * \returns SDL_TRUE if events matching `type` are present, or SDL_FALSE if + * events matching `type` are not present. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HasEvents + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasEvent(Uint32 type); + + +/** + * Check for the existence of certain event types in the event queue. + * + * If you need to check for a single event type, use SDL_HasEvent() instead. + * + * \param minType the low end of event type to be queried, inclusive; see + * SDL_EventType for details + * \param maxType the high end of event type to be queried, inclusive; see + * SDL_EventType for details + * \returns SDL_TRUE if events with type >= `minType` and <= `maxType` are + * present, or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HasEvents + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasEvents(Uint32 minType, Uint32 maxType); + +/** + * Clear events of a specific type from the event queue. + * + * This will unconditionally remove any events from the queue that match + * `type`. If you need to remove a range of event types, use SDL_FlushEvents() + * instead. + * + * It's also normal to just ignore events you don't care about in your event + * loop without calling this function. + * + * This function only affects currently queued events. If you want to make + * sure that all pending OS events are flushed, you can call SDL_PumpEvents() + * on the main thread immediately before the flush call. + * + * \param type the type of event to be cleared; see SDL_EventType for details + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_FlushEvents + */ +extern DECLSPEC void SDLCALL SDL_FlushEvent(Uint32 type); + +/** + * Clear events of a range of types from the event queue. + * + * This will unconditionally remove any events from the queue that are in the + * range of `minType` to `maxType`, inclusive. If you need to remove a single + * event type, use SDL_FlushEvent() instead. + * + * It's also normal to just ignore events you don't care about in your event + * loop without calling this function. + * + * This function only affects currently queued events. If you want to make + * sure that all pending OS events are flushed, you can call SDL_PumpEvents() + * on the main thread immediately before the flush call. + * + * \param minType the low end of event type to be cleared, inclusive; see + * SDL_EventType for details + * \param maxType the high end of event type to be cleared, inclusive; see + * SDL_EventType for details + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_FlushEvent + */ +extern DECLSPEC void SDLCALL SDL_FlushEvents(Uint32 minType, Uint32 maxType); + +/** + * Poll for currently pending events. + * + * If `event` is not NULL, the next event is removed from the queue and stored + * in the SDL_Event structure pointed to by `event`. The 1 returned refers to + * this event, immediately stored in the SDL Event structure -- not an event + * to follow. + * + * If `event` is NULL, it simply returns 1 if there is an event in the queue, + * but will not remove it from the queue. + * + * As this function may implicitly call SDL_PumpEvents(), you can only call + * this function in the thread that set the video mode. + * + * SDL_PollEvent() is the favored way of receiving system events since it can + * be done from the main loop and does not suspend the main loop while waiting + * on an event to be posted. + * + * The common practice is to fully process the event queue once every frame, + * usually as a first step before updating the game's state: + * + * ```c + * while (game_is_still_running) { + * SDL_Event event; + * while (SDL_PollEvent(&event)) { // poll until all events are handled! + * // decide what to do with this event. + * } + * + * // update game state, draw the current frame + * } + * ``` + * + * \param event the SDL_Event structure to be filled with the next event from + * the queue, or NULL + * \returns 1 if there is a pending event or 0 if there are none available. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetEventFilter + * \sa SDL_PeepEvents + * \sa SDL_PushEvent + * \sa SDL_SetEventFilter + * \sa SDL_WaitEvent + * \sa SDL_WaitEventTimeout + */ +extern DECLSPEC int SDLCALL SDL_PollEvent(SDL_Event * event); + +/** + * Wait indefinitely for the next available event. + * + * If `event` is not NULL, the next event is removed from the queue and stored + * in the SDL_Event structure pointed to by `event`. + * + * As this function may implicitly call SDL_PumpEvents(), you can only call + * this function in the thread that initialized the video subsystem. + * + * \param event the SDL_Event structure to be filled in with the next event + * from the queue, or NULL + * \returns 1 on success or 0 if there was an error while waiting for events; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_PollEvent + * \sa SDL_PumpEvents + * \sa SDL_WaitEventTimeout + */ +extern DECLSPEC int SDLCALL SDL_WaitEvent(SDL_Event * event); + +/** + * Wait until the specified timeout (in milliseconds) for the next available + * event. + * + * If `event` is not NULL, the next event is removed from the queue and stored + * in the SDL_Event structure pointed to by `event`. + * + * As this function may implicitly call SDL_PumpEvents(), you can only call + * this function in the thread that initialized the video subsystem. + * + * \param event the SDL_Event structure to be filled in with the next event + * from the queue, or NULL + * \param timeout the maximum number of milliseconds to wait for the next + * available event + * \returns 1 on success or 0 if there was an error while waiting for events; + * call SDL_GetError() for more information. This also returns 0 if + * the timeout elapsed without an event arriving. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_PollEvent + * \sa SDL_PumpEvents + * \sa SDL_WaitEvent + */ +extern DECLSPEC int SDLCALL SDL_WaitEventTimeout(SDL_Event * event, + int timeout); + +/** + * Add an event to the event queue. + * + * The event queue can actually be used as a two way communication channel. + * Not only can events be read from the queue, but the user can also push + * their own events onto it. `event` is a pointer to the event structure you + * wish to push onto the queue. The event is copied into the queue, and the + * caller may dispose of the memory pointed to after SDL_PushEvent() returns. + * + * Note: Pushing device input events onto the queue doesn't modify the state + * of the device within SDL. + * + * This function is thread-safe, and can be called from other threads safely. + * + * Note: Events pushed onto the queue with SDL_PushEvent() get passed through + * the event filter but events added with SDL_PeepEvents() do not. + * + * For pushing application-specific events, please use SDL_RegisterEvents() to + * get an event type that does not conflict with other code that also wants + * its own custom event types. + * + * \param event the SDL_Event to be added to the queue + * \returns 1 on success, 0 if the event was filtered, or a negative error + * code on failure; call SDL_GetError() for more information. A + * common reason for error is the event queue being full. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_PeepEvents + * \sa SDL_PollEvent + * \sa SDL_RegisterEvents + */ +extern DECLSPEC int SDLCALL SDL_PushEvent(SDL_Event * event); + +/** + * A function pointer used for callbacks that watch the event queue. + * + * \param userdata what was passed as `userdata` to SDL_SetEventFilter() + * or SDL_AddEventWatch, etc + * \param event the event that triggered the callback + * \returns 1 to permit event to be added to the queue, and 0 to disallow + * it. When used with SDL_AddEventWatch, the return value is ignored. + * + * \sa SDL_SetEventFilter + * \sa SDL_AddEventWatch + */ +typedef int (SDLCALL * SDL_EventFilter) (void *userdata, SDL_Event * event); + +/** + * Set up a filter to process all events before they change internal state and + * are posted to the internal event queue. + * + * If the filter function returns 1 when called, then the event will be added + * to the internal queue. If it returns 0, then the event will be dropped from + * the queue, but the internal state will still be updated. This allows + * selective filtering of dynamically arriving events. + * + * **WARNING**: Be very careful of what you do in the event filter function, + * as it may run in a different thread! + * + * On platforms that support it, if the quit event is generated by an + * interrupt signal (e.g. pressing Ctrl-C), it will be delivered to the + * application at the next event poll. + * + * There is one caveat when dealing with the ::SDL_QuitEvent event type. The + * event filter is only called when the window manager desires to close the + * application window. If the event filter returns 1, then the window will be + * closed, otherwise the window will remain open if possible. + * + * Note: Disabled events never make it to the event filter function; see + * SDL_EventState(). + * + * Note: If you just want to inspect events without filtering, you should use + * SDL_AddEventWatch() instead. + * + * Note: Events pushed onto the queue with SDL_PushEvent() get passed through + * the event filter, but events pushed onto the queue with SDL_PeepEvents() do + * not. + * + * \param filter An SDL_EventFilter function to call when an event happens + * \param userdata a pointer that is passed to `filter` + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AddEventWatch + * \sa SDL_EventState + * \sa SDL_GetEventFilter + * \sa SDL_PeepEvents + * \sa SDL_PushEvent + */ +extern DECLSPEC void SDLCALL SDL_SetEventFilter(SDL_EventFilter filter, + void *userdata); + +/** + * Query the current event filter. + * + * This function can be used to "chain" filters, by saving the existing filter + * before replacing it with a function that will call that saved filter. + * + * \param filter the current callback function will be stored here + * \param userdata the pointer that is passed to the current event filter will + * be stored here + * \returns SDL_TRUE on success or SDL_FALSE if there is no event filter set. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetEventFilter + */ +extern DECLSPEC SDL_bool SDLCALL SDL_GetEventFilter(SDL_EventFilter * filter, + void **userdata); + +/** + * Add a callback to be triggered when an event is added to the event queue. + * + * `filter` will be called when an event happens, and its return value is + * ignored. + * + * **WARNING**: Be very careful of what you do in the event filter function, + * as it may run in a different thread! + * + * If the quit event is generated by a signal (e.g. SIGINT), it will bypass + * the internal queue and be delivered to the watch callback immediately, and + * arrive at the next event poll. + * + * Note: the callback is called for events posted by the user through + * SDL_PushEvent(), but not for disabled events, nor for events by a filter + * callback set with SDL_SetEventFilter(), nor for events posted by the user + * through SDL_PeepEvents(). + * + * \param filter an SDL_EventFilter function to call when an event happens. + * \param userdata a pointer that is passed to `filter` + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_DelEventWatch + * \sa SDL_SetEventFilter + */ +extern DECLSPEC void SDLCALL SDL_AddEventWatch(SDL_EventFilter filter, + void *userdata); + +/** + * Remove an event watch callback added with SDL_AddEventWatch(). + * + * This function takes the same input as SDL_AddEventWatch() to identify and + * delete the corresponding callback. + * + * \param filter the function originally passed to SDL_AddEventWatch() + * \param userdata the pointer originally passed to SDL_AddEventWatch() + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AddEventWatch + */ +extern DECLSPEC void SDLCALL SDL_DelEventWatch(SDL_EventFilter filter, + void *userdata); + +/** + * Run a specific filter function on the current event queue, removing any + * events for which the filter returns 0. + * + * See SDL_SetEventFilter() for more information. Unlike SDL_SetEventFilter(), + * this function does not change the filter permanently, it only uses the + * supplied filter until this function returns. + * + * \param filter the SDL_EventFilter function to call when an event happens + * \param userdata a pointer that is passed to `filter` + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetEventFilter + * \sa SDL_SetEventFilter + */ +extern DECLSPEC void SDLCALL SDL_FilterEvents(SDL_EventFilter filter, + void *userdata); + +/* @{ */ +#define SDL_QUERY -1 +#define SDL_IGNORE 0 +#define SDL_DISABLE 0 +#define SDL_ENABLE 1 + +/** + * Set the state of processing events by type. + * + * `state` may be any of the following: + * + * - `SDL_QUERY`: returns the current processing state of the specified event + * - `SDL_IGNORE` (aka `SDL_DISABLE`): the event will automatically be dropped + * from the event queue and will not be filtered + * - `SDL_ENABLE`: the event will be processed normally + * + * \param type the type of event; see SDL_EventType for details + * \param state how to process the event + * \returns `SDL_DISABLE` or `SDL_ENABLE`, representing the processing state + * of the event before this function makes any changes to it. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetEventState + */ +extern DECLSPEC Uint8 SDLCALL SDL_EventState(Uint32 type, int state); +/* @} */ +#define SDL_GetEventState(type) SDL_EventState(type, SDL_QUERY) + +/** + * Allocate a set of user-defined events, and return the beginning event + * number for that set of events. + * + * Calling this function with `numevents` <= 0 is an error and will return + * (Uint32)-1. + * + * Note, (Uint32)-1 means the maximum unsigned 32-bit integer value (or + * 0xFFFFFFFF), but is clearer to write. + * + * \param numevents the number of events to be allocated + * \returns the beginning event number, or (Uint32)-1 if there are not enough + * user-defined events left. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_PushEvent + */ +extern DECLSPEC Uint32 SDLCALL SDL_RegisterEvents(int numevents); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_events_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_filesystem.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_filesystem.h new file mode 100644 index 00000000..4cad657e --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_filesystem.h @@ -0,0 +1,149 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_filesystem.h + * + * \brief Include file for filesystem SDL API functions + */ + +#ifndef SDL_filesystem_h_ +#define SDL_filesystem_h_ + +#include "SDL_stdinc.h" + +#include "begin_code.h" + +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * Get the directory where the application was run from. + * + * This is not necessarily a fast call, so you should call this once near + * startup and save the string if you need it. + * + * **Mac OS X and iOS Specific Functionality**: If the application is in a + * ".app" bundle, this function returns the Resource directory (e.g. + * MyApp.app/Contents/Resources/). This behaviour can be overridden by adding + * a property to the Info.plist file. Adding a string key with the name + * SDL_FILESYSTEM_BASE_DIR_TYPE with a supported value will change the + * behaviour. + * + * Supported values for the SDL_FILESYSTEM_BASE_DIR_TYPE property (Given an + * application in /Applications/SDLApp/MyApp.app): + * + * - `resource`: bundle resource directory (the default). For example: + * `/Applications/SDLApp/MyApp.app/Contents/Resources` + * - `bundle`: the Bundle directory. For example: + * `/Applications/SDLApp/MyApp.app/` + * - `parent`: the containing directory of the bundle. For example: + * `/Applications/SDLApp/` + * + * **Nintendo 3DS Specific Functionality**: This function returns "romfs" + * directory of the application as it is uncommon to store resources outside + * the executable. As such it is not a writable directory. + * + * The returned path is guaranteed to end with a path separator ('\' on + * Windows, '/' on most other platforms). + * + * The pointer returned is owned by the caller. Please call SDL_free() on the + * pointer when done with it. + * + * \returns an absolute path in UTF-8 encoding to the application data + * directory. NULL will be returned on error or when the platform + * doesn't implement this functionality, call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.1. + * + * \sa SDL_GetPrefPath + */ +extern DECLSPEC char *SDLCALL SDL_GetBasePath(void); + +/** + * Get the user-and-app-specific path where files can be written. + * + * Get the "pref dir". This is meant to be where users can write personal + * files (preferences and save games, etc) that are specific to your + * application. This directory is unique per user, per application. + * + * This function will decide the appropriate location in the native + * filesystem, create the directory if necessary, and return a string of the + * absolute path to the directory in UTF-8 encoding. + * + * On Windows, the string might look like: + * + * `C:\\Users\\bob\\AppData\\Roaming\\My Company\\My Program Name\\` + * + * On Linux, the string might look like: + * + * `/home/bob/.local/share/My Program Name/` + * + * On Mac OS X, the string might look like: + * + * `/Users/bob/Library/Application Support/My Program Name/` + * + * You should assume the path returned by this function is the only safe place + * to write files (and that SDL_GetBasePath(), while it might be writable, or + * even the parent of the returned path, isn't where you should be writing + * things). + * + * Both the org and app strings may become part of a directory name, so please + * follow these rules: + * + * - Try to use the same org string (_including case-sensitivity_) for all + * your applications that use this function. + * - Always use a unique app string for each one, and make sure it never + * changes for an app once you've decided on it. + * - Unicode characters are legal, as long as it's UTF-8 encoded, but... + * - ...only use letters, numbers, and spaces. Avoid punctuation like "Game + * Name 2: Bad Guy's Revenge!" ... "Game Name 2" is sufficient. + * + * The returned path is guaranteed to end with a path separator ('\' on + * Windows, '/' on most other platforms). + * + * The pointer returned is owned by the caller. Please call SDL_free() on the + * pointer when done with it. + * + * \param org the name of your organization + * \param app the name of your application + * \returns a UTF-8 string of the user directory in platform-dependent + * notation. NULL if there's a problem (creating directory failed, + * etc.). + * + * \since This function is available since SDL 2.0.1. + * + * \sa SDL_GetBasePath + */ +extern DECLSPEC char *SDLCALL SDL_GetPrefPath(const char *org, const char *app); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_filesystem_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_gamecontroller.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_gamecontroller.h new file mode 100644 index 00000000..140054d3 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_gamecontroller.h @@ -0,0 +1,1074 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_gamecontroller.h + * + * Include file for SDL game controller event handling + */ + +#ifndef SDL_gamecontroller_h_ +#define SDL_gamecontroller_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_rwops.h" +#include "SDL_sensor.h" +#include "SDL_joystick.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \file SDL_gamecontroller.h + * + * In order to use these functions, SDL_Init() must have been called + * with the ::SDL_INIT_GAMECONTROLLER flag. This causes SDL to scan the system + * for game controllers, and load appropriate drivers. + * + * If you would like to receive controller updates while the application + * is in the background, you should set the following hint before calling + * SDL_Init(): SDL_HINT_JOYSTICK_ALLOW_BACKGROUND_EVENTS + */ + +/** + * The gamecontroller structure used to identify an SDL game controller + */ +struct _SDL_GameController; +typedef struct _SDL_GameController SDL_GameController; + +typedef enum +{ + SDL_CONTROLLER_TYPE_UNKNOWN = 0, + SDL_CONTROLLER_TYPE_XBOX360, + SDL_CONTROLLER_TYPE_XBOXONE, + SDL_CONTROLLER_TYPE_PS3, + SDL_CONTROLLER_TYPE_PS4, + SDL_CONTROLLER_TYPE_NINTENDO_SWITCH_PRO, + SDL_CONTROLLER_TYPE_VIRTUAL, + SDL_CONTROLLER_TYPE_PS5, + SDL_CONTROLLER_TYPE_AMAZON_LUNA, + SDL_CONTROLLER_TYPE_GOOGLE_STADIA, + SDL_CONTROLLER_TYPE_NVIDIA_SHIELD, + SDL_CONTROLLER_TYPE_NINTENDO_SWITCH_JOYCON_LEFT, + SDL_CONTROLLER_TYPE_NINTENDO_SWITCH_JOYCON_RIGHT, + SDL_CONTROLLER_TYPE_NINTENDO_SWITCH_JOYCON_PAIR +} SDL_GameControllerType; + +typedef enum +{ + SDL_CONTROLLER_BINDTYPE_NONE = 0, + SDL_CONTROLLER_BINDTYPE_BUTTON, + SDL_CONTROLLER_BINDTYPE_AXIS, + SDL_CONTROLLER_BINDTYPE_HAT +} SDL_GameControllerBindType; + +/** + * Get the SDL joystick layer binding for this controller button/axis mapping + */ +typedef struct SDL_GameControllerButtonBind +{ + SDL_GameControllerBindType bindType; + union + { + int button; + int axis; + struct { + int hat; + int hat_mask; + } hat; + } value; + +} SDL_GameControllerButtonBind; + + +/** + * To count the number of game controllers in the system for the following: + * + * ```c + * int nJoysticks = SDL_NumJoysticks(); + * int nGameControllers = 0; + * for (int i = 0; i < nJoysticks; i++) { + * if (SDL_IsGameController(i)) { + * nGameControllers++; + * } + * } + * ``` + * + * Using the SDL_HINT_GAMECONTROLLERCONFIG hint or the SDL_GameControllerAddMapping() you can add support for controllers SDL is unaware of or cause an existing controller to have a different binding. The format is: + * guid,name,mappings + * + * Where GUID is the string value from SDL_JoystickGetGUIDString(), name is the human readable string for the device and mappings are controller mappings to joystick ones. + * Under Windows there is a reserved GUID of "xinput" that covers any XInput devices. + * The mapping format for joystick is: + * bX - a joystick button, index X + * hX.Y - hat X with value Y + * aX - axis X of the joystick + * Buttons can be used as a controller axis and vice versa. + * + * This string shows an example of a valid mapping for a controller + * + * ```c + * "03000000341a00003608000000000000,PS3 Controller,a:b1,b:b2,y:b3,x:b0,start:b9,guide:b12,back:b8,dpup:h0.1,dpleft:h0.8,dpdown:h0.4,dpright:h0.2,leftshoulder:b4,rightshoulder:b5,leftstick:b10,rightstick:b11,leftx:a0,lefty:a1,rightx:a2,righty:a3,lefttrigger:b6,righttrigger:b7", + * ``` + */ + +/** + * Load a set of Game Controller mappings from a seekable SDL data stream. + * + * You can call this function several times, if needed, to load different + * database files. + * + * If a new mapping is loaded for an already known controller GUID, the later + * version will overwrite the one currently loaded. + * + * Mappings not belonging to the current platform or with no platform field + * specified will be ignored (i.e. mappings for Linux will be ignored in + * Windows, etc). + * + * This function will load the text database entirely in memory before + * processing it, so take this into consideration if you are in a memory + * constrained environment. + * + * \param rw the data stream for the mappings to be added + * \param freerw non-zero to close the stream after being read + * \returns the number of mappings added or -1 on error; call SDL_GetError() + * for more information. + * + * \since This function is available since SDL 2.0.2. + * + * \sa SDL_GameControllerAddMapping + * \sa SDL_GameControllerAddMappingsFromFile + * \sa SDL_GameControllerMappingForGUID + */ +extern DECLSPEC int SDLCALL SDL_GameControllerAddMappingsFromRW(SDL_RWops * rw, int freerw); + +/** + * Load a set of mappings from a file, filtered by the current SDL_GetPlatform() + * + * Convenience macro. + */ +#define SDL_GameControllerAddMappingsFromFile(file) SDL_GameControllerAddMappingsFromRW(SDL_RWFromFile(file, "rb"), 1) + +/** + * Add support for controllers that SDL is unaware of or to cause an existing + * controller to have a different binding. + * + * The mapping string has the format "GUID,name,mapping", where GUID is the + * string value from SDL_JoystickGetGUIDString(), name is the human readable + * string for the device and mappings are controller mappings to joystick + * ones. Under Windows there is a reserved GUID of "xinput" that covers all + * XInput devices. The mapping format for joystick is: {| |bX |a joystick + * button, index X |- |hX.Y |hat X with value Y |- |aX |axis X of the joystick + * |} Buttons can be used as a controller axes and vice versa. + * + * This string shows an example of a valid mapping for a controller: + * + * ```c + * "341a3608000000000000504944564944,Afterglow PS3 Controller,a:b1,b:b2,y:b3,x:b0,start:b9,guide:b12,back:b8,dpup:h0.1,dpleft:h0.8,dpdown:h0.4,dpright:h0.2,leftshoulder:b4,rightshoulder:b5,leftstick:b10,rightstick:b11,leftx:a0,lefty:a1,rightx:a2,righty:a3,lefttrigger:b6,righttrigger:b7" + * ``` + * + * \param mappingString the mapping string + * \returns 1 if a new mapping is added, 0 if an existing mapping is updated, + * -1 on error; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerMapping + * \sa SDL_GameControllerMappingForGUID + */ +extern DECLSPEC int SDLCALL SDL_GameControllerAddMapping(const char* mappingString); + +/** + * Get the number of mappings installed. + * + * \returns the number of mappings. + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC int SDLCALL SDL_GameControllerNumMappings(void); + +/** + * Get the mapping at a particular index. + * + * \returns the mapping string. Must be freed with SDL_free(). Returns NULL if + * the index is out of range. + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC char * SDLCALL SDL_GameControllerMappingForIndex(int mapping_index); + +/** + * Get the game controller mapping string for a given GUID. + * + * The returned string must be freed with SDL_free(). + * + * \param guid a structure containing the GUID for which a mapping is desired + * \returns a mapping string or NULL on error; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickGetDeviceGUID + * \sa SDL_JoystickGetGUID + */ +extern DECLSPEC char * SDLCALL SDL_GameControllerMappingForGUID(SDL_JoystickGUID guid); + +/** + * Get the current mapping of a Game Controller. + * + * The returned string must be freed with SDL_free(). + * + * Details about mappings are discussed with SDL_GameControllerAddMapping(). + * + * \param gamecontroller the game controller you want to get the current + * mapping for + * \returns a string that has the controller's mapping or NULL if no mapping + * is available; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerAddMapping + * \sa SDL_GameControllerMappingForGUID + */ +extern DECLSPEC char * SDLCALL SDL_GameControllerMapping(SDL_GameController *gamecontroller); + +/** + * Check if the given joystick is supported by the game controller interface. + * + * `joystick_index` is the same as the `device_index` passed to + * SDL_JoystickOpen(). + * + * \param joystick_index the device_index of a device, up to + * SDL_NumJoysticks() + * \returns SDL_TRUE if the given joystick is supported by the game controller + * interface, SDL_FALSE if it isn't or it's an invalid index. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerNameForIndex + * \sa SDL_GameControllerOpen + */ +extern DECLSPEC SDL_bool SDLCALL SDL_IsGameController(int joystick_index); + +/** + * Get the implementation dependent name for the game controller. + * + * This function can be called before any controllers are opened. + * + * `joystick_index` is the same as the `device_index` passed to + * SDL_JoystickOpen(). + * + * \param joystick_index the device_index of a device, from zero to + * SDL_NumJoysticks()-1 + * \returns the implementation-dependent name for the game controller, or NULL + * if there is no name or the index is invalid. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerName + * \sa SDL_GameControllerOpen + * \sa SDL_IsGameController + */ +extern DECLSPEC const char *SDLCALL SDL_GameControllerNameForIndex(int joystick_index); + +/** + * Get the implementation dependent path for the game controller. + * + * This function can be called before any controllers are opened. + * + * `joystick_index` is the same as the `device_index` passed to + * SDL_JoystickOpen(). + * + * \param joystick_index the device_index of a device, from zero to + * SDL_NumJoysticks()-1 + * \returns the implementation-dependent path for the game controller, or NULL + * if there is no path or the index is invalid. + * + * \since This function is available since SDL 2.24.0. + * + * \sa SDL_GameControllerPath + */ +extern DECLSPEC const char *SDLCALL SDL_GameControllerPathForIndex(int joystick_index); + +/** + * Get the type of a game controller. + * + * This can be called before any controllers are opened. + * + * \param joystick_index the device_index of a device, from zero to + * SDL_NumJoysticks()-1 + * \returns the controller type. + * + * \since This function is available since SDL 2.0.12. + */ +extern DECLSPEC SDL_GameControllerType SDLCALL SDL_GameControllerTypeForIndex(int joystick_index); + +/** + * Get the mapping of a game controller. + * + * This can be called before any controllers are opened. + * + * \param joystick_index the device_index of a device, from zero to + * SDL_NumJoysticks()-1 + * \returns the mapping string. Must be freed with SDL_free(). Returns NULL if + * no mapping is available. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC char *SDLCALL SDL_GameControllerMappingForDeviceIndex(int joystick_index); + +/** + * Open a game controller for use. + * + * `joystick_index` is the same as the `device_index` passed to + * SDL_JoystickOpen(). + * + * The index passed as an argument refers to the N'th game controller on the + * system. This index is not the value which will identify this controller in + * future controller events. The joystick's instance id (SDL_JoystickID) will + * be used there instead. + * + * \param joystick_index the device_index of a device, up to + * SDL_NumJoysticks() + * \returns a gamecontroller identifier or NULL if an error occurred; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerClose + * \sa SDL_GameControllerNameForIndex + * \sa SDL_IsGameController + */ +extern DECLSPEC SDL_GameController *SDLCALL SDL_GameControllerOpen(int joystick_index); + +/** + * Get the SDL_GameController associated with an instance id. + * + * \param joyid the instance id to get the SDL_GameController for + * \returns an SDL_GameController on success or NULL on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.4. + */ +extern DECLSPEC SDL_GameController *SDLCALL SDL_GameControllerFromInstanceID(SDL_JoystickID joyid); + +/** + * Get the SDL_GameController associated with a player index. + * + * Please note that the player index is _not_ the device index, nor is it the + * instance id! + * + * \param player_index the player index, which is not the device index or the + * instance id! + * \returns the SDL_GameController associated with a player index. + * + * \since This function is available since SDL 2.0.12. + * + * \sa SDL_GameControllerGetPlayerIndex + * \sa SDL_GameControllerSetPlayerIndex + */ +extern DECLSPEC SDL_GameController *SDLCALL SDL_GameControllerFromPlayerIndex(int player_index); + +/** + * Get the implementation-dependent name for an opened game controller. + * + * This is the same name as returned by SDL_GameControllerNameForIndex(), but + * it takes a controller identifier instead of the (unstable) device index. + * + * \param gamecontroller a game controller identifier previously returned by + * SDL_GameControllerOpen() + * \returns the implementation dependent name for the game controller, or NULL + * if there is no name or the identifier passed is invalid. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerNameForIndex + * \sa SDL_GameControllerOpen + */ +extern DECLSPEC const char *SDLCALL SDL_GameControllerName(SDL_GameController *gamecontroller); + +/** + * Get the implementation-dependent path for an opened game controller. + * + * This is the same path as returned by SDL_GameControllerNameForIndex(), but + * it takes a controller identifier instead of the (unstable) device index. + * + * \param gamecontroller a game controller identifier previously returned by + * SDL_GameControllerOpen() + * \returns the implementation dependent path for the game controller, or NULL + * if there is no path or the identifier passed is invalid. + * + * \since This function is available since SDL 2.24.0. + * + * \sa SDL_GameControllerPathForIndex + */ +extern DECLSPEC const char *SDLCALL SDL_GameControllerPath(SDL_GameController *gamecontroller); + +/** + * Get the type of this currently opened controller + * + * This is the same name as returned by SDL_GameControllerTypeForIndex(), but + * it takes a controller identifier instead of the (unstable) device index. + * + * \param gamecontroller the game controller object to query. + * \returns the controller type. + * + * \since This function is available since SDL 2.0.12. + */ +extern DECLSPEC SDL_GameControllerType SDLCALL SDL_GameControllerGetType(SDL_GameController *gamecontroller); + +/** + * Get the player index of an opened game controller. + * + * For XInput controllers this returns the XInput user index. + * + * \param gamecontroller the game controller object to query. + * \returns the player index for controller, or -1 if it's not available. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC int SDLCALL SDL_GameControllerGetPlayerIndex(SDL_GameController *gamecontroller); + +/** + * Set the player index of an opened game controller. + * + * \param gamecontroller the game controller object to adjust. + * \param player_index Player index to assign to this controller, or -1 to + * clear the player index and turn off player LEDs. + * + * \since This function is available since SDL 2.0.12. + */ +extern DECLSPEC void SDLCALL SDL_GameControllerSetPlayerIndex(SDL_GameController *gamecontroller, int player_index); + +/** + * Get the USB vendor ID of an opened controller, if available. + * + * If the vendor ID isn't available this function returns 0. + * + * \param gamecontroller the game controller object to query. + * \return the USB vendor ID, or zero if unavailable. + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC Uint16 SDLCALL SDL_GameControllerGetVendor(SDL_GameController *gamecontroller); + +/** + * Get the USB product ID of an opened controller, if available. + * + * If the product ID isn't available this function returns 0. + * + * \param gamecontroller the game controller object to query. + * \return the USB product ID, or zero if unavailable. + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC Uint16 SDLCALL SDL_GameControllerGetProduct(SDL_GameController *gamecontroller); + +/** + * Get the product version of an opened controller, if available. + * + * If the product version isn't available this function returns 0. + * + * \param gamecontroller the game controller object to query. + * \return the USB product version, or zero if unavailable. + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC Uint16 SDLCALL SDL_GameControllerGetProductVersion(SDL_GameController *gamecontroller); + +/** + * Get the firmware version of an opened controller, if available. + * + * If the firmware version isn't available this function returns 0. + * + * \param gamecontroller the game controller object to query. + * \return the controller firmware version, or zero if unavailable. + * + * \since This function is available since SDL 2.24.0. + */ +extern DECLSPEC Uint16 SDLCALL SDL_GameControllerGetFirmwareVersion(SDL_GameController *gamecontroller); + +/** + * Get the serial number of an opened controller, if available. + * + * Returns the serial number of the controller, or NULL if it is not + * available. + * + * \param gamecontroller the game controller object to query. + * \return the serial number, or NULL if unavailable. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC const char * SDLCALL SDL_GameControllerGetSerial(SDL_GameController *gamecontroller); + +/** + * Check if a controller has been opened and is currently connected. + * + * \param gamecontroller a game controller identifier previously returned by + * SDL_GameControllerOpen() + * \returns SDL_TRUE if the controller has been opened and is currently + * connected, or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerClose + * \sa SDL_GameControllerOpen + */ +extern DECLSPEC SDL_bool SDLCALL SDL_GameControllerGetAttached(SDL_GameController *gamecontroller); + +/** + * Get the Joystick ID from a Game Controller. + * + * This function will give you a SDL_Joystick object, which allows you to use + * the SDL_Joystick functions with a SDL_GameController object. This would be + * useful for getting a joystick's position at any given time, even if it + * hasn't moved (moving it would produce an event, which would have the axis' + * value). + * + * The pointer returned is owned by the SDL_GameController. You should not + * call SDL_JoystickClose() on it, for example, since doing so will likely + * cause SDL to crash. + * + * \param gamecontroller the game controller object that you want to get a + * joystick from + * \returns a SDL_Joystick object; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC SDL_Joystick *SDLCALL SDL_GameControllerGetJoystick(SDL_GameController *gamecontroller); + +/** + * Query or change current state of Game Controller events. + * + * If controller events are disabled, you must call SDL_GameControllerUpdate() + * yourself and check the state of the controller when you want controller + * information. + * + * Any number can be passed to SDL_GameControllerEventState(), but only -1, 0, + * and 1 will have any effect. Other numbers will just be returned. + * + * \param state can be one of `SDL_QUERY`, `SDL_IGNORE`, or `SDL_ENABLE` + * \returns the same value passed to the function, with exception to -1 + * (SDL_QUERY), which will return the current state. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickEventState + */ +extern DECLSPEC int SDLCALL SDL_GameControllerEventState(int state); + +/** + * Manually pump game controller updates if not using the loop. + * + * This function is called automatically by the event loop if events are + * enabled. Under such circumstances, it will not be necessary to call this + * function. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC void SDLCALL SDL_GameControllerUpdate(void); + + +/** + * The list of axes available from a controller + * + * Thumbstick axis values range from SDL_JOYSTICK_AXIS_MIN to SDL_JOYSTICK_AXIS_MAX, + * and are centered within ~8000 of zero, though advanced UI will allow users to set + * or autodetect the dead zone, which varies between controllers. + * + * Trigger axis values range from 0 to SDL_JOYSTICK_AXIS_MAX. + */ +typedef enum +{ + SDL_CONTROLLER_AXIS_INVALID = -1, + SDL_CONTROLLER_AXIS_LEFTX, + SDL_CONTROLLER_AXIS_LEFTY, + SDL_CONTROLLER_AXIS_RIGHTX, + SDL_CONTROLLER_AXIS_RIGHTY, + SDL_CONTROLLER_AXIS_TRIGGERLEFT, + SDL_CONTROLLER_AXIS_TRIGGERRIGHT, + SDL_CONTROLLER_AXIS_MAX +} SDL_GameControllerAxis; + +/** + * Convert a string into SDL_GameControllerAxis enum. + * + * This function is called internally to translate SDL_GameController mapping + * strings for the underlying joystick device into the consistent + * SDL_GameController mapping. You do not normally need to call this function + * unless you are parsing SDL_GameController mappings in your own code. + * + * Note specially that "righttrigger" and "lefttrigger" map to + * `SDL_CONTROLLER_AXIS_TRIGGERRIGHT` and `SDL_CONTROLLER_AXIS_TRIGGERLEFT`, + * respectively. + * + * \param str string representing a SDL_GameController axis + * \returns the SDL_GameControllerAxis enum corresponding to the input string, + * or `SDL_CONTROLLER_AXIS_INVALID` if no match was found. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerGetStringForAxis + */ +extern DECLSPEC SDL_GameControllerAxis SDLCALL SDL_GameControllerGetAxisFromString(const char *str); + +/** + * Convert from an SDL_GameControllerAxis enum to a string. + * + * The caller should not SDL_free() the returned string. + * + * \param axis an enum value for a given SDL_GameControllerAxis + * \returns a string for the given axis, or NULL if an invalid axis is + * specified. The string returned is of the format used by + * SDL_GameController mapping strings. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerGetAxisFromString + */ +extern DECLSPEC const char* SDLCALL SDL_GameControllerGetStringForAxis(SDL_GameControllerAxis axis); + +/** + * Get the SDL joystick layer binding for a controller axis mapping. + * + * \param gamecontroller a game controller + * \param axis an axis enum value (one of the SDL_GameControllerAxis values) + * \returns a SDL_GameControllerButtonBind describing the bind. On failure + * (like the given Controller axis doesn't exist on the device), its + * `.bindType` will be `SDL_CONTROLLER_BINDTYPE_NONE`. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerGetBindForButton + */ +extern DECLSPEC SDL_GameControllerButtonBind SDLCALL +SDL_GameControllerGetBindForAxis(SDL_GameController *gamecontroller, + SDL_GameControllerAxis axis); + +/** + * Query whether a game controller has a given axis. + * + * This merely reports whether the controller's mapping defined this axis, as + * that is all the information SDL has about the physical device. + * + * \param gamecontroller a game controller + * \param axis an axis enum value (an SDL_GameControllerAxis value) + * \returns SDL_TRUE if the controller has this axis, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC SDL_bool SDLCALL +SDL_GameControllerHasAxis(SDL_GameController *gamecontroller, SDL_GameControllerAxis axis); + +/** + * Get the current state of an axis control on a game controller. + * + * The axis indices start at index 0. + * + * The state is a value ranging from -32768 to 32767. Triggers, however, range + * from 0 to 32767 (they never return a negative value). + * + * \param gamecontroller a game controller + * \param axis an axis index (one of the SDL_GameControllerAxis values) + * \returns axis state (including 0) on success or 0 (also) on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerGetButton + */ +extern DECLSPEC Sint16 SDLCALL +SDL_GameControllerGetAxis(SDL_GameController *gamecontroller, SDL_GameControllerAxis axis); + +/** + * The list of buttons available from a controller + */ +typedef enum +{ + SDL_CONTROLLER_BUTTON_INVALID = -1, + SDL_CONTROLLER_BUTTON_A, + SDL_CONTROLLER_BUTTON_B, + SDL_CONTROLLER_BUTTON_X, + SDL_CONTROLLER_BUTTON_Y, + SDL_CONTROLLER_BUTTON_BACK, + SDL_CONTROLLER_BUTTON_GUIDE, + SDL_CONTROLLER_BUTTON_START, + SDL_CONTROLLER_BUTTON_LEFTSTICK, + SDL_CONTROLLER_BUTTON_RIGHTSTICK, + SDL_CONTROLLER_BUTTON_LEFTSHOULDER, + SDL_CONTROLLER_BUTTON_RIGHTSHOULDER, + SDL_CONTROLLER_BUTTON_DPAD_UP, + SDL_CONTROLLER_BUTTON_DPAD_DOWN, + SDL_CONTROLLER_BUTTON_DPAD_LEFT, + SDL_CONTROLLER_BUTTON_DPAD_RIGHT, + SDL_CONTROLLER_BUTTON_MISC1, /* Xbox Series X share button, PS5 microphone button, Nintendo Switch Pro capture button, Amazon Luna microphone button */ + SDL_CONTROLLER_BUTTON_PADDLE1, /* Xbox Elite paddle P1 (upper left, facing the back) */ + SDL_CONTROLLER_BUTTON_PADDLE2, /* Xbox Elite paddle P3 (upper right, facing the back) */ + SDL_CONTROLLER_BUTTON_PADDLE3, /* Xbox Elite paddle P2 (lower left, facing the back) */ + SDL_CONTROLLER_BUTTON_PADDLE4, /* Xbox Elite paddle P4 (lower right, facing the back) */ + SDL_CONTROLLER_BUTTON_TOUCHPAD, /* PS4/PS5 touchpad button */ + SDL_CONTROLLER_BUTTON_MAX +} SDL_GameControllerButton; + +/** + * Convert a string into an SDL_GameControllerButton enum. + * + * This function is called internally to translate SDL_GameController mapping + * strings for the underlying joystick device into the consistent + * SDL_GameController mapping. You do not normally need to call this function + * unless you are parsing SDL_GameController mappings in your own code. + * + * \param str string representing a SDL_GameController axis + * \returns the SDL_GameControllerButton enum corresponding to the input + * string, or `SDL_CONTROLLER_AXIS_INVALID` if no match was found. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC SDL_GameControllerButton SDLCALL SDL_GameControllerGetButtonFromString(const char *str); + +/** + * Convert from an SDL_GameControllerButton enum to a string. + * + * The caller should not SDL_free() the returned string. + * + * \param button an enum value for a given SDL_GameControllerButton + * \returns a string for the given button, or NULL if an invalid button is + * specified. The string returned is of the format used by + * SDL_GameController mapping strings. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerGetButtonFromString + */ +extern DECLSPEC const char* SDLCALL SDL_GameControllerGetStringForButton(SDL_GameControllerButton button); + +/** + * Get the SDL joystick layer binding for a controller button mapping. + * + * \param gamecontroller a game controller + * \param button an button enum value (an SDL_GameControllerButton value) + * \returns a SDL_GameControllerButtonBind describing the bind. On failure + * (like the given Controller button doesn't exist on the device), + * its `.bindType` will be `SDL_CONTROLLER_BINDTYPE_NONE`. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerGetBindForAxis + */ +extern DECLSPEC SDL_GameControllerButtonBind SDLCALL +SDL_GameControllerGetBindForButton(SDL_GameController *gamecontroller, + SDL_GameControllerButton button); + +/** + * Query whether a game controller has a given button. + * + * This merely reports whether the controller's mapping defined this button, + * as that is all the information SDL has about the physical device. + * + * \param gamecontroller a game controller + * \param button a button enum value (an SDL_GameControllerButton value) + * \returns SDL_TRUE if the controller has this button, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_GameControllerHasButton(SDL_GameController *gamecontroller, + SDL_GameControllerButton button); + +/** + * Get the current state of a button on a game controller. + * + * \param gamecontroller a game controller + * \param button a button index (one of the SDL_GameControllerButton values) + * \returns 1 for pressed state or 0 for not pressed state or error; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerGetAxis + */ +extern DECLSPEC Uint8 SDLCALL SDL_GameControllerGetButton(SDL_GameController *gamecontroller, + SDL_GameControllerButton button); + +/** + * Get the number of touchpads on a game controller. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC int SDLCALL SDL_GameControllerGetNumTouchpads(SDL_GameController *gamecontroller); + +/** + * Get the number of supported simultaneous fingers on a touchpad on a game + * controller. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC int SDLCALL SDL_GameControllerGetNumTouchpadFingers(SDL_GameController *gamecontroller, int touchpad); + +/** + * Get the current state of a finger on a touchpad on a game controller. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC int SDLCALL SDL_GameControllerGetTouchpadFinger(SDL_GameController *gamecontroller, int touchpad, int finger, Uint8 *state, float *x, float *y, float *pressure); + +/** + * Return whether a game controller has a particular sensor. + * + * \param gamecontroller The controller to query + * \param type The type of sensor to query + * \returns SDL_TRUE if the sensor exists, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_GameControllerHasSensor(SDL_GameController *gamecontroller, SDL_SensorType type); + +/** + * Set whether data reporting for a game controller sensor is enabled. + * + * \param gamecontroller The controller to update + * \param type The type of sensor to enable/disable + * \param enabled Whether data reporting should be enabled + * \returns 0 or -1 if an error occurred. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC int SDLCALL SDL_GameControllerSetSensorEnabled(SDL_GameController *gamecontroller, SDL_SensorType type, SDL_bool enabled); + +/** + * Query whether sensor data reporting is enabled for a game controller. + * + * \param gamecontroller The controller to query + * \param type The type of sensor to query + * \returns SDL_TRUE if the sensor is enabled, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_GameControllerIsSensorEnabled(SDL_GameController *gamecontroller, SDL_SensorType type); + +/** + * Get the data rate (number of events per second) of a game controller + * sensor. + * + * \param gamecontroller The controller to query + * \param type The type of sensor to query + * \return the data rate, or 0.0f if the data rate is not available. + * + * \since This function is available since SDL 2.0.16. + */ +extern DECLSPEC float SDLCALL SDL_GameControllerGetSensorDataRate(SDL_GameController *gamecontroller, SDL_SensorType type); + +/** + * Get the current state of a game controller sensor. + * + * The number of values and interpretation of the data is sensor dependent. + * See SDL_sensor.h for the details for each type of sensor. + * + * \param gamecontroller The controller to query + * \param type The type of sensor to query + * \param data A pointer filled with the current sensor state + * \param num_values The number of values to write to data + * \return 0 or -1 if an error occurred. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC int SDLCALL SDL_GameControllerGetSensorData(SDL_GameController *gamecontroller, SDL_SensorType type, float *data, int num_values); + +/** + * Get the current state of a game controller sensor with the timestamp of the + * last update. + * + * The number of values and interpretation of the data is sensor dependent. + * See SDL_sensor.h for the details for each type of sensor. + * + * \param gamecontroller The controller to query + * \param type The type of sensor to query + * \param timestamp A pointer filled with the timestamp in microseconds of the + * current sensor reading if available, or 0 if not + * \param data A pointer filled with the current sensor state + * \param num_values The number of values to write to data + * \return 0 or -1 if an error occurred. + * + * \since This function is available since SDL 2.26.0. + */ +extern DECLSPEC int SDLCALL SDL_GameControllerGetSensorDataWithTimestamp(SDL_GameController *gamecontroller, SDL_SensorType type, Uint64 *timestamp, float *data, int num_values); + +/** + * Start a rumble effect on a game controller. + * + * Each call to this function cancels any previous rumble effect, and calling + * it with 0 intensity stops any rumbling. + * + * \param gamecontroller The controller to vibrate + * \param low_frequency_rumble The intensity of the low frequency (left) + * rumble motor, from 0 to 0xFFFF + * \param high_frequency_rumble The intensity of the high frequency (right) + * rumble motor, from 0 to 0xFFFF + * \param duration_ms The duration of the rumble effect, in milliseconds + * \returns 0, or -1 if rumble isn't supported on this controller + * + * \since This function is available since SDL 2.0.9. + * + * \sa SDL_GameControllerHasRumble + */ +extern DECLSPEC int SDLCALL SDL_GameControllerRumble(SDL_GameController *gamecontroller, Uint16 low_frequency_rumble, Uint16 high_frequency_rumble, Uint32 duration_ms); + +/** + * Start a rumble effect in the game controller's triggers. + * + * Each call to this function cancels any previous trigger rumble effect, and + * calling it with 0 intensity stops any rumbling. + * + * Note that this is rumbling of the _triggers_ and not the game controller as + * a whole. This is currently only supported on Xbox One controllers. If you + * want the (more common) whole-controller rumble, use + * SDL_GameControllerRumble() instead. + * + * \param gamecontroller The controller to vibrate + * \param left_rumble The intensity of the left trigger rumble motor, from 0 + * to 0xFFFF + * \param right_rumble The intensity of the right trigger rumble motor, from 0 + * to 0xFFFF + * \param duration_ms The duration of the rumble effect, in milliseconds + * \returns 0, or -1 if trigger rumble isn't supported on this controller + * + * \since This function is available since SDL 2.0.14. + * + * \sa SDL_GameControllerHasRumbleTriggers + */ +extern DECLSPEC int SDLCALL SDL_GameControllerRumbleTriggers(SDL_GameController *gamecontroller, Uint16 left_rumble, Uint16 right_rumble, Uint32 duration_ms); + +/** + * Query whether a game controller has an LED. + * + * \param gamecontroller The controller to query + * \returns SDL_TRUE, or SDL_FALSE if this controller does not have a + * modifiable LED + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_GameControllerHasLED(SDL_GameController *gamecontroller); + +/** + * Query whether a game controller has rumble support. + * + * \param gamecontroller The controller to query + * \returns SDL_TRUE, or SDL_FALSE if this controller does not have rumble + * support + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_GameControllerRumble + */ +extern DECLSPEC SDL_bool SDLCALL SDL_GameControllerHasRumble(SDL_GameController *gamecontroller); + +/** + * Query whether a game controller has rumble support on triggers. + * + * \param gamecontroller The controller to query + * \returns SDL_TRUE, or SDL_FALSE if this controller does not have trigger + * rumble support + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_GameControllerRumbleTriggers + */ +extern DECLSPEC SDL_bool SDLCALL SDL_GameControllerHasRumbleTriggers(SDL_GameController *gamecontroller); + +/** + * Update a game controller's LED color. + * + * \param gamecontroller The controller to update + * \param red The intensity of the red LED + * \param green The intensity of the green LED + * \param blue The intensity of the blue LED + * \returns 0, or -1 if this controller does not have a modifiable LED + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC int SDLCALL SDL_GameControllerSetLED(SDL_GameController *gamecontroller, Uint8 red, Uint8 green, Uint8 blue); + +/** + * Send a controller specific effect packet + * + * \param gamecontroller The controller to affect + * \param data The data to send to the controller + * \param size The size of the data to send to the controller + * \returns 0, or -1 if this controller or driver doesn't support effect + * packets + * + * \since This function is available since SDL 2.0.16. + */ +extern DECLSPEC int SDLCALL SDL_GameControllerSendEffect(SDL_GameController *gamecontroller, const void *data, int size); + +/** + * Close a game controller previously opened with SDL_GameControllerOpen(). + * + * \param gamecontroller a game controller identifier previously returned by + * SDL_GameControllerOpen() + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerOpen + */ +extern DECLSPEC void SDLCALL SDL_GameControllerClose(SDL_GameController *gamecontroller); + +/** + * Return the sfSymbolsName for a given button on a game controller on Apple + * platforms. + * + * \param gamecontroller the controller to query + * \param button a button on the game controller + * \returns the sfSymbolsName or NULL if the name can't be found + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_GameControllerGetAppleSFSymbolsNameForAxis + */ +extern DECLSPEC const char* SDLCALL SDL_GameControllerGetAppleSFSymbolsNameForButton(SDL_GameController *gamecontroller, SDL_GameControllerButton button); + +/** + * Return the sfSymbolsName for a given axis on a game controller on Apple + * platforms. + * + * \param gamecontroller the controller to query + * \param axis an axis on the game controller + * \returns the sfSymbolsName or NULL if the name can't be found + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_GameControllerGetAppleSFSymbolsNameForButton + */ +extern DECLSPEC const char* SDLCALL SDL_GameControllerGetAppleSFSymbolsNameForAxis(SDL_GameController *gamecontroller, SDL_GameControllerAxis axis); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_gamecontroller_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_gesture.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_gesture.h new file mode 100644 index 00000000..db70b4dd --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_gesture.h @@ -0,0 +1,117 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_gesture.h + * + * Include file for SDL gesture event handling. + */ + +#ifndef SDL_gesture_h_ +#define SDL_gesture_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_video.h" + +#include "SDL_touch.h" + + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +typedef Sint64 SDL_GestureID; + +/* Function prototypes */ + +/** + * Begin recording a gesture on a specified touch device or all touch devices. + * + * If the parameter `touchId` is -1 (i.e., all devices), this function will + * always return 1, regardless of whether there actually are any devices. + * + * \param touchId the touch device id, or -1 for all touch devices + * \returns 1 on success or 0 if the specified device could not be found. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetTouchDevice + */ +extern DECLSPEC int SDLCALL SDL_RecordGesture(SDL_TouchID touchId); + + +/** + * Save all currently loaded Dollar Gesture templates. + * + * \param dst a SDL_RWops to save to + * \returns the number of saved templates on success or 0 on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LoadDollarTemplates + * \sa SDL_SaveDollarTemplate + */ +extern DECLSPEC int SDLCALL SDL_SaveAllDollarTemplates(SDL_RWops *dst); + +/** + * Save a currently loaded Dollar Gesture template. + * + * \param gestureId a gesture id + * \param dst a SDL_RWops to save to + * \returns 1 on success or 0 on failure; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LoadDollarTemplates + * \sa SDL_SaveAllDollarTemplates + */ +extern DECLSPEC int SDLCALL SDL_SaveDollarTemplate(SDL_GestureID gestureId,SDL_RWops *dst); + + +/** + * Load Dollar Gesture templates from a file. + * + * \param touchId a touch id + * \param src a SDL_RWops to load from + * \returns the number of loaded templates on success or a negative error code + * (or 0) on failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SaveAllDollarTemplates + * \sa SDL_SaveDollarTemplate + */ +extern DECLSPEC int SDLCALL SDL_LoadDollarTemplates(SDL_TouchID touchId, SDL_RWops *src); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_gesture_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_guid.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_guid.h new file mode 100644 index 00000000..d964223c --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_guid.h @@ -0,0 +1,100 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_guid.h + * + * Include file for handling ::SDL_GUID values. + */ + +#ifndef SDL_guid_h_ +#define SDL_guid_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * An SDL_GUID is a 128-bit identifier for an input device that + * identifies that device across runs of SDL programs on the same + * platform. If the device is detached and then re-attached to a + * different port, or if the base system is rebooted, the device + * should still report the same GUID. + * + * GUIDs are as precise as possible but are not guaranteed to + * distinguish physically distinct but equivalent devices. For + * example, two game controllers from the same vendor with the same + * product ID and revision may have the same GUID. + * + * GUIDs may be platform-dependent (i.e., the same device may report + * different GUIDs on different operating systems). + */ +typedef struct { + Uint8 data[16]; +} SDL_GUID; + +/* Function prototypes */ + +/** + * Get an ASCII string representation for a given ::SDL_GUID. + * + * You should supply at least 33 bytes for pszGUID. + * + * \param guid the ::SDL_GUID you wish to convert to string + * \param pszGUID buffer in which to write the ASCII string + * \param cbGUID the size of pszGUID + * + * \since This function is available since SDL 2.24.0. + * + * \sa SDL_GUIDFromString + */ +extern DECLSPEC void SDLCALL SDL_GUIDToString(SDL_GUID guid, char *pszGUID, int cbGUID); + +/** + * Convert a GUID string into a ::SDL_GUID structure. + * + * Performs no error checking. If this function is given a string containing + * an invalid GUID, the function will silently succeed, but the GUID generated + * will not be useful. + * + * \param pchGUID string containing an ASCII representation of a GUID + * \returns a ::SDL_GUID structure. + * + * \since This function is available since SDL 2.24.0. + * + * \sa SDL_GUIDToString + */ +extern DECLSPEC SDL_GUID SDLCALL SDL_GUIDFromString(const char *pchGUID); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_guid_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_haptic.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_haptic.h new file mode 100644 index 00000000..2462a1e4 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_haptic.h @@ -0,0 +1,1341 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_haptic.h + * + * \brief The SDL haptic subsystem allows you to control haptic (force feedback) + * devices. + * + * The basic usage is as follows: + * - Initialize the subsystem (::SDL_INIT_HAPTIC). + * - Open a haptic device. + * - SDL_HapticOpen() to open from index. + * - SDL_HapticOpenFromJoystick() to open from an existing joystick. + * - Create an effect (::SDL_HapticEffect). + * - Upload the effect with SDL_HapticNewEffect(). + * - Run the effect with SDL_HapticRunEffect(). + * - (optional) Free the effect with SDL_HapticDestroyEffect(). + * - Close the haptic device with SDL_HapticClose(). + * + * \par Simple rumble example: + * \code + * SDL_Haptic *haptic; + * + * // Open the device + * haptic = SDL_HapticOpen( 0 ); + * if (haptic == NULL) + * return -1; + * + * // Initialize simple rumble + * if (SDL_HapticRumbleInit( haptic ) != 0) + * return -1; + * + * // Play effect at 50% strength for 2 seconds + * if (SDL_HapticRumblePlay( haptic, 0.5, 2000 ) != 0) + * return -1; + * SDL_Delay( 2000 ); + * + * // Clean up + * SDL_HapticClose( haptic ); + * \endcode + * + * \par Complete example: + * \code + * int test_haptic( SDL_Joystick * joystick ) { + * SDL_Haptic *haptic; + * SDL_HapticEffect effect; + * int effect_id; + * + * // Open the device + * haptic = SDL_HapticOpenFromJoystick( joystick ); + * if (haptic == NULL) return -1; // Most likely joystick isn't haptic + * + * // See if it can do sine waves + * if ((SDL_HapticQuery(haptic) & SDL_HAPTIC_SINE)==0) { + * SDL_HapticClose(haptic); // No sine effect + * return -1; + * } + * + * // Create the effect + * SDL_memset( &effect, 0, sizeof(SDL_HapticEffect) ); // 0 is safe default + * effect.type = SDL_HAPTIC_SINE; + * effect.periodic.direction.type = SDL_HAPTIC_POLAR; // Polar coordinates + * effect.periodic.direction.dir[0] = 18000; // Force comes from south + * effect.periodic.period = 1000; // 1000 ms + * effect.periodic.magnitude = 20000; // 20000/32767 strength + * effect.periodic.length = 5000; // 5 seconds long + * effect.periodic.attack_length = 1000; // Takes 1 second to get max strength + * effect.periodic.fade_length = 1000; // Takes 1 second to fade away + * + * // Upload the effect + * effect_id = SDL_HapticNewEffect( haptic, &effect ); + * + * // Test the effect + * SDL_HapticRunEffect( haptic, effect_id, 1 ); + * SDL_Delay( 5000); // Wait for the effect to finish + * + * // We destroy the effect, although closing the device also does this + * SDL_HapticDestroyEffect( haptic, effect_id ); + * + * // Close the device + * SDL_HapticClose(haptic); + * + * return 0; // Success + * } + * \endcode + */ + +#ifndef SDL_haptic_h_ +#define SDL_haptic_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_joystick.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif /* __cplusplus */ + +/* FIXME: For SDL 2.1, adjust all the magnitude variables to be Uint16 (0xFFFF). + * + * At the moment the magnitude variables are mixed between signed/unsigned, and + * it is also not made clear that ALL of those variables expect a max of 0x7FFF. + * + * Some platforms may have higher precision than that (Linux FF, Windows XInput) + * so we should fix the inconsistency in favor of higher possible precision, + * adjusting for platforms that use different scales. + * -flibit + */ + +/** + * \typedef SDL_Haptic + * + * \brief The haptic structure used to identify an SDL haptic. + * + * \sa SDL_HapticOpen + * \sa SDL_HapticOpenFromJoystick + * \sa SDL_HapticClose + */ +struct _SDL_Haptic; +typedef struct _SDL_Haptic SDL_Haptic; + + +/** + * \name Haptic features + * + * Different haptic features a device can have. + */ +/* @{ */ + +/** + * \name Haptic effects + */ +/* @{ */ + +/** + * \brief Constant effect supported. + * + * Constant haptic effect. + * + * \sa SDL_HapticCondition + */ +#define SDL_HAPTIC_CONSTANT (1u<<0) + +/** + * \brief Sine wave effect supported. + * + * Periodic haptic effect that simulates sine waves. + * + * \sa SDL_HapticPeriodic + */ +#define SDL_HAPTIC_SINE (1u<<1) + +/** + * \brief Left/Right effect supported. + * + * Haptic effect for direct control over high/low frequency motors. + * + * \sa SDL_HapticLeftRight + * \warning this value was SDL_HAPTIC_SQUARE right before 2.0.0 shipped. Sorry, + * we ran out of bits, and this is important for XInput devices. + */ +#define SDL_HAPTIC_LEFTRIGHT (1u<<2) + +/* !!! FIXME: put this back when we have more bits in 2.1 */ +/* #define SDL_HAPTIC_SQUARE (1<<2) */ + +/** + * \brief Triangle wave effect supported. + * + * Periodic haptic effect that simulates triangular waves. + * + * \sa SDL_HapticPeriodic + */ +#define SDL_HAPTIC_TRIANGLE (1u<<3) + +/** + * \brief Sawtoothup wave effect supported. + * + * Periodic haptic effect that simulates saw tooth up waves. + * + * \sa SDL_HapticPeriodic + */ +#define SDL_HAPTIC_SAWTOOTHUP (1u<<4) + +/** + * \brief Sawtoothdown wave effect supported. + * + * Periodic haptic effect that simulates saw tooth down waves. + * + * \sa SDL_HapticPeriodic + */ +#define SDL_HAPTIC_SAWTOOTHDOWN (1u<<5) + +/** + * \brief Ramp effect supported. + * + * Ramp haptic effect. + * + * \sa SDL_HapticRamp + */ +#define SDL_HAPTIC_RAMP (1u<<6) + +/** + * \brief Spring effect supported - uses axes position. + * + * Condition haptic effect that simulates a spring. Effect is based on the + * axes position. + * + * \sa SDL_HapticCondition + */ +#define SDL_HAPTIC_SPRING (1u<<7) + +/** + * \brief Damper effect supported - uses axes velocity. + * + * Condition haptic effect that simulates dampening. Effect is based on the + * axes velocity. + * + * \sa SDL_HapticCondition + */ +#define SDL_HAPTIC_DAMPER (1u<<8) + +/** + * \brief Inertia effect supported - uses axes acceleration. + * + * Condition haptic effect that simulates inertia. Effect is based on the axes + * acceleration. + * + * \sa SDL_HapticCondition + */ +#define SDL_HAPTIC_INERTIA (1u<<9) + +/** + * \brief Friction effect supported - uses axes movement. + * + * Condition haptic effect that simulates friction. Effect is based on the + * axes movement. + * + * \sa SDL_HapticCondition + */ +#define SDL_HAPTIC_FRICTION (1u<<10) + +/** + * \brief Custom effect is supported. + * + * User defined custom haptic effect. + */ +#define SDL_HAPTIC_CUSTOM (1u<<11) + +/* @} *//* Haptic effects */ + +/* These last few are features the device has, not effects */ + +/** + * \brief Device can set global gain. + * + * Device supports setting the global gain. + * + * \sa SDL_HapticSetGain + */ +#define SDL_HAPTIC_GAIN (1u<<12) + +/** + * \brief Device can set autocenter. + * + * Device supports setting autocenter. + * + * \sa SDL_HapticSetAutocenter + */ +#define SDL_HAPTIC_AUTOCENTER (1u<<13) + +/** + * \brief Device can be queried for effect status. + * + * Device supports querying effect status. + * + * \sa SDL_HapticGetEffectStatus + */ +#define SDL_HAPTIC_STATUS (1u<<14) + +/** + * \brief Device can be paused. + * + * Devices supports being paused. + * + * \sa SDL_HapticPause + * \sa SDL_HapticUnpause + */ +#define SDL_HAPTIC_PAUSE (1u<<15) + + +/** + * \name Direction encodings + */ +/* @{ */ + +/** + * \brief Uses polar coordinates for the direction. + * + * \sa SDL_HapticDirection + */ +#define SDL_HAPTIC_POLAR 0 + +/** + * \brief Uses cartesian coordinates for the direction. + * + * \sa SDL_HapticDirection + */ +#define SDL_HAPTIC_CARTESIAN 1 + +/** + * \brief Uses spherical coordinates for the direction. + * + * \sa SDL_HapticDirection + */ +#define SDL_HAPTIC_SPHERICAL 2 + +/** + * \brief Use this value to play an effect on the steering wheel axis. This + * provides better compatibility across platforms and devices as SDL will guess + * the correct axis. + * \sa SDL_HapticDirection + */ +#define SDL_HAPTIC_STEERING_AXIS 3 + +/* @} *//* Direction encodings */ + +/* @} *//* Haptic features */ + +/* + * Misc defines. + */ + +/** + * \brief Used to play a device an infinite number of times. + * + * \sa SDL_HapticRunEffect + */ +#define SDL_HAPTIC_INFINITY 4294967295U + + +/** + * \brief Structure that represents a haptic direction. + * + * This is the direction where the force comes from, + * instead of the direction in which the force is exerted. + * + * Directions can be specified by: + * - ::SDL_HAPTIC_POLAR : Specified by polar coordinates. + * - ::SDL_HAPTIC_CARTESIAN : Specified by cartesian coordinates. + * - ::SDL_HAPTIC_SPHERICAL : Specified by spherical coordinates. + * + * Cardinal directions of the haptic device are relative to the positioning + * of the device. North is considered to be away from the user. + * + * The following diagram represents the cardinal directions: + * \verbatim + .--. + |__| .-------. + |=.| |.-----.| + |--| || || + | | |'-----'| + |__|~')_____(' + [ COMPUTER ] + + + North (0,-1) + ^ + | + | + (-1,0) West <----[ HAPTIC ]----> East (1,0) + | + | + v + South (0,1) + + + [ USER ] + \|||/ + (o o) + ---ooO-(_)-Ooo--- + \endverbatim + * + * If type is ::SDL_HAPTIC_POLAR, direction is encoded by hundredths of a + * degree starting north and turning clockwise. ::SDL_HAPTIC_POLAR only uses + * the first \c dir parameter. The cardinal directions would be: + * - North: 0 (0 degrees) + * - East: 9000 (90 degrees) + * - South: 18000 (180 degrees) + * - West: 27000 (270 degrees) + * + * If type is ::SDL_HAPTIC_CARTESIAN, direction is encoded by three positions + * (X axis, Y axis and Z axis (with 3 axes)). ::SDL_HAPTIC_CARTESIAN uses + * the first three \c dir parameters. The cardinal directions would be: + * - North: 0,-1, 0 + * - East: 1, 0, 0 + * - South: 0, 1, 0 + * - West: -1, 0, 0 + * + * The Z axis represents the height of the effect if supported, otherwise + * it's unused. In cartesian encoding (1, 2) would be the same as (2, 4), you + * can use any multiple you want, only the direction matters. + * + * If type is ::SDL_HAPTIC_SPHERICAL, direction is encoded by two rotations. + * The first two \c dir parameters are used. The \c dir parameters are as + * follows (all values are in hundredths of degrees): + * - Degrees from (1, 0) rotated towards (0, 1). + * - Degrees towards (0, 0, 1) (device needs at least 3 axes). + * + * + * Example of force coming from the south with all encodings (force coming + * from the south means the user will have to pull the stick to counteract): + * \code + * SDL_HapticDirection direction; + * + * // Cartesian directions + * direction.type = SDL_HAPTIC_CARTESIAN; // Using cartesian direction encoding. + * direction.dir[0] = 0; // X position + * direction.dir[1] = 1; // Y position + * // Assuming the device has 2 axes, we don't need to specify third parameter. + * + * // Polar directions + * direction.type = SDL_HAPTIC_POLAR; // We'll be using polar direction encoding. + * direction.dir[0] = 18000; // Polar only uses first parameter + * + * // Spherical coordinates + * direction.type = SDL_HAPTIC_SPHERICAL; // Spherical encoding + * direction.dir[0] = 9000; // Since we only have two axes we don't need more parameters. + * \endcode + * + * \sa SDL_HAPTIC_POLAR + * \sa SDL_HAPTIC_CARTESIAN + * \sa SDL_HAPTIC_SPHERICAL + * \sa SDL_HAPTIC_STEERING_AXIS + * \sa SDL_HapticEffect + * \sa SDL_HapticNumAxes + */ +typedef struct SDL_HapticDirection +{ + Uint8 type; /**< The type of encoding. */ + Sint32 dir[3]; /**< The encoded direction. */ +} SDL_HapticDirection; + + +/** + * \brief A structure containing a template for a Constant effect. + * + * This struct is exclusively for the ::SDL_HAPTIC_CONSTANT effect. + * + * A constant effect applies a constant force in the specified direction + * to the joystick. + * + * \sa SDL_HAPTIC_CONSTANT + * \sa SDL_HapticEffect + */ +typedef struct SDL_HapticConstant +{ + /* Header */ + Uint16 type; /**< ::SDL_HAPTIC_CONSTANT */ + SDL_HapticDirection direction; /**< Direction of the effect. */ + + /* Replay */ + Uint32 length; /**< Duration of the effect. */ + Uint16 delay; /**< Delay before starting the effect. */ + + /* Trigger */ + Uint16 button; /**< Button that triggers the effect. */ + Uint16 interval; /**< How soon it can be triggered again after button. */ + + /* Constant */ + Sint16 level; /**< Strength of the constant effect. */ + + /* Envelope */ + Uint16 attack_length; /**< Duration of the attack. */ + Uint16 attack_level; /**< Level at the start of the attack. */ + Uint16 fade_length; /**< Duration of the fade. */ + Uint16 fade_level; /**< Level at the end of the fade. */ +} SDL_HapticConstant; + +/** + * \brief A structure containing a template for a Periodic effect. + * + * The struct handles the following effects: + * - ::SDL_HAPTIC_SINE + * - ::SDL_HAPTIC_LEFTRIGHT + * - ::SDL_HAPTIC_TRIANGLE + * - ::SDL_HAPTIC_SAWTOOTHUP + * - ::SDL_HAPTIC_SAWTOOTHDOWN + * + * A periodic effect consists in a wave-shaped effect that repeats itself + * over time. The type determines the shape of the wave and the parameters + * determine the dimensions of the wave. + * + * Phase is given by hundredth of a degree meaning that giving the phase a value + * of 9000 will displace it 25% of its period. Here are sample values: + * - 0: No phase displacement. + * - 9000: Displaced 25% of its period. + * - 18000: Displaced 50% of its period. + * - 27000: Displaced 75% of its period. + * - 36000: Displaced 100% of its period, same as 0, but 0 is preferred. + * + * Examples: + * \verbatim + SDL_HAPTIC_SINE + __ __ __ __ + / \ / \ / \ / + / \__/ \__/ \__/ + + SDL_HAPTIC_SQUARE + __ __ __ __ __ + | | | | | | | | | | + | |__| |__| |__| |__| | + + SDL_HAPTIC_TRIANGLE + /\ /\ /\ /\ /\ + / \ / \ / \ / \ / + / \/ \/ \/ \/ + + SDL_HAPTIC_SAWTOOTHUP + /| /| /| /| /| /| /| + / | / | / | / | / | / | / | + / |/ |/ |/ |/ |/ |/ | + + SDL_HAPTIC_SAWTOOTHDOWN + \ |\ |\ |\ |\ |\ |\ | + \ | \ | \ | \ | \ | \ | \ | + \| \| \| \| \| \| \| + \endverbatim + * + * \sa SDL_HAPTIC_SINE + * \sa SDL_HAPTIC_LEFTRIGHT + * \sa SDL_HAPTIC_TRIANGLE + * \sa SDL_HAPTIC_SAWTOOTHUP + * \sa SDL_HAPTIC_SAWTOOTHDOWN + * \sa SDL_HapticEffect + */ +typedef struct SDL_HapticPeriodic +{ + /* Header */ + Uint16 type; /**< ::SDL_HAPTIC_SINE, ::SDL_HAPTIC_LEFTRIGHT, + ::SDL_HAPTIC_TRIANGLE, ::SDL_HAPTIC_SAWTOOTHUP or + ::SDL_HAPTIC_SAWTOOTHDOWN */ + SDL_HapticDirection direction; /**< Direction of the effect. */ + + /* Replay */ + Uint32 length; /**< Duration of the effect. */ + Uint16 delay; /**< Delay before starting the effect. */ + + /* Trigger */ + Uint16 button; /**< Button that triggers the effect. */ + Uint16 interval; /**< How soon it can be triggered again after button. */ + + /* Periodic */ + Uint16 period; /**< Period of the wave. */ + Sint16 magnitude; /**< Peak value; if negative, equivalent to 180 degrees extra phase shift. */ + Sint16 offset; /**< Mean value of the wave. */ + Uint16 phase; /**< Positive phase shift given by hundredth of a degree. */ + + /* Envelope */ + Uint16 attack_length; /**< Duration of the attack. */ + Uint16 attack_level; /**< Level at the start of the attack. */ + Uint16 fade_length; /**< Duration of the fade. */ + Uint16 fade_level; /**< Level at the end of the fade. */ +} SDL_HapticPeriodic; + +/** + * \brief A structure containing a template for a Condition effect. + * + * The struct handles the following effects: + * - ::SDL_HAPTIC_SPRING: Effect based on axes position. + * - ::SDL_HAPTIC_DAMPER: Effect based on axes velocity. + * - ::SDL_HAPTIC_INERTIA: Effect based on axes acceleration. + * - ::SDL_HAPTIC_FRICTION: Effect based on axes movement. + * + * Direction is handled by condition internals instead of a direction member. + * The condition effect specific members have three parameters. The first + * refers to the X axis, the second refers to the Y axis and the third + * refers to the Z axis. The right terms refer to the positive side of the + * axis and the left terms refer to the negative side of the axis. Please + * refer to the ::SDL_HapticDirection diagram for which side is positive and + * which is negative. + * + * \sa SDL_HapticDirection + * \sa SDL_HAPTIC_SPRING + * \sa SDL_HAPTIC_DAMPER + * \sa SDL_HAPTIC_INERTIA + * \sa SDL_HAPTIC_FRICTION + * \sa SDL_HapticEffect + */ +typedef struct SDL_HapticCondition +{ + /* Header */ + Uint16 type; /**< ::SDL_HAPTIC_SPRING, ::SDL_HAPTIC_DAMPER, + ::SDL_HAPTIC_INERTIA or ::SDL_HAPTIC_FRICTION */ + SDL_HapticDirection direction; /**< Direction of the effect - Not used ATM. */ + + /* Replay */ + Uint32 length; /**< Duration of the effect. */ + Uint16 delay; /**< Delay before starting the effect. */ + + /* Trigger */ + Uint16 button; /**< Button that triggers the effect. */ + Uint16 interval; /**< How soon it can be triggered again after button. */ + + /* Condition */ + Uint16 right_sat[3]; /**< Level when joystick is to the positive side; max 0xFFFF. */ + Uint16 left_sat[3]; /**< Level when joystick is to the negative side; max 0xFFFF. */ + Sint16 right_coeff[3]; /**< How fast to increase the force towards the positive side. */ + Sint16 left_coeff[3]; /**< How fast to increase the force towards the negative side. */ + Uint16 deadband[3]; /**< Size of the dead zone; max 0xFFFF: whole axis-range when 0-centered. */ + Sint16 center[3]; /**< Position of the dead zone. */ +} SDL_HapticCondition; + +/** + * \brief A structure containing a template for a Ramp effect. + * + * This struct is exclusively for the ::SDL_HAPTIC_RAMP effect. + * + * The ramp effect starts at start strength and ends at end strength. + * It augments in linear fashion. If you use attack and fade with a ramp + * the effects get added to the ramp effect making the effect become + * quadratic instead of linear. + * + * \sa SDL_HAPTIC_RAMP + * \sa SDL_HapticEffect + */ +typedef struct SDL_HapticRamp +{ + /* Header */ + Uint16 type; /**< ::SDL_HAPTIC_RAMP */ + SDL_HapticDirection direction; /**< Direction of the effect. */ + + /* Replay */ + Uint32 length; /**< Duration of the effect. */ + Uint16 delay; /**< Delay before starting the effect. */ + + /* Trigger */ + Uint16 button; /**< Button that triggers the effect. */ + Uint16 interval; /**< How soon it can be triggered again after button. */ + + /* Ramp */ + Sint16 start; /**< Beginning strength level. */ + Sint16 end; /**< Ending strength level. */ + + /* Envelope */ + Uint16 attack_length; /**< Duration of the attack. */ + Uint16 attack_level; /**< Level at the start of the attack. */ + Uint16 fade_length; /**< Duration of the fade. */ + Uint16 fade_level; /**< Level at the end of the fade. */ +} SDL_HapticRamp; + +/** + * \brief A structure containing a template for a Left/Right effect. + * + * This struct is exclusively for the ::SDL_HAPTIC_LEFTRIGHT effect. + * + * The Left/Right effect is used to explicitly control the large and small + * motors, commonly found in modern game controllers. The small (right) motor + * is high frequency, and the large (left) motor is low frequency. + * + * \sa SDL_HAPTIC_LEFTRIGHT + * \sa SDL_HapticEffect + */ +typedef struct SDL_HapticLeftRight +{ + /* Header */ + Uint16 type; /**< ::SDL_HAPTIC_LEFTRIGHT */ + + /* Replay */ + Uint32 length; /**< Duration of the effect in milliseconds. */ + + /* Rumble */ + Uint16 large_magnitude; /**< Control of the large controller motor. */ + Uint16 small_magnitude; /**< Control of the small controller motor. */ +} SDL_HapticLeftRight; + +/** + * \brief A structure containing a template for the ::SDL_HAPTIC_CUSTOM effect. + * + * This struct is exclusively for the ::SDL_HAPTIC_CUSTOM effect. + * + * A custom force feedback effect is much like a periodic effect, where the + * application can define its exact shape. You will have to allocate the + * data yourself. Data should consist of channels * samples Uint16 samples. + * + * If channels is one, the effect is rotated using the defined direction. + * Otherwise it uses the samples in data for the different axes. + * + * \sa SDL_HAPTIC_CUSTOM + * \sa SDL_HapticEffect + */ +typedef struct SDL_HapticCustom +{ + /* Header */ + Uint16 type; /**< ::SDL_HAPTIC_CUSTOM */ + SDL_HapticDirection direction; /**< Direction of the effect. */ + + /* Replay */ + Uint32 length; /**< Duration of the effect. */ + Uint16 delay; /**< Delay before starting the effect. */ + + /* Trigger */ + Uint16 button; /**< Button that triggers the effect. */ + Uint16 interval; /**< How soon it can be triggered again after button. */ + + /* Custom */ + Uint8 channels; /**< Axes to use, minimum of one. */ + Uint16 period; /**< Sample periods. */ + Uint16 samples; /**< Amount of samples. */ + Uint16 *data; /**< Should contain channels*samples items. */ + + /* Envelope */ + Uint16 attack_length; /**< Duration of the attack. */ + Uint16 attack_level; /**< Level at the start of the attack. */ + Uint16 fade_length; /**< Duration of the fade. */ + Uint16 fade_level; /**< Level at the end of the fade. */ +} SDL_HapticCustom; + +/** + * \brief The generic template for any haptic effect. + * + * All values max at 32767 (0x7FFF). Signed values also can be negative. + * Time values unless specified otherwise are in milliseconds. + * + * You can also pass ::SDL_HAPTIC_INFINITY to length instead of a 0-32767 + * value. Neither delay, interval, attack_length nor fade_length support + * ::SDL_HAPTIC_INFINITY. Fade will also not be used since effect never ends. + * + * Additionally, the ::SDL_HAPTIC_RAMP effect does not support a duration of + * ::SDL_HAPTIC_INFINITY. + * + * Button triggers may not be supported on all devices, it is advised to not + * use them if possible. Buttons start at index 1 instead of index 0 like + * the joystick. + * + * If both attack_length and fade_level are 0, the envelope is not used, + * otherwise both values are used. + * + * Common parts: + * \code + * // Replay - All effects have this + * Uint32 length; // Duration of effect (ms). + * Uint16 delay; // Delay before starting effect. + * + * // Trigger - All effects have this + * Uint16 button; // Button that triggers effect. + * Uint16 interval; // How soon before effect can be triggered again. + * + * // Envelope - All effects except condition effects have this + * Uint16 attack_length; // Duration of the attack (ms). + * Uint16 attack_level; // Level at the start of the attack. + * Uint16 fade_length; // Duration of the fade out (ms). + * Uint16 fade_level; // Level at the end of the fade. + * \endcode + * + * + * Here we have an example of a constant effect evolution in time: + * \verbatim + Strength + ^ + | + | effect level --> _________________ + | / \ + | / \ + | / \ + | / \ + | attack_level --> | \ + | | | <--- fade_level + | + +--------------------------------------------------> Time + [--] [---] + attack_length fade_length + + [------------------][-----------------------] + delay length + \endverbatim + * + * Note either the attack_level or the fade_level may be above the actual + * effect level. + * + * \sa SDL_HapticConstant + * \sa SDL_HapticPeriodic + * \sa SDL_HapticCondition + * \sa SDL_HapticRamp + * \sa SDL_HapticLeftRight + * \sa SDL_HapticCustom + */ +typedef union SDL_HapticEffect +{ + /* Common for all force feedback effects */ + Uint16 type; /**< Effect type. */ + SDL_HapticConstant constant; /**< Constant effect. */ + SDL_HapticPeriodic periodic; /**< Periodic effect. */ + SDL_HapticCondition condition; /**< Condition effect. */ + SDL_HapticRamp ramp; /**< Ramp effect. */ + SDL_HapticLeftRight leftright; /**< Left/Right effect. */ + SDL_HapticCustom custom; /**< Custom effect. */ +} SDL_HapticEffect; + + +/* Function prototypes */ + +/** + * Count the number of haptic devices attached to the system. + * + * \returns the number of haptic devices detected on the system or a negative + * error code on failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticName + */ +extern DECLSPEC int SDLCALL SDL_NumHaptics(void); + +/** + * Get the implementation dependent name of a haptic device. + * + * This can be called before any joysticks are opened. If no name can be + * found, this function returns NULL. + * + * \param device_index index of the device to query. + * \returns the name of the device or NULL on failure; call SDL_GetError() for + * more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_NumHaptics + */ +extern DECLSPEC const char *SDLCALL SDL_HapticName(int device_index); + +/** + * Open a haptic device for use. + * + * The index passed as an argument refers to the N'th haptic device on this + * system. + * + * When opening a haptic device, its gain will be set to maximum and + * autocenter will be disabled. To modify these values use SDL_HapticSetGain() + * and SDL_HapticSetAutocenter(). + * + * \param device_index index of the device to open + * \returns the device identifier or NULL on failure; call SDL_GetError() for + * more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticClose + * \sa SDL_HapticIndex + * \sa SDL_HapticOpenFromJoystick + * \sa SDL_HapticOpenFromMouse + * \sa SDL_HapticPause + * \sa SDL_HapticSetAutocenter + * \sa SDL_HapticSetGain + * \sa SDL_HapticStopAll + */ +extern DECLSPEC SDL_Haptic *SDLCALL SDL_HapticOpen(int device_index); + +/** + * Check if the haptic device at the designated index has been opened. + * + * \param device_index the index of the device to query + * \returns 1 if it has been opened, 0 if it hasn't or on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticIndex + * \sa SDL_HapticOpen + */ +extern DECLSPEC int SDLCALL SDL_HapticOpened(int device_index); + +/** + * Get the index of a haptic device. + * + * \param haptic the SDL_Haptic device to query + * \returns the index of the specified haptic device or a negative error code + * on failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticOpen + * \sa SDL_HapticOpened + */ +extern DECLSPEC int SDLCALL SDL_HapticIndex(SDL_Haptic * haptic); + +/** + * Query whether or not the current mouse has haptic capabilities. + * + * \returns SDL_TRUE if the mouse is haptic or SDL_FALSE if it isn't. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticOpenFromMouse + */ +extern DECLSPEC int SDLCALL SDL_MouseIsHaptic(void); + +/** + * Try to open a haptic device from the current mouse. + * + * \returns the haptic device identifier or NULL on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticOpen + * \sa SDL_MouseIsHaptic + */ +extern DECLSPEC SDL_Haptic *SDLCALL SDL_HapticOpenFromMouse(void); + +/** + * Query if a joystick has haptic features. + * + * \param joystick the SDL_Joystick to test for haptic capabilities + * \returns SDL_TRUE if the joystick is haptic, SDL_FALSE if it isn't, or a + * negative error code on failure; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticOpenFromJoystick + */ +extern DECLSPEC int SDLCALL SDL_JoystickIsHaptic(SDL_Joystick * joystick); + +/** + * Open a haptic device for use from a joystick device. + * + * You must still close the haptic device separately. It will not be closed + * with the joystick. + * + * When opened from a joystick you should first close the haptic device before + * closing the joystick device. If not, on some implementations the haptic + * device will also get unallocated and you'll be unable to use force feedback + * on that device. + * + * \param joystick the SDL_Joystick to create a haptic device from + * \returns a valid haptic device identifier on success or NULL on failure; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticClose + * \sa SDL_HapticOpen + * \sa SDL_JoystickIsHaptic + */ +extern DECLSPEC SDL_Haptic *SDLCALL SDL_HapticOpenFromJoystick(SDL_Joystick * + joystick); + +/** + * Close a haptic device previously opened with SDL_HapticOpen(). + * + * \param haptic the SDL_Haptic device to close + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticOpen + */ +extern DECLSPEC void SDLCALL SDL_HapticClose(SDL_Haptic * haptic); + +/** + * Get the number of effects a haptic device can store. + * + * On some platforms this isn't fully supported, and therefore is an + * approximation. Always check to see if your created effect was actually + * created and do not rely solely on SDL_HapticNumEffects(). + * + * \param haptic the SDL_Haptic device to query + * \returns the number of effects the haptic device can store or a negative + * error code on failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticNumEffectsPlaying + * \sa SDL_HapticQuery + */ +extern DECLSPEC int SDLCALL SDL_HapticNumEffects(SDL_Haptic * haptic); + +/** + * Get the number of effects a haptic device can play at the same time. + * + * This is not supported on all platforms, but will always return a value. + * + * \param haptic the SDL_Haptic device to query maximum playing effects + * \returns the number of effects the haptic device can play at the same time + * or a negative error code on failure; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticNumEffects + * \sa SDL_HapticQuery + */ +extern DECLSPEC int SDLCALL SDL_HapticNumEffectsPlaying(SDL_Haptic * haptic); + +/** + * Get the haptic device's supported features in bitwise manner. + * + * \param haptic the SDL_Haptic device to query + * \returns a list of supported haptic features in bitwise manner (OR'd), or 0 + * on failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticEffectSupported + * \sa SDL_HapticNumEffects + */ +extern DECLSPEC unsigned int SDLCALL SDL_HapticQuery(SDL_Haptic * haptic); + + +/** + * Get the number of haptic axes the device has. + * + * The number of haptic axes might be useful if working with the + * SDL_HapticDirection effect. + * + * \param haptic the SDL_Haptic device to query + * \returns the number of axes on success or a negative error code on failure; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC int SDLCALL SDL_HapticNumAxes(SDL_Haptic * haptic); + +/** + * Check to see if an effect is supported by a haptic device. + * + * \param haptic the SDL_Haptic device to query + * \param effect the desired effect to query + * \returns SDL_TRUE if effect is supported, SDL_FALSE if it isn't, or a + * negative error code on failure; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticNewEffect + * \sa SDL_HapticQuery + */ +extern DECLSPEC int SDLCALL SDL_HapticEffectSupported(SDL_Haptic * haptic, + SDL_HapticEffect * + effect); + +/** + * Create a new haptic effect on a specified device. + * + * \param haptic an SDL_Haptic device to create the effect on + * \param effect an SDL_HapticEffect structure containing the properties of + * the effect to create + * \returns the ID of the effect on success or a negative error code on + * failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticDestroyEffect + * \sa SDL_HapticRunEffect + * \sa SDL_HapticUpdateEffect + */ +extern DECLSPEC int SDLCALL SDL_HapticNewEffect(SDL_Haptic * haptic, + SDL_HapticEffect * effect); + +/** + * Update the properties of an effect. + * + * Can be used dynamically, although behavior when dynamically changing + * direction may be strange. Specifically the effect may re-upload itself and + * start playing from the start. You also cannot change the type either when + * running SDL_HapticUpdateEffect(). + * + * \param haptic the SDL_Haptic device that has the effect + * \param effect the identifier of the effect to update + * \param data an SDL_HapticEffect structure containing the new effect + * properties to use + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticDestroyEffect + * \sa SDL_HapticNewEffect + * \sa SDL_HapticRunEffect + */ +extern DECLSPEC int SDLCALL SDL_HapticUpdateEffect(SDL_Haptic * haptic, + int effect, + SDL_HapticEffect * data); + +/** + * Run the haptic effect on its associated haptic device. + * + * To repeat the effect over and over indefinitely, set `iterations` to + * `SDL_HAPTIC_INFINITY`. (Repeats the envelope - attack and fade.) To make + * one instance of the effect last indefinitely (so the effect does not fade), + * set the effect's `length` in its structure/union to `SDL_HAPTIC_INFINITY` + * instead. + * + * \param haptic the SDL_Haptic device to run the effect on + * \param effect the ID of the haptic effect to run + * \param iterations the number of iterations to run the effect; use + * `SDL_HAPTIC_INFINITY` to repeat forever + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticDestroyEffect + * \sa SDL_HapticGetEffectStatus + * \sa SDL_HapticStopEffect + */ +extern DECLSPEC int SDLCALL SDL_HapticRunEffect(SDL_Haptic * haptic, + int effect, + Uint32 iterations); + +/** + * Stop the haptic effect on its associated haptic device. + * + * * + * + * \param haptic the SDL_Haptic device to stop the effect on + * \param effect the ID of the haptic effect to stop + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticDestroyEffect + * \sa SDL_HapticRunEffect + */ +extern DECLSPEC int SDLCALL SDL_HapticStopEffect(SDL_Haptic * haptic, + int effect); + +/** + * Destroy a haptic effect on the device. + * + * This will stop the effect if it's running. Effects are automatically + * destroyed when the device is closed. + * + * \param haptic the SDL_Haptic device to destroy the effect on + * \param effect the ID of the haptic effect to destroy + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticNewEffect + */ +extern DECLSPEC void SDLCALL SDL_HapticDestroyEffect(SDL_Haptic * haptic, + int effect); + +/** + * Get the status of the current effect on the specified haptic device. + * + * Device must support the SDL_HAPTIC_STATUS feature. + * + * \param haptic the SDL_Haptic device to query for the effect status on + * \param effect the ID of the haptic effect to query its status + * \returns 0 if it isn't playing, 1 if it is playing, or a negative error + * code on failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticRunEffect + * \sa SDL_HapticStopEffect + */ +extern DECLSPEC int SDLCALL SDL_HapticGetEffectStatus(SDL_Haptic * haptic, + int effect); + +/** + * Set the global gain of the specified haptic device. + * + * Device must support the SDL_HAPTIC_GAIN feature. + * + * The user may specify the maximum gain by setting the environment variable + * `SDL_HAPTIC_GAIN_MAX` which should be between 0 and 100. All calls to + * SDL_HapticSetGain() will scale linearly using `SDL_HAPTIC_GAIN_MAX` as the + * maximum. + * + * \param haptic the SDL_Haptic device to set the gain on + * \param gain value to set the gain to, should be between 0 and 100 (0 - 100) + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticQuery + */ +extern DECLSPEC int SDLCALL SDL_HapticSetGain(SDL_Haptic * haptic, int gain); + +/** + * Set the global autocenter of the device. + * + * Autocenter should be between 0 and 100. Setting it to 0 will disable + * autocentering. + * + * Device must support the SDL_HAPTIC_AUTOCENTER feature. + * + * \param haptic the SDL_Haptic device to set autocentering on + * \param autocenter value to set autocenter to (0-100) + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticQuery + */ +extern DECLSPEC int SDLCALL SDL_HapticSetAutocenter(SDL_Haptic * haptic, + int autocenter); + +/** + * Pause a haptic device. + * + * Device must support the `SDL_HAPTIC_PAUSE` feature. Call + * SDL_HapticUnpause() to resume playback. + * + * Do not modify the effects nor add new ones while the device is paused. That + * can cause all sorts of weird errors. + * + * \param haptic the SDL_Haptic device to pause + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticUnpause + */ +extern DECLSPEC int SDLCALL SDL_HapticPause(SDL_Haptic * haptic); + +/** + * Unpause a haptic device. + * + * Call to unpause after SDL_HapticPause(). + * + * \param haptic the SDL_Haptic device to unpause + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticPause + */ +extern DECLSPEC int SDLCALL SDL_HapticUnpause(SDL_Haptic * haptic); + +/** + * Stop all the currently playing effects on a haptic device. + * + * \param haptic the SDL_Haptic device to stop + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC int SDLCALL SDL_HapticStopAll(SDL_Haptic * haptic); + +/** + * Check whether rumble is supported on a haptic device. + * + * \param haptic haptic device to check for rumble support + * \returns SDL_TRUE if effect is supported, SDL_FALSE if it isn't, or a + * negative error code on failure; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticRumbleInit + * \sa SDL_HapticRumblePlay + * \sa SDL_HapticRumbleStop + */ +extern DECLSPEC int SDLCALL SDL_HapticRumbleSupported(SDL_Haptic * haptic); + +/** + * Initialize a haptic device for simple rumble playback. + * + * \param haptic the haptic device to initialize for simple rumble playback + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticOpen + * \sa SDL_HapticRumblePlay + * \sa SDL_HapticRumbleStop + * \sa SDL_HapticRumbleSupported + */ +extern DECLSPEC int SDLCALL SDL_HapticRumbleInit(SDL_Haptic * haptic); + +/** + * Run a simple rumble effect on a haptic device. + * + * \param haptic the haptic device to play the rumble effect on + * \param strength strength of the rumble to play as a 0-1 float value + * \param length length of the rumble to play in milliseconds + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticRumbleInit + * \sa SDL_HapticRumbleStop + * \sa SDL_HapticRumbleSupported + */ +extern DECLSPEC int SDLCALL SDL_HapticRumblePlay(SDL_Haptic * haptic, float strength, Uint32 length ); + +/** + * Stop the simple rumble on a haptic device. + * + * \param haptic the haptic device to stop the rumble effect on + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HapticRumbleInit + * \sa SDL_HapticRumblePlay + * \sa SDL_HapticRumbleSupported + */ +extern DECLSPEC int SDLCALL SDL_HapticRumbleStop(SDL_Haptic * haptic); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_haptic_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_hidapi.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_hidapi.h new file mode 100644 index 00000000..05751003 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_hidapi.h @@ -0,0 +1,451 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_hidapi.h + * + * Header file for SDL HIDAPI functions. + * + * This is an adaptation of the original HIDAPI interface by Alan Ott, + * and includes source code licensed under the following BSD license: + * + Copyright (c) 2010, Alan Ott, Signal 11 Software + All rights reserved. + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are met: + + * Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + * Neither the name of Signal 11 Software nor the names of its + contributors may be used to endorse or promote products derived from + this software without specific prior written permission. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE + LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + POSSIBILITY OF SUCH DAMAGE. + * + * If you would like a version of SDL without this code, you can build SDL + * with SDL_HIDAPI_DISABLED defined to 1. You might want to do this for example + * on iOS or tvOS to avoid a dependency on the CoreBluetooth framework. + */ + +#ifndef SDL_hidapi_h_ +#define SDL_hidapi_h_ + +#include "SDL_stdinc.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \brief A handle representing an open HID device + */ +struct SDL_hid_device_; +typedef struct SDL_hid_device_ SDL_hid_device; /**< opaque hidapi structure */ + +/** hidapi info structure */ +/** + * \brief Information about a connected HID device + */ +typedef struct SDL_hid_device_info +{ + /** Platform-specific device path */ + char *path; + /** Device Vendor ID */ + unsigned short vendor_id; + /** Device Product ID */ + unsigned short product_id; + /** Serial Number */ + wchar_t *serial_number; + /** Device Release Number in binary-coded decimal, + also known as Device Version Number */ + unsigned short release_number; + /** Manufacturer String */ + wchar_t *manufacturer_string; + /** Product string */ + wchar_t *product_string; + /** Usage Page for this Device/Interface + (Windows/Mac only). */ + unsigned short usage_page; + /** Usage for this Device/Interface + (Windows/Mac only).*/ + unsigned short usage; + /** The USB interface which this logical device + represents. + + * Valid on both Linux implementations in all cases. + * Valid on the Windows implementation only if the device + contains more than one interface. */ + int interface_number; + + /** Additional information about the USB interface. + Valid on libusb and Android implementations. */ + int interface_class; + int interface_subclass; + int interface_protocol; + + /** Pointer to the next device */ + struct SDL_hid_device_info *next; +} SDL_hid_device_info; + + +/** + * Initialize the HIDAPI library. + * + * This function initializes the HIDAPI library. Calling it is not strictly + * necessary, as it will be called automatically by SDL_hid_enumerate() and + * any of the SDL_hid_open_*() functions if it is needed. This function should + * be called at the beginning of execution however, if there is a chance of + * HIDAPI handles being opened by different threads simultaneously. + * + * Each call to this function should have a matching call to SDL_hid_exit() + * + * \returns 0 on success and -1 on error. + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_hid_exit + */ +extern DECLSPEC int SDLCALL SDL_hid_init(void); + +/** + * Finalize the HIDAPI library. + * + * This function frees all of the static data associated with HIDAPI. It + * should be called at the end of execution to avoid memory leaks. + * + * \returns 0 on success and -1 on error. + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_hid_init + */ +extern DECLSPEC int SDLCALL SDL_hid_exit(void); + +/** + * Check to see if devices may have been added or removed. + * + * Enumerating the HID devices is an expensive operation, so you can call this + * to see if there have been any system device changes since the last call to + * this function. A change in the counter returned doesn't necessarily mean + * that anything has changed, but you can call SDL_hid_enumerate() to get an + * updated device list. + * + * Calling this function for the first time may cause a thread or other system + * resource to be allocated to track device change notifications. + * + * \returns a change counter that is incremented with each potential device + * change, or 0 if device change detection isn't available. + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_hid_enumerate + */ +extern DECLSPEC Uint32 SDLCALL SDL_hid_device_change_count(void); + +/** + * Enumerate the HID Devices. + * + * This function returns a linked list of all the HID devices attached to the + * system which match vendor_id and product_id. If `vendor_id` is set to 0 + * then any vendor matches. If `product_id` is set to 0 then any product + * matches. If `vendor_id` and `product_id` are both set to 0, then all HID + * devices will be returned. + * + * \param vendor_id The Vendor ID (VID) of the types of device to open. + * \param product_id The Product ID (PID) of the types of device to open. + * \returns a pointer to a linked list of type SDL_hid_device_info, containing + * information about the HID devices attached to the system, or NULL + * in the case of failure. Free this linked list by calling + * SDL_hid_free_enumeration(). + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_hid_device_change_count + */ +extern DECLSPEC SDL_hid_device_info * SDLCALL SDL_hid_enumerate(unsigned short vendor_id, unsigned short product_id); + +/** + * Free an enumeration Linked List + * + * This function frees a linked list created by SDL_hid_enumerate(). + * + * \param devs Pointer to a list of struct_device returned from + * SDL_hid_enumerate(). + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC void SDLCALL SDL_hid_free_enumeration(SDL_hid_device_info *devs); + +/** + * Open a HID device using a Vendor ID (VID), Product ID (PID) and optionally + * a serial number. + * + * If `serial_number` is NULL, the first device with the specified VID and PID + * is opened. + * + * \param vendor_id The Vendor ID (VID) of the device to open. + * \param product_id The Product ID (PID) of the device to open. + * \param serial_number The Serial Number of the device to open (Optionally + * NULL). + * \returns a pointer to a SDL_hid_device object on success or NULL on + * failure. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC SDL_hid_device * SDLCALL SDL_hid_open(unsigned short vendor_id, unsigned short product_id, const wchar_t *serial_number); + +/** + * Open a HID device by its path name. + * + * The path name be determined by calling SDL_hid_enumerate(), or a + * platform-specific path name can be used (eg: /dev/hidraw0 on Linux). + * + * \param path The path name of the device to open + * \returns a pointer to a SDL_hid_device object on success or NULL on + * failure. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC SDL_hid_device * SDLCALL SDL_hid_open_path(const char *path, int bExclusive /* = false */); + +/** + * Write an Output report to a HID device. + * + * The first byte of `data` must contain the Report ID. For devices which only + * support a single report, this must be set to 0x0. The remaining bytes + * contain the report data. Since the Report ID is mandatory, calls to + * SDL_hid_write() will always contain one more byte than the report contains. + * For example, if a hid report is 16 bytes long, 17 bytes must be passed to + * SDL_hid_write(), the Report ID (or 0x0, for devices with a single report), + * followed by the report data (16 bytes). In this example, the length passed + * in would be 17. + * + * SDL_hid_write() will send the data on the first OUT endpoint, if one + * exists. If it does not, it will send the data through the Control Endpoint + * (Endpoint 0). + * + * \param dev A device handle returned from SDL_hid_open(). + * \param data The data to send, including the report number as the first + * byte. + * \param length The length in bytes of the data to send. + * \returns the actual number of bytes written and -1 on error. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC int SDLCALL SDL_hid_write(SDL_hid_device *dev, const unsigned char *data, size_t length); + +/** + * Read an Input report from a HID device with timeout. + * + * Input reports are returned to the host through the INTERRUPT IN endpoint. + * The first byte will contain the Report number if the device uses numbered + * reports. + * + * \param dev A device handle returned from SDL_hid_open(). + * \param data A buffer to put the read data into. + * \param length The number of bytes to read. For devices with multiple + * reports, make sure to read an extra byte for the report + * number. + * \param milliseconds timeout in milliseconds or -1 for blocking wait. + * \returns the actual number of bytes read and -1 on error. If no packet was + * available to be read within the timeout period, this function + * returns 0. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC int SDLCALL SDL_hid_read_timeout(SDL_hid_device *dev, unsigned char *data, size_t length, int milliseconds); + +/** + * Read an Input report from a HID device. + * + * Input reports are returned to the host through the INTERRUPT IN endpoint. + * The first byte will contain the Report number if the device uses numbered + * reports. + * + * \param dev A device handle returned from SDL_hid_open(). + * \param data A buffer to put the read data into. + * \param length The number of bytes to read. For devices with multiple + * reports, make sure to read an extra byte for the report + * number. + * \returns the actual number of bytes read and -1 on error. If no packet was + * available to be read and the handle is in non-blocking mode, this + * function returns 0. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC int SDLCALL SDL_hid_read(SDL_hid_device *dev, unsigned char *data, size_t length); + +/** + * Set the device handle to be non-blocking. + * + * In non-blocking mode calls to SDL_hid_read() will return immediately with a + * value of 0 if there is no data to be read. In blocking mode, SDL_hid_read() + * will wait (block) until there is data to read before returning. + * + * Nonblocking can be turned on and off at any time. + * + * \param dev A device handle returned from SDL_hid_open(). + * \param nonblock enable or not the nonblocking reads - 1 to enable + * nonblocking - 0 to disable nonblocking. + * \returns 0 on success and -1 on error. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC int SDLCALL SDL_hid_set_nonblocking(SDL_hid_device *dev, int nonblock); + +/** + * Send a Feature report to the device. + * + * Feature reports are sent over the Control endpoint as a Set_Report + * transfer. The first byte of `data` must contain the Report ID. For devices + * which only support a single report, this must be set to 0x0. The remaining + * bytes contain the report data. Since the Report ID is mandatory, calls to + * SDL_hid_send_feature_report() will always contain one more byte than the + * report contains. For example, if a hid report is 16 bytes long, 17 bytes + * must be passed to SDL_hid_send_feature_report(): the Report ID (or 0x0, for + * devices which do not use numbered reports), followed by the report data (16 + * bytes). In this example, the length passed in would be 17. + * + * \param dev A device handle returned from SDL_hid_open(). + * \param data The data to send, including the report number as the first + * byte. + * \param length The length in bytes of the data to send, including the report + * number. + * \returns the actual number of bytes written and -1 on error. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC int SDLCALL SDL_hid_send_feature_report(SDL_hid_device *dev, const unsigned char *data, size_t length); + +/** + * Get a feature report from a HID device. + * + * Set the first byte of `data` to the Report ID of the report to be read. + * Make sure to allow space for this extra byte in `data`. Upon return, the + * first byte will still contain the Report ID, and the report data will start + * in data[1]. + * + * \param dev A device handle returned from SDL_hid_open(). + * \param data A buffer to put the read data into, including the Report ID. + * Set the first byte of `data` to the Report ID of the report to + * be read, or set it to zero if your device does not use numbered + * reports. + * \param length The number of bytes to read, including an extra byte for the + * report ID. The buffer can be longer than the actual report. + * \returns the number of bytes read plus one for the report ID (which is + * still in the first byte), or -1 on error. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC int SDLCALL SDL_hid_get_feature_report(SDL_hid_device *dev, unsigned char *data, size_t length); + +/** + * Close a HID device. + * + * \param dev A device handle returned from SDL_hid_open(). + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC void SDLCALL SDL_hid_close(SDL_hid_device *dev); + +/** + * Get The Manufacturer String from a HID device. + * + * \param dev A device handle returned from SDL_hid_open(). + * \param string A wide string buffer to put the data into. + * \param maxlen The length of the buffer in multiples of wchar_t. + * \returns 0 on success and -1 on error. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC int SDLCALL SDL_hid_get_manufacturer_string(SDL_hid_device *dev, wchar_t *string, size_t maxlen); + +/** + * Get The Product String from a HID device. + * + * \param dev A device handle returned from SDL_hid_open(). + * \param string A wide string buffer to put the data into. + * \param maxlen The length of the buffer in multiples of wchar_t. + * \returns 0 on success and -1 on error. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC int SDLCALL SDL_hid_get_product_string(SDL_hid_device *dev, wchar_t *string, size_t maxlen); + +/** + * Get The Serial Number String from a HID device. + * + * \param dev A device handle returned from SDL_hid_open(). + * \param string A wide string buffer to put the data into. + * \param maxlen The length of the buffer in multiples of wchar_t. + * \returns 0 on success and -1 on error. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC int SDLCALL SDL_hid_get_serial_number_string(SDL_hid_device *dev, wchar_t *string, size_t maxlen); + +/** + * Get a string from a HID device, based on its string index. + * + * \param dev A device handle returned from SDL_hid_open(). + * \param string_index The index of the string to get. + * \param string A wide string buffer to put the data into. + * \param maxlen The length of the buffer in multiples of wchar_t. + * \returns 0 on success and -1 on error. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC int SDLCALL SDL_hid_get_indexed_string(SDL_hid_device *dev, int string_index, wchar_t *string, size_t maxlen); + +/** + * Start or stop a BLE scan on iOS and tvOS to pair Steam Controllers + * + * \param active SDL_TRUE to start the scan, SDL_FALSE to stop the scan + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC void SDLCALL SDL_hid_ble_scan(SDL_bool active); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_hidapi_h_ */ + +/* vi: set sts=4 ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_hints.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_hints.h new file mode 100644 index 00000000..00beef51 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_hints.h @@ -0,0 +1,2624 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_hints.h + * + * Official documentation for SDL configuration variables + * + * This file contains functions to set and get configuration hints, + * as well as listing each of them alphabetically. + * + * The convention for naming hints is SDL_HINT_X, where "SDL_X" is + * the environment variable that can be used to override the default. + * + * In general these hints are just that - they may or may not be + * supported or applicable on any given platform, but they provide + * a way for an application or user to give the library a hint as + * to how they would like the library to work. + */ + +#ifndef SDL_hints_h_ +#define SDL_hints_h_ + +#include "SDL_stdinc.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \brief A variable controlling whether the Android / iOS built-in + * accelerometer should be listed as a joystick device. + * + * This variable can be set to the following values: + * "0" - The accelerometer is not listed as a joystick + * "1" - The accelerometer is available as a 3 axis joystick (the default). + */ +#define SDL_HINT_ACCELEROMETER_AS_JOYSTICK "SDL_ACCELEROMETER_AS_JOYSTICK" + +/** + * \brief Specify the behavior of Alt+Tab while the keyboard is grabbed. + * + * By default, SDL emulates Alt+Tab functionality while the keyboard is grabbed + * and your window is full-screen. This prevents the user from getting stuck in + * your application if you've enabled keyboard grab. + * + * The variable can be set to the following values: + * "0" - SDL will not handle Alt+Tab. Your application is responsible + for handling Alt+Tab while the keyboard is grabbed. + * "1" - SDL will minimize your window when Alt+Tab is pressed (default) +*/ +#define SDL_HINT_ALLOW_ALT_TAB_WHILE_GRABBED "SDL_ALLOW_ALT_TAB_WHILE_GRABBED" + +/** + * \brief If set to "0" then never set the top most bit on a SDL Window, even if the video mode expects it. + * This is a debugging aid for developers and not expected to be used by end users. The default is "1" + * + * This variable can be set to the following values: + * "0" - don't allow topmost + * "1" - allow topmost + */ +#define SDL_HINT_ALLOW_TOPMOST "SDL_ALLOW_TOPMOST" + +/** + * \brief Android APK expansion main file version. Should be a string number like "1", "2" etc. + * + * Must be set together with SDL_HINT_ANDROID_APK_EXPANSION_PATCH_FILE_VERSION. + * + * If both hints were set then SDL_RWFromFile() will look into expansion files + * after a given relative path was not found in the internal storage and assets. + * + * By default this hint is not set and the APK expansion files are not searched. + */ +#define SDL_HINT_ANDROID_APK_EXPANSION_MAIN_FILE_VERSION "SDL_ANDROID_APK_EXPANSION_MAIN_FILE_VERSION" + +/** + * \brief Android APK expansion patch file version. Should be a string number like "1", "2" etc. + * + * Must be set together with SDL_HINT_ANDROID_APK_EXPANSION_MAIN_FILE_VERSION. + * + * If both hints were set then SDL_RWFromFile() will look into expansion files + * after a given relative path was not found in the internal storage and assets. + * + * By default this hint is not set and the APK expansion files are not searched. + */ +#define SDL_HINT_ANDROID_APK_EXPANSION_PATCH_FILE_VERSION "SDL_ANDROID_APK_EXPANSION_PATCH_FILE_VERSION" + +/** + * \brief A variable to control whether the event loop will block itself when the app is paused. + * + * The variable can be set to the following values: + * "0" - Non blocking. + * "1" - Blocking. (default) + * + * The value should be set before SDL is initialized. + */ +#define SDL_HINT_ANDROID_BLOCK_ON_PAUSE "SDL_ANDROID_BLOCK_ON_PAUSE" + +/** + * \brief A variable to control whether SDL will pause audio in background + * (Requires SDL_ANDROID_BLOCK_ON_PAUSE as "Non blocking") + * + * The variable can be set to the following values: + * "0" - Non paused. + * "1" - Paused. (default) + * + * The value should be set before SDL is initialized. + */ +#define SDL_HINT_ANDROID_BLOCK_ON_PAUSE_PAUSEAUDIO "SDL_ANDROID_BLOCK_ON_PAUSE_PAUSEAUDIO" + +/** + * \brief A variable to control whether we trap the Android back button to handle it manually. + * This is necessary for the right mouse button to work on some Android devices, or + * to be able to trap the back button for use in your code reliably. If set to true, + * the back button will show up as an SDL_KEYDOWN / SDL_KEYUP pair with a keycode of + * SDL_SCANCODE_AC_BACK. + * + * The variable can be set to the following values: + * "0" - Back button will be handled as usual for system. (default) + * "1" - Back button will be trapped, allowing you to handle the key press + * manually. (This will also let right mouse click work on systems + * where the right mouse button functions as back.) + * + * The value of this hint is used at runtime, so it can be changed at any time. + */ +#define SDL_HINT_ANDROID_TRAP_BACK_BUTTON "SDL_ANDROID_TRAP_BACK_BUTTON" + +/** + * \brief Specify an application name. + * + * This hint lets you specify the application name sent to the OS when + * required. For example, this will often appear in volume control applets for + * audio streams, and in lists of applications which are inhibiting the + * screensaver. You should use a string that describes your program ("My Game + * 2: The Revenge") + * + * Setting this to "" or leaving it unset will have SDL use a reasonable + * default: probably the application's name or "SDL Application" if SDL + * doesn't have any better information. + * + * Note that, for audio streams, this can be overridden with + * SDL_HINT_AUDIO_DEVICE_APP_NAME. + * + * On targets where this is not supported, this hint does nothing. + */ +#define SDL_HINT_APP_NAME "SDL_APP_NAME" + +/** + * \brief A variable controlling whether controllers used with the Apple TV + * generate UI events. + * + * When UI events are generated by controller input, the app will be + * backgrounded when the Apple TV remote's menu button is pressed, and when the + * pause or B buttons on gamepads are pressed. + * + * More information about properly making use of controllers for the Apple TV + * can be found here: + * https://developer.apple.com/tvos/human-interface-guidelines/remote-and-controllers/ + * + * This variable can be set to the following values: + * "0" - Controller input does not generate UI events (the default). + * "1" - Controller input generates UI events. + */ +#define SDL_HINT_APPLE_TV_CONTROLLER_UI_EVENTS "SDL_APPLE_TV_CONTROLLER_UI_EVENTS" + +/** + * \brief A variable controlling whether the Apple TV remote's joystick axes + * will automatically match the rotation of the remote. + * + * This variable can be set to the following values: + * "0" - Remote orientation does not affect joystick axes (the default). + * "1" - Joystick axes are based on the orientation of the remote. + */ +#define SDL_HINT_APPLE_TV_REMOTE_ALLOW_ROTATION "SDL_APPLE_TV_REMOTE_ALLOW_ROTATION" + +/** + * \brief A variable controlling the audio category on iOS and Mac OS X + * + * This variable can be set to the following values: + * + * "ambient" - Use the AVAudioSessionCategoryAmbient audio category, will be muted by the phone mute switch (default) + * "playback" - Use the AVAudioSessionCategoryPlayback category + * + * For more information, see Apple's documentation: + * https://developer.apple.com/library/content/documentation/Audio/Conceptual/AudioSessionProgrammingGuide/AudioSessionCategoriesandModes/AudioSessionCategoriesandModes.html + */ +#define SDL_HINT_AUDIO_CATEGORY "SDL_AUDIO_CATEGORY" + +/** + * \brief Specify an application name for an audio device. + * + * Some audio backends (such as PulseAudio) allow you to describe your audio + * stream. Among other things, this description might show up in a system + * control panel that lets the user adjust the volume on specific audio + * streams instead of using one giant master volume slider. + * + * This hints lets you transmit that information to the OS. The contents of + * this hint are used while opening an audio device. You should use a string + * that describes your program ("My Game 2: The Revenge") + * + * Setting this to "" or leaving it unset will have SDL use a reasonable + * default: this will be the name set with SDL_HINT_APP_NAME, if that hint is + * set. Otherwise, it'll probably the application's name or "SDL Application" + * if SDL doesn't have any better information. + * + * On targets where this is not supported, this hint does nothing. + */ +#define SDL_HINT_AUDIO_DEVICE_APP_NAME "SDL_AUDIO_DEVICE_APP_NAME" + +/** + * \brief Specify an application name for an audio device. + * + * Some audio backends (such as PulseAudio) allow you to describe your audio + * stream. Among other things, this description might show up in a system + * control panel that lets the user adjust the volume on specific audio + * streams instead of using one giant master volume slider. + * + * This hints lets you transmit that information to the OS. The contents of + * this hint are used while opening an audio device. You should use a string + * that describes your what your program is playing ("audio stream" is + * probably sufficient in many cases, but this could be useful for something + * like "team chat" if you have a headset playing VoIP audio separately). + * + * Setting this to "" or leaving it unset will have SDL use a reasonable + * default: "audio stream" or something similar. + * + * On targets where this is not supported, this hint does nothing. + */ +#define SDL_HINT_AUDIO_DEVICE_STREAM_NAME "SDL_AUDIO_DEVICE_STREAM_NAME" + +/** + * \brief Specify an application role for an audio device. + * + * Some audio backends (such as Pipewire) allow you to describe the role of + * your audio stream. Among other things, this description might show up in + * a system control panel or software for displaying and manipulating media + * playback/capture graphs. + * + * This hints lets you transmit that information to the OS. The contents of + * this hint are used while opening an audio device. You should use a string + * that describes your what your program is playing (Game, Music, Movie, + * etc...). + * + * Setting this to "" or leaving it unset will have SDL use a reasonable + * default: "Game" or something similar. + * + * On targets where this is not supported, this hint does nothing. + */ +#define SDL_HINT_AUDIO_DEVICE_STREAM_ROLE "SDL_AUDIO_DEVICE_STREAM_ROLE" + +/** + * \brief A variable controlling speed/quality tradeoff of audio resampling. + * + * If available, SDL can use libsamplerate ( http://www.mega-nerd.com/SRC/ ) + * to handle audio resampling. There are different resampling modes available + * that produce different levels of quality, using more CPU. + * + * If this hint isn't specified to a valid setting, or libsamplerate isn't + * available, SDL will use the default, internal resampling algorithm. + * + * As of SDL 2.26, SDL_ConvertAudio() respects this hint when libsamplerate is available. + * + * This hint is currently only checked at audio subsystem initialization. + * + * This variable can be set to the following values: + * + * "0" or "default" - Use SDL's internal resampling (Default when not set - low quality, fast) + * "1" or "fast" - Use fast, slightly higher quality resampling, if available + * "2" or "medium" - Use medium quality resampling, if available + * "3" or "best" - Use high quality resampling, if available + */ +#define SDL_HINT_AUDIO_RESAMPLING_MODE "SDL_AUDIO_RESAMPLING_MODE" + +/** + * \brief A variable controlling whether SDL updates joystick state when getting input events + * + * This variable can be set to the following values: + * + * "0" - You'll call SDL_JoystickUpdate() manually + * "1" - SDL will automatically call SDL_JoystickUpdate() (default) + * + * This hint can be toggled on and off at runtime. + */ +#define SDL_HINT_AUTO_UPDATE_JOYSTICKS "SDL_AUTO_UPDATE_JOYSTICKS" + +/** + * \brief A variable controlling whether SDL updates sensor state when getting input events + * + * This variable can be set to the following values: + * + * "0" - You'll call SDL_SensorUpdate() manually + * "1" - SDL will automatically call SDL_SensorUpdate() (default) + * + * This hint can be toggled on and off at runtime. + */ +#define SDL_HINT_AUTO_UPDATE_SENSORS "SDL_AUTO_UPDATE_SENSORS" + +/** + * \brief Prevent SDL from using version 4 of the bitmap header when saving BMPs. + * + * The bitmap header version 4 is required for proper alpha channel support and + * SDL will use it when required. Should this not be desired, this hint can + * force the use of the 40 byte header version which is supported everywhere. + * + * The variable can be set to the following values: + * "0" - Surfaces with a colorkey or an alpha channel are saved to a + * 32-bit BMP file with an alpha mask. SDL will use the bitmap + * header version 4 and set the alpha mask accordingly. + * "1" - Surfaces with a colorkey or an alpha channel are saved to a + * 32-bit BMP file without an alpha mask. The alpha channel data + * will be in the file, but applications are going to ignore it. + * + * The default value is "0". + */ +#define SDL_HINT_BMP_SAVE_LEGACY_FORMAT "SDL_BMP_SAVE_LEGACY_FORMAT" + +/** + * \brief Override for SDL_GetDisplayUsableBounds() + * + * If set, this hint will override the expected results for + * SDL_GetDisplayUsableBounds() for display index 0. Generally you don't want + * to do this, but this allows an embedded system to request that some of the + * screen be reserved for other uses when paired with a well-behaved + * application. + * + * The contents of this hint must be 4 comma-separated integers, the first + * is the bounds x, then y, width and height, in that order. + */ +#define SDL_HINT_DISPLAY_USABLE_BOUNDS "SDL_DISPLAY_USABLE_BOUNDS" + +/** + * \brief Disable giving back control to the browser automatically + * when running with asyncify + * + * With -s ASYNCIFY, SDL2 calls emscripten_sleep during operations + * such as refreshing the screen or polling events. + * + * This hint only applies to the emscripten platform + * + * The variable can be set to the following values: + * "0" - Disable emscripten_sleep calls (if you give back browser control manually or use asyncify for other purposes) + * "1" - Enable emscripten_sleep calls (the default) + */ +#define SDL_HINT_EMSCRIPTEN_ASYNCIFY "SDL_EMSCRIPTEN_ASYNCIFY" + +/** + * \brief override the binding element for keyboard inputs for Emscripten builds + * + * This hint only applies to the emscripten platform + * + * The variable can be one of + * "#window" - The javascript window object (this is the default) + * "#document" - The javascript document object + * "#screen" - the javascript window.screen object + * "#canvas" - the WebGL canvas element + * any other string without a leading # sign applies to the element on the page with that ID. + */ +#define SDL_HINT_EMSCRIPTEN_KEYBOARD_ELEMENT "SDL_EMSCRIPTEN_KEYBOARD_ELEMENT" + +/** + * \brief A variable that controls whether the on-screen keyboard should be shown when text input is active + * + * The variable can be set to the following values: + * "0" - Do not show the on-screen keyboard + * "1" - Show the on-screen keyboard + * + * The default value is "1". This hint must be set before text input is activated. + */ +#define SDL_HINT_ENABLE_SCREEN_KEYBOARD "SDL_ENABLE_SCREEN_KEYBOARD" + +/** + * \brief A variable that controls whether Steam Controllers should be exposed using the SDL joystick and game controller APIs + * + * The variable can be set to the following values: + * "0" - Do not scan for Steam Controllers + * "1" - Scan for Steam Controllers (the default) + * + * The default value is "1". This hint must be set before initializing the joystick subsystem. + */ +#define SDL_HINT_ENABLE_STEAM_CONTROLLERS "SDL_ENABLE_STEAM_CONTROLLERS" + +/** + * \brief A variable controlling verbosity of the logging of SDL events pushed onto the internal queue. + * + * This variable can be set to the following values, from least to most verbose: + * + * "0" - Don't log any events (default) + * "1" - Log most events (other than the really spammy ones). + * "2" - Include mouse and finger motion events. + * "3" - Include SDL_SysWMEvent events. + * + * This is generally meant to be used to debug SDL itself, but can be useful + * for application developers that need better visibility into what is going + * on in the event queue. Logged events are sent through SDL_Log(), which + * means by default they appear on stdout on most platforms or maybe + * OutputDebugString() on Windows, and can be funneled by the app with + * SDL_LogSetOutputFunction(), etc. + * + * This hint can be toggled on and off at runtime, if you only need to log + * events for a small subset of program execution. + */ +#define SDL_HINT_EVENT_LOGGING "SDL_EVENT_LOGGING" + +/** + * \brief A variable controlling whether raising the window should be done more forcefully + * + * This variable can be set to the following values: + * "0" - No forcing (the default) + * "1" - Extra level of forcing + * + * At present, this is only an issue under MS Windows, which makes it nearly impossible to + * programmatically move a window to the foreground, for "security" reasons. See + * http://stackoverflow.com/a/34414846 for a discussion. + */ +#define SDL_HINT_FORCE_RAISEWINDOW "SDL_HINT_FORCE_RAISEWINDOW" + +/** + * \brief A variable controlling how 3D acceleration is used to accelerate the SDL screen surface. + * + * SDL can try to accelerate the SDL screen surface by using streaming + * textures with a 3D rendering engine. This variable controls whether and + * how this is done. + * + * This variable can be set to the following values: + * "0" - Disable 3D acceleration + * "1" - Enable 3D acceleration, using the default renderer. + * "X" - Enable 3D acceleration, using X where X is one of the valid rendering drivers. (e.g. "direct3d", "opengl", etc.) + * + * By default SDL tries to make a best guess for each platform whether + * to use acceleration or not. + */ +#define SDL_HINT_FRAMEBUFFER_ACCELERATION "SDL_FRAMEBUFFER_ACCELERATION" + +/** + * \brief A variable that lets you manually hint extra gamecontroller db entries. + * + * The variable should be newline delimited rows of gamecontroller config data, see SDL_gamecontroller.h + * + * This hint must be set before calling SDL_Init(SDL_INIT_GAMECONTROLLER) + * You can update mappings after the system is initialized with SDL_GameControllerMappingForGUID() and SDL_GameControllerAddMapping() + */ +#define SDL_HINT_GAMECONTROLLERCONFIG "SDL_GAMECONTROLLERCONFIG" + +/** + * \brief A variable that lets you provide a file with extra gamecontroller db entries. + * + * The file should contain lines of gamecontroller config data, see SDL_gamecontroller.h + * + * This hint must be set before calling SDL_Init(SDL_INIT_GAMECONTROLLER) + * You can update mappings after the system is initialized with SDL_GameControllerMappingForGUID() and SDL_GameControllerAddMapping() + */ +#define SDL_HINT_GAMECONTROLLERCONFIG_FILE "SDL_GAMECONTROLLERCONFIG_FILE" + +/** + * \brief A variable that overrides the automatic controller type detection + * + * The variable should be comma separated entries, in the form: VID/PID=type + * + * The VID and PID should be hexadecimal with exactly 4 digits, e.g. 0x00fd + * + * The type should be one of: + * Xbox360 + * XboxOne + * PS3 + * PS4 + * PS5 + * SwitchPro + * + * This hint affects what driver is used, and must be set before calling SDL_Init(SDL_INIT_GAMECONTROLLER) + */ +#define SDL_HINT_GAMECONTROLLERTYPE "SDL_GAMECONTROLLERTYPE" + +/** + * \brief A variable containing a list of devices to skip when scanning for game controllers. + * + * The format of the string is a comma separated list of USB VID/PID pairs + * in hexadecimal form, e.g. + * + * 0xAAAA/0xBBBB,0xCCCC/0xDDDD + * + * The variable can also take the form of @file, in which case the named + * file will be loaded and interpreted as the value of the variable. + */ +#define SDL_HINT_GAMECONTROLLER_IGNORE_DEVICES "SDL_GAMECONTROLLER_IGNORE_DEVICES" + +/** + * \brief If set, all devices will be skipped when scanning for game controllers except for the ones listed in this variable. + * + * The format of the string is a comma separated list of USB VID/PID pairs + * in hexadecimal form, e.g. + * + * 0xAAAA/0xBBBB,0xCCCC/0xDDDD + * + * The variable can also take the form of @file, in which case the named + * file will be loaded and interpreted as the value of the variable. + */ +#define SDL_HINT_GAMECONTROLLER_IGNORE_DEVICES_EXCEPT "SDL_GAMECONTROLLER_IGNORE_DEVICES_EXCEPT" + +/** + * \brief If set, game controller face buttons report their values according to their labels instead of their positional layout. + * + * For example, on Nintendo Switch controllers, normally you'd get: + * + * (Y) + * (X) (B) + * (A) + * + * but if this hint is set, you'll get: + * + * (X) + * (Y) (A) + * (B) + * + * The variable can be set to the following values: + * "0" - Report the face buttons by position, as though they were on an Xbox controller. + * "1" - Report the face buttons by label instead of position + * + * The default value is "1". This hint may be set at any time. + */ +#define SDL_HINT_GAMECONTROLLER_USE_BUTTON_LABELS "SDL_GAMECONTROLLER_USE_BUTTON_LABELS" + +/** + * \brief A variable controlling whether grabbing input grabs the keyboard + * + * This variable can be set to the following values: + * "0" - Grab will affect only the mouse + * "1" - Grab will affect mouse and keyboard + * + * By default SDL will not grab the keyboard so system shortcuts still work. + */ +#define SDL_HINT_GRAB_KEYBOARD "SDL_GRAB_KEYBOARD" + +/** + * \brief A variable containing a list of devices to ignore in SDL_hid_enumerate() + * + * For example, to ignore the Shanwan DS3 controller and any Valve controller, you might + * have the string "0x2563/0x0523,0x28de/0x0000" + */ +#define SDL_HINT_HIDAPI_IGNORE_DEVICES "SDL_HIDAPI_IGNORE_DEVICES" + +/** + * \brief A variable controlling whether the idle timer is disabled on iOS. + * + * When an iOS app does not receive touches for some time, the screen is + * dimmed automatically. For games where the accelerometer is the only input + * this is problematic. This functionality can be disabled by setting this + * hint. + * + * As of SDL 2.0.4, SDL_EnableScreenSaver() and SDL_DisableScreenSaver() + * accomplish the same thing on iOS. They should be preferred over this hint. + * + * This variable can be set to the following values: + * "0" - Enable idle timer + * "1" - Disable idle timer + */ +#define SDL_HINT_IDLE_TIMER_DISABLED "SDL_IOS_IDLE_TIMER_DISABLED" + +/** + * \brief A variable to control whether certain IMEs should handle text editing internally instead of sending SDL_TEXTEDITING events. + * + * The variable can be set to the following values: + * "0" - SDL_TEXTEDITING events are sent, and it is the application's + * responsibility to render the text from these events and + * differentiate it somehow from committed text. (default) + * "1" - If supported by the IME then SDL_TEXTEDITING events are not sent, + * and text that is being composed will be rendered in its own UI. + */ +#define SDL_HINT_IME_INTERNAL_EDITING "SDL_IME_INTERNAL_EDITING" + +/** + * \brief A variable to control whether certain IMEs should show native UI components (such as the Candidate List) instead of suppressing them. + * + * The variable can be set to the following values: + * "0" - Native UI components are not display. (default) + * "1" - Native UI components are displayed. + */ +#define SDL_HINT_IME_SHOW_UI "SDL_IME_SHOW_UI" + +/** + * \brief A variable to control if extended IME text support is enabled. + * If enabled then SDL_TextEditingExtEvent will be issued if the text would be truncated otherwise. + * Additionally SDL_TextInputEvent will be dispatched multiple times so that it is not truncated. + * + * The variable can be set to the following values: + * "0" - Legacy behavior. Text can be truncated, no heap allocations. (default) + * "1" - Modern behavior. + */ +#define SDL_HINT_IME_SUPPORT_EXTENDED_TEXT "SDL_IME_SUPPORT_EXTENDED_TEXT" + +/** + * \brief A variable controlling whether the home indicator bar on iPhone X + * should be hidden. + * + * This variable can be set to the following values: + * "0" - The indicator bar is not hidden (default for windowed applications) + * "1" - The indicator bar is hidden and is shown when the screen is touched (useful for movie playback applications) + * "2" - The indicator bar is dim and the first swipe makes it visible and the second swipe performs the "home" action (default for fullscreen applications) + */ +#define SDL_HINT_IOS_HIDE_HOME_INDICATOR "SDL_IOS_HIDE_HOME_INDICATOR" + +/** + * \brief A variable that lets you enable joystick (and gamecontroller) events even when your app is in the background. + * + * The variable can be set to the following values: + * "0" - Disable joystick & gamecontroller input events when the + * application is in the background. + * "1" - Enable joystick & gamecontroller input events when the + * application is in the background. + * + * The default value is "0". This hint may be set at any time. + */ +#define SDL_HINT_JOYSTICK_ALLOW_BACKGROUND_EVENTS "SDL_JOYSTICK_ALLOW_BACKGROUND_EVENTS" + +/** + * \brief A variable controlling whether the HIDAPI joystick drivers should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI drivers are not used + * "1" - HIDAPI drivers are used (the default) + * + * This variable is the default for all drivers, but can be overridden by the hints for specific drivers below. + */ +#define SDL_HINT_JOYSTICK_HIDAPI "SDL_JOYSTICK_HIDAPI" + +/** + * \brief A variable controlling whether the HIDAPI driver for Nintendo GameCube controllers should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used + * + * The default is the value of SDL_HINT_JOYSTICK_HIDAPI + */ +#define SDL_HINT_JOYSTICK_HIDAPI_GAMECUBE "SDL_JOYSTICK_HIDAPI_GAMECUBE" + +/** + * \brief A variable controlling whether "low_frequency_rumble" and "high_frequency_rumble" is used to implement + * the GameCube controller's 3 rumble modes, Stop(0), Rumble(1), and StopHard(2) + * this is useful for applications that need full compatibility for things like ADSR envelopes. + * Stop is implemented by setting "low_frequency_rumble" to "0" and "high_frequency_rumble" ">0" + * Rumble is both at any arbitrary value, + * StopHard is implemented by setting both "low_frequency_rumble" and "high_frequency_rumble" to "0" + * + * This variable can be set to the following values: + * "0" - Normal rumble behavior is behavior is used (default) + * "1" - Proper GameCube controller rumble behavior is used + * + */ +#define SDL_HINT_JOYSTICK_GAMECUBE_RUMBLE_BRAKE "SDL_JOYSTICK_GAMECUBE_RUMBLE_BRAKE" + +/** + * \brief A variable controlling whether the HIDAPI driver for Nintendo Switch Joy-Cons should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used + * + * The default is the value of SDL_HINT_JOYSTICK_HIDAPI + */ +#define SDL_HINT_JOYSTICK_HIDAPI_JOY_CONS "SDL_JOYSTICK_HIDAPI_JOY_CONS" + +/** + * \brief A variable controlling whether Nintendo Switch Joy-Con controllers will be combined into a single Pro-like controller when using the HIDAPI driver + * + * This variable can be set to the following values: + * "0" - Left and right Joy-Con controllers will not be combined and each will be a mini-gamepad + * "1" - Left and right Joy-Con controllers will be combined into a single controller (the default) + */ +#define SDL_HINT_JOYSTICK_HIDAPI_COMBINE_JOY_CONS "SDL_JOYSTICK_HIDAPI_COMBINE_JOY_CONS" + +/** + * \brief A variable controlling whether Nintendo Switch Joy-Con controllers will be in vertical mode when using the HIDAPI driver + * + * This variable can be set to the following values: + * "0" - Left and right Joy-Con controllers will not be in vertical mode (the default) + * "1" - Left and right Joy-Con controllers will be in vertical mode + * + * This hint must be set before calling SDL_Init(SDL_INIT_GAMECONTROLLER) + */ +#define SDL_HINT_JOYSTICK_HIDAPI_VERTICAL_JOY_CONS "SDL_JOYSTICK_HIDAPI_VERTICAL_JOY_CONS" + +/** + * \brief A variable controlling whether the HIDAPI driver for Amazon Luna controllers connected via Bluetooth should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used + * + * The default is the value of SDL_HINT_JOYSTICK_HIDAPI + */ +#define SDL_HINT_JOYSTICK_HIDAPI_LUNA "SDL_JOYSTICK_HIDAPI_LUNA" + +/** + * \brief A variable controlling whether the HIDAPI driver for Nintendo Online classic controllers should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used + * + * The default is the value of SDL_HINT_JOYSTICK_HIDAPI + */ +#define SDL_HINT_JOYSTICK_HIDAPI_NINTENDO_CLASSIC "SDL_JOYSTICK_HIDAPI_NINTENDO_CLASSIC" + +/** + * \brief A variable controlling whether the HIDAPI driver for NVIDIA SHIELD controllers should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used + * + * The default is the value of SDL_HINT_JOYSTICK_HIDAPI + */ +#define SDL_HINT_JOYSTICK_HIDAPI_SHIELD "SDL_JOYSTICK_HIDAPI_SHIELD" + +/** + * \brief A variable controlling whether the HIDAPI driver for PS3 controllers should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used + * + * The default is the value of SDL_HINT_JOYSTICK_HIDAPI on macOS, and "0" on other platforms. + * + * It is not possible to use this driver on Windows, due to limitations in the default drivers + * installed. See https://github.com/ViGEm/DsHidMini for an alternative driver on Windows. + */ +#define SDL_HINT_JOYSTICK_HIDAPI_PS3 "SDL_JOYSTICK_HIDAPI_PS3" + +/** + * \brief A variable controlling whether the HIDAPI driver for PS4 controllers should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used + * + * The default is the value of SDL_HINT_JOYSTICK_HIDAPI + */ +#define SDL_HINT_JOYSTICK_HIDAPI_PS4 "SDL_JOYSTICK_HIDAPI_PS4" + +/** + * \brief A variable controlling whether extended input reports should be used for PS4 controllers when using the HIDAPI driver. + * + * This variable can be set to the following values: + * "0" - extended reports are not enabled (the default) + * "1" - extended reports + * + * Extended input reports allow rumble on Bluetooth PS4 controllers, but + * break DirectInput handling for applications that don't use SDL. + * + * Once extended reports are enabled, they can not be disabled without + * power cycling the controller. + * + * For compatibility with applications written for versions of SDL prior + * to the introduction of PS5 controller support, this value will also + * control the state of extended reports on PS5 controllers when the + * SDL_HINT_JOYSTICK_HIDAPI_PS5_RUMBLE hint is not explicitly set. + */ +#define SDL_HINT_JOYSTICK_HIDAPI_PS4_RUMBLE "SDL_JOYSTICK_HIDAPI_PS4_RUMBLE" + +/** + * \brief A variable controlling whether the HIDAPI driver for PS5 controllers should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used + * + * The default is the value of SDL_HINT_JOYSTICK_HIDAPI + */ +#define SDL_HINT_JOYSTICK_HIDAPI_PS5 "SDL_JOYSTICK_HIDAPI_PS5" + +/** + * \brief A variable controlling whether the player LEDs should be lit to indicate which player is associated with a PS5 controller. + * + * This variable can be set to the following values: + * "0" - player LEDs are not enabled + * "1" - player LEDs are enabled (the default) + */ +#define SDL_HINT_JOYSTICK_HIDAPI_PS5_PLAYER_LED "SDL_JOYSTICK_HIDAPI_PS5_PLAYER_LED" + +/** + * \brief A variable controlling whether extended input reports should be used for PS5 controllers when using the HIDAPI driver. + * + * This variable can be set to the following values: + * "0" - extended reports are not enabled (the default) + * "1" - extended reports + * + * Extended input reports allow rumble on Bluetooth PS5 controllers, but + * break DirectInput handling for applications that don't use SDL. + * + * Once extended reports are enabled, they can not be disabled without + * power cycling the controller. + * + * For compatibility with applications written for versions of SDL prior + * to the introduction of PS5 controller support, this value defaults to + * the value of SDL_HINT_JOYSTICK_HIDAPI_PS4_RUMBLE. + */ +#define SDL_HINT_JOYSTICK_HIDAPI_PS5_RUMBLE "SDL_JOYSTICK_HIDAPI_PS5_RUMBLE" + +/** + * \brief A variable controlling whether the HIDAPI driver for Google Stadia controllers should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used + * + * The default is the value of SDL_HINT_JOYSTICK_HIDAPI + */ +#define SDL_HINT_JOYSTICK_HIDAPI_STADIA "SDL_JOYSTICK_HIDAPI_STADIA" + +/** + * \brief A variable controlling whether the HIDAPI driver for Bluetooth Steam Controllers should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used for Steam Controllers, which requires Bluetooth access + * and may prompt the user for permission on iOS and Android. + * + * The default is "0" + */ +#define SDL_HINT_JOYSTICK_HIDAPI_STEAM "SDL_JOYSTICK_HIDAPI_STEAM" + +/** + * \brief A variable controlling whether the HIDAPI driver for Nintendo Switch controllers should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used + * + * The default is the value of SDL_HINT_JOYSTICK_HIDAPI + */ +#define SDL_HINT_JOYSTICK_HIDAPI_SWITCH "SDL_JOYSTICK_HIDAPI_SWITCH" + +/** + * \brief A variable controlling whether the Home button LED should be turned on when a Nintendo Switch Pro controller is opened + * + * This variable can be set to the following values: + * "0" - home button LED is turned off + * "1" - home button LED is turned on + * + * By default the Home button LED state is not changed. This hint can also be set to a floating point value between 0.0 and 1.0 which controls the brightness of the Home button LED. + */ +#define SDL_HINT_JOYSTICK_HIDAPI_SWITCH_HOME_LED "SDL_JOYSTICK_HIDAPI_SWITCH_HOME_LED" + +/** + * \brief A variable controlling whether the Home button LED should be turned on when a Nintendo Switch Joy-Con controller is opened + * + * This variable can be set to the following values: + * "0" - home button LED is turned off + * "1" - home button LED is turned on + * + * By default the Home button LED state is not changed. This hint can also be set to a floating point value between 0.0 and 1.0 which controls the brightness of the Home button LED. + */ +#define SDL_HINT_JOYSTICK_HIDAPI_JOYCON_HOME_LED "SDL_JOYSTICK_HIDAPI_JOYCON_HOME_LED" + +/** + * \brief A variable controlling whether the player LEDs should be lit to indicate which player is associated with a Nintendo Switch controller. + * + * This variable can be set to the following values: + * "0" - player LEDs are not enabled + * "1" - player LEDs are enabled (the default) + */ +#define SDL_HINT_JOYSTICK_HIDAPI_SWITCH_PLAYER_LED "SDL_JOYSTICK_HIDAPI_SWITCH_PLAYER_LED" + +/** + * \brief A variable controlling whether the HIDAPI driver for Nintendo Wii and Wii U controllers should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used + * + * This driver doesn't work with the dolphinbar, so the default is SDL_FALSE for now. + */ +#define SDL_HINT_JOYSTICK_HIDAPI_WII "SDL_JOYSTICK_HIDAPI_WII" + +/** + * \brief A variable controlling whether the player LEDs should be lit to indicate which player is associated with a Wii controller. + * + * This variable can be set to the following values: + * "0" - player LEDs are not enabled + * "1" - player LEDs are enabled (the default) + */ +#define SDL_HINT_JOYSTICK_HIDAPI_WII_PLAYER_LED "SDL_JOYSTICK_HIDAPI_WII_PLAYER_LED" + +/** + * \brief A variable controlling whether the HIDAPI driver for XBox controllers should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used + * + * The default is "0" on Windows, otherwise the value of SDL_HINT_JOYSTICK_HIDAPI + */ +#define SDL_HINT_JOYSTICK_HIDAPI_XBOX "SDL_JOYSTICK_HIDAPI_XBOX" + +/** + * \brief A variable controlling whether the HIDAPI driver for XBox 360 controllers should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used + * + * The default is the value of SDL_HINT_JOYSTICK_HIDAPI_XBOX + */ +#define SDL_HINT_JOYSTICK_HIDAPI_XBOX_360 "SDL_JOYSTICK_HIDAPI_XBOX_360" + +/** + * \brief A variable controlling whether the player LEDs should be lit to indicate which player is associated with an Xbox 360 controller. + * + * This variable can be set to the following values: + * "0" - player LEDs are not enabled + * "1" - player LEDs are enabled (the default) + */ +#define SDL_HINT_JOYSTICK_HIDAPI_XBOX_360_PLAYER_LED "SDL_JOYSTICK_HIDAPI_XBOX_360_PLAYER_LED" + +/** + * \brief A variable controlling whether the HIDAPI driver for XBox 360 wireless controllers should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used + * + * The default is the value of SDL_HINT_JOYSTICK_HIDAPI_XBOX_360 + */ +#define SDL_HINT_JOYSTICK_HIDAPI_XBOX_360_WIRELESS "SDL_JOYSTICK_HIDAPI_XBOX_360_WIRELESS" + +/** + * \brief A variable controlling whether the HIDAPI driver for XBox One controllers should be used. + * + * This variable can be set to the following values: + * "0" - HIDAPI driver is not used + * "1" - HIDAPI driver is used + * + * The default is the value of SDL_HINT_JOYSTICK_HIDAPI_XBOX + */ +#define SDL_HINT_JOYSTICK_HIDAPI_XBOX_ONE "SDL_JOYSTICK_HIDAPI_XBOX_ONE" + +/** + * \brief A variable controlling whether the Home button LED should be turned on when an Xbox One controller is opened + * + * This variable can be set to the following values: + * "0" - home button LED is turned off + * "1" - home button LED is turned on + * + * By default the Home button LED state is not changed. This hint can also be set to a floating point value between 0.0 and 1.0 which controls the brightness of the Home button LED. The default brightness is 0.4. + */ +#define SDL_HINT_JOYSTICK_HIDAPI_XBOX_ONE_HOME_LED "SDL_JOYSTICK_HIDAPI_XBOX_ONE_HOME_LED" + +/** + * \brief A variable controlling whether the RAWINPUT joystick drivers should be used for better handling XInput-capable devices. + * + * This variable can be set to the following values: + * "0" - RAWINPUT drivers are not used + * "1" - RAWINPUT drivers are used (the default) + */ +#define SDL_HINT_JOYSTICK_RAWINPUT "SDL_JOYSTICK_RAWINPUT" + +/** + * \brief A variable controlling whether the RAWINPUT driver should pull correlated data from XInput. + * + * This variable can be set to the following values: + * "0" - RAWINPUT driver will only use data from raw input APIs + * "1" - RAWINPUT driver will also pull data from XInput, providing + * better trigger axes, guide button presses, and rumble support + * for Xbox controllers + * + * The default is "1". This hint applies to any joysticks opened after setting the hint. + */ +#define SDL_HINT_JOYSTICK_RAWINPUT_CORRELATE_XINPUT "SDL_JOYSTICK_RAWINPUT_CORRELATE_XINPUT" + +/** + * \brief A variable controlling whether the ROG Chakram mice should show up as joysticks + * + * This variable can be set to the following values: + * "0" - ROG Chakram mice do not show up as joysticks (the default) + * "1" - ROG Chakram mice show up as joysticks + */ +#define SDL_HINT_JOYSTICK_ROG_CHAKRAM "SDL_JOYSTICK_ROG_CHAKRAM" + +/** + * \brief A variable controlling whether a separate thread should be used + * for handling joystick detection and raw input messages on Windows + * + * This variable can be set to the following values: + * "0" - A separate thread is not used (the default) + * "1" - A separate thread is used for handling raw input messages + * + */ +#define SDL_HINT_JOYSTICK_THREAD "SDL_JOYSTICK_THREAD" + +/** + * \brief A variable controlling whether Windows.Gaming.Input should be used for controller handling. + * + * This variable can be set to the following values: + * "0" - WGI is not used + * "1" - WGI is used (the default) + */ +#define SDL_HINT_JOYSTICK_WGI "SDL_JOYSTICK_WGI" + +/** + * \brief Determines whether SDL enforces that DRM master is required in order + * to initialize the KMSDRM video backend. + * + * The DRM subsystem has a concept of a "DRM master" which is a DRM client that + * has the ability to set planes, set cursor, etc. When SDL is DRM master, it + * can draw to the screen using the SDL rendering APIs. Without DRM master, SDL + * is still able to process input and query attributes of attached displays, + * but it cannot change display state or draw to the screen directly. + * + * In some cases, it can be useful to have the KMSDRM backend even if it cannot + * be used for rendering. An app may want to use SDL for input processing while + * using another rendering API (such as an MMAL overlay on Raspberry Pi) or + * using its own code to render to DRM overlays that SDL doesn't support. + * + * This hint must be set before initializing the video subsystem. + * + * This variable can be set to the following values: + * "0" - SDL will allow usage of the KMSDRM backend without DRM master + * "1" - SDL Will require DRM master to use the KMSDRM backend (default) + */ +#define SDL_HINT_KMSDRM_REQUIRE_DRM_MASTER "SDL_KMSDRM_REQUIRE_DRM_MASTER" + +/** + * \brief A comma separated list of devices to open as joysticks + * + * This variable is currently only used by the Linux joystick driver. + */ +#define SDL_HINT_JOYSTICK_DEVICE "SDL_JOYSTICK_DEVICE" + +/** + * \brief A variable controlling whether joysticks on Linux will always treat 'hat' axis inputs (ABS_HAT0X - ABS_HAT3Y) as 8-way digital hats without checking whether they may be analog. + * + * This variable can be set to the following values: + * "0" - Only map hat axis inputs to digital hat outputs if the input axes appear to actually be digital (the default) + * "1" - Always handle the input axes numbered ABS_HAT0X to ABS_HAT3Y as digital hats + */ +#define SDL_HINT_LINUX_DIGITAL_HATS "SDL_LINUX_DIGITAL_HATS" + +/** + * \brief A variable controlling whether digital hats on Linux will apply deadzones to their underlying input axes or use unfiltered values. + * + * This variable can be set to the following values: + * "0" - Return digital hat values based on unfiltered input axis values + * "1" - Return digital hat values with deadzones on the input axes taken into account (the default) + */ +#define SDL_HINT_LINUX_HAT_DEADZONES "SDL_LINUX_HAT_DEADZONES" + +/** + * \brief A variable controlling whether to use the classic /dev/input/js* joystick interface or the newer /dev/input/event* joystick interface on Linux + * + * This variable can be set to the following values: + * "0" - Use /dev/input/event* + * "1" - Use /dev/input/js* + * + * By default the /dev/input/event* interfaces are used + */ +#define SDL_HINT_LINUX_JOYSTICK_CLASSIC "SDL_LINUX_JOYSTICK_CLASSIC" + +/** + * \brief A variable controlling whether joysticks on Linux adhere to their HID-defined deadzones or return unfiltered values. + * + * This variable can be set to the following values: + * "0" - Return unfiltered joystick axis values (the default) + * "1" - Return axis values with deadzones taken into account + */ +#define SDL_HINT_LINUX_JOYSTICK_DEADZONES "SDL_LINUX_JOYSTICK_DEADZONES" + +/** +* \brief When set don't force the SDL app to become a foreground process +* +* This hint only applies to Mac OS X. +* +*/ +#define SDL_HINT_MAC_BACKGROUND_APP "SDL_MAC_BACKGROUND_APP" + +/** + * \brief A variable that determines whether ctrl+click should generate a right-click event on Mac + * + * If present, holding ctrl while left clicking will generate a right click + * event when on Mac. + */ +#define SDL_HINT_MAC_CTRL_CLICK_EMULATE_RIGHT_CLICK "SDL_MAC_CTRL_CLICK_EMULATE_RIGHT_CLICK" + +/** + * \brief A variable controlling whether dispatching OpenGL context updates should block the dispatching thread until the main thread finishes processing + * + * This variable can be set to the following values: + * "0" - Dispatching OpenGL context updates will block the dispatching thread until the main thread finishes processing (default). + * "1" - Dispatching OpenGL context updates will allow the dispatching thread to continue execution. + * + * Generally you want the default, but if you have OpenGL code in a background thread on a Mac, and the main thread + * hangs because it's waiting for that background thread, but that background thread is also hanging because it's + * waiting for the main thread to do an update, this might fix your issue. + * + * This hint only applies to macOS. + * + * This hint is available since SDL 2.24.0. + * + */ +#define SDL_HINT_MAC_OPENGL_ASYNC_DISPATCH "SDL_MAC_OPENGL_ASYNC_DISPATCH" + +/** + * \brief A variable setting the double click radius, in pixels. + */ +#define SDL_HINT_MOUSE_DOUBLE_CLICK_RADIUS "SDL_MOUSE_DOUBLE_CLICK_RADIUS" + +/** + * \brief A variable setting the double click time, in milliseconds. + */ +#define SDL_HINT_MOUSE_DOUBLE_CLICK_TIME "SDL_MOUSE_DOUBLE_CLICK_TIME" + +/** + * \brief Allow mouse click events when clicking to focus an SDL window + * + * This variable can be set to the following values: + * "0" - Ignore mouse clicks that activate a window + * "1" - Generate events for mouse clicks that activate a window + * + * By default SDL will ignore mouse clicks that activate a window + */ +#define SDL_HINT_MOUSE_FOCUS_CLICKTHROUGH "SDL_MOUSE_FOCUS_CLICKTHROUGH" + +/** + * \brief A variable setting the speed scale for mouse motion, in floating point, when the mouse is not in relative mode + */ +#define SDL_HINT_MOUSE_NORMAL_SPEED_SCALE "SDL_MOUSE_NORMAL_SPEED_SCALE" + +/** + * \brief A variable controlling whether relative mouse mode constrains the mouse to the center of the window + * + * This variable can be set to the following values: + * "0" - Relative mouse mode constrains the mouse to the window + * "1" - Relative mouse mode constrains the mouse to the center of the window + * + * Constraining to the center of the window works better for FPS games and when the + * application is running over RDP. Constraining to the whole window works better + * for 2D games and increases the chance that the mouse will be in the correct + * position when using high DPI mice. + * + * By default SDL will constrain the mouse to the center of the window + */ +#define SDL_HINT_MOUSE_RELATIVE_MODE_CENTER "SDL_MOUSE_RELATIVE_MODE_CENTER" + +/** + * \brief A variable controlling whether relative mouse mode is implemented using mouse warping + * + * This variable can be set to the following values: + * "0" - Relative mouse mode uses raw input + * "1" - Relative mouse mode uses mouse warping + * + * By default SDL will use raw input for relative mouse mode + */ +#define SDL_HINT_MOUSE_RELATIVE_MODE_WARP "SDL_MOUSE_RELATIVE_MODE_WARP" + +/** + * \brief A variable controlling whether relative mouse motion is affected by renderer scaling + * + * This variable can be set to the following values: + * "0" - Relative motion is unaffected by DPI or renderer's logical size + * "1" - Relative motion is scaled according to DPI scaling and logical size + * + * By default relative mouse deltas are affected by DPI and renderer scaling + */ +#define SDL_HINT_MOUSE_RELATIVE_SCALING "SDL_MOUSE_RELATIVE_SCALING" + +/** + * \brief A variable setting the scale for mouse motion, in floating point, when the mouse is in relative mode + */ +#define SDL_HINT_MOUSE_RELATIVE_SPEED_SCALE "SDL_MOUSE_RELATIVE_SPEED_SCALE" + +/** + * \brief A variable controlling whether the system mouse acceleration curve is used for relative mouse motion. + * + * This variable can be set to the following values: + * "0" - Relative mouse motion will be unscaled (the default) + * "1" - Relative mouse motion will be scaled using the system mouse acceleration curve. + * + * If SDL_HINT_MOUSE_RELATIVE_SPEED_SCALE is set, that will override the system speed scale. + */ +#define SDL_HINT_MOUSE_RELATIVE_SYSTEM_SCALE "SDL_MOUSE_RELATIVE_SYSTEM_SCALE" + +/** + * \brief A variable controlling whether a motion event should be generated for mouse warping in relative mode. + * + * This variable can be set to the following values: + * "0" - Warping the mouse will not generate a motion event in relative mode + * "1" - Warping the mouse will generate a motion event in relative mode + * + * By default warping the mouse will not generate motion events in relative mode. This avoids the application having to filter out large relative motion due to warping. + */ +#define SDL_HINT_MOUSE_RELATIVE_WARP_MOTION "SDL_MOUSE_RELATIVE_WARP_MOTION" + +/** + * \brief A variable controlling whether mouse events should generate synthetic touch events + * + * This variable can be set to the following values: + * "0" - Mouse events will not generate touch events (default for desktop platforms) + * "1" - Mouse events will generate touch events (default for mobile platforms, such as Android and iOS) + */ +#define SDL_HINT_MOUSE_TOUCH_EVENTS "SDL_MOUSE_TOUCH_EVENTS" + +/** + * \brief A variable controlling whether the mouse is captured while mouse buttons are pressed + * + * This variable can be set to the following values: + * "0" - The mouse is not captured while mouse buttons are pressed + * "1" - The mouse is captured while mouse buttons are pressed + * + * By default the mouse is captured while mouse buttons are pressed so if the mouse is dragged + * outside the window, the application continues to receive mouse events until the button is + * released. + */ +#define SDL_HINT_MOUSE_AUTO_CAPTURE "SDL_MOUSE_AUTO_CAPTURE" + +/** + * \brief Tell SDL not to catch the SIGINT or SIGTERM signals. + * + * This hint only applies to Unix-like platforms, and should set before + * any calls to SDL_Init() + * + * The variable can be set to the following values: + * "0" - SDL will install a SIGINT and SIGTERM handler, and when it + * catches a signal, convert it into an SDL_QUIT event. + * "1" - SDL will not install a signal handler at all. + */ +#define SDL_HINT_NO_SIGNAL_HANDLERS "SDL_NO_SIGNAL_HANDLERS" + +/** + * \brief A variable controlling what driver to use for OpenGL ES contexts. + * + * On some platforms, currently Windows and X11, OpenGL drivers may support + * creating contexts with an OpenGL ES profile. By default SDL uses these + * profiles, when available, otherwise it attempts to load an OpenGL ES + * library, e.g. that provided by the ANGLE project. This variable controls + * whether SDL follows this default behaviour or will always load an + * OpenGL ES library. + * + * Circumstances where this is useful include + * - Testing an app with a particular OpenGL ES implementation, e.g ANGLE, + * or emulator, e.g. those from ARM, Imagination or Qualcomm. + * - Resolving OpenGL ES function addresses at link time by linking with + * the OpenGL ES library instead of querying them at run time with + * SDL_GL_GetProcAddress(). + * + * Caution: for an application to work with the default behaviour across + * different OpenGL drivers it must query the OpenGL ES function + * addresses at run time using SDL_GL_GetProcAddress(). + * + * This variable is ignored on most platforms because OpenGL ES is native + * or not supported. + * + * This variable can be set to the following values: + * "0" - Use ES profile of OpenGL, if available. (Default when not set.) + * "1" - Load OpenGL ES library using the default library names. + * + */ +#define SDL_HINT_OPENGL_ES_DRIVER "SDL_OPENGL_ES_DRIVER" + +/** + * \brief A variable controlling which orientations are allowed on iOS/Android. + * + * In some circumstances it is necessary to be able to explicitly control + * which UI orientations are allowed. + * + * This variable is a space delimited list of the following values: + * "LandscapeLeft", "LandscapeRight", "Portrait" "PortraitUpsideDown" + */ +#define SDL_HINT_ORIENTATIONS "SDL_IOS_ORIENTATIONS" + +/** + * \brief A variable controlling the use of a sentinel event when polling the event queue + * + * This variable can be set to the following values: + * "0" - Disable poll sentinels + * "1" - Enable poll sentinels + * + * When polling for events, SDL_PumpEvents is used to gather new events from devices. + * If a device keeps producing new events between calls to SDL_PumpEvents, a poll loop will + * become stuck until the new events stop. + * This is most noticeable when moving a high frequency mouse. + * + * By default, poll sentinels are enabled. + */ +#define SDL_HINT_POLL_SENTINEL "SDL_POLL_SENTINEL" + +/** + * \brief Override for SDL_GetPreferredLocales() + * + * If set, this will be favored over anything the OS might report for the + * user's preferred locales. Changing this hint at runtime will not generate + * a SDL_LOCALECHANGED event (but if you can change the hint, you can push + * your own event, if you want). + * + * The format of this hint is a comma-separated list of language and locale, + * combined with an underscore, as is a common format: "en_GB". Locale is + * optional: "en". So you might have a list like this: "en_GB,jp,es_PT" + */ +#define SDL_HINT_PREFERRED_LOCALES "SDL_PREFERRED_LOCALES" + +/** + * \brief A variable describing the content orientation on QtWayland-based platforms. + * + * On QtWayland platforms, windows are rotated client-side to allow for custom + * transitions. In order to correctly position overlays (e.g. volume bar) and + * gestures (e.g. events view, close/minimize gestures), the system needs to + * know in which orientation the application is currently drawing its contents. + * + * This does not cause the window to be rotated or resized, the application + * needs to take care of drawing the content in the right orientation (the + * framebuffer is always in portrait mode). + * + * This variable can be one of the following values: + * "primary" (default), "portrait", "landscape", "inverted-portrait", "inverted-landscape" + * + * Since SDL 2.0.22 this variable accepts a comma-separated list of values above. + */ +#define SDL_HINT_QTWAYLAND_CONTENT_ORIENTATION "SDL_QTWAYLAND_CONTENT_ORIENTATION" + +/** + * \brief Flags to set on QtWayland windows to integrate with the native window manager. + * + * On QtWayland platforms, this hint controls the flags to set on the windows. + * For example, on Sailfish OS "OverridesSystemGestures" disables swipe gestures. + * + * This variable is a space-separated list of the following values (empty = no flags): + * "OverridesSystemGestures", "StaysOnTop", "BypassWindowManager" + */ +#define SDL_HINT_QTWAYLAND_WINDOW_FLAGS "SDL_QTWAYLAND_WINDOW_FLAGS" + +/** + * \brief A variable controlling whether the 2D render API is compatible or efficient. + * + * This variable can be set to the following values: + * + * "0" - Don't use batching to make rendering more efficient. + * "1" - Use batching, but might cause problems if app makes its own direct OpenGL calls. + * + * Up to SDL 2.0.9, the render API would draw immediately when requested. Now + * it batches up draw requests and sends them all to the GPU only when forced + * to (during SDL_RenderPresent, when changing render targets, by updating a + * texture that the batch needs, etc). This is significantly more efficient, + * but it can cause problems for apps that expect to render on top of the + * render API's output. As such, SDL will disable batching if a specific + * render backend is requested (since this might indicate that the app is + * planning to use the underlying graphics API directly). This hint can + * be used to explicitly request batching in this instance. It is a contract + * that you will either never use the underlying graphics API directly, or + * if you do, you will call SDL_RenderFlush() before you do so any current + * batch goes to the GPU before your work begins. Not following this contract + * will result in undefined behavior. + */ +#define SDL_HINT_RENDER_BATCHING "SDL_RENDER_BATCHING" + +/** + * \brief A variable controlling how the 2D render API renders lines + * + * This variable can be set to the following values: + * "0" - Use the default line drawing method (Bresenham's line algorithm as of SDL 2.0.20) + * "1" - Use the driver point API using Bresenham's line algorithm (correct, draws many points) + * "2" - Use the driver line API (occasionally misses line endpoints based on hardware driver quirks, was the default before 2.0.20) + * "3" - Use the driver geometry API (correct, draws thicker diagonal lines) + * + * This variable should be set when the renderer is created. + */ +#define SDL_HINT_RENDER_LINE_METHOD "SDL_RENDER_LINE_METHOD" + +/** + * \brief A variable controlling whether to enable Direct3D 11+'s Debug Layer. + * + * This variable does not have any effect on the Direct3D 9 based renderer. + * + * This variable can be set to the following values: + * "0" - Disable Debug Layer use + * "1" - Enable Debug Layer use + * + * By default, SDL does not use Direct3D Debug Layer. + */ +#define SDL_HINT_RENDER_DIRECT3D11_DEBUG "SDL_RENDER_DIRECT3D11_DEBUG" + +/** + * \brief A variable controlling whether the Direct3D device is initialized for thread-safe operations. + * + * This variable can be set to the following values: + * "0" - Thread-safety is not enabled (faster) + * "1" - Thread-safety is enabled + * + * By default the Direct3D device is created with thread-safety disabled. + */ +#define SDL_HINT_RENDER_DIRECT3D_THREADSAFE "SDL_RENDER_DIRECT3D_THREADSAFE" + +/** + * \brief A variable specifying which render driver to use. + * + * If the application doesn't pick a specific renderer to use, this variable + * specifies the name of the preferred renderer. If the preferred renderer + * can't be initialized, the normal default renderer is used. + * + * This variable is case insensitive and can be set to the following values: + * "direct3d" + * "direct3d11" + * "direct3d12" + * "opengl" + * "opengles2" + * "opengles" + * "metal" + * "software" + * + * The default varies by platform, but it's the first one in the list that + * is available on the current platform. + */ +#define SDL_HINT_RENDER_DRIVER "SDL_RENDER_DRIVER" + +/** + * \brief A variable controlling the scaling policy for SDL_RenderSetLogicalSize. + * + * This variable can be set to the following values: + * "0" or "letterbox" - Uses letterbox/sidebars to fit the entire rendering on screen + * "1" or "overscan" - Will zoom the rendering so it fills the entire screen, allowing edges to be drawn offscreen + * + * By default letterbox is used + */ +#define SDL_HINT_RENDER_LOGICAL_SIZE_MODE "SDL_RENDER_LOGICAL_SIZE_MODE" + +/** + * \brief A variable controlling whether the OpenGL render driver uses shaders if they are available. + * + * This variable can be set to the following values: + * "0" - Disable shaders + * "1" - Enable shaders + * + * By default shaders are used if OpenGL supports them. + */ +#define SDL_HINT_RENDER_OPENGL_SHADERS "SDL_RENDER_OPENGL_SHADERS" + +/** + * \brief A variable controlling the scaling quality + * + * This variable can be set to the following values: + * "0" or "nearest" - Nearest pixel sampling + * "1" or "linear" - Linear filtering (supported by OpenGL and Direct3D) + * "2" or "best" - Currently this is the same as "linear" + * + * By default nearest pixel sampling is used + */ +#define SDL_HINT_RENDER_SCALE_QUALITY "SDL_RENDER_SCALE_QUALITY" + +/** + * \brief A variable controlling whether updates to the SDL screen surface should be synchronized with the vertical refresh, to avoid tearing. + * + * This variable can be set to the following values: + * "0" - Disable vsync + * "1" - Enable vsync + * + * By default SDL does not sync screen surface updates with vertical refresh. + */ +#define SDL_HINT_RENDER_VSYNC "SDL_RENDER_VSYNC" + +/** + * \brief A variable controlling whether the Metal render driver select low power device over default one + * + * This variable can be set to the following values: + * "0" - Use the prefered OS device + * "1" - Select a low power one + * + * By default the prefered OS device is used. + */ +#define SDL_HINT_RENDER_METAL_PREFER_LOW_POWER_DEVICE "SDL_RENDER_METAL_PREFER_LOW_POWER_DEVICE" + +/** + * \brief A variable controlling if VSYNC is automatically disable if doesn't reach the enough FPS + * + * This variable can be set to the following values: + * "0" - It will be using VSYNC as defined in the main flag. Default + * "1" - If VSYNC was previously enabled, then it will disable VSYNC if doesn't reach enough speed + * + * By default SDL does not enable the automatic VSYNC + */ +#define SDL_HINT_PS2_DYNAMIC_VSYNC "SDL_PS2_DYNAMIC_VSYNC" + +/** + * \brief A variable to control whether the return key on the soft keyboard + * should hide the soft keyboard on Android and iOS. + * + * The variable can be set to the following values: + * "0" - The return key will be handled as a key event. This is the behaviour of SDL <= 2.0.3. (default) + * "1" - The return key will hide the keyboard. + * + * The value of this hint is used at runtime, so it can be changed at any time. + */ +#define SDL_HINT_RETURN_KEY_HIDES_IME "SDL_RETURN_KEY_HIDES_IME" + +/** + * \brief Tell SDL which Dispmanx layer to use on a Raspberry PI + * + * Also known as Z-order. The variable can take a negative or positive value. + * The default is 10000. + */ +#define SDL_HINT_RPI_VIDEO_LAYER "SDL_RPI_VIDEO_LAYER" + +/** + * \brief Specify an "activity name" for screensaver inhibition. + * + * Some platforms, notably Linux desktops, list the applications which are + * inhibiting the screensaver or other power-saving features. + * + * This hint lets you specify the "activity name" sent to the OS when + * SDL_DisableScreenSaver() is used (or the screensaver is automatically + * disabled). The contents of this hint are used when the screensaver is + * disabled. You should use a string that describes what your program is doing + * (and, therefore, why the screensaver is disabled). For example, "Playing a + * game" or "Watching a video". + * + * Setting this to "" or leaving it unset will have SDL use a reasonable + * default: "Playing a game" or something similar. + * + * On targets where this is not supported, this hint does nothing. + */ +#define SDL_HINT_SCREENSAVER_INHIBIT_ACTIVITY_NAME "SDL_SCREENSAVER_INHIBIT_ACTIVITY_NAME" + +/** + * \brief Specifies whether SDL_THREAD_PRIORITY_TIME_CRITICAL should be treated as realtime. + * + * On some platforms, like Linux, a realtime priority thread may be subject to restrictions + * that require special handling by the application. This hint exists to let SDL know that + * the app is prepared to handle said restrictions. + * + * On Linux, SDL will apply the following configuration to any thread that becomes realtime: + * * The SCHED_RESET_ON_FORK bit will be set on the scheduling policy, + * * An RLIMIT_RTTIME budget will be configured to the rtkit specified limit. + * * Exceeding this limit will result in the kernel sending SIGKILL to the app, + * * Refer to the man pages for more information. + * + * This variable can be set to the following values: + * "0" - default platform specific behaviour + * "1" - Force SDL_THREAD_PRIORITY_TIME_CRITICAL to a realtime scheduling policy + */ +#define SDL_HINT_THREAD_FORCE_REALTIME_TIME_CRITICAL "SDL_THREAD_FORCE_REALTIME_TIME_CRITICAL" + +/** +* \brief A string specifying additional information to use with SDL_SetThreadPriority. +* +* By default SDL_SetThreadPriority will make appropriate system changes in order to +* apply a thread priority. For example on systems using pthreads the scheduler policy +* is changed automatically to a policy that works well with a given priority. +* Code which has specific requirements can override SDL's default behavior with this hint. +* +* pthread hint values are "current", "other", "fifo" and "rr". +* Currently no other platform hint values are defined but may be in the future. +* +* \note On Linux, the kernel may send SIGKILL to realtime tasks which exceed the distro +* configured execution budget for rtkit. This budget can be queried through RLIMIT_RTTIME +* after calling SDL_SetThreadPriority(). +*/ +#define SDL_HINT_THREAD_PRIORITY_POLICY "SDL_THREAD_PRIORITY_POLICY" + +/** +* \brief A string specifying SDL's threads stack size in bytes or "0" for the backend's default size +* +* Use this hint in case you need to set SDL's threads stack size to other than the default. +* This is specially useful if you build SDL against a non glibc libc library (such as musl) which +* provides a relatively small default thread stack size (a few kilobytes versus the default 8MB glibc uses). +* Support for this hint is currently available only in the pthread, Windows, and PSP backend. +* +* Instead of this hint, in 2.0.9 and later, you can use +* SDL_CreateThreadWithStackSize(). This hint only works with the classic +* SDL_CreateThread(). +*/ +#define SDL_HINT_THREAD_STACK_SIZE "SDL_THREAD_STACK_SIZE" + +/** + * \brief A variable that controls the timer resolution, in milliseconds. + * + * The higher resolution the timer, the more frequently the CPU services + * timer interrupts, and the more precise delays are, but this takes up + * power and CPU time. This hint is only used on Windows. + * + * See this blog post for more information: + * http://randomascii.wordpress.com/2013/07/08/windows-timer-resolution-megawatts-wasted/ + * + * If this variable is set to "0", the system timer resolution is not set. + * + * The default value is "1". This hint may be set at any time. + */ +#define SDL_HINT_TIMER_RESOLUTION "SDL_TIMER_RESOLUTION" + +/** + * \brief A variable controlling whether touch events should generate synthetic mouse events + * + * This variable can be set to the following values: + * "0" - Touch events will not generate mouse events + * "1" - Touch events will generate mouse events + * + * By default SDL will generate mouse events for touch events + */ +#define SDL_HINT_TOUCH_MOUSE_EVENTS "SDL_TOUCH_MOUSE_EVENTS" + +/** + * \brief A variable controlling which touchpad should generate synthetic mouse events + * + * This variable can be set to the following values: + * "0" - Only front touchpad should generate mouse events. Default + * "1" - Only back touchpad should generate mouse events. + * "2" - Both touchpads should generate mouse events. + * + * By default SDL will generate mouse events for all touch devices + */ +#define SDL_HINT_VITA_TOUCH_MOUSE_DEVICE "SDL_HINT_VITA_TOUCH_MOUSE_DEVICE" + +/** + * \brief A variable controlling whether the Android / tvOS remotes + * should be listed as joystick devices, instead of sending keyboard events. + * + * This variable can be set to the following values: + * "0" - Remotes send enter/escape/arrow key events + * "1" - Remotes are available as 2 axis, 2 button joysticks (the default). + */ +#define SDL_HINT_TV_REMOTE_AS_JOYSTICK "SDL_TV_REMOTE_AS_JOYSTICK" + +/** + * \brief A variable controlling whether the screensaver is enabled. + * + * This variable can be set to the following values: + * "0" - Disable screensaver + * "1" - Enable screensaver + * + * By default SDL will disable the screensaver. + */ +#define SDL_HINT_VIDEO_ALLOW_SCREENSAVER "SDL_VIDEO_ALLOW_SCREENSAVER" + +/** + * \brief Tell the video driver that we only want a double buffer. + * + * By default, most lowlevel 2D APIs will use a triple buffer scheme that + * wastes no CPU time on waiting for vsync after issuing a flip, but + * introduces a frame of latency. On the other hand, using a double buffer + * scheme instead is recommended for cases where low latency is an important + * factor because we save a whole frame of latency. + * We do so by waiting for vsync immediately after issuing a flip, usually just + * after eglSwapBuffers call in the backend's *_SwapWindow function. + * + * Since it's driver-specific, it's only supported where possible and + * implemented. Currently supported the following drivers: + * + * - KMSDRM (kmsdrm) + * - Raspberry Pi (raspberrypi) + */ +#define SDL_HINT_VIDEO_DOUBLE_BUFFER "SDL_VIDEO_DOUBLE_BUFFER" + +/** + * \brief A variable controlling whether the EGL window is allowed to be + * composited as transparent, rather than opaque. + * + * Most window systems will always render windows opaque, even if the surface + * format has an alpha channel. This is not always true, however, so by default + * SDL will try to enforce opaque composition. To override this behavior, you + * can set this hint to "1". + */ +#define SDL_HINT_VIDEO_EGL_ALLOW_TRANSPARENCY "SDL_VIDEO_EGL_ALLOW_TRANSPARENCY" + +/** + * \brief A variable controlling whether the graphics context is externally managed. + * + * This variable can be set to the following values: + * "0" - SDL will manage graphics contexts that are attached to windows. + * "1" - Disable graphics context management on windows. + * + * By default SDL will manage OpenGL contexts in certain situations. For example, on Android the + * context will be automatically saved and restored when pausing the application. Additionally, some + * platforms will assume usage of OpenGL if Vulkan isn't used. Setting this to "1" will prevent this + * behavior, which is desireable when the application manages the graphics context, such as + * an externally managed OpenGL context or attaching a Vulkan surface to the window. + */ +#define SDL_HINT_VIDEO_EXTERNAL_CONTEXT "SDL_VIDEO_EXTERNAL_CONTEXT" + +/** + * \brief If set to 1, then do not allow high-DPI windows. ("Retina" on Mac and iOS) + */ +#define SDL_HINT_VIDEO_HIGHDPI_DISABLED "SDL_VIDEO_HIGHDPI_DISABLED" + +/** + * \brief A variable that dictates policy for fullscreen Spaces on Mac OS X. + * + * This hint only applies to Mac OS X. + * + * The variable can be set to the following values: + * "0" - Disable Spaces support (FULLSCREEN_DESKTOP won't use them and + * SDL_WINDOW_RESIZABLE windows won't offer the "fullscreen" + * button on their titlebars). + * "1" - Enable Spaces support (FULLSCREEN_DESKTOP will use them and + * SDL_WINDOW_RESIZABLE windows will offer the "fullscreen" + * button on their titlebars). + * + * The default value is "1". This hint must be set before any windows are created. + */ +#define SDL_HINT_VIDEO_MAC_FULLSCREEN_SPACES "SDL_VIDEO_MAC_FULLSCREEN_SPACES" + +/** + * \brief Minimize your SDL_Window if it loses key focus when in fullscreen mode. Defaults to false. + * \warning Before SDL 2.0.14, this defaulted to true! In 2.0.14, we're + * seeing if "true" causes more problems than it solves in modern times. + * + */ +#define SDL_HINT_VIDEO_MINIMIZE_ON_FOCUS_LOSS "SDL_VIDEO_MINIMIZE_ON_FOCUS_LOSS" + +/** + * \brief A variable controlling whether the libdecor Wayland backend is allowed to be used. + * + * This variable can be set to the following values: + * "0" - libdecor use is disabled. + * "1" - libdecor use is enabled (default). + * + * libdecor is used over xdg-shell when xdg-decoration protocol is unavailable. + */ +#define SDL_HINT_VIDEO_WAYLAND_ALLOW_LIBDECOR "SDL_VIDEO_WAYLAND_ALLOW_LIBDECOR" + +/** + * \brief A variable controlling whether the libdecor Wayland backend is preferred over native decrations. + * + * When this hint is set, libdecor will be used to provide window decorations, even if xdg-decoration is + * available. (Note that, by default, libdecor will use xdg-decoration itself if available). + * + * This variable can be set to the following values: + * "0" - libdecor is enabled only if server-side decorations are unavailable. + * "1" - libdecor is always enabled if available. + * + * libdecor is used over xdg-shell when xdg-decoration protocol is unavailable. + */ +#define SDL_HINT_VIDEO_WAYLAND_PREFER_LIBDECOR "SDL_VIDEO_WAYLAND_PREFER_LIBDECOR" + +/** + * \brief A variable controlling whether video mode emulation is enabled under Wayland. + * + * When this hint is set, a standard set of emulated CVT video modes will be exposed for use by the application. + * If it is disabled, the only modes exposed will be the logical desktop size and, in the case of a scaled + * desktop, the native display resolution. + * + * This variable can be set to the following values: + * "0" - Video mode emulation is disabled. + * "1" - Video mode emulation is enabled. + * + * By default video mode emulation is enabled. + */ +#define SDL_HINT_VIDEO_WAYLAND_MODE_EMULATION "SDL_VIDEO_WAYLAND_MODE_EMULATION" + +/** + * \brief Enable or disable mouse pointer warp emulation, needed by some older games. + * + * When this hint is set, any SDL will emulate mouse warps using relative mouse mode. + * This is required for some older games (such as Source engine games), which warp the + * mouse to the centre of the screen rather than using relative mouse motion. Note that + * relative mouse mode may have different mouse acceleration behaviour than pointer warps. + * + * This variable can be set to the following values: + * "0" - All mouse warps fail, as mouse warping is not available under wayland. + * "1" - Some mouse warps will be emulated by forcing relative mouse mode. + * + * If not set, this is automatically enabled unless an application uses relative mouse + * mode directly. + */ +#define SDL_HINT_VIDEO_WAYLAND_EMULATE_MOUSE_WARP "SDL_VIDEO_WAYLAND_EMULATE_MOUSE_WARP" + +/** +* \brief A variable that is the address of another SDL_Window* (as a hex string formatted with "%p"). +* +* If this hint is set before SDL_CreateWindowFrom() and the SDL_Window* it is set to has +* SDL_WINDOW_OPENGL set (and running on WGL only, currently), then two things will occur on the newly +* created SDL_Window: +* +* 1. Its pixel format will be set to the same pixel format as this SDL_Window. This is +* needed for example when sharing an OpenGL context across multiple windows. +* +* 2. The flag SDL_WINDOW_OPENGL will be set on the new window so it can be used for +* OpenGL rendering. +* +* This variable can be set to the following values: +* The address (as a string "%p") of the SDL_Window* that new windows created with SDL_CreateWindowFrom() should +* share a pixel format with. +*/ +#define SDL_HINT_VIDEO_WINDOW_SHARE_PIXEL_FORMAT "SDL_VIDEO_WINDOW_SHARE_PIXEL_FORMAT" + +/** + * \brief When calling SDL_CreateWindowFrom(), make the window compatible with OpenGL. + * + * This variable can be set to the following values: + * "0" - Don't add any graphics flags to the SDL_WindowFlags + * "1" - Add SDL_WINDOW_OPENGL to the SDL_WindowFlags + * + * By default SDL will not make the foreign window compatible with OpenGL. + */ +#define SDL_HINT_VIDEO_FOREIGN_WINDOW_OPENGL "SDL_VIDEO_FOREIGN_WINDOW_OPENGL" + +/** + * \brief When calling SDL_CreateWindowFrom(), make the window compatible with Vulkan. + * + * This variable can be set to the following values: + * "0" - Don't add any graphics flags to the SDL_WindowFlags + * "1" - Add SDL_WINDOW_VULKAN to the SDL_WindowFlags + * + * By default SDL will not make the foreign window compatible with Vulkan. + */ +#define SDL_HINT_VIDEO_FOREIGN_WINDOW_VULKAN "SDL_VIDEO_FOREIGN_WINDOW_VULKAN" + +/** +* \brief A variable specifying which shader compiler to preload when using the Chrome ANGLE binaries +* +* SDL has EGL and OpenGL ES2 support on Windows via the ANGLE project. It +* can use two different sets of binaries, those compiled by the user from source +* or those provided by the Chrome browser. In the later case, these binaries require +* that SDL loads a DLL providing the shader compiler. +* +* This variable can be set to the following values: +* "d3dcompiler_46.dll" - default, best for Vista or later. +* "d3dcompiler_43.dll" - for XP support. +* "none" - do not load any library, useful if you compiled ANGLE from source and included the compiler in your binaries. +* +*/ +#define SDL_HINT_VIDEO_WIN_D3DCOMPILER "SDL_VIDEO_WIN_D3DCOMPILER" + +/** + * \brief A variable controlling whether X11 should use GLX or EGL by default + * + * This variable can be set to the following values: + * "0" - Use GLX + * "1" - Use EGL + * + * By default SDL will use GLX when both are present. + */ +#define SDL_HINT_VIDEO_X11_FORCE_EGL "SDL_VIDEO_X11_FORCE_EGL" + +/** + * \brief A variable controlling whether the X11 _NET_WM_BYPASS_COMPOSITOR hint should be used. + * + * This variable can be set to the following values: + * "0" - Disable _NET_WM_BYPASS_COMPOSITOR + * "1" - Enable _NET_WM_BYPASS_COMPOSITOR + * + * By default SDL will use _NET_WM_BYPASS_COMPOSITOR + * + */ +#define SDL_HINT_VIDEO_X11_NET_WM_BYPASS_COMPOSITOR "SDL_VIDEO_X11_NET_WM_BYPASS_COMPOSITOR" + +/** + * \brief A variable controlling whether the X11 _NET_WM_PING protocol should be supported. + * + * This variable can be set to the following values: + * "0" - Disable _NET_WM_PING + * "1" - Enable _NET_WM_PING + * + * By default SDL will use _NET_WM_PING, but for applications that know they + * will not always be able to respond to ping requests in a timely manner they can + * turn it off to avoid the window manager thinking the app is hung. + * The hint is checked in CreateWindow. + */ +#define SDL_HINT_VIDEO_X11_NET_WM_PING "SDL_VIDEO_X11_NET_WM_PING" + +/** + * \brief A variable forcing the visual ID chosen for new X11 windows + * + */ +#define SDL_HINT_VIDEO_X11_WINDOW_VISUALID "SDL_VIDEO_X11_WINDOW_VISUALID" + +/** + * \brief A no-longer-used variable controlling whether the X11 Xinerama extension should be used. + * + * Before SDL 2.0.24, this would let apps and users disable Xinerama support on X11. + * Now SDL never uses Xinerama, and does not check for this hint at all. + * The preprocessor define is left here for source compatibility. + */ +#define SDL_HINT_VIDEO_X11_XINERAMA "SDL_VIDEO_X11_XINERAMA" + +/** + * \brief A variable controlling whether the X11 XRandR extension should be used. + * + * This variable can be set to the following values: + * "0" - Disable XRandR + * "1" - Enable XRandR + * + * By default SDL will use XRandR. + */ +#define SDL_HINT_VIDEO_X11_XRANDR "SDL_VIDEO_X11_XRANDR" + +/** + * \brief A no-longer-used variable controlling whether the X11 VidMode extension should be used. + * + * Before SDL 2.0.24, this would let apps and users disable XVidMode support on X11. + * Now SDL never uses XVidMode, and does not check for this hint at all. + * The preprocessor define is left here for source compatibility. + */ +#define SDL_HINT_VIDEO_X11_XVIDMODE "SDL_VIDEO_X11_XVIDMODE" + +/** + * \brief Controls how the fact chunk affects the loading of a WAVE file. + * + * The fact chunk stores information about the number of samples of a WAVE + * file. The Standards Update from Microsoft notes that this value can be used + * to 'determine the length of the data in seconds'. This is especially useful + * for compressed formats (for which this is a mandatory chunk) if they produce + * multiple sample frames per block and truncating the block is not allowed. + * The fact chunk can exactly specify how many sample frames there should be + * in this case. + * + * Unfortunately, most application seem to ignore the fact chunk and so SDL + * ignores it by default as well. + * + * This variable can be set to the following values: + * + * "truncate" - Use the number of samples to truncate the wave data if + * the fact chunk is present and valid + * "strict" - Like "truncate", but raise an error if the fact chunk + * is invalid, not present for non-PCM formats, or if the + * data chunk doesn't have that many samples + * "ignorezero" - Like "truncate", but ignore fact chunk if the number of + * samples is zero + * "ignore" - Ignore fact chunk entirely (default) + */ +#define SDL_HINT_WAVE_FACT_CHUNK "SDL_WAVE_FACT_CHUNK" + +/** + * \brief Controls how the size of the RIFF chunk affects the loading of a WAVE file. + * + * The size of the RIFF chunk (which includes all the sub-chunks of the WAVE + * file) is not always reliable. In case the size is wrong, it's possible to + * just ignore it and step through the chunks until a fixed limit is reached. + * + * Note that files that have trailing data unrelated to the WAVE file or + * corrupt files may slow down the loading process without a reliable boundary. + * By default, SDL stops after 10000 chunks to prevent wasting time. Use the + * environment variable SDL_WAVE_CHUNK_LIMIT to adjust this value. + * + * This variable can be set to the following values: + * + * "force" - Always use the RIFF chunk size as a boundary for the chunk search + * "ignorezero" - Like "force", but a zero size searches up to 4 GiB (default) + * "ignore" - Ignore the RIFF chunk size and always search up to 4 GiB + * "maximum" - Search for chunks until the end of file (not recommended) + */ +#define SDL_HINT_WAVE_RIFF_CHUNK_SIZE "SDL_WAVE_RIFF_CHUNK_SIZE" + +/** + * \brief Controls how a truncated WAVE file is handled. + * + * A WAVE file is considered truncated if any of the chunks are incomplete or + * the data chunk size is not a multiple of the block size. By default, SDL + * decodes until the first incomplete block, as most applications seem to do. + * + * This variable can be set to the following values: + * + * "verystrict" - Raise an error if the file is truncated + * "strict" - Like "verystrict", but the size of the RIFF chunk is ignored + * "dropframe" - Decode until the first incomplete sample frame + * "dropblock" - Decode until the first incomplete block (default) + */ +#define SDL_HINT_WAVE_TRUNCATION "SDL_WAVE_TRUNCATION" + +/** + * \brief Tell SDL not to name threads on Windows with the 0x406D1388 Exception. + * The 0x406D1388 Exception is a trick used to inform Visual Studio of a + * thread's name, but it tends to cause problems with other debuggers, + * and the .NET runtime. Note that SDL 2.0.6 and later will still use + * the (safer) SetThreadDescription API, introduced in the Windows 10 + * Creators Update, if available. + * + * The variable can be set to the following values: + * "0" - SDL will raise the 0x406D1388 Exception to name threads. + * This is the default behavior of SDL <= 2.0.4. + * "1" - SDL will not raise this exception, and threads will be unnamed. (default) + * This is necessary with .NET languages or debuggers that aren't Visual Studio. + */ +#define SDL_HINT_WINDOWS_DISABLE_THREAD_NAMING "SDL_WINDOWS_DISABLE_THREAD_NAMING" + +/** + * \brief Controls whether menus can be opened with their keyboard shortcut (Alt+mnemonic). + * + * If the mnemonics are enabled, then menus can be opened by pressing the Alt + * key and the corresponding mnemonic (for example, Alt+F opens the File menu). + * However, in case an invalid mnemonic is pressed, Windows makes an audible + * beep to convey that nothing happened. This is true even if the window has + * no menu at all! + * + * Because most SDL applications don't have menus, and some want to use the Alt + * key for other purposes, SDL disables mnemonics (and the beeping) by default. + * + * Note: This also affects keyboard events: with mnemonics enabled, when a + * menu is opened from the keyboard, you will not receive a KEYUP event for + * the mnemonic key, and *might* not receive one for Alt. + * + * This variable can be set to the following values: + * "0" - Alt+mnemonic does nothing, no beeping. (default) + * "1" - Alt+mnemonic opens menus, invalid mnemonics produce a beep. + */ +#define SDL_HINT_WINDOWS_ENABLE_MENU_MNEMONICS "SDL_WINDOWS_ENABLE_MENU_MNEMONICS" + +/** + * \brief A variable controlling whether the windows message loop is processed by SDL + * + * This variable can be set to the following values: + * "0" - The window message loop is not run + * "1" - The window message loop is processed in SDL_PumpEvents() + * + * By default SDL will process the windows message loop + */ +#define SDL_HINT_WINDOWS_ENABLE_MESSAGELOOP "SDL_WINDOWS_ENABLE_MESSAGELOOP" + +/** + * \brief Force SDL to use Critical Sections for mutexes on Windows. + * On Windows 7 and newer, Slim Reader/Writer Locks are available. + * They offer better performance, allocate no kernel ressources and + * use less memory. SDL will fall back to Critical Sections on older + * OS versions or if forced to by this hint. + * + * This variable can be set to the following values: + * "0" - Use SRW Locks when available. If not, fall back to Critical Sections. (default) + * "1" - Force the use of Critical Sections in all cases. + * + */ +#define SDL_HINT_WINDOWS_FORCE_MUTEX_CRITICAL_SECTIONS "SDL_WINDOWS_FORCE_MUTEX_CRITICAL_SECTIONS" + +/** + * \brief Force SDL to use Kernel Semaphores on Windows. + * Kernel Semaphores are inter-process and require a context + * switch on every interaction. On Windows 8 and newer, the + * WaitOnAddress API is available. Using that and atomics to + * implement semaphores increases performance. + * SDL will fall back to Kernel Objects on older OS versions + * or if forced to by this hint. + * + * This variable can be set to the following values: + * "0" - Use Atomics and WaitOnAddress API when available. If not, fall back to Kernel Objects. (default) + * "1" - Force the use of Kernel Objects in all cases. + * + */ +#define SDL_HINT_WINDOWS_FORCE_SEMAPHORE_KERNEL "SDL_WINDOWS_FORCE_SEMAPHORE_KERNEL" + +/** + * \brief A variable to specify custom icon resource id from RC file on Windows platform + */ +#define SDL_HINT_WINDOWS_INTRESOURCE_ICON "SDL_WINDOWS_INTRESOURCE_ICON" +#define SDL_HINT_WINDOWS_INTRESOURCE_ICON_SMALL "SDL_WINDOWS_INTRESOURCE_ICON_SMALL" + +/** + * \brief Tell SDL not to generate window-close events for Alt+F4 on Windows. + * + * The variable can be set to the following values: + * "0" - SDL will generate a window-close event when it sees Alt+F4. + * "1" - SDL will only do normal key handling for Alt+F4. + */ +#define SDL_HINT_WINDOWS_NO_CLOSE_ON_ALT_F4 "SDL_WINDOWS_NO_CLOSE_ON_ALT_F4" + +/** + * \brief Use the D3D9Ex API introduced in Windows Vista, instead of normal D3D9. + * Direct3D 9Ex contains changes to state management that can eliminate device + * loss errors during scenarios like Alt+Tab or UAC prompts. D3D9Ex may require + * some changes to your application to cope with the new behavior, so this + * is disabled by default. + * + * This hint must be set before initializing the video subsystem. + * + * For more information on Direct3D 9Ex, see: + * - https://docs.microsoft.com/en-us/windows/win32/direct3darticles/graphics-apis-in-windows-vista#direct3d-9ex + * - https://docs.microsoft.com/en-us/windows/win32/direct3darticles/direct3d-9ex-improvements + * + * This variable can be set to the following values: + * "0" - Use the original Direct3D 9 API (default) + * "1" - Use the Direct3D 9Ex API on Vista and later (and fall back if D3D9Ex is unavailable) + * + */ +#define SDL_HINT_WINDOWS_USE_D3D9EX "SDL_WINDOWS_USE_D3D9EX" + +/** + * \brief Controls whether SDL will declare the process to be DPI aware. + * + * This hint must be set before initializing the video subsystem. + * + * The main purpose of declaring DPI awareness is to disable OS bitmap scaling of SDL windows on monitors with + * a DPI scale factor. + * + * This hint is equivalent to requesting DPI awareness via external means (e.g. calling SetProcessDpiAwarenessContext) + * and does not cause SDL to use a virtualized coordinate system, so it will generally give you 1 SDL coordinate = 1 pixel + * even on high-DPI displays. + * + * For more information, see: + * https://docs.microsoft.com/en-us/windows/win32/hidpi/high-dpi-desktop-application-development-on-windows + * + * This variable can be set to the following values: + * "" - Do not change the DPI awareness (default). + * "unaware" - Declare the process as DPI unaware. (Windows 8.1 and later). + * "system" - Request system DPI awareness. (Vista and later). + * "permonitor" - Request per-monitor DPI awareness. (Windows 8.1 and later). + * "permonitorv2" - Request per-monitor V2 DPI awareness. (Windows 10, version 1607 and later). + * The most visible difference from "permonitor" is that window title bar will be scaled + * to the visually correct size when dragging between monitors with different scale factors. + * This is the preferred DPI awareness level. + * + * If the requested DPI awareness is not available on the currently running OS, SDL will try to request the best + * available match. + */ +#define SDL_HINT_WINDOWS_DPI_AWARENESS "SDL_WINDOWS_DPI_AWARENESS" + +/** + * \brief Uses DPI-scaled points as the SDL coordinate system on Windows. + * + * This changes the SDL coordinate system units to be DPI-scaled points, rather than pixels everywhere. + * This means windows will be appropriately sized, even when created on high-DPI displays with scaling. + * + * e.g. requesting a 640x480 window from SDL, on a display with 125% scaling in Windows display settings, + * will create a window with an 800x600 client area (in pixels). + * + * Setting this to "1" implicitly requests process DPI awareness (setting SDL_WINDOWS_DPI_AWARENESS is unnecessary), + * and forces SDL_WINDOW_ALLOW_HIGHDPI on all windows. + * + * This variable can be set to the following values: + * "0" - SDL coordinates equal Windows coordinates. No automatic window resizing when dragging + * between monitors with different scale factors (unless this is performed by + * Windows itself, which is the case when the process is DPI unaware). + * "1" - SDL coordinates are in DPI-scaled points. Automatically resize windows as needed on + * displays with non-100% scale factors. + */ +#define SDL_HINT_WINDOWS_DPI_SCALING "SDL_WINDOWS_DPI_SCALING" + +/** + * \brief A variable controlling whether the window frame and title bar are interactive when the cursor is hidden + * + * This variable can be set to the following values: + * "0" - The window frame is not interactive when the cursor is hidden (no move, resize, etc) + * "1" - The window frame is interactive when the cursor is hidden + * + * By default SDL will allow interaction with the window frame when the cursor is hidden + */ +#define SDL_HINT_WINDOW_FRAME_USABLE_WHILE_CURSOR_HIDDEN "SDL_WINDOW_FRAME_USABLE_WHILE_CURSOR_HIDDEN" + +/** +* \brief A variable controlling whether the window is activated when the SDL_ShowWindow function is called +* +* This variable can be set to the following values: +* "0" - The window is activated when the SDL_ShowWindow function is called +* "1" - The window is not activated when the SDL_ShowWindow function is called +* +* By default SDL will activate the window when the SDL_ShowWindow function is called +*/ +#define SDL_HINT_WINDOW_NO_ACTIVATION_WHEN_SHOWN "SDL_WINDOW_NO_ACTIVATION_WHEN_SHOWN" + +/** \brief Allows back-button-press events on Windows Phone to be marked as handled + * + * Windows Phone devices typically feature a Back button. When pressed, + * the OS will emit back-button-press events, which apps are expected to + * handle in an appropriate manner. If apps do not explicitly mark these + * events as 'Handled', then the OS will invoke its default behavior for + * unhandled back-button-press events, which on Windows Phone 8 and 8.1 is to + * terminate the app (and attempt to switch to the previous app, or to the + * device's home screen). + * + * Setting the SDL_HINT_WINRT_HANDLE_BACK_BUTTON hint to "1" will cause SDL + * to mark back-button-press events as Handled, if and when one is sent to + * the app. + * + * Internally, Windows Phone sends back button events as parameters to + * special back-button-press callback functions. Apps that need to respond + * to back-button-press events are expected to register one or more + * callback functions for such, shortly after being launched (during the + * app's initialization phase). After the back button is pressed, the OS + * will invoke these callbacks. If the app's callback(s) do not explicitly + * mark the event as handled by the time they return, or if the app never + * registers one of these callback, the OS will consider the event + * un-handled, and it will apply its default back button behavior (terminate + * the app). + * + * SDL registers its own back-button-press callback with the Windows Phone + * OS. This callback will emit a pair of SDL key-press events (SDL_KEYDOWN + * and SDL_KEYUP), each with a scancode of SDL_SCANCODE_AC_BACK, after which + * it will check the contents of the hint, SDL_HINT_WINRT_HANDLE_BACK_BUTTON. + * If the hint's value is set to "1", the back button event's Handled + * property will get set to 'true'. If the hint's value is set to something + * else, or if it is unset, SDL will leave the event's Handled property + * alone. (By default, the OS sets this property to 'false', to note.) + * + * SDL apps can either set SDL_HINT_WINRT_HANDLE_BACK_BUTTON well before a + * back button is pressed, or can set it in direct-response to a back button + * being pressed. + * + * In order to get notified when a back button is pressed, SDL apps should + * register a callback function with SDL_AddEventWatch(), and have it listen + * for SDL_KEYDOWN events that have a scancode of SDL_SCANCODE_AC_BACK. + * (Alternatively, SDL_KEYUP events can be listened-for. Listening for + * either event type is suitable.) Any value of SDL_HINT_WINRT_HANDLE_BACK_BUTTON + * set by such a callback, will be applied to the OS' current + * back-button-press event. + * + * More details on back button behavior in Windows Phone apps can be found + * at the following page, on Microsoft's developer site: + * http://msdn.microsoft.com/en-us/library/windowsphone/develop/jj247550(v=vs.105).aspx + */ +#define SDL_HINT_WINRT_HANDLE_BACK_BUTTON "SDL_WINRT_HANDLE_BACK_BUTTON" + +/** \brief Label text for a WinRT app's privacy policy link + * + * Network-enabled WinRT apps must include a privacy policy. On Windows 8, 8.1, and RT, + * Microsoft mandates that this policy be available via the Windows Settings charm. + * SDL provides code to add a link there, with its label text being set via the + * optional hint, SDL_HINT_WINRT_PRIVACY_POLICY_LABEL. + * + * Please note that a privacy policy's contents are not set via this hint. A separate + * hint, SDL_HINT_WINRT_PRIVACY_POLICY_URL, is used to link to the actual text of the + * policy. + * + * The contents of this hint should be encoded as a UTF8 string. + * + * The default value is "Privacy Policy". This hint should only be set during app + * initialization, preferably before any calls to SDL_Init(). + * + * For additional information on linking to a privacy policy, see the documentation for + * SDL_HINT_WINRT_PRIVACY_POLICY_URL. + */ +#define SDL_HINT_WINRT_PRIVACY_POLICY_LABEL "SDL_WINRT_PRIVACY_POLICY_LABEL" + +/** + * \brief A URL to a WinRT app's privacy policy + * + * All network-enabled WinRT apps must make a privacy policy available to its + * users. On Windows 8, 8.1, and RT, Microsoft mandates that this policy be + * be available in the Windows Settings charm, as accessed from within the app. + * SDL provides code to add a URL-based link there, which can point to the app's + * privacy policy. + * + * To setup a URL to an app's privacy policy, set SDL_HINT_WINRT_PRIVACY_POLICY_URL + * before calling any SDL_Init() functions. The contents of the hint should + * be a valid URL. For example, "http://www.example.com". + * + * The default value is "", which will prevent SDL from adding a privacy policy + * link to the Settings charm. This hint should only be set during app init. + * + * The label text of an app's "Privacy Policy" link may be customized via another + * hint, SDL_HINT_WINRT_PRIVACY_POLICY_LABEL. + * + * Please note that on Windows Phone, Microsoft does not provide standard UI + * for displaying a privacy policy link, and as such, SDL_HINT_WINRT_PRIVACY_POLICY_URL + * will not get used on that platform. Network-enabled phone apps should display + * their privacy policy through some other, in-app means. + */ +#define SDL_HINT_WINRT_PRIVACY_POLICY_URL "SDL_WINRT_PRIVACY_POLICY_URL" + +/** + * \brief Mark X11 windows as override-redirect. + * + * If set, this _might_ increase framerate at the expense of the desktop + * not working as expected. Override-redirect windows aren't noticed by the + * window manager at all. + * + * You should probably only use this for fullscreen windows, and you probably + * shouldn't even use it for that. But it's here if you want to try! + */ +#define SDL_HINT_X11_FORCE_OVERRIDE_REDIRECT "SDL_X11_FORCE_OVERRIDE_REDIRECT" + +/** + * \brief A variable that lets you disable the detection and use of Xinput gamepad devices + * + * The variable can be set to the following values: + * "0" - Disable XInput detection (only uses direct input) + * "1" - Enable XInput detection (the default) + */ +#define SDL_HINT_XINPUT_ENABLED "SDL_XINPUT_ENABLED" + + /** + * \brief A variable that lets you disable the detection and use of DirectInput gamepad devices + * + * The variable can be set to the following values: + * "0" - Disable DirectInput detection (only uses XInput) + * "1" - Enable DirectInput detection (the default) + */ +#define SDL_HINT_DIRECTINPUT_ENABLED "SDL_DIRECTINPUT_ENABLED" + +/** + * \brief A variable that causes SDL to use the old axis and button mapping for XInput devices. + * + * This hint is for backwards compatibility only and will be removed in SDL 2.1 + * + * The default value is "0". This hint must be set before SDL_Init() + */ +#define SDL_HINT_XINPUT_USE_OLD_JOYSTICK_MAPPING "SDL_XINPUT_USE_OLD_JOYSTICK_MAPPING" + +/** + * \brief A variable that causes SDL to not ignore audio "monitors" + * + * This is currently only used for PulseAudio and ignored elsewhere. + * + * By default, SDL ignores audio devices that aren't associated with physical + * hardware. Changing this hint to "1" will expose anything SDL sees that + * appears to be an audio source or sink. This will add "devices" to the list + * that the user probably doesn't want or need, but it can be useful in + * scenarios where you want to hook up SDL to some sort of virtual device, + * etc. + * + * The default value is "0". This hint must be set before SDL_Init(). + * + * This hint is available since SDL 2.0.16. Before then, virtual devices are + * always ignored. + */ +#define SDL_HINT_AUDIO_INCLUDE_MONITORS "SDL_AUDIO_INCLUDE_MONITORS" + +/** + * \brief A variable that forces X11 windows to create as a custom type. + * + * This is currently only used for X11 and ignored elsewhere. + * + * During SDL_CreateWindow, SDL uses the _NET_WM_WINDOW_TYPE X11 property + * to report to the window manager the type of window it wants to create. + * This might be set to various things if SDL_WINDOW_TOOLTIP or + * SDL_WINDOW_POPUP_MENU, etc, were specified. For "normal" windows that + * haven't set a specific type, this hint can be used to specify a custom + * type. For example, a dock window might set this to + * "_NET_WM_WINDOW_TYPE_DOCK". + * + * If not set or set to "", this hint is ignored. This hint must be set + * before the SDL_CreateWindow() call that it is intended to affect. + * + * This hint is available since SDL 2.0.22. + */ +#define SDL_HINT_X11_WINDOW_TYPE "SDL_X11_WINDOW_TYPE" + +/** + * \brief A variable that decides whether to send SDL_QUIT when closing the final window. + * + * By default, SDL sends an SDL_QUIT event when there is only one window + * and it receives an SDL_WINDOWEVENT_CLOSE event, under the assumption most + * apps would also take the loss of this window as a signal to terminate the + * program. + * + * However, it's not unreasonable in some cases to have the program continue + * to live on, perhaps to create new windows later. + * + * Changing this hint to "0" will cause SDL to not send an SDL_QUIT event + * when the final window is requesting to close. Note that in this case, + * there are still other legitimate reasons one might get an SDL_QUIT + * event: choosing "Quit" from the macOS menu bar, sending a SIGINT (ctrl-c) + * on Unix, etc. + * + * The default value is "1". This hint can be changed at any time. + * + * This hint is available since SDL 2.0.22. Before then, you always get + * an SDL_QUIT event when closing the final window. + */ +#define SDL_HINT_QUIT_ON_LAST_WINDOW_CLOSE "SDL_QUIT_ON_LAST_WINDOW_CLOSE" + + +/** + * \brief A variable that decides what video backend to use. + * + * By default, SDL will try all available video backends in a reasonable + * order until it finds one that can work, but this hint allows the app + * or user to force a specific target, such as "x11" if, say, you are + * on Wayland but want to try talking to the X server instead. + * + * This functionality has existed since SDL 2.0.0 (indeed, before that) + * but before 2.0.22 this was an environment variable only. In 2.0.22, + * it was upgraded to a full SDL hint, so you can set the environment + * variable as usual or programatically set the hint with SDL_SetHint, + * which won't propagate to child processes. + * + * The default value is unset, in which case SDL will try to figure out + * the best video backend on your behalf. This hint needs to be set + * before SDL_Init() is called to be useful. + * + * This hint is available since SDL 2.0.22. Before then, you could set + * the environment variable to get the same effect. + */ +#define SDL_HINT_VIDEODRIVER "SDL_VIDEODRIVER" + +/** + * \brief A variable that decides what audio backend to use. + * + * By default, SDL will try all available audio backends in a reasonable + * order until it finds one that can work, but this hint allows the app + * or user to force a specific target, such as "alsa" if, say, you are + * on PulseAudio but want to try talking to the lower level instead. + * + * This functionality has existed since SDL 2.0.0 (indeed, before that) + * but before 2.0.22 this was an environment variable only. In 2.0.22, + * it was upgraded to a full SDL hint, so you can set the environment + * variable as usual or programatically set the hint with SDL_SetHint, + * which won't propagate to child processes. + * + * The default value is unset, in which case SDL will try to figure out + * the best audio backend on your behalf. This hint needs to be set + * before SDL_Init() is called to be useful. + * + * This hint is available since SDL 2.0.22. Before then, you could set + * the environment variable to get the same effect. + */ +#define SDL_HINT_AUDIODRIVER "SDL_AUDIODRIVER" + +/** + * \brief A variable that decides what KMSDRM device to use. + * + * Internally, SDL might open something like "/dev/dri/cardNN" to + * access KMSDRM functionality, where "NN" is a device index number. + * + * SDL makes a guess at the best index to use (usually zero), but the + * app or user can set this hint to a number between 0 and 99 to + * force selection. + * + * This hint is available since SDL 2.24.0. + */ +#define SDL_HINT_KMSDRM_DEVICE_INDEX "SDL_KMSDRM_DEVICE_INDEX" + + +/** + * \brief A variable that treats trackpads as touch devices. + * + * On macOS (and possibly other platforms in the future), SDL will report + * touches on a trackpad as mouse input, which is generally what users + * expect from this device; however, these are often actually full + * multitouch-capable touch devices, so it might be preferable to some apps + * to treat them as such. + * + * Setting this hint to true will make the trackpad input report as a + * multitouch device instead of a mouse. The default is false. + * + * Note that most platforms don't support this hint. As of 2.24.0, it + * only supports MacBooks' trackpads on macOS. Others may follow later. + * + * This hint is checked during SDL_Init and can not be changed after. + * + * This hint is available since SDL 2.24.0. + */ +#define SDL_HINT_TRACKPAD_IS_TOUCH_ONLY "SDL_TRACKPAD_IS_TOUCH_ONLY" + + +/** + * \brief An enumeration of hint priorities + */ +typedef enum +{ + SDL_HINT_DEFAULT, + SDL_HINT_NORMAL, + SDL_HINT_OVERRIDE +} SDL_HintPriority; + + +/** + * Set a hint with a specific priority. + * + * The priority controls the behavior when setting a hint that already has a + * value. Hints will replace existing hints of their priority and lower. + * Environment variables are considered to have override priority. + * + * \param name the hint to set + * \param value the value of the hint variable + * \param priority the SDL_HintPriority level for the hint + * \returns SDL_TRUE if the hint was set, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetHint + * \sa SDL_SetHint + */ +extern DECLSPEC SDL_bool SDLCALL SDL_SetHintWithPriority(const char *name, + const char *value, + SDL_HintPriority priority); + +/** + * Set a hint with normal priority. + * + * Hints will not be set if there is an existing override hint or environment + * variable that takes precedence. You can use SDL_SetHintWithPriority() to + * set the hint with override priority instead. + * + * \param name the hint to set + * \param value the value of the hint variable + * \returns SDL_TRUE if the hint was set, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetHint + * \sa SDL_SetHintWithPriority + */ +extern DECLSPEC SDL_bool SDLCALL SDL_SetHint(const char *name, + const char *value); + +/** + * Reset a hint to the default value. + * + * This will reset a hint to the value of the environment variable, or NULL if + * the environment isn't set. Callbacks will be called normally with this + * change. + * + * \param name the hint to set + * \returns SDL_TRUE if the hint was set, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.24.0. + * + * \sa SDL_GetHint + * \sa SDL_SetHint + */ +extern DECLSPEC SDL_bool SDLCALL SDL_ResetHint(const char *name); + +/** + * Reset all hints to the default values. + * + * This will reset all hints to the value of the associated environment + * variable, or NULL if the environment isn't set. Callbacks will be called + * normally with this change. + * + * \since This function is available since SDL 2.26.0. + * + * \sa SDL_GetHint + * \sa SDL_SetHint + * \sa SDL_ResetHint + */ +extern DECLSPEC void SDLCALL SDL_ResetHints(void); + +/** + * Get the value of a hint. + * + * \param name the hint to query + * \returns the string value of a hint or NULL if the hint isn't set. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetHint + * \sa SDL_SetHintWithPriority + */ +extern DECLSPEC const char * SDLCALL SDL_GetHint(const char *name); + +/** + * Get the boolean value of a hint variable. + * + * \param name the name of the hint to get the boolean value from + * \param default_value the value to return if the hint does not exist + * \returns the boolean value of a hint or the provided default value if the + * hint does not exist. + * + * \since This function is available since SDL 2.0.5. + * + * \sa SDL_GetHint + * \sa SDL_SetHint + */ +extern DECLSPEC SDL_bool SDLCALL SDL_GetHintBoolean(const char *name, SDL_bool default_value); + +/** + * Type definition of the hint callback function. + * + * \param userdata what was passed as `userdata` to SDL_AddHintCallback() + * \param name what was passed as `name` to SDL_AddHintCallback() + * \param oldValue the previous hint value + * \param newValue the new value hint is to be set to + */ +typedef void (SDLCALL *SDL_HintCallback)(void *userdata, const char *name, const char *oldValue, const char *newValue); + +/** + * Add a function to watch a particular hint. + * + * \param name the hint to watch + * \param callback An SDL_HintCallback function that will be called when the + * hint value changes + * \param userdata a pointer to pass to the callback function + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_DelHintCallback + */ +extern DECLSPEC void SDLCALL SDL_AddHintCallback(const char *name, + SDL_HintCallback callback, + void *userdata); + +/** + * Remove a function watching a particular hint. + * + * \param name the hint being watched + * \param callback An SDL_HintCallback function that will be called when the + * hint value changes + * \param userdata a pointer being passed to the callback function + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AddHintCallback + */ +extern DECLSPEC void SDLCALL SDL_DelHintCallback(const char *name, + SDL_HintCallback callback, + void *userdata); + +/** + * Clear all hints. + * + * This function is automatically called during SDL_Quit(), and deletes all + * callbacks without calling them and frees all memory associated with hints. + * If you're calling this from application code you probably want to call + * SDL_ResetHints() instead. + * + * This function will be removed from the API the next time we rev the ABI. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ResetHints + */ +extern DECLSPEC void SDLCALL SDL_ClearHints(void); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_hints_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_joystick.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_joystick.h new file mode 100644 index 00000000..b9b4f622 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_joystick.h @@ -0,0 +1,1069 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_joystick.h + * + * Include file for SDL joystick event handling + * + * The term "device_index" identifies currently plugged in joystick devices between 0 and SDL_NumJoysticks(), with the exact joystick + * behind a device_index changing as joysticks are plugged and unplugged. + * + * The term "instance_id" is the current instantiation of a joystick device in the system, if the joystick is removed and then re-inserted + * then it will get a new instance_id, instance_id's are monotonically increasing identifiers of a joystick plugged in. + * + * The term "player_index" is the number assigned to a player on a specific + * controller. For XInput controllers this returns the XInput user index. + * Many joysticks will not be able to supply this information. + * + * The term JoystickGUID is a stable 128-bit identifier for a joystick device that does not change over time, it identifies class of + * the device (a X360 wired controller for example). This identifier is platform dependent. + */ + +#ifndef SDL_joystick_h_ +#define SDL_joystick_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_guid.h" +#include "SDL_mutex.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \file SDL_joystick.h + * + * In order to use these functions, SDL_Init() must have been called + * with the ::SDL_INIT_JOYSTICK flag. This causes SDL to scan the system + * for joysticks, and load appropriate drivers. + * + * If you would like to receive joystick updates while the application + * is in the background, you should set the following hint before calling + * SDL_Init(): SDL_HINT_JOYSTICK_ALLOW_BACKGROUND_EVENTS + */ + +/** + * The joystick structure used to identify an SDL joystick + */ +#ifdef SDL_THREAD_SAFETY_ANALYSIS +extern SDL_mutex *SDL_joystick_lock; +#endif +struct _SDL_Joystick; +typedef struct _SDL_Joystick SDL_Joystick; + +/* A structure that encodes the stable unique id for a joystick device */ +typedef SDL_GUID SDL_JoystickGUID; + +/** + * This is a unique ID for a joystick for the time it is connected to the system, + * and is never reused for the lifetime of the application. If the joystick is + * disconnected and reconnected, it will get a new ID. + * + * The ID value starts at 0 and increments from there. The value -1 is an invalid ID. + */ +typedef Sint32 SDL_JoystickID; + +typedef enum +{ + SDL_JOYSTICK_TYPE_UNKNOWN, + SDL_JOYSTICK_TYPE_GAMECONTROLLER, + SDL_JOYSTICK_TYPE_WHEEL, + SDL_JOYSTICK_TYPE_ARCADE_STICK, + SDL_JOYSTICK_TYPE_FLIGHT_STICK, + SDL_JOYSTICK_TYPE_DANCE_PAD, + SDL_JOYSTICK_TYPE_GUITAR, + SDL_JOYSTICK_TYPE_DRUM_KIT, + SDL_JOYSTICK_TYPE_ARCADE_PAD, + SDL_JOYSTICK_TYPE_THROTTLE +} SDL_JoystickType; + +typedef enum +{ + SDL_JOYSTICK_POWER_UNKNOWN = -1, + SDL_JOYSTICK_POWER_EMPTY, /* <= 5% */ + SDL_JOYSTICK_POWER_LOW, /* <= 20% */ + SDL_JOYSTICK_POWER_MEDIUM, /* <= 70% */ + SDL_JOYSTICK_POWER_FULL, /* <= 100% */ + SDL_JOYSTICK_POWER_WIRED, + SDL_JOYSTICK_POWER_MAX +} SDL_JoystickPowerLevel; + +/* Set max recognized G-force from accelerometer + See src/joystick/uikit/SDL_sysjoystick.m for notes on why this is needed + */ +#define SDL_IPHONE_MAX_GFORCE 5.0 + + +/* Function prototypes */ + +/** + * Locking for multi-threaded access to the joystick API + * + * If you are using the joystick API or handling events from multiple threads + * you should use these locking functions to protect access to the joysticks. + * + * In particular, you are guaranteed that the joystick list won't change, so + * the API functions that take a joystick index will be valid, and joystick + * and game controller events will not be delivered. + * + * As of SDL 2.26.0, you can take the joystick lock around reinitializing the + * joystick subsystem, to prevent other threads from seeing joysticks in an + * uninitialized state. However, all open joysticks will be closed and SDL + * functions called with them will fail. + * + * \since This function is available since SDL 2.0.7. + */ +extern DECLSPEC void SDLCALL SDL_LockJoysticks(void) SDL_ACQUIRE(SDL_joystick_lock); + + +/** + * Unlocking for multi-threaded access to the joystick API + * + * If you are using the joystick API or handling events from multiple threads + * you should use these locking functions to protect access to the joysticks. + * + * In particular, you are guaranteed that the joystick list won't change, so + * the API functions that take a joystick index will be valid, and joystick + * and game controller events will not be delivered. + * + * \since This function is available since SDL 2.0.7. + */ +extern DECLSPEC void SDLCALL SDL_UnlockJoysticks(void) SDL_RELEASE(SDL_joystick_lock); + +/** + * Count the number of joysticks attached to the system. + * + * \returns the number of attached joysticks on success or a negative error + * code on failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickName + * \sa SDL_JoystickPath + * \sa SDL_JoystickOpen + */ +extern DECLSPEC int SDLCALL SDL_NumJoysticks(void); + +/** + * Get the implementation dependent name of a joystick. + * + * This can be called before any joysticks are opened. + * + * \param device_index the index of the joystick to query (the N'th joystick + * on the system) + * \returns the name of the selected joystick. If no name can be found, this + * function returns NULL; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickName + * \sa SDL_JoystickOpen + */ +extern DECLSPEC const char *SDLCALL SDL_JoystickNameForIndex(int device_index); + +/** + * Get the implementation dependent path of a joystick. + * + * This can be called before any joysticks are opened. + * + * \param device_index the index of the joystick to query (the N'th joystick + * on the system) + * \returns the path of the selected joystick. If no path can be found, this + * function returns NULL; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.24.0. + * + * \sa SDL_JoystickPath + * \sa SDL_JoystickOpen + */ +extern DECLSPEC const char *SDLCALL SDL_JoystickPathForIndex(int device_index); + +/** + * Get the player index of a joystick, or -1 if it's not available This can be + * called before any joysticks are opened. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC int SDLCALL SDL_JoystickGetDevicePlayerIndex(int device_index); + +/** + * Get the implementation-dependent GUID for the joystick at a given device + * index. + * + * This function can be called before any joysticks are opened. + * + * \param device_index the index of the joystick to query (the N'th joystick + * on the system + * \returns the GUID of the selected joystick. If called on an invalid index, + * this function returns a zero GUID + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickGetGUID + * \sa SDL_JoystickGetGUIDString + */ +extern DECLSPEC SDL_JoystickGUID SDLCALL SDL_JoystickGetDeviceGUID(int device_index); + +/** + * Get the USB vendor ID of a joystick, if available. + * + * This can be called before any joysticks are opened. If the vendor ID isn't + * available this function returns 0. + * + * \param device_index the index of the joystick to query (the N'th joystick + * on the system + * \returns the USB vendor ID of the selected joystick. If called on an + * invalid index, this function returns zero + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC Uint16 SDLCALL SDL_JoystickGetDeviceVendor(int device_index); + +/** + * Get the USB product ID of a joystick, if available. + * + * This can be called before any joysticks are opened. If the product ID isn't + * available this function returns 0. + * + * \param device_index the index of the joystick to query (the N'th joystick + * on the system + * \returns the USB product ID of the selected joystick. If called on an + * invalid index, this function returns zero + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC Uint16 SDLCALL SDL_JoystickGetDeviceProduct(int device_index); + +/** + * Get the product version of a joystick, if available. + * + * This can be called before any joysticks are opened. If the product version + * isn't available this function returns 0. + * + * \param device_index the index of the joystick to query (the N'th joystick + * on the system + * \returns the product version of the selected joystick. If called on an + * invalid index, this function returns zero + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC Uint16 SDLCALL SDL_JoystickGetDeviceProductVersion(int device_index); + +/** + * Get the type of a joystick, if available. + * + * This can be called before any joysticks are opened. + * + * \param device_index the index of the joystick to query (the N'th joystick + * on the system + * \returns the SDL_JoystickType of the selected joystick. If called on an + * invalid index, this function returns `SDL_JOYSTICK_TYPE_UNKNOWN` + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC SDL_JoystickType SDLCALL SDL_JoystickGetDeviceType(int device_index); + +/** + * Get the instance ID of a joystick. + * + * This can be called before any joysticks are opened. + * + * \param device_index the index of the joystick to query (the N'th joystick + * on the system + * \returns the instance id of the selected joystick. If called on an invalid + * index, this function returns -1. + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC SDL_JoystickID SDLCALL SDL_JoystickGetDeviceInstanceID(int device_index); + +/** + * Open a joystick for use. + * + * The `device_index` argument refers to the N'th joystick presently + * recognized by SDL on the system. It is **NOT** the same as the instance ID + * used to identify the joystick in future events. See + * SDL_JoystickInstanceID() for more details about instance IDs. + * + * The joystick subsystem must be initialized before a joystick can be opened + * for use. + * + * \param device_index the index of the joystick to query + * \returns a joystick identifier or NULL if an error occurred; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickClose + * \sa SDL_JoystickInstanceID + */ +extern DECLSPEC SDL_Joystick *SDLCALL SDL_JoystickOpen(int device_index); + +/** + * Get the SDL_Joystick associated with an instance id. + * + * \param instance_id the instance id to get the SDL_Joystick for + * \returns an SDL_Joystick on success or NULL on failure; call SDL_GetError() + * for more information. + * + * \since This function is available since SDL 2.0.4. + */ +extern DECLSPEC SDL_Joystick *SDLCALL SDL_JoystickFromInstanceID(SDL_JoystickID instance_id); + +/** + * Get the SDL_Joystick associated with a player index. + * + * \param player_index the player index to get the SDL_Joystick for + * \returns an SDL_Joystick on success or NULL on failure; call SDL_GetError() + * for more information. + * + * \since This function is available since SDL 2.0.12. + */ +extern DECLSPEC SDL_Joystick *SDLCALL SDL_JoystickFromPlayerIndex(int player_index); + +/** + * Attach a new virtual joystick. + * + * \returns the joystick's device index, or -1 if an error occurred. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC int SDLCALL SDL_JoystickAttachVirtual(SDL_JoystickType type, + int naxes, + int nbuttons, + int nhats); + +/** + * The structure that defines an extended virtual joystick description + * + * The caller must zero the structure and then initialize the version with `SDL_VIRTUAL_JOYSTICK_DESC_VERSION` before passing it to SDL_JoystickAttachVirtualEx() + * All other elements of this structure are optional and can be left 0. + * + * \sa SDL_JoystickAttachVirtualEx + */ +typedef struct SDL_VirtualJoystickDesc +{ + Uint16 version; /**< `SDL_VIRTUAL_JOYSTICK_DESC_VERSION` */ + Uint16 type; /**< `SDL_JoystickType` */ + Uint16 naxes; /**< the number of axes on this joystick */ + Uint16 nbuttons; /**< the number of buttons on this joystick */ + Uint16 nhats; /**< the number of hats on this joystick */ + Uint16 vendor_id; /**< the USB vendor ID of this joystick */ + Uint16 product_id; /**< the USB product ID of this joystick */ + Uint16 padding; /**< unused */ + Uint32 button_mask; /**< A mask of which buttons are valid for this controller + e.g. (1 << SDL_CONTROLLER_BUTTON_A) */ + Uint32 axis_mask; /**< A mask of which axes are valid for this controller + e.g. (1 << SDL_CONTROLLER_AXIS_LEFTX) */ + const char *name; /**< the name of the joystick */ + + void *userdata; /**< User data pointer passed to callbacks */ + void (SDLCALL *Update)(void *userdata); /**< Called when the joystick state should be updated */ + void (SDLCALL *SetPlayerIndex)(void *userdata, int player_index); /**< Called when the player index is set */ + int (SDLCALL *Rumble)(void *userdata, Uint16 low_frequency_rumble, Uint16 high_frequency_rumble); /**< Implements SDL_JoystickRumble() */ + int (SDLCALL *RumbleTriggers)(void *userdata, Uint16 left_rumble, Uint16 right_rumble); /**< Implements SDL_JoystickRumbleTriggers() */ + int (SDLCALL *SetLED)(void *userdata, Uint8 red, Uint8 green, Uint8 blue); /**< Implements SDL_JoystickSetLED() */ + int (SDLCALL *SendEffect)(void *userdata, const void *data, int size); /**< Implements SDL_JoystickSendEffect() */ + +} SDL_VirtualJoystickDesc; + +/** + * \brief The current version of the SDL_VirtualJoystickDesc structure + */ +#define SDL_VIRTUAL_JOYSTICK_DESC_VERSION 1 + +/** + * Attach a new virtual joystick with extended properties. + * + * \returns the joystick's device index, or -1 if an error occurred. + * + * \since This function is available since SDL 2.24.0. + */ +extern DECLSPEC int SDLCALL SDL_JoystickAttachVirtualEx(const SDL_VirtualJoystickDesc *desc); + +/** + * Detach a virtual joystick. + * + * \param device_index a value previously returned from + * SDL_JoystickAttachVirtual() + * \returns 0 on success, or -1 if an error occurred. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC int SDLCALL SDL_JoystickDetachVirtual(int device_index); + +/** + * Query whether or not the joystick at a given device index is virtual. + * + * \param device_index a joystick device index. + * \returns SDL_TRUE if the joystick is virtual, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_JoystickIsVirtual(int device_index); + +/** + * Set values on an opened, virtual-joystick's axis. + * + * Please note that values set here will not be applied until the next call to + * SDL_JoystickUpdate, which can either be called directly, or can be called + * indirectly through various other SDL APIs, including, but not limited to + * the following: SDL_PollEvent, SDL_PumpEvents, SDL_WaitEventTimeout, + * SDL_WaitEvent. + * + * Note that when sending trigger axes, you should scale the value to the full + * range of Sint16. For example, a trigger at rest would have the value of + * `SDL_JOYSTICK_AXIS_MIN`. + * + * \param joystick the virtual joystick on which to set state. + * \param axis the specific axis on the virtual joystick to set. + * \param value the new value for the specified axis. + * \returns 0 on success, -1 on error. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC int SDLCALL SDL_JoystickSetVirtualAxis(SDL_Joystick *joystick, int axis, Sint16 value); + +/** + * Set values on an opened, virtual-joystick's button. + * + * Please note that values set here will not be applied until the next call to + * SDL_JoystickUpdate, which can either be called directly, or can be called + * indirectly through various other SDL APIs, including, but not limited to + * the following: SDL_PollEvent, SDL_PumpEvents, SDL_WaitEventTimeout, + * SDL_WaitEvent. + * + * \param joystick the virtual joystick on which to set state. + * \param button the specific button on the virtual joystick to set. + * \param value the new value for the specified button. + * \returns 0 on success, -1 on error. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC int SDLCALL SDL_JoystickSetVirtualButton(SDL_Joystick *joystick, int button, Uint8 value); + +/** + * Set values on an opened, virtual-joystick's hat. + * + * Please note that values set here will not be applied until the next call to + * SDL_JoystickUpdate, which can either be called directly, or can be called + * indirectly through various other SDL APIs, including, but not limited to + * the following: SDL_PollEvent, SDL_PumpEvents, SDL_WaitEventTimeout, + * SDL_WaitEvent. + * + * \param joystick the virtual joystick on which to set state. + * \param hat the specific hat on the virtual joystick to set. + * \param value the new value for the specified hat. + * \returns 0 on success, -1 on error. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC int SDLCALL SDL_JoystickSetVirtualHat(SDL_Joystick *joystick, int hat, Uint8 value); + +/** + * Get the implementation dependent name of a joystick. + * + * \param joystick the SDL_Joystick obtained from SDL_JoystickOpen() + * \returns the name of the selected joystick. If no name can be found, this + * function returns NULL; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickNameForIndex + * \sa SDL_JoystickOpen + */ +extern DECLSPEC const char *SDLCALL SDL_JoystickName(SDL_Joystick *joystick); + +/** + * Get the implementation dependent path of a joystick. + * + * \param joystick the SDL_Joystick obtained from SDL_JoystickOpen() + * \returns the path of the selected joystick. If no path can be found, this + * function returns NULL; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.24.0. + * + * \sa SDL_JoystickPathForIndex + */ +extern DECLSPEC const char *SDLCALL SDL_JoystickPath(SDL_Joystick *joystick); + +/** + * Get the player index of an opened joystick. + * + * For XInput controllers this returns the XInput user index. Many joysticks + * will not be able to supply this information. + * + * \param joystick the SDL_Joystick obtained from SDL_JoystickOpen() + * \returns the player index, or -1 if it's not available. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC int SDLCALL SDL_JoystickGetPlayerIndex(SDL_Joystick *joystick); + +/** + * Set the player index of an opened joystick. + * + * \param joystick the SDL_Joystick obtained from SDL_JoystickOpen() + * \param player_index Player index to assign to this joystick, or -1 to clear + * the player index and turn off player LEDs. + * + * \since This function is available since SDL 2.0.12. + */ +extern DECLSPEC void SDLCALL SDL_JoystickSetPlayerIndex(SDL_Joystick *joystick, int player_index); + +/** + * Get the implementation-dependent GUID for the joystick. + * + * This function requires an open joystick. + * + * \param joystick the SDL_Joystick obtained from SDL_JoystickOpen() + * \returns the GUID of the given joystick. If called on an invalid index, + * this function returns a zero GUID; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickGetDeviceGUID + * \sa SDL_JoystickGetGUIDString + */ +extern DECLSPEC SDL_JoystickGUID SDLCALL SDL_JoystickGetGUID(SDL_Joystick *joystick); + +/** + * Get the USB vendor ID of an opened joystick, if available. + * + * If the vendor ID isn't available this function returns 0. + * + * \param joystick the SDL_Joystick obtained from SDL_JoystickOpen() + * \returns the USB vendor ID of the selected joystick, or 0 if unavailable. + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC Uint16 SDLCALL SDL_JoystickGetVendor(SDL_Joystick *joystick); + +/** + * Get the USB product ID of an opened joystick, if available. + * + * If the product ID isn't available this function returns 0. + * + * \param joystick the SDL_Joystick obtained from SDL_JoystickOpen() + * \returns the USB product ID of the selected joystick, or 0 if unavailable. + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC Uint16 SDLCALL SDL_JoystickGetProduct(SDL_Joystick *joystick); + +/** + * Get the product version of an opened joystick, if available. + * + * If the product version isn't available this function returns 0. + * + * \param joystick the SDL_Joystick obtained from SDL_JoystickOpen() + * \returns the product version of the selected joystick, or 0 if unavailable. + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC Uint16 SDLCALL SDL_JoystickGetProductVersion(SDL_Joystick *joystick); + +/** + * Get the firmware version of an opened joystick, if available. + * + * If the firmware version isn't available this function returns 0. + * + * \param joystick the SDL_Joystick obtained from SDL_JoystickOpen() + * \returns the firmware version of the selected joystick, or 0 if + * unavailable. + * + * \since This function is available since SDL 2.24.0. + */ +extern DECLSPEC Uint16 SDLCALL SDL_JoystickGetFirmwareVersion(SDL_Joystick *joystick); + +/** + * Get the serial number of an opened joystick, if available. + * + * Returns the serial number of the joystick, or NULL if it is not available. + * + * \param joystick the SDL_Joystick obtained from SDL_JoystickOpen() + * \returns the serial number of the selected joystick, or NULL if + * unavailable. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC const char * SDLCALL SDL_JoystickGetSerial(SDL_Joystick *joystick); + +/** + * Get the type of an opened joystick. + * + * \param joystick the SDL_Joystick obtained from SDL_JoystickOpen() + * \returns the SDL_JoystickType of the selected joystick. + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC SDL_JoystickType SDLCALL SDL_JoystickGetType(SDL_Joystick *joystick); + +/** + * Get an ASCII string representation for a given SDL_JoystickGUID. + * + * You should supply at least 33 bytes for pszGUID. + * + * \param guid the SDL_JoystickGUID you wish to convert to string + * \param pszGUID buffer in which to write the ASCII string + * \param cbGUID the size of pszGUID + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickGetDeviceGUID + * \sa SDL_JoystickGetGUID + * \sa SDL_JoystickGetGUIDFromString + */ +extern DECLSPEC void SDLCALL SDL_JoystickGetGUIDString(SDL_JoystickGUID guid, char *pszGUID, int cbGUID); + +/** + * Convert a GUID string into a SDL_JoystickGUID structure. + * + * Performs no error checking. If this function is given a string containing + * an invalid GUID, the function will silently succeed, but the GUID generated + * will not be useful. + * + * \param pchGUID string containing an ASCII representation of a GUID + * \returns a SDL_JoystickGUID structure. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickGetGUIDString + */ +extern DECLSPEC SDL_JoystickGUID SDLCALL SDL_JoystickGetGUIDFromString(const char *pchGUID); + +/** + * Get the device information encoded in a SDL_JoystickGUID structure + * + * \param guid the SDL_JoystickGUID you wish to get info about + * \param vendor A pointer filled in with the device VID, or 0 if not + * available + * \param product A pointer filled in with the device PID, or 0 if not + * available + * \param version A pointer filled in with the device version, or 0 if not + * available + * \param crc16 A pointer filled in with a CRC used to distinguish different + * products with the same VID/PID, or 0 if not available + * + * \since This function is available since SDL 2.26.0. + * + * \sa SDL_JoystickGetDeviceGUID + */ +extern DECLSPEC void SDLCALL SDL_GetJoystickGUIDInfo(SDL_JoystickGUID guid, Uint16 *vendor, Uint16 *product, Uint16 *version, Uint16 *crc16); + +/** + * Get the status of a specified joystick. + * + * \param joystick the joystick to query + * \returns SDL_TRUE if the joystick has been opened, SDL_FALSE if it has not; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickClose + * \sa SDL_JoystickOpen + */ +extern DECLSPEC SDL_bool SDLCALL SDL_JoystickGetAttached(SDL_Joystick *joystick); + +/** + * Get the instance ID of an opened joystick. + * + * \param joystick an SDL_Joystick structure containing joystick information + * \returns the instance ID of the specified joystick on success or a negative + * error code on failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickOpen + */ +extern DECLSPEC SDL_JoystickID SDLCALL SDL_JoystickInstanceID(SDL_Joystick *joystick); + +/** + * Get the number of general axis controls on a joystick. + * + * Often, the directional pad on a game controller will either look like 4 + * separate buttons or a POV hat, and not axes, but all of this is up to the + * device and platform. + * + * \param joystick an SDL_Joystick structure containing joystick information + * \returns the number of axis controls/number of axes on success or a + * negative error code on failure; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickGetAxis + * \sa SDL_JoystickOpen + */ +extern DECLSPEC int SDLCALL SDL_JoystickNumAxes(SDL_Joystick *joystick); + +/** + * Get the number of trackballs on a joystick. + * + * Joystick trackballs have only relative motion events associated with them + * and their state cannot be polled. + * + * Most joysticks do not have trackballs. + * + * \param joystick an SDL_Joystick structure containing joystick information + * \returns the number of trackballs on success or a negative error code on + * failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickGetBall + */ +extern DECLSPEC int SDLCALL SDL_JoystickNumBalls(SDL_Joystick *joystick); + +/** + * Get the number of POV hats on a joystick. + * + * \param joystick an SDL_Joystick structure containing joystick information + * \returns the number of POV hats on success or a negative error code on + * failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickGetHat + * \sa SDL_JoystickOpen + */ +extern DECLSPEC int SDLCALL SDL_JoystickNumHats(SDL_Joystick *joystick); + +/** + * Get the number of buttons on a joystick. + * + * \param joystick an SDL_Joystick structure containing joystick information + * \returns the number of buttons on success or a negative error code on + * failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickGetButton + * \sa SDL_JoystickOpen + */ +extern DECLSPEC int SDLCALL SDL_JoystickNumButtons(SDL_Joystick *joystick); + +/** + * Update the current state of the open joysticks. + * + * This is called automatically by the event loop if any joystick events are + * enabled. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickEventState + */ +extern DECLSPEC void SDLCALL SDL_JoystickUpdate(void); + +/** + * Enable/disable joystick event polling. + * + * If joystick events are disabled, you must call SDL_JoystickUpdate() + * yourself and manually check the state of the joystick when you want + * joystick information. + * + * It is recommended that you leave joystick event handling enabled. + * + * **WARNING**: Calling this function may delete all events currently in SDL's + * event queue. + * + * \param state can be one of `SDL_QUERY`, `SDL_IGNORE`, or `SDL_ENABLE` + * \returns 1 if enabled, 0 if disabled, or a negative error code on failure; + * call SDL_GetError() for more information. + * + * If `state` is `SDL_QUERY` then the current state is returned, + * otherwise the new processing state is returned. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GameControllerEventState + */ +extern DECLSPEC int SDLCALL SDL_JoystickEventState(int state); + +#define SDL_JOYSTICK_AXIS_MAX 32767 +#define SDL_JOYSTICK_AXIS_MIN -32768 + +/** + * Get the current state of an axis control on a joystick. + * + * SDL makes no promises about what part of the joystick any given axis refers + * to. Your game should have some sort of configuration UI to let users + * specify what each axis should be bound to. Alternately, SDL's higher-level + * Game Controller API makes a great effort to apply order to this lower-level + * interface, so you know that a specific axis is the "left thumb stick," etc. + * + * The value returned by SDL_JoystickGetAxis() is a signed integer (-32768 to + * 32767) representing the current position of the axis. It may be necessary + * to impose certain tolerances on these values to account for jitter. + * + * \param joystick an SDL_Joystick structure containing joystick information + * \param axis the axis to query; the axis indices start at index 0 + * \returns a 16-bit signed integer representing the current position of the + * axis or 0 on failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickNumAxes + */ +extern DECLSPEC Sint16 SDLCALL SDL_JoystickGetAxis(SDL_Joystick *joystick, + int axis); + +/** + * Get the initial state of an axis control on a joystick. + * + * The state is a value ranging from -32768 to 32767. + * + * The axis indices start at index 0. + * + * \param joystick an SDL_Joystick structure containing joystick information + * \param axis the axis to query; the axis indices start at index 0 + * \param state Upon return, the initial value is supplied here. + * \return SDL_TRUE if this axis has any initial value, or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_JoystickGetAxisInitialState(SDL_Joystick *joystick, + int axis, Sint16 *state); + +/** + * \name Hat positions + */ +/* @{ */ +#define SDL_HAT_CENTERED 0x00 +#define SDL_HAT_UP 0x01 +#define SDL_HAT_RIGHT 0x02 +#define SDL_HAT_DOWN 0x04 +#define SDL_HAT_LEFT 0x08 +#define SDL_HAT_RIGHTUP (SDL_HAT_RIGHT|SDL_HAT_UP) +#define SDL_HAT_RIGHTDOWN (SDL_HAT_RIGHT|SDL_HAT_DOWN) +#define SDL_HAT_LEFTUP (SDL_HAT_LEFT|SDL_HAT_UP) +#define SDL_HAT_LEFTDOWN (SDL_HAT_LEFT|SDL_HAT_DOWN) +/* @} */ + +/** + * Get the current state of a POV hat on a joystick. + * + * The returned value will be one of the following positions: + * + * - `SDL_HAT_CENTERED` + * - `SDL_HAT_UP` + * - `SDL_HAT_RIGHT` + * - `SDL_HAT_DOWN` + * - `SDL_HAT_LEFT` + * - `SDL_HAT_RIGHTUP` + * - `SDL_HAT_RIGHTDOWN` + * - `SDL_HAT_LEFTUP` + * - `SDL_HAT_LEFTDOWN` + * + * \param joystick an SDL_Joystick structure containing joystick information + * \param hat the hat index to get the state from; indices start at index 0 + * \returns the current hat position. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickNumHats + */ +extern DECLSPEC Uint8 SDLCALL SDL_JoystickGetHat(SDL_Joystick *joystick, + int hat); + +/** + * Get the ball axis change since the last poll. + * + * Trackballs can only return relative motion since the last call to + * SDL_JoystickGetBall(), these motion deltas are placed into `dx` and `dy`. + * + * Most joysticks do not have trackballs. + * + * \param joystick the SDL_Joystick to query + * \param ball the ball index to query; ball indices start at index 0 + * \param dx stores the difference in the x axis position since the last poll + * \param dy stores the difference in the y axis position since the last poll + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickNumBalls + */ +extern DECLSPEC int SDLCALL SDL_JoystickGetBall(SDL_Joystick *joystick, + int ball, int *dx, int *dy); + +/** + * Get the current state of a button on a joystick. + * + * \param joystick an SDL_Joystick structure containing joystick information + * \param button the button index to get the state from; indices start at + * index 0 + * \returns 1 if the specified button is pressed, 0 otherwise. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickNumButtons + */ +extern DECLSPEC Uint8 SDLCALL SDL_JoystickGetButton(SDL_Joystick *joystick, + int button); + +/** + * Start a rumble effect. + * + * Each call to this function cancels any previous rumble effect, and calling + * it with 0 intensity stops any rumbling. + * + * \param joystick The joystick to vibrate + * \param low_frequency_rumble The intensity of the low frequency (left) + * rumble motor, from 0 to 0xFFFF + * \param high_frequency_rumble The intensity of the high frequency (right) + * rumble motor, from 0 to 0xFFFF + * \param duration_ms The duration of the rumble effect, in milliseconds + * \returns 0, or -1 if rumble isn't supported on this joystick + * + * \since This function is available since SDL 2.0.9. + * + * \sa SDL_JoystickHasRumble + */ +extern DECLSPEC int SDLCALL SDL_JoystickRumble(SDL_Joystick *joystick, Uint16 low_frequency_rumble, Uint16 high_frequency_rumble, Uint32 duration_ms); + +/** + * Start a rumble effect in the joystick's triggers + * + * Each call to this function cancels any previous trigger rumble effect, and + * calling it with 0 intensity stops any rumbling. + * + * Note that this is rumbling of the _triggers_ and not the game controller as + * a whole. This is currently only supported on Xbox One controllers. If you + * want the (more common) whole-controller rumble, use SDL_JoystickRumble() + * instead. + * + * \param joystick The joystick to vibrate + * \param left_rumble The intensity of the left trigger rumble motor, from 0 + * to 0xFFFF + * \param right_rumble The intensity of the right trigger rumble motor, from 0 + * to 0xFFFF + * \param duration_ms The duration of the rumble effect, in milliseconds + * \returns 0, or -1 if trigger rumble isn't supported on this joystick + * + * \since This function is available since SDL 2.0.14. + * + * \sa SDL_JoystickHasRumbleTriggers + */ +extern DECLSPEC int SDLCALL SDL_JoystickRumbleTriggers(SDL_Joystick *joystick, Uint16 left_rumble, Uint16 right_rumble, Uint32 duration_ms); + +/** + * Query whether a joystick has an LED. + * + * An example of a joystick LED is the light on the back of a PlayStation 4's + * DualShock 4 controller. + * + * \param joystick The joystick to query + * \return SDL_TRUE if the joystick has a modifiable LED, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_JoystickHasLED(SDL_Joystick *joystick); + +/** + * Query whether a joystick has rumble support. + * + * \param joystick The joystick to query + * \return SDL_TRUE if the joystick has rumble, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_JoystickRumble + */ +extern DECLSPEC SDL_bool SDLCALL SDL_JoystickHasRumble(SDL_Joystick *joystick); + +/** + * Query whether a joystick has rumble support on triggers. + * + * \param joystick The joystick to query + * \return SDL_TRUE if the joystick has trigger rumble, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_JoystickRumbleTriggers + */ +extern DECLSPEC SDL_bool SDLCALL SDL_JoystickHasRumbleTriggers(SDL_Joystick *joystick); + +/** + * Update a joystick's LED color. + * + * An example of a joystick LED is the light on the back of a PlayStation 4's + * DualShock 4 controller. + * + * \param joystick The joystick to update + * \param red The intensity of the red LED + * \param green The intensity of the green LED + * \param blue The intensity of the blue LED + * \returns 0 on success, -1 if this joystick does not have a modifiable LED + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC int SDLCALL SDL_JoystickSetLED(SDL_Joystick *joystick, Uint8 red, Uint8 green, Uint8 blue); + +/** + * Send a joystick specific effect packet + * + * \param joystick The joystick to affect + * \param data The data to send to the joystick + * \param size The size of the data to send to the joystick + * \returns 0, or -1 if this joystick or driver doesn't support effect packets + * + * \since This function is available since SDL 2.0.16. + */ +extern DECLSPEC int SDLCALL SDL_JoystickSendEffect(SDL_Joystick *joystick, const void *data, int size); + +/** + * Close a joystick previously opened with SDL_JoystickOpen(). + * + * \param joystick The joystick device to close + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_JoystickOpen + */ +extern DECLSPEC void SDLCALL SDL_JoystickClose(SDL_Joystick *joystick); + +/** + * Get the battery level of a joystick as SDL_JoystickPowerLevel. + * + * \param joystick the SDL_Joystick to query + * \returns the current battery level as SDL_JoystickPowerLevel on success or + * `SDL_JOYSTICK_POWER_UNKNOWN` if it is unknown + * + * \since This function is available since SDL 2.0.4. + */ +extern DECLSPEC SDL_JoystickPowerLevel SDLCALL SDL_JoystickCurrentPowerLevel(SDL_Joystick *joystick); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_joystick_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_keyboard.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_keyboard.h new file mode 100644 index 00000000..86a37ad1 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_keyboard.h @@ -0,0 +1,353 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_keyboard.h + * + * Include file for SDL keyboard event handling + */ + +#ifndef SDL_keyboard_h_ +#define SDL_keyboard_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_keycode.h" +#include "SDL_video.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \brief The SDL keysym structure, used in key events. + * + * \note If you are looking for translated character input, see the ::SDL_TEXTINPUT event. + */ +typedef struct SDL_Keysym +{ + SDL_Scancode scancode; /**< SDL physical key code - see ::SDL_Scancode for details */ + SDL_Keycode sym; /**< SDL virtual key code - see ::SDL_Keycode for details */ + Uint16 mod; /**< current key modifiers */ + Uint32 unused; +} SDL_Keysym; + +/* Function prototypes */ + +/** + * Query the window which currently has keyboard focus. + * + * \returns the window with keyboard focus. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC SDL_Window * SDLCALL SDL_GetKeyboardFocus(void); + +/** + * Get a snapshot of the current state of the keyboard. + * + * The pointer returned is a pointer to an internal SDL array. It will be + * valid for the whole lifetime of the application and should not be freed by + * the caller. + * + * A array element with a value of 1 means that the key is pressed and a value + * of 0 means that it is not. Indexes into this array are obtained by using + * SDL_Scancode values. + * + * Use SDL_PumpEvents() to update the state array. + * + * This function gives you the current state after all events have been + * processed, so if a key or button has been pressed and released before you + * process events, then the pressed state will never show up in the + * SDL_GetKeyboardState() calls. + * + * Note: This function doesn't take into account whether shift has been + * pressed or not. + * + * \param numkeys if non-NULL, receives the length of the returned array + * \returns a pointer to an array of key states. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_PumpEvents + * \sa SDL_ResetKeyboard + */ +extern DECLSPEC const Uint8 *SDLCALL SDL_GetKeyboardState(int *numkeys); + +/** + * Clear the state of the keyboard + * + * This function will generate key up events for all pressed keys. + * + * \since This function is available since SDL 2.24.0. + * + * \sa SDL_GetKeyboardState + */ +extern DECLSPEC void SDLCALL SDL_ResetKeyboard(void); + +/** + * Get the current key modifier state for the keyboard. + * + * \returns an OR'd combination of the modifier keys for the keyboard. See + * SDL_Keymod for details. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetKeyboardState + * \sa SDL_SetModState + */ +extern DECLSPEC SDL_Keymod SDLCALL SDL_GetModState(void); + +/** + * Set the current key modifier state for the keyboard. + * + * The inverse of SDL_GetModState(), SDL_SetModState() allows you to impose + * modifier key states on your application. Simply pass your desired modifier + * states into `modstate`. This value may be a bitwise, OR'd combination of + * SDL_Keymod values. + * + * This does not change the keyboard state, only the key modifier flags that + * SDL reports. + * + * \param modstate the desired SDL_Keymod for the keyboard + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetModState + */ +extern DECLSPEC void SDLCALL SDL_SetModState(SDL_Keymod modstate); + +/** + * Get the key code corresponding to the given scancode according to the + * current keyboard layout. + * + * See SDL_Keycode for details. + * + * \param scancode the desired SDL_Scancode to query + * \returns the SDL_Keycode that corresponds to the given SDL_Scancode. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetKeyName + * \sa SDL_GetScancodeFromKey + */ +extern DECLSPEC SDL_Keycode SDLCALL SDL_GetKeyFromScancode(SDL_Scancode scancode); + +/** + * Get the scancode corresponding to the given key code according to the + * current keyboard layout. + * + * See SDL_Scancode for details. + * + * \param key the desired SDL_Keycode to query + * \returns the SDL_Scancode that corresponds to the given SDL_Keycode. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetKeyFromScancode + * \sa SDL_GetScancodeName + */ +extern DECLSPEC SDL_Scancode SDLCALL SDL_GetScancodeFromKey(SDL_Keycode key); + +/** + * Get a human-readable name for a scancode. + * + * See SDL_Scancode for details. + * + * **Warning**: The returned name is by design not stable across platforms, + * e.g. the name for `SDL_SCANCODE_LGUI` is "Left GUI" under Linux but "Left + * Windows" under Microsoft Windows, and some scancodes like + * `SDL_SCANCODE_NONUSBACKSLASH` don't have any name at all. There are even + * scancodes that share names, e.g. `SDL_SCANCODE_RETURN` and + * `SDL_SCANCODE_RETURN2` (both called "Return"). This function is therefore + * unsuitable for creating a stable cross-platform two-way mapping between + * strings and scancodes. + * + * \param scancode the desired SDL_Scancode to query + * \returns a pointer to the name for the scancode. If the scancode doesn't + * have a name this function returns an empty string (""). + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetScancodeFromKey + * \sa SDL_GetScancodeFromName + */ +extern DECLSPEC const char *SDLCALL SDL_GetScancodeName(SDL_Scancode scancode); + +/** + * Get a scancode from a human-readable name. + * + * \param name the human-readable scancode name + * \returns the SDL_Scancode, or `SDL_SCANCODE_UNKNOWN` if the name wasn't + * recognized; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetKeyFromName + * \sa SDL_GetScancodeFromKey + * \sa SDL_GetScancodeName + */ +extern DECLSPEC SDL_Scancode SDLCALL SDL_GetScancodeFromName(const char *name); + +/** + * Get a human-readable name for a key. + * + * See SDL_Scancode and SDL_Keycode for details. + * + * \param key the desired SDL_Keycode to query + * \returns a pointer to a UTF-8 string that stays valid at least until the + * next call to this function. If you need it around any longer, you + * must copy it. If the key doesn't have a name, this function + * returns an empty string (""). + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetKeyFromName + * \sa SDL_GetKeyFromScancode + * \sa SDL_GetScancodeFromKey + */ +extern DECLSPEC const char *SDLCALL SDL_GetKeyName(SDL_Keycode key); + +/** + * Get a key code from a human-readable name. + * + * \param name the human-readable key name + * \returns key code, or `SDLK_UNKNOWN` if the name wasn't recognized; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetKeyFromScancode + * \sa SDL_GetKeyName + * \sa SDL_GetScancodeFromName + */ +extern DECLSPEC SDL_Keycode SDLCALL SDL_GetKeyFromName(const char *name); + +/** + * Start accepting Unicode text input events. + * + * This function will start accepting Unicode text input events in the focused + * SDL window, and start emitting SDL_TextInputEvent (SDL_TEXTINPUT) and + * SDL_TextEditingEvent (SDL_TEXTEDITING) events. Please use this function in + * pair with SDL_StopTextInput(). + * + * On some platforms using this function activates the screen keyboard. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetTextInputRect + * \sa SDL_StopTextInput + */ +extern DECLSPEC void SDLCALL SDL_StartTextInput(void); + +/** + * Check whether or not Unicode text input events are enabled. + * + * \returns SDL_TRUE if text input events are enabled else SDL_FALSE. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_StartTextInput + */ +extern DECLSPEC SDL_bool SDLCALL SDL_IsTextInputActive(void); + +/** + * Stop receiving any text input events. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_StartTextInput + */ +extern DECLSPEC void SDLCALL SDL_StopTextInput(void); + +/** + * Dismiss the composition window/IME without disabling the subsystem. + * + * \since This function is available since SDL 2.0.22. + * + * \sa SDL_StartTextInput + * \sa SDL_StopTextInput + */ +extern DECLSPEC void SDLCALL SDL_ClearComposition(void); + +/** + * Returns if an IME Composite or Candidate window is currently shown. + * + * \since This function is available since SDL 2.0.22. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_IsTextInputShown(void); + +/** + * Set the rectangle used to type Unicode text inputs. + * + * To start text input in a given location, this function is intended to be + * called before SDL_StartTextInput, although some platforms support moving + * the rectangle even while text input (and a composition) is active. + * + * Note: If you want to use the system native IME window, try setting hint + * **SDL_HINT_IME_SHOW_UI** to **1**, otherwise this function won't give you + * any feedback. + * + * \param rect the SDL_Rect structure representing the rectangle to receive + * text (ignored if NULL) + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_StartTextInput + */ +extern DECLSPEC void SDLCALL SDL_SetTextInputRect(const SDL_Rect *rect); + +/** + * Check whether the platform has screen keyboard support. + * + * \returns SDL_TRUE if the platform has some screen keyboard support or + * SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_StartTextInput + * \sa SDL_IsScreenKeyboardShown + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasScreenKeyboardSupport(void); + +/** + * Check whether the screen keyboard is shown for given window. + * + * \param window the window for which screen keyboard should be queried + * \returns SDL_TRUE if screen keyboard is shown or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HasScreenKeyboardSupport + */ +extern DECLSPEC SDL_bool SDLCALL SDL_IsScreenKeyboardShown(SDL_Window *window); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_keyboard_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_keycode.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_keycode.h new file mode 100644 index 00000000..71062230 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_keycode.h @@ -0,0 +1,358 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_keycode.h + * + * Defines constants which identify keyboard keys and modifiers. + */ + +#ifndef SDL_keycode_h_ +#define SDL_keycode_h_ + +#include "SDL_stdinc.h" +#include "SDL_scancode.h" + +/** + * \brief The SDL virtual key representation. + * + * Values of this type are used to represent keyboard keys using the current + * layout of the keyboard. These values include Unicode values representing + * the unmodified character that would be generated by pressing the key, or + * an SDLK_* constant for those keys that do not generate characters. + * + * A special exception is the number keys at the top of the keyboard which + * map to SDLK_0...SDLK_9 on AZERTY layouts. + */ +typedef Sint32 SDL_Keycode; + +#define SDLK_SCANCODE_MASK (1<<30) +#define SDL_SCANCODE_TO_KEYCODE(X) (X | SDLK_SCANCODE_MASK) + +typedef enum +{ + SDLK_UNKNOWN = 0, + + SDLK_RETURN = '\r', + SDLK_ESCAPE = '\x1B', + SDLK_BACKSPACE = '\b', + SDLK_TAB = '\t', + SDLK_SPACE = ' ', + SDLK_EXCLAIM = '!', + SDLK_QUOTEDBL = '"', + SDLK_HASH = '#', + SDLK_PERCENT = '%', + SDLK_DOLLAR = '$', + SDLK_AMPERSAND = '&', + SDLK_QUOTE = '\'', + SDLK_LEFTPAREN = '(', + SDLK_RIGHTPAREN = ')', + SDLK_ASTERISK = '*', + SDLK_PLUS = '+', + SDLK_COMMA = ',', + SDLK_MINUS = '-', + SDLK_PERIOD = '.', + SDLK_SLASH = '/', + SDLK_0 = '0', + SDLK_1 = '1', + SDLK_2 = '2', + SDLK_3 = '3', + SDLK_4 = '4', + SDLK_5 = '5', + SDLK_6 = '6', + SDLK_7 = '7', + SDLK_8 = '8', + SDLK_9 = '9', + SDLK_COLON = ':', + SDLK_SEMICOLON = ';', + SDLK_LESS = '<', + SDLK_EQUALS = '=', + SDLK_GREATER = '>', + SDLK_QUESTION = '?', + SDLK_AT = '@', + + /* + Skip uppercase letters + */ + + SDLK_LEFTBRACKET = '[', + SDLK_BACKSLASH = '\\', + SDLK_RIGHTBRACKET = ']', + SDLK_CARET = '^', + SDLK_UNDERSCORE = '_', + SDLK_BACKQUOTE = '`', + SDLK_a = 'a', + SDLK_b = 'b', + SDLK_c = 'c', + SDLK_d = 'd', + SDLK_e = 'e', + SDLK_f = 'f', + SDLK_g = 'g', + SDLK_h = 'h', + SDLK_i = 'i', + SDLK_j = 'j', + SDLK_k = 'k', + SDLK_l = 'l', + SDLK_m = 'm', + SDLK_n = 'n', + SDLK_o = 'o', + SDLK_p = 'p', + SDLK_q = 'q', + SDLK_r = 'r', + SDLK_s = 's', + SDLK_t = 't', + SDLK_u = 'u', + SDLK_v = 'v', + SDLK_w = 'w', + SDLK_x = 'x', + SDLK_y = 'y', + SDLK_z = 'z', + + SDLK_CAPSLOCK = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_CAPSLOCK), + + SDLK_F1 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F1), + SDLK_F2 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F2), + SDLK_F3 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F3), + SDLK_F4 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F4), + SDLK_F5 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F5), + SDLK_F6 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F6), + SDLK_F7 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F7), + SDLK_F8 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F8), + SDLK_F9 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F9), + SDLK_F10 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F10), + SDLK_F11 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F11), + SDLK_F12 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F12), + + SDLK_PRINTSCREEN = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_PRINTSCREEN), + SDLK_SCROLLLOCK = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_SCROLLLOCK), + SDLK_PAUSE = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_PAUSE), + SDLK_INSERT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_INSERT), + SDLK_HOME = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_HOME), + SDLK_PAGEUP = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_PAGEUP), + SDLK_DELETE = '\x7F', + SDLK_END = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_END), + SDLK_PAGEDOWN = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_PAGEDOWN), + SDLK_RIGHT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_RIGHT), + SDLK_LEFT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_LEFT), + SDLK_DOWN = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_DOWN), + SDLK_UP = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_UP), + + SDLK_NUMLOCKCLEAR = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_NUMLOCKCLEAR), + SDLK_KP_DIVIDE = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_DIVIDE), + SDLK_KP_MULTIPLY = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_MULTIPLY), + SDLK_KP_MINUS = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_MINUS), + SDLK_KP_PLUS = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_PLUS), + SDLK_KP_ENTER = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_ENTER), + SDLK_KP_1 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_1), + SDLK_KP_2 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_2), + SDLK_KP_3 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_3), + SDLK_KP_4 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_4), + SDLK_KP_5 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_5), + SDLK_KP_6 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_6), + SDLK_KP_7 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_7), + SDLK_KP_8 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_8), + SDLK_KP_9 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_9), + SDLK_KP_0 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_0), + SDLK_KP_PERIOD = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_PERIOD), + + SDLK_APPLICATION = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_APPLICATION), + SDLK_POWER = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_POWER), + SDLK_KP_EQUALS = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_EQUALS), + SDLK_F13 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F13), + SDLK_F14 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F14), + SDLK_F15 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F15), + SDLK_F16 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F16), + SDLK_F17 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F17), + SDLK_F18 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F18), + SDLK_F19 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F19), + SDLK_F20 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F20), + SDLK_F21 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F21), + SDLK_F22 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F22), + SDLK_F23 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F23), + SDLK_F24 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_F24), + SDLK_EXECUTE = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_EXECUTE), + SDLK_HELP = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_HELP), + SDLK_MENU = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_MENU), + SDLK_SELECT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_SELECT), + SDLK_STOP = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_STOP), + SDLK_AGAIN = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_AGAIN), + SDLK_UNDO = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_UNDO), + SDLK_CUT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_CUT), + SDLK_COPY = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_COPY), + SDLK_PASTE = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_PASTE), + SDLK_FIND = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_FIND), + SDLK_MUTE = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_MUTE), + SDLK_VOLUMEUP = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_VOLUMEUP), + SDLK_VOLUMEDOWN = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_VOLUMEDOWN), + SDLK_KP_COMMA = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_COMMA), + SDLK_KP_EQUALSAS400 = + SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_EQUALSAS400), + + SDLK_ALTERASE = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_ALTERASE), + SDLK_SYSREQ = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_SYSREQ), + SDLK_CANCEL = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_CANCEL), + SDLK_CLEAR = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_CLEAR), + SDLK_PRIOR = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_PRIOR), + SDLK_RETURN2 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_RETURN2), + SDLK_SEPARATOR = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_SEPARATOR), + SDLK_OUT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_OUT), + SDLK_OPER = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_OPER), + SDLK_CLEARAGAIN = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_CLEARAGAIN), + SDLK_CRSEL = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_CRSEL), + SDLK_EXSEL = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_EXSEL), + + SDLK_KP_00 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_00), + SDLK_KP_000 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_000), + SDLK_THOUSANDSSEPARATOR = + SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_THOUSANDSSEPARATOR), + SDLK_DECIMALSEPARATOR = + SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_DECIMALSEPARATOR), + SDLK_CURRENCYUNIT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_CURRENCYUNIT), + SDLK_CURRENCYSUBUNIT = + SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_CURRENCYSUBUNIT), + SDLK_KP_LEFTPAREN = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_LEFTPAREN), + SDLK_KP_RIGHTPAREN = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_RIGHTPAREN), + SDLK_KP_LEFTBRACE = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_LEFTBRACE), + SDLK_KP_RIGHTBRACE = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_RIGHTBRACE), + SDLK_KP_TAB = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_TAB), + SDLK_KP_BACKSPACE = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_BACKSPACE), + SDLK_KP_A = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_A), + SDLK_KP_B = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_B), + SDLK_KP_C = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_C), + SDLK_KP_D = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_D), + SDLK_KP_E = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_E), + SDLK_KP_F = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_F), + SDLK_KP_XOR = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_XOR), + SDLK_KP_POWER = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_POWER), + SDLK_KP_PERCENT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_PERCENT), + SDLK_KP_LESS = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_LESS), + SDLK_KP_GREATER = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_GREATER), + SDLK_KP_AMPERSAND = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_AMPERSAND), + SDLK_KP_DBLAMPERSAND = + SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_DBLAMPERSAND), + SDLK_KP_VERTICALBAR = + SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_VERTICALBAR), + SDLK_KP_DBLVERTICALBAR = + SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_DBLVERTICALBAR), + SDLK_KP_COLON = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_COLON), + SDLK_KP_HASH = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_HASH), + SDLK_KP_SPACE = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_SPACE), + SDLK_KP_AT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_AT), + SDLK_KP_EXCLAM = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_EXCLAM), + SDLK_KP_MEMSTORE = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_MEMSTORE), + SDLK_KP_MEMRECALL = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_MEMRECALL), + SDLK_KP_MEMCLEAR = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_MEMCLEAR), + SDLK_KP_MEMADD = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_MEMADD), + SDLK_KP_MEMSUBTRACT = + SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_MEMSUBTRACT), + SDLK_KP_MEMMULTIPLY = + SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_MEMMULTIPLY), + SDLK_KP_MEMDIVIDE = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_MEMDIVIDE), + SDLK_KP_PLUSMINUS = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_PLUSMINUS), + SDLK_KP_CLEAR = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_CLEAR), + SDLK_KP_CLEARENTRY = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_CLEARENTRY), + SDLK_KP_BINARY = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_BINARY), + SDLK_KP_OCTAL = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_OCTAL), + SDLK_KP_DECIMAL = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_DECIMAL), + SDLK_KP_HEXADECIMAL = + SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KP_HEXADECIMAL), + + SDLK_LCTRL = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_LCTRL), + SDLK_LSHIFT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_LSHIFT), + SDLK_LALT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_LALT), + SDLK_LGUI = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_LGUI), + SDLK_RCTRL = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_RCTRL), + SDLK_RSHIFT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_RSHIFT), + SDLK_RALT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_RALT), + SDLK_RGUI = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_RGUI), + + SDLK_MODE = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_MODE), + + SDLK_AUDIONEXT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_AUDIONEXT), + SDLK_AUDIOPREV = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_AUDIOPREV), + SDLK_AUDIOSTOP = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_AUDIOSTOP), + SDLK_AUDIOPLAY = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_AUDIOPLAY), + SDLK_AUDIOMUTE = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_AUDIOMUTE), + SDLK_MEDIASELECT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_MEDIASELECT), + SDLK_WWW = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_WWW), + SDLK_MAIL = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_MAIL), + SDLK_CALCULATOR = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_CALCULATOR), + SDLK_COMPUTER = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_COMPUTER), + SDLK_AC_SEARCH = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_AC_SEARCH), + SDLK_AC_HOME = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_AC_HOME), + SDLK_AC_BACK = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_AC_BACK), + SDLK_AC_FORWARD = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_AC_FORWARD), + SDLK_AC_STOP = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_AC_STOP), + SDLK_AC_REFRESH = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_AC_REFRESH), + SDLK_AC_BOOKMARKS = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_AC_BOOKMARKS), + + SDLK_BRIGHTNESSDOWN = + SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_BRIGHTNESSDOWN), + SDLK_BRIGHTNESSUP = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_BRIGHTNESSUP), + SDLK_DISPLAYSWITCH = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_DISPLAYSWITCH), + SDLK_KBDILLUMTOGGLE = + SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KBDILLUMTOGGLE), + SDLK_KBDILLUMDOWN = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KBDILLUMDOWN), + SDLK_KBDILLUMUP = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_KBDILLUMUP), + SDLK_EJECT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_EJECT), + SDLK_SLEEP = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_SLEEP), + SDLK_APP1 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_APP1), + SDLK_APP2 = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_APP2), + + SDLK_AUDIOREWIND = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_AUDIOREWIND), + SDLK_AUDIOFASTFORWARD = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_AUDIOFASTFORWARD), + + SDLK_SOFTLEFT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_SOFTLEFT), + SDLK_SOFTRIGHT = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_SOFTRIGHT), + SDLK_CALL = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_CALL), + SDLK_ENDCALL = SDL_SCANCODE_TO_KEYCODE(SDL_SCANCODE_ENDCALL) +} SDL_KeyCode; + +/** + * \brief Enumeration of valid key mods (possibly OR'd together). + */ +typedef enum +{ + KMOD_NONE = 0x0000, + KMOD_LSHIFT = 0x0001, + KMOD_RSHIFT = 0x0002, + KMOD_LCTRL = 0x0040, + KMOD_RCTRL = 0x0080, + KMOD_LALT = 0x0100, + KMOD_RALT = 0x0200, + KMOD_LGUI = 0x0400, + KMOD_RGUI = 0x0800, + KMOD_NUM = 0x1000, + KMOD_CAPS = 0x2000, + KMOD_MODE = 0x4000, + KMOD_SCROLL = 0x8000, + + KMOD_CTRL = KMOD_LCTRL | KMOD_RCTRL, + KMOD_SHIFT = KMOD_LSHIFT | KMOD_RSHIFT, + KMOD_ALT = KMOD_LALT | KMOD_RALT, + KMOD_GUI = KMOD_LGUI | KMOD_RGUI, + + KMOD_RESERVED = KMOD_SCROLL /* This is for source-level compatibility with SDL 2.0.0. */ +} SDL_Keymod; + +#endif /* SDL_keycode_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_loadso.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_loadso.h new file mode 100644 index 00000000..ca59b681 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_loadso.h @@ -0,0 +1,115 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_loadso.h + * + * System dependent library loading routines + * + * Some things to keep in mind: + * \li These functions only work on C function names. Other languages may + * have name mangling and intrinsic language support that varies from + * compiler to compiler. + * \li Make sure you declare your function pointers with the same calling + * convention as the actual library function. Your code will crash + * mysteriously if you do not do this. + * \li Avoid namespace collisions. If you load a symbol from the library, + * it is not defined whether or not it goes into the global symbol + * namespace for the application. If it does and it conflicts with + * symbols in your code or other shared libraries, you will not get + * the results you expect. :) + */ + +#ifndef SDL_loadso_h_ +#define SDL_loadso_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * Dynamically load a shared object. + * + * \param sofile a system-dependent name of the object file + * \returns an opaque pointer to the object handle or NULL if there was an + * error; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LoadFunction + * \sa SDL_UnloadObject + */ +extern DECLSPEC void *SDLCALL SDL_LoadObject(const char *sofile); + +/** + * Look up the address of the named function in a shared object. + * + * This function pointer is no longer valid after calling SDL_UnloadObject(). + * + * This function can only look up C function names. Other languages may have + * name mangling and intrinsic language support that varies from compiler to + * compiler. + * + * Make sure you declare your function pointers with the same calling + * convention as the actual library function. Your code will crash + * mysteriously if you do not do this. + * + * If the requested function doesn't exist, NULL is returned. + * + * \param handle a valid shared object handle returned by SDL_LoadObject() + * \param name the name of the function to look up + * \returns a pointer to the function or NULL if there was an error; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LoadObject + * \sa SDL_UnloadObject + */ +extern DECLSPEC void *SDLCALL SDL_LoadFunction(void *handle, + const char *name); + +/** + * Unload a shared object from memory. + * + * \param handle a valid shared object handle returned by SDL_LoadObject() + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LoadFunction + * \sa SDL_LoadObject + */ +extern DECLSPEC void SDLCALL SDL_UnloadObject(void *handle); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_loadso_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_locale.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_locale.h new file mode 100644 index 00000000..482dbefe --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_locale.h @@ -0,0 +1,103 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_locale.h + * + * Include file for SDL locale services + */ + +#ifndef _SDL_locale_h +#define _SDL_locale_h + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +/* *INDENT-OFF* */ +extern "C" { +/* *INDENT-ON* */ +#endif + + +typedef struct SDL_Locale +{ + const char *language; /**< A language name, like "en" for English. */ + const char *country; /**< A country, like "US" for America. Can be NULL. */ +} SDL_Locale; + +/** + * Report the user's preferred locale. + * + * This returns an array of SDL_Locale structs, the final item zeroed out. + * When the caller is done with this array, it should call SDL_free() on the + * returned value; all the memory involved is allocated in a single block, so + * a single SDL_free() will suffice. + * + * Returned language strings are in the format xx, where 'xx' is an ISO-639 + * language specifier (such as "en" for English, "de" for German, etc). + * Country strings are in the format YY, where "YY" is an ISO-3166 country + * code (such as "US" for the United States, "CA" for Canada, etc). Country + * might be NULL if there's no specific guidance on them (so you might get { + * "en", "US" } for American English, but { "en", NULL } means "English + * language, generically"). Language strings are never NULL, except to + * terminate the array. + * + * Please note that not all of these strings are 2 characters; some are three + * or more. + * + * The returned list of locales are in the order of the user's preference. For + * example, a German citizen that is fluent in US English and knows enough + * Japanese to navigate around Tokyo might have a list like: { "de", "en_US", + * "jp", NULL }. Someone from England might prefer British English (where + * "color" is spelled "colour", etc), but will settle for anything like it: { + * "en_GB", "en", NULL }. + * + * This function returns NULL on error, including when the platform does not + * supply this information at all. + * + * This might be a "slow" call that has to query the operating system. It's + * best to ask for this once and save the results. However, this list can + * change, usually because the user has changed a system preference outside of + * your program; SDL will send an SDL_LOCALECHANGED event in this case, if + * possible, and you can call this function again to get an updated copy of + * preferred locales. + * + * \return array of locales, terminated with a locale with a NULL language + * field. Will return NULL on error. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC SDL_Locale * SDLCALL SDL_GetPreferredLocales(void); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +/* *INDENT-OFF* */ +} +/* *INDENT-ON* */ +#endif +#include "close_code.h" + +#endif /* _SDL_locale_h */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_log.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_log.h new file mode 100644 index 00000000..da733c40 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_log.h @@ -0,0 +1,404 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_log.h + * + * Simple log messages with categories and priorities. + * + * By default logs are quiet, but if you're debugging SDL you might want: + * + * SDL_LogSetAllPriority(SDL_LOG_PRIORITY_WARN); + * + * Here's where the messages go on different platforms: + * Windows: debug output stream + * Android: log output + * Others: standard error output (stderr) + */ + +#ifndef SDL_log_h_ +#define SDL_log_h_ + +#include "SDL_stdinc.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * \brief The maximum size of a log message prior to SDL 2.0.24 + * + * As of 2.0.24 there is no limit to the length of SDL log messages. + */ +#define SDL_MAX_LOG_MESSAGE 4096 + +/** + * \brief The predefined log categories + * + * By default the application category is enabled at the INFO level, + * the assert category is enabled at the WARN level, test is enabled + * at the VERBOSE level and all other categories are enabled at the + * CRITICAL level. + */ +typedef enum +{ + SDL_LOG_CATEGORY_APPLICATION, + SDL_LOG_CATEGORY_ERROR, + SDL_LOG_CATEGORY_ASSERT, + SDL_LOG_CATEGORY_SYSTEM, + SDL_LOG_CATEGORY_AUDIO, + SDL_LOG_CATEGORY_VIDEO, + SDL_LOG_CATEGORY_RENDER, + SDL_LOG_CATEGORY_INPUT, + SDL_LOG_CATEGORY_TEST, + + /* Reserved for future SDL library use */ + SDL_LOG_CATEGORY_RESERVED1, + SDL_LOG_CATEGORY_RESERVED2, + SDL_LOG_CATEGORY_RESERVED3, + SDL_LOG_CATEGORY_RESERVED4, + SDL_LOG_CATEGORY_RESERVED5, + SDL_LOG_CATEGORY_RESERVED6, + SDL_LOG_CATEGORY_RESERVED7, + SDL_LOG_CATEGORY_RESERVED8, + SDL_LOG_CATEGORY_RESERVED9, + SDL_LOG_CATEGORY_RESERVED10, + + /* Beyond this point is reserved for application use, e.g. + enum { + MYAPP_CATEGORY_AWESOME1 = SDL_LOG_CATEGORY_CUSTOM, + MYAPP_CATEGORY_AWESOME2, + MYAPP_CATEGORY_AWESOME3, + ... + }; + */ + SDL_LOG_CATEGORY_CUSTOM +} SDL_LogCategory; + +/** + * \brief The predefined log priorities + */ +typedef enum +{ + SDL_LOG_PRIORITY_VERBOSE = 1, + SDL_LOG_PRIORITY_DEBUG, + SDL_LOG_PRIORITY_INFO, + SDL_LOG_PRIORITY_WARN, + SDL_LOG_PRIORITY_ERROR, + SDL_LOG_PRIORITY_CRITICAL, + SDL_NUM_LOG_PRIORITIES +} SDL_LogPriority; + + +/** + * Set the priority of all log categories. + * + * \param priority the SDL_LogPriority to assign + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LogSetPriority + */ +extern DECLSPEC void SDLCALL SDL_LogSetAllPriority(SDL_LogPriority priority); + +/** + * Set the priority of a particular log category. + * + * \param category the category to assign a priority to + * \param priority the SDL_LogPriority to assign + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LogGetPriority + * \sa SDL_LogSetAllPriority + */ +extern DECLSPEC void SDLCALL SDL_LogSetPriority(int category, + SDL_LogPriority priority); + +/** + * Get the priority of a particular log category. + * + * \param category the category to query + * \returns the SDL_LogPriority for the requested category + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LogSetPriority + */ +extern DECLSPEC SDL_LogPriority SDLCALL SDL_LogGetPriority(int category); + +/** + * Reset all priorities to default. + * + * This is called by SDL_Quit(). + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LogSetAllPriority + * \sa SDL_LogSetPriority + */ +extern DECLSPEC void SDLCALL SDL_LogResetPriorities(void); + +/** + * Log a message with SDL_LOG_CATEGORY_APPLICATION and SDL_LOG_PRIORITY_INFO. + * + * = * \param fmt a printf() style message format string + * + * \param ... additional parameters matching % tokens in the `fmt` string, if + * any + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LogCritical + * \sa SDL_LogDebug + * \sa SDL_LogError + * \sa SDL_LogInfo + * \sa SDL_LogMessage + * \sa SDL_LogMessageV + * \sa SDL_LogVerbose + * \sa SDL_LogWarn + */ +extern DECLSPEC void SDLCALL SDL_Log(SDL_PRINTF_FORMAT_STRING const char *fmt, ...) SDL_PRINTF_VARARG_FUNC(1); + +/** + * Log a message with SDL_LOG_PRIORITY_VERBOSE. + * + * \param category the category of the message + * \param fmt a printf() style message format string + * \param ... additional parameters matching % tokens in the **fmt** string, + * if any + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Log + * \sa SDL_LogCritical + * \sa SDL_LogDebug + * \sa SDL_LogError + * \sa SDL_LogInfo + * \sa SDL_LogMessage + * \sa SDL_LogMessageV + * \sa SDL_LogWarn + */ +extern DECLSPEC void SDLCALL SDL_LogVerbose(int category, SDL_PRINTF_FORMAT_STRING const char *fmt, ...) SDL_PRINTF_VARARG_FUNC(2); + +/** + * Log a message with SDL_LOG_PRIORITY_DEBUG. + * + * \param category the category of the message + * \param fmt a printf() style message format string + * \param ... additional parameters matching % tokens in the **fmt** string, + * if any + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Log + * \sa SDL_LogCritical + * \sa SDL_LogError + * \sa SDL_LogInfo + * \sa SDL_LogMessage + * \sa SDL_LogMessageV + * \sa SDL_LogVerbose + * \sa SDL_LogWarn + */ +extern DECLSPEC void SDLCALL SDL_LogDebug(int category, SDL_PRINTF_FORMAT_STRING const char *fmt, ...) SDL_PRINTF_VARARG_FUNC(2); + +/** + * Log a message with SDL_LOG_PRIORITY_INFO. + * + * \param category the category of the message + * \param fmt a printf() style message format string + * \param ... additional parameters matching % tokens in the **fmt** string, + * if any + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Log + * \sa SDL_LogCritical + * \sa SDL_LogDebug + * \sa SDL_LogError + * \sa SDL_LogMessage + * \sa SDL_LogMessageV + * \sa SDL_LogVerbose + * \sa SDL_LogWarn + */ +extern DECLSPEC void SDLCALL SDL_LogInfo(int category, SDL_PRINTF_FORMAT_STRING const char *fmt, ...) SDL_PRINTF_VARARG_FUNC(2); + +/** + * Log a message with SDL_LOG_PRIORITY_WARN. + * + * \param category the category of the message + * \param fmt a printf() style message format string + * \param ... additional parameters matching % tokens in the **fmt** string, + * if any + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Log + * \sa SDL_LogCritical + * \sa SDL_LogDebug + * \sa SDL_LogError + * \sa SDL_LogInfo + * \sa SDL_LogMessage + * \sa SDL_LogMessageV + * \sa SDL_LogVerbose + */ +extern DECLSPEC void SDLCALL SDL_LogWarn(int category, SDL_PRINTF_FORMAT_STRING const char *fmt, ...) SDL_PRINTF_VARARG_FUNC(2); + +/** + * Log a message with SDL_LOG_PRIORITY_ERROR. + * + * \param category the category of the message + * \param fmt a printf() style message format string + * \param ... additional parameters matching % tokens in the **fmt** string, + * if any + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Log + * \sa SDL_LogCritical + * \sa SDL_LogDebug + * \sa SDL_LogInfo + * \sa SDL_LogMessage + * \sa SDL_LogMessageV + * \sa SDL_LogVerbose + * \sa SDL_LogWarn + */ +extern DECLSPEC void SDLCALL SDL_LogError(int category, SDL_PRINTF_FORMAT_STRING const char *fmt, ...) SDL_PRINTF_VARARG_FUNC(2); + +/** + * Log a message with SDL_LOG_PRIORITY_CRITICAL. + * + * \param category the category of the message + * \param fmt a printf() style message format string + * \param ... additional parameters matching % tokens in the **fmt** string, + * if any + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Log + * \sa SDL_LogDebug + * \sa SDL_LogError + * \sa SDL_LogInfo + * \sa SDL_LogMessage + * \sa SDL_LogMessageV + * \sa SDL_LogVerbose + * \sa SDL_LogWarn + */ +extern DECLSPEC void SDLCALL SDL_LogCritical(int category, SDL_PRINTF_FORMAT_STRING const char *fmt, ...) SDL_PRINTF_VARARG_FUNC(2); + +/** + * Log a message with the specified category and priority. + * + * \param category the category of the message + * \param priority the priority of the message + * \param fmt a printf() style message format string + * \param ... additional parameters matching % tokens in the **fmt** string, + * if any + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Log + * \sa SDL_LogCritical + * \sa SDL_LogDebug + * \sa SDL_LogError + * \sa SDL_LogInfo + * \sa SDL_LogMessageV + * \sa SDL_LogVerbose + * \sa SDL_LogWarn + */ +extern DECLSPEC void SDLCALL SDL_LogMessage(int category, + SDL_LogPriority priority, + SDL_PRINTF_FORMAT_STRING const char *fmt, ...) SDL_PRINTF_VARARG_FUNC(3); + +/** + * Log a message with the specified category and priority. + * + * \param category the category of the message + * \param priority the priority of the message + * \param fmt a printf() style message format string + * \param ap a variable argument list + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Log + * \sa SDL_LogCritical + * \sa SDL_LogDebug + * \sa SDL_LogError + * \sa SDL_LogInfo + * \sa SDL_LogMessage + * \sa SDL_LogVerbose + * \sa SDL_LogWarn + */ +extern DECLSPEC void SDLCALL SDL_LogMessageV(int category, + SDL_LogPriority priority, + const char *fmt, va_list ap); + +/** + * The prototype for the log output callback function. + * + * This function is called by SDL when there is new text to be logged. + * + * \param userdata what was passed as `userdata` to SDL_LogSetOutputFunction() + * \param category the category of the message + * \param priority the priority of the message + * \param message the message being output + */ +typedef void (SDLCALL *SDL_LogOutputFunction)(void *userdata, int category, SDL_LogPriority priority, const char *message); + +/** + * Get the current log output function. + * + * \param callback an SDL_LogOutputFunction filled in with the current log + * callback + * \param userdata a pointer filled in with the pointer that is passed to + * `callback` + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LogSetOutputFunction + */ +extern DECLSPEC void SDLCALL SDL_LogGetOutputFunction(SDL_LogOutputFunction *callback, void **userdata); + +/** + * Replace the default log output function with one of your own. + * + * \param callback an SDL_LogOutputFunction to call instead of the default + * \param userdata a pointer that is passed to `callback` + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LogGetOutputFunction + */ +extern DECLSPEC void SDLCALL SDL_LogSetOutputFunction(SDL_LogOutputFunction callback, void *userdata); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_log_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_main.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_main.h new file mode 100644 index 00000000..5cc8e591 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_main.h @@ -0,0 +1,282 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +#ifndef SDL_main_h_ +#define SDL_main_h_ + +#include "SDL_stdinc.h" + +/** + * \file SDL_main.h + * + * Redefine main() on some platforms so that it is called by SDL. + */ + +#ifndef SDL_MAIN_HANDLED +#if defined(__WIN32__) +/* On Windows SDL provides WinMain(), which parses the command line and passes + the arguments to your main function. + + If you provide your own WinMain(), you may define SDL_MAIN_HANDLED + */ +#define SDL_MAIN_AVAILABLE + +#elif defined(__WINRT__) +/* On WinRT, SDL provides a main function that initializes CoreApplication, + creating an instance of IFrameworkView in the process. + + Please note that #include'ing SDL_main.h is not enough to get a main() + function working. In non-XAML apps, the file, + src/main/winrt/SDL_WinRT_main_NonXAML.cpp, or a copy of it, must be compiled + into the app itself. In XAML apps, the function, SDL_WinRTRunApp must be + called, with a pointer to the Direct3D-hosted XAML control passed in. +*/ +#define SDL_MAIN_NEEDED + +#elif defined(__GDK__) +/* On GDK, SDL provides a main function that initializes the game runtime. + + Please note that #include'ing SDL_main.h is not enough to get a main() + function working. You must either link against SDL2main or, if not possible, + call the SDL_GDKRunApp function from your entry point. +*/ +#define SDL_MAIN_NEEDED + +#elif defined(__IPHONEOS__) +/* On iOS SDL provides a main function that creates an application delegate + and starts the iOS application run loop. + + If you link with SDL dynamically on iOS, the main function can't be in a + shared library, so you need to link with libSDLmain.a, which includes a + stub main function that calls into the shared library to start execution. + + See src/video/uikit/SDL_uikitappdelegate.m for more details. + */ +#define SDL_MAIN_NEEDED + +#elif defined(__ANDROID__) +/* On Android SDL provides a Java class in SDLActivity.java that is the + main activity entry point. + + See docs/README-android.md for more details on extending that class. + */ +#define SDL_MAIN_NEEDED + +/* We need to export SDL_main so it can be launched from Java */ +#define SDLMAIN_DECLSPEC DECLSPEC + +#elif defined(__NACL__) +/* On NACL we use ppapi_simple to set up the application helper code, + then wait for the first PSE_INSTANCE_DIDCHANGEVIEW event before + starting the user main function. + All user code is run in a separate thread by ppapi_simple, thus + allowing for blocking io to take place via nacl_io +*/ +#define SDL_MAIN_NEEDED + +#elif defined(__PSP__) +/* On PSP SDL provides a main function that sets the module info, + activates the GPU and starts the thread required to be able to exit + the software. + + If you provide this yourself, you may define SDL_MAIN_HANDLED + */ +#define SDL_MAIN_AVAILABLE + +#elif defined(__PS2__) +#define SDL_MAIN_AVAILABLE + +#define SDL_PS2_SKIP_IOP_RESET() \ + void reset_IOP(); \ + void reset_IOP() {} + +#elif defined(__3DS__) +/* + On N3DS, SDL provides a main function that sets up the screens + and storage. + + If you provide this yourself, you may define SDL_MAIN_HANDLED +*/ +#define SDL_MAIN_AVAILABLE + +#endif +#endif /* SDL_MAIN_HANDLED */ + +#ifndef SDLMAIN_DECLSPEC +#define SDLMAIN_DECLSPEC +#endif + +/** + * \file SDL_main.h + * + * The application's main() function must be called with C linkage, + * and should be declared like this: + * \code + * #ifdef __cplusplus + * extern "C" + * #endif + * int main(int argc, char *argv[]) + * { + * } + * \endcode + */ + +#if defined(SDL_MAIN_NEEDED) || defined(SDL_MAIN_AVAILABLE) +#define main SDL_main +#endif + +#include "begin_code.h" +#ifdef __cplusplus +extern "C" { +#endif + +/** + * The prototype for the application's main() function + */ +typedef int (*SDL_main_func)(int argc, char *argv[]); +extern SDLMAIN_DECLSPEC int SDL_main(int argc, char *argv[]); + + +/** + * Circumvent failure of SDL_Init() when not using SDL_main() as an entry + * point. + * + * This function is defined in SDL_main.h, along with the preprocessor rule to + * redefine main() as SDL_main(). Thus to ensure that your main() function + * will not be changed it is necessary to define SDL_MAIN_HANDLED before + * including SDL.h. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_Init + */ +extern DECLSPEC void SDLCALL SDL_SetMainReady(void); + +#if defined(__WIN32__) || defined(__GDK__) + +/** + * Register a win32 window class for SDL's use. + * + * This can be called to set the application window class at startup. It is + * safe to call this multiple times, as long as every call is eventually + * paired with a call to SDL_UnregisterApp, but a second registration attempt + * while a previous registration is still active will be ignored, other than + * to increment a counter. + * + * Most applications do not need to, and should not, call this directly; SDL + * will call it when initializing the video subsystem. + * + * \param name the window class name, in UTF-8 encoding. If NULL, SDL + * currently uses "SDL_app" but this isn't guaranteed. + * \param style the value to use in WNDCLASSEX::style. If `name` is NULL, SDL + * currently uses `(CS_BYTEALIGNCLIENT | CS_OWNDC)` regardless of + * what is specified here. + * \param hInst the HINSTANCE to use in WNDCLASSEX::hInstance. If zero, SDL + * will use `GetModuleHandle(NULL)` instead. + * \returns 0 on success, -1 on error. SDL_GetError() may have details. + * + * \since This function is available since SDL 2.0.2. + */ +extern DECLSPEC int SDLCALL SDL_RegisterApp(const char *name, Uint32 style, void *hInst); + +/** + * Deregister the win32 window class from an SDL_RegisterApp call. + * + * This can be called to undo the effects of SDL_RegisterApp. + * + * Most applications do not need to, and should not, call this directly; SDL + * will call it when deinitializing the video subsystem. + * + * It is safe to call this multiple times, as long as every call is eventually + * paired with a prior call to SDL_RegisterApp. The window class will only be + * deregistered when the registration counter in SDL_RegisterApp decrements to + * zero through calls to this function. + * + * \since This function is available since SDL 2.0.2. + */ +extern DECLSPEC void SDLCALL SDL_UnregisterApp(void); + +#endif /* defined(__WIN32__) || defined(__GDK__) */ + + +#ifdef __WINRT__ + +/** + * Initialize and launch an SDL/WinRT application. + * + * \param mainFunction the SDL app's C-style main(), an SDL_main_func + * \param reserved reserved for future use; should be NULL + * \returns 0 on success or -1 on failure; call SDL_GetError() to retrieve + * more information on the failure. + * + * \since This function is available since SDL 2.0.3. + */ +extern DECLSPEC int SDLCALL SDL_WinRTRunApp(SDL_main_func mainFunction, void * reserved); + +#endif /* __WINRT__ */ + +#if defined(__IPHONEOS__) + +/** + * Initializes and launches an SDL application. + * + * \param argc The argc parameter from the application's main() function + * \param argv The argv parameter from the application's main() function + * \param mainFunction The SDL app's C-style main(), an SDL_main_func + * \return the return value from mainFunction + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC int SDLCALL SDL_UIKitRunApp(int argc, char *argv[], SDL_main_func mainFunction); + +#endif /* __IPHONEOS__ */ + +#ifdef __GDK__ + +/** + * Initialize and launch an SDL GDK application. + * + * \param mainFunction the SDL app's C-style main(), an SDL_main_func + * \param reserved reserved for future use; should be NULL + * \returns 0 on success or -1 on failure; call SDL_GetError() to retrieve + * more information on the failure. + * + * \since This function is available since SDL 2.24.0. + */ +extern DECLSPEC int SDLCALL SDL_GDKRunApp(SDL_main_func mainFunction, void *reserved); + +/** + * Callback from the application to let the suspend continue. + * + * \since This function is available since SDL 2.28.0. + */ +extern DECLSPEC void SDLCALL SDL_GDKSuspendComplete(void); + +#endif /* __GDK__ */ + +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_main_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_messagebox.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_messagebox.h new file mode 100644 index 00000000..7896fd12 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_messagebox.h @@ -0,0 +1,193 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +#ifndef SDL_messagebox_h_ +#define SDL_messagebox_h_ + +#include "SDL_stdinc.h" +#include "SDL_video.h" /* For SDL_Window */ + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * SDL_MessageBox flags. If supported will display warning icon, etc. + */ +typedef enum +{ + SDL_MESSAGEBOX_ERROR = 0x00000010, /**< error dialog */ + SDL_MESSAGEBOX_WARNING = 0x00000020, /**< warning dialog */ + SDL_MESSAGEBOX_INFORMATION = 0x00000040, /**< informational dialog */ + SDL_MESSAGEBOX_BUTTONS_LEFT_TO_RIGHT = 0x00000080, /**< buttons placed left to right */ + SDL_MESSAGEBOX_BUTTONS_RIGHT_TO_LEFT = 0x00000100 /**< buttons placed right to left */ +} SDL_MessageBoxFlags; + +/** + * Flags for SDL_MessageBoxButtonData. + */ +typedef enum +{ + SDL_MESSAGEBOX_BUTTON_RETURNKEY_DEFAULT = 0x00000001, /**< Marks the default button when return is hit */ + SDL_MESSAGEBOX_BUTTON_ESCAPEKEY_DEFAULT = 0x00000002 /**< Marks the default button when escape is hit */ +} SDL_MessageBoxButtonFlags; + +/** + * Individual button data. + */ +typedef struct +{ + Uint32 flags; /**< ::SDL_MessageBoxButtonFlags */ + int buttonid; /**< User defined button id (value returned via SDL_ShowMessageBox) */ + const char * text; /**< The UTF-8 button text */ +} SDL_MessageBoxButtonData; + +/** + * RGB value used in a message box color scheme + */ +typedef struct +{ + Uint8 r, g, b; +} SDL_MessageBoxColor; + +typedef enum +{ + SDL_MESSAGEBOX_COLOR_BACKGROUND, + SDL_MESSAGEBOX_COLOR_TEXT, + SDL_MESSAGEBOX_COLOR_BUTTON_BORDER, + SDL_MESSAGEBOX_COLOR_BUTTON_BACKGROUND, + SDL_MESSAGEBOX_COLOR_BUTTON_SELECTED, + SDL_MESSAGEBOX_COLOR_MAX +} SDL_MessageBoxColorType; + +/** + * A set of colors to use for message box dialogs + */ +typedef struct +{ + SDL_MessageBoxColor colors[SDL_MESSAGEBOX_COLOR_MAX]; +} SDL_MessageBoxColorScheme; + +/** + * MessageBox structure containing title, text, window, etc. + */ +typedef struct +{ + Uint32 flags; /**< ::SDL_MessageBoxFlags */ + SDL_Window *window; /**< Parent window, can be NULL */ + const char *title; /**< UTF-8 title */ + const char *message; /**< UTF-8 message text */ + + int numbuttons; + const SDL_MessageBoxButtonData *buttons; + + const SDL_MessageBoxColorScheme *colorScheme; /**< ::SDL_MessageBoxColorScheme, can be NULL to use system settings */ +} SDL_MessageBoxData; + +/** + * Create a modal message box. + * + * If your needs aren't complex, it might be easier to use + * SDL_ShowSimpleMessageBox. + * + * This function should be called on the thread that created the parent + * window, or on the main thread if the messagebox has no parent. It will + * block execution of that thread until the user clicks a button or closes the + * messagebox. + * + * This function may be called at any time, even before SDL_Init(). This makes + * it useful for reporting errors like a failure to create a renderer or + * OpenGL context. + * + * On X11, SDL rolls its own dialog box with X11 primitives instead of a + * formal toolkit like GTK+ or Qt. + * + * Note that if SDL_Init() would fail because there isn't any available video + * target, this function is likely to fail for the same reasons. If this is a + * concern, check the return value from this function and fall back to writing + * to stderr if you can. + * + * \param messageboxdata the SDL_MessageBoxData structure with title, text and + * other options + * \param buttonid the pointer to which user id of hit button should be copied + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ShowSimpleMessageBox + */ +extern DECLSPEC int SDLCALL SDL_ShowMessageBox(const SDL_MessageBoxData *messageboxdata, int *buttonid); + +/** + * Display a simple modal message box. + * + * If your needs aren't complex, this function is preferred over + * SDL_ShowMessageBox. + * + * `flags` may be any of the following: + * + * - `SDL_MESSAGEBOX_ERROR`: error dialog + * - `SDL_MESSAGEBOX_WARNING`: warning dialog + * - `SDL_MESSAGEBOX_INFORMATION`: informational dialog + * + * This function should be called on the thread that created the parent + * window, or on the main thread if the messagebox has no parent. It will + * block execution of that thread until the user clicks a button or closes the + * messagebox. + * + * This function may be called at any time, even before SDL_Init(). This makes + * it useful for reporting errors like a failure to create a renderer or + * OpenGL context. + * + * On X11, SDL rolls its own dialog box with X11 primitives instead of a + * formal toolkit like GTK+ or Qt. + * + * Note that if SDL_Init() would fail because there isn't any available video + * target, this function is likely to fail for the same reasons. If this is a + * concern, check the return value from this function and fall back to writing + * to stderr if you can. + * + * \param flags an SDL_MessageBoxFlags value + * \param title UTF-8 title text + * \param message UTF-8 message text + * \param window the parent window, or NULL for no parent + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ShowMessageBox + */ +extern DECLSPEC int SDLCALL SDL_ShowSimpleMessageBox(Uint32 flags, const char *title, const char *message, SDL_Window *window); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_messagebox_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_metal.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_metal.h new file mode 100644 index 00000000..f36e3487 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_metal.h @@ -0,0 +1,113 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_metal.h + * + * Header file for functions to creating Metal layers and views on SDL windows. + */ + +#ifndef SDL_metal_h_ +#define SDL_metal_h_ + +#include "SDL_video.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \brief A handle to a CAMetalLayer-backed NSView (macOS) or UIView (iOS/tvOS). + * + * \note This can be cast directly to an NSView or UIView. + */ +typedef void *SDL_MetalView; + +/** + * \name Metal support functions + */ +/* @{ */ + +/** + * Create a CAMetalLayer-backed NSView/UIView and attach it to the specified + * window. + * + * On macOS, this does *not* associate a MTLDevice with the CAMetalLayer on + * its own. It is up to user code to do that. + * + * The returned handle can be casted directly to a NSView or UIView. To access + * the backing CAMetalLayer, call SDL_Metal_GetLayer(). + * + * \since This function is available since SDL 2.0.12. + * + * \sa SDL_Metal_DestroyView + * \sa SDL_Metal_GetLayer + */ +extern DECLSPEC SDL_MetalView SDLCALL SDL_Metal_CreateView(SDL_Window * window); + +/** + * Destroy an existing SDL_MetalView object. + * + * This should be called before SDL_DestroyWindow, if SDL_Metal_CreateView was + * called after SDL_CreateWindow. + * + * \since This function is available since SDL 2.0.12. + * + * \sa SDL_Metal_CreateView + */ +extern DECLSPEC void SDLCALL SDL_Metal_DestroyView(SDL_MetalView view); + +/** + * Get a pointer to the backing CAMetalLayer for the given view. + * + * \since This function is available since SDL 2.0.14. + * + * \sa SDL_Metal_CreateView + */ +extern DECLSPEC void *SDLCALL SDL_Metal_GetLayer(SDL_MetalView view); + +/** + * Get the size of a window's underlying drawable in pixels (for use with + * setting viewport, scissor & etc). + * + * \param window SDL_Window from which the drawable size should be queried + * \param w Pointer to variable for storing the width in pixels, may be NULL + * \param h Pointer to variable for storing the height in pixels, may be NULL + * + * \since This function is available since SDL 2.0.14. + * + * \sa SDL_GetWindowSize + * \sa SDL_CreateWindow + */ +extern DECLSPEC void SDLCALL SDL_Metal_GetDrawableSize(SDL_Window* window, int *w, + int *h); + +/* @} *//* Metal support functions */ + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_metal_h_ */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_misc.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_misc.h new file mode 100644 index 00000000..13ed9c77 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_misc.h @@ -0,0 +1,79 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_misc.h + * + * \brief Include file for SDL API functions that don't fit elsewhere. + */ + +#ifndef SDL_misc_h_ +#define SDL_misc_h_ + +#include "SDL_stdinc.h" + +#include "begin_code.h" + +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * Open a URL/URI in the browser or other appropriate external application. + * + * Open a URL in a separate, system-provided application. How this works will + * vary wildly depending on the platform. This will likely launch what makes + * sense to handle a specific URL's protocol (a web browser for `http://`, + * etc), but it might also be able to launch file managers for directories and + * other things. + * + * What happens when you open a URL varies wildly as well: your game window + * may lose focus (and may or may not lose focus if your game was fullscreen + * or grabbing input at the time). On mobile devices, your app will likely + * move to the background or your process might be paused. Any given platform + * may or may not handle a given URL. + * + * If this is unimplemented (or simply unavailable) for a platform, this will + * fail with an error. A successful result does not mean the URL loaded, just + * that we launched _something_ to handle it (or at least believe we did). + * + * All this to say: this function can be useful, but you should definitely + * test it on every platform you target. + * + * \param url A valid URL/URI to open. Use `file:///full/path/to/file` for + * local files, if supported. + * \returns 0 on success, or -1 on error; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC int SDLCALL SDL_OpenURL(const char *url); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_misc_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_mouse.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_mouse.h new file mode 100644 index 00000000..aa075757 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_mouse.h @@ -0,0 +1,464 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_mouse.h + * + * Include file for SDL mouse event handling. + */ + +#ifndef SDL_mouse_h_ +#define SDL_mouse_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_video.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +typedef struct SDL_Cursor SDL_Cursor; /**< Implementation dependent */ + +/** + * \brief Cursor types for SDL_CreateSystemCursor(). + */ +typedef enum +{ + SDL_SYSTEM_CURSOR_ARROW, /**< Arrow */ + SDL_SYSTEM_CURSOR_IBEAM, /**< I-beam */ + SDL_SYSTEM_CURSOR_WAIT, /**< Wait */ + SDL_SYSTEM_CURSOR_CROSSHAIR, /**< Crosshair */ + SDL_SYSTEM_CURSOR_WAITARROW, /**< Small wait cursor (or Wait if not available) */ + SDL_SYSTEM_CURSOR_SIZENWSE, /**< Double arrow pointing northwest and southeast */ + SDL_SYSTEM_CURSOR_SIZENESW, /**< Double arrow pointing northeast and southwest */ + SDL_SYSTEM_CURSOR_SIZEWE, /**< Double arrow pointing west and east */ + SDL_SYSTEM_CURSOR_SIZENS, /**< Double arrow pointing north and south */ + SDL_SYSTEM_CURSOR_SIZEALL, /**< Four pointed arrow pointing north, south, east, and west */ + SDL_SYSTEM_CURSOR_NO, /**< Slashed circle or crossbones */ + SDL_SYSTEM_CURSOR_HAND, /**< Hand */ + SDL_NUM_SYSTEM_CURSORS +} SDL_SystemCursor; + +/** + * \brief Scroll direction types for the Scroll event + */ +typedef enum +{ + SDL_MOUSEWHEEL_NORMAL, /**< The scroll direction is normal */ + SDL_MOUSEWHEEL_FLIPPED /**< The scroll direction is flipped / natural */ +} SDL_MouseWheelDirection; + +/* Function prototypes */ + +/** + * Get the window which currently has mouse focus. + * + * \returns the window with mouse focus. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC SDL_Window * SDLCALL SDL_GetMouseFocus(void); + +/** + * Retrieve the current state of the mouse. + * + * The current button state is returned as a button bitmask, which can be + * tested using the `SDL_BUTTON(X)` macros (where `X` is generally 1 for the + * left, 2 for middle, 3 for the right button), and `x` and `y` are set to the + * mouse cursor position relative to the focus window. You can pass NULL for + * either `x` or `y`. + * + * \param x the x coordinate of the mouse cursor position relative to the + * focus window + * \param y the y coordinate of the mouse cursor position relative to the + * focus window + * \returns a 32-bit button bitmask of the current button state. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetGlobalMouseState + * \sa SDL_GetRelativeMouseState + * \sa SDL_PumpEvents + */ +extern DECLSPEC Uint32 SDLCALL SDL_GetMouseState(int *x, int *y); + +/** + * Get the current state of the mouse in relation to the desktop. + * + * This works similarly to SDL_GetMouseState(), but the coordinates will be + * reported relative to the top-left of the desktop. This can be useful if you + * need to track the mouse outside of a specific window and SDL_CaptureMouse() + * doesn't fit your needs. For example, it could be useful if you need to + * track the mouse while dragging a window, where coordinates relative to a + * window might not be in sync at all times. + * + * Note: SDL_GetMouseState() returns the mouse position as SDL understands it + * from the last pump of the event queue. This function, however, queries the + * OS for the current mouse position, and as such, might be a slightly less + * efficient function. Unless you know what you're doing and have a good + * reason to use this function, you probably want SDL_GetMouseState() instead. + * + * \param x filled in with the current X coord relative to the desktop; can be + * NULL + * \param y filled in with the current Y coord relative to the desktop; can be + * NULL + * \returns the current button state as a bitmask which can be tested using + * the SDL_BUTTON(X) macros. + * + * \since This function is available since SDL 2.0.4. + * + * \sa SDL_CaptureMouse + */ +extern DECLSPEC Uint32 SDLCALL SDL_GetGlobalMouseState(int *x, int *y); + +/** + * Retrieve the relative state of the mouse. + * + * The current button state is returned as a button bitmask, which can be + * tested using the `SDL_BUTTON(X)` macros (where `X` is generally 1 for the + * left, 2 for middle, 3 for the right button), and `x` and `y` are set to the + * mouse deltas since the last call to SDL_GetRelativeMouseState() or since + * event initialization. You can pass NULL for either `x` or `y`. + * + * \param x a pointer filled with the last recorded x coordinate of the mouse + * \param y a pointer filled with the last recorded y coordinate of the mouse + * \returns a 32-bit button bitmask of the relative button state. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetMouseState + */ +extern DECLSPEC Uint32 SDLCALL SDL_GetRelativeMouseState(int *x, int *y); + +/** + * Move the mouse cursor to the given position within the window. + * + * This function generates a mouse motion event if relative mode is not + * enabled. If relative mode is enabled, you can force mouse events for the + * warp by setting the SDL_HINT_MOUSE_RELATIVE_WARP_MOTION hint. + * + * Note that this function will appear to succeed, but not actually move the + * mouse when used over Microsoft Remote Desktop. + * + * \param window the window to move the mouse into, or NULL for the current + * mouse focus + * \param x the x coordinate within the window + * \param y the y coordinate within the window + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_WarpMouseGlobal + */ +extern DECLSPEC void SDLCALL SDL_WarpMouseInWindow(SDL_Window * window, + int x, int y); + +/** + * Move the mouse to the given position in global screen space. + * + * This function generates a mouse motion event. + * + * A failure of this function usually means that it is unsupported by a + * platform. + * + * Note that this function will appear to succeed, but not actually move the + * mouse when used over Microsoft Remote Desktop. + * + * \param x the x coordinate + * \param y the y coordinate + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.4. + * + * \sa SDL_WarpMouseInWindow + */ +extern DECLSPEC int SDLCALL SDL_WarpMouseGlobal(int x, int y); + +/** + * Set relative mouse mode. + * + * While the mouse is in relative mode, the cursor is hidden, the mouse + * position is constrained to the window, and SDL will report continuous + * relative mouse motion even if the mouse is at the edge of the window. + * + * This function will flush any pending mouse motion. + * + * \param enabled SDL_TRUE to enable relative mode, SDL_FALSE to disable. + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * If relative mode is not supported, this returns -1. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetRelativeMouseMode + */ +extern DECLSPEC int SDLCALL SDL_SetRelativeMouseMode(SDL_bool enabled); + +/** + * Capture the mouse and to track input outside an SDL window. + * + * Capturing enables your app to obtain mouse events globally, instead of just + * within your window. Not all video targets support this function. When + * capturing is enabled, the current window will get all mouse events, but + * unlike relative mode, no change is made to the cursor and it is not + * restrained to your window. + * + * This function may also deny mouse input to other windows--both those in + * your application and others on the system--so you should use this function + * sparingly, and in small bursts. For example, you might want to track the + * mouse while the user is dragging something, until the user releases a mouse + * button. It is not recommended that you capture the mouse for long periods + * of time, such as the entire time your app is running. For that, you should + * probably use SDL_SetRelativeMouseMode() or SDL_SetWindowGrab(), depending + * on your goals. + * + * While captured, mouse events still report coordinates relative to the + * current (foreground) window, but those coordinates may be outside the + * bounds of the window (including negative values). Capturing is only allowed + * for the foreground window. If the window loses focus while capturing, the + * capture will be disabled automatically. + * + * While capturing is enabled, the current window will have the + * `SDL_WINDOW_MOUSE_CAPTURE` flag set. + * + * Please note that as of SDL 2.0.22, SDL will attempt to "auto capture" the + * mouse while the user is pressing a button; this is to try and make mouse + * behavior more consistent between platforms, and deal with the common case + * of a user dragging the mouse outside of the window. This means that if you + * are calling SDL_CaptureMouse() only to deal with this situation, you no + * longer have to (although it is safe to do so). If this causes problems for + * your app, you can disable auto capture by setting the + * `SDL_HINT_MOUSE_AUTO_CAPTURE` hint to zero. + * + * \param enabled SDL_TRUE to enable capturing, SDL_FALSE to disable. + * \returns 0 on success or -1 if not supported; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.4. + * + * \sa SDL_GetGlobalMouseState + */ +extern DECLSPEC int SDLCALL SDL_CaptureMouse(SDL_bool enabled); + +/** + * Query whether relative mouse mode is enabled. + * + * \returns SDL_TRUE if relative mode is enabled or SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetRelativeMouseMode + */ +extern DECLSPEC SDL_bool SDLCALL SDL_GetRelativeMouseMode(void); + +/** + * Create a cursor using the specified bitmap data and mask (in MSB format). + * + * `mask` has to be in MSB (Most Significant Bit) format. + * + * The cursor width (`w`) must be a multiple of 8 bits. + * + * The cursor is created in black and white according to the following: + * + * - data=0, mask=1: white + * - data=1, mask=1: black + * - data=0, mask=0: transparent + * - data=1, mask=0: inverted color if possible, black if not. + * + * Cursors created with this function must be freed with SDL_FreeCursor(). + * + * If you want to have a color cursor, or create your cursor from an + * SDL_Surface, you should use SDL_CreateColorCursor(). Alternately, you can + * hide the cursor and draw your own as part of your game's rendering, but it + * will be bound to the framerate. + * + * Also, since SDL 2.0.0, SDL_CreateSystemCursor() is available, which + * provides twelve readily available system cursors to pick from. + * + * \param data the color value for each pixel of the cursor + * \param mask the mask value for each pixel of the cursor + * \param w the width of the cursor + * \param h the height of the cursor + * \param hot_x the X-axis location of the upper left corner of the cursor + * relative to the actual mouse position + * \param hot_y the Y-axis location of the upper left corner of the cursor + * relative to the actual mouse position + * \returns a new cursor with the specified parameters on success or NULL on + * failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_FreeCursor + * \sa SDL_SetCursor + * \sa SDL_ShowCursor + */ +extern DECLSPEC SDL_Cursor *SDLCALL SDL_CreateCursor(const Uint8 * data, + const Uint8 * mask, + int w, int h, int hot_x, + int hot_y); + +/** + * Create a color cursor. + * + * \param surface an SDL_Surface structure representing the cursor image + * \param hot_x the x position of the cursor hot spot + * \param hot_y the y position of the cursor hot spot + * \returns the new cursor on success or NULL on failure; call SDL_GetError() + * for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateCursor + * \sa SDL_FreeCursor + */ +extern DECLSPEC SDL_Cursor *SDLCALL SDL_CreateColorCursor(SDL_Surface *surface, + int hot_x, + int hot_y); + +/** + * Create a system cursor. + * + * \param id an SDL_SystemCursor enum value + * \returns a cursor on success or NULL on failure; call SDL_GetError() for + * more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_FreeCursor + */ +extern DECLSPEC SDL_Cursor *SDLCALL SDL_CreateSystemCursor(SDL_SystemCursor id); + +/** + * Set the active cursor. + * + * This function sets the currently active cursor to the specified one. If the + * cursor is currently visible, the change will be immediately represented on + * the display. SDL_SetCursor(NULL) can be used to force cursor redraw, if + * this is desired for any reason. + * + * \param cursor a cursor to make active + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateCursor + * \sa SDL_GetCursor + * \sa SDL_ShowCursor + */ +extern DECLSPEC void SDLCALL SDL_SetCursor(SDL_Cursor * cursor); + +/** + * Get the active cursor. + * + * This function returns a pointer to the current cursor which is owned by the + * library. It is not necessary to free the cursor with SDL_FreeCursor(). + * + * \returns the active cursor or NULL if there is no mouse. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetCursor + */ +extern DECLSPEC SDL_Cursor *SDLCALL SDL_GetCursor(void); + +/** + * Get the default cursor. + * + * You do not have to call SDL_FreeCursor() on the return value, but it is + * safe to do so. + * + * \returns the default cursor on success or NULL on failure. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateSystemCursor + */ +extern DECLSPEC SDL_Cursor *SDLCALL SDL_GetDefaultCursor(void); + +/** + * Free a previously-created cursor. + * + * Use this function to free cursor resources created with SDL_CreateCursor(), + * SDL_CreateColorCursor() or SDL_CreateSystemCursor(). + * + * \param cursor the cursor to free + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateColorCursor + * \sa SDL_CreateCursor + * \sa SDL_CreateSystemCursor + */ +extern DECLSPEC void SDLCALL SDL_FreeCursor(SDL_Cursor * cursor); + +/** + * Toggle whether or not the cursor is shown. + * + * The cursor starts off displayed but can be turned off. Passing `SDL_ENABLE` + * displays the cursor and passing `SDL_DISABLE` hides it. + * + * The current state of the mouse cursor can be queried by passing + * `SDL_QUERY`; either `SDL_DISABLE` or `SDL_ENABLE` will be returned. + * + * \param toggle `SDL_ENABLE` to show the cursor, `SDL_DISABLE` to hide it, + * `SDL_QUERY` to query the current state without changing it. + * \returns `SDL_ENABLE` if the cursor is shown, or `SDL_DISABLE` if the + * cursor is hidden, or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateCursor + * \sa SDL_SetCursor + */ +extern DECLSPEC int SDLCALL SDL_ShowCursor(int toggle); + +/** + * Used as a mask when testing buttons in buttonstate. + * + * - Button 1: Left mouse button + * - Button 2: Middle mouse button + * - Button 3: Right mouse button + */ +#define SDL_BUTTON(X) (1 << ((X)-1)) +#define SDL_BUTTON_LEFT 1 +#define SDL_BUTTON_MIDDLE 2 +#define SDL_BUTTON_RIGHT 3 +#define SDL_BUTTON_X1 4 +#define SDL_BUTTON_X2 5 +#define SDL_BUTTON_LMASK SDL_BUTTON(SDL_BUTTON_LEFT) +#define SDL_BUTTON_MMASK SDL_BUTTON(SDL_BUTTON_MIDDLE) +#define SDL_BUTTON_RMASK SDL_BUTTON(SDL_BUTTON_RIGHT) +#define SDL_BUTTON_X1MASK SDL_BUTTON(SDL_BUTTON_X1) +#define SDL_BUTTON_X2MASK SDL_BUTTON(SDL_BUTTON_X2) + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_mouse_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_mutex.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_mutex.h new file mode 100644 index 00000000..e679d380 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_mutex.h @@ -0,0 +1,545 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +#ifndef SDL_mutex_h_ +#define SDL_mutex_h_ + +/** + * \file SDL_mutex.h + * + * Functions to provide thread synchronization primitives. + */ + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +/******************************************************************************/ +/* Enable thread safety attributes only with clang. + * The attributes can be safely erased when compiling with other compilers. + */ +#if defined(SDL_THREAD_SAFETY_ANALYSIS) && \ + defined(__clang__) && (!defined(SWIG)) +#define SDL_THREAD_ANNOTATION_ATTRIBUTE__(x) __attribute__((x)) +#else +#define SDL_THREAD_ANNOTATION_ATTRIBUTE__(x) /* no-op */ +#endif + +#define SDL_CAPABILITY(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(capability(x)) + +#define SDL_SCOPED_CAPABILITY \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(scoped_lockable) + +#define SDL_GUARDED_BY(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(guarded_by(x)) + +#define SDL_PT_GUARDED_BY(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(pt_guarded_by(x)) + +#define SDL_ACQUIRED_BEFORE(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(acquired_before(x)) + +#define SDL_ACQUIRED_AFTER(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(acquired_after(x)) + +#define SDL_REQUIRES(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(requires_capability(x)) + +#define SDL_REQUIRES_SHARED(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(requires_shared_capability(x)) + +#define SDL_ACQUIRE(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(acquire_capability(x)) + +#define SDL_ACQUIRE_SHARED(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(acquire_shared_capability(x)) + +#define SDL_RELEASE(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(release_capability(x)) + +#define SDL_RELEASE_SHARED(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(release_shared_capability(x)) + +#define SDL_RELEASE_GENERIC(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(release_generic_capability(x)) + +#define SDL_TRY_ACQUIRE(x, y) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(try_acquire_capability(x, y)) + +#define SDL_TRY_ACQUIRE_SHARED(x, y) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(try_acquire_shared_capability(x, y)) + +#define SDL_EXCLUDES(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(locks_excluded(x)) + +#define SDL_ASSERT_CAPABILITY(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(assert_capability(x)) + +#define SDL_ASSERT_SHARED_CAPABILITY(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(assert_shared_capability(x)) + +#define SDL_RETURN_CAPABILITY(x) \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(lock_returned(x)) + +#define SDL_NO_THREAD_SAFETY_ANALYSIS \ + SDL_THREAD_ANNOTATION_ATTRIBUTE__(no_thread_safety_analysis) + +/******************************************************************************/ + + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * Synchronization functions which can time out return this value + * if they time out. + */ +#define SDL_MUTEX_TIMEDOUT 1 + +/** + * This is the timeout value which corresponds to never time out. + */ +#define SDL_MUTEX_MAXWAIT (~(Uint32)0) + + +/** + * \name Mutex functions + */ +/* @{ */ + +/* The SDL mutex structure, defined in SDL_sysmutex.c */ +struct SDL_mutex; +typedef struct SDL_mutex SDL_mutex; + +/** + * Create a new mutex. + * + * All newly-created mutexes begin in the _unlocked_ state. + * + * Calls to SDL_LockMutex() will not return while the mutex is locked by + * another thread. See SDL_TryLockMutex() to attempt to lock without blocking. + * + * SDL mutexes are reentrant. + * + * \returns the initialized and unlocked mutex or NULL on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_DestroyMutex + * \sa SDL_LockMutex + * \sa SDL_TryLockMutex + * \sa SDL_UnlockMutex + */ +extern DECLSPEC SDL_mutex *SDLCALL SDL_CreateMutex(void); + +/** + * Lock the mutex. + * + * This will block until the mutex is available, which is to say it is in the + * unlocked state and the OS has chosen the caller as the next thread to lock + * it. Of all threads waiting to lock the mutex, only one may do so at a time. + * + * It is legal for the owning thread to lock an already-locked mutex. It must + * unlock it the same number of times before it is actually made available for + * other threads in the system (this is known as a "recursive mutex"). + * + * \param mutex the mutex to lock + * \return 0, or -1 on error. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC int SDLCALL SDL_LockMutex(SDL_mutex * mutex) SDL_ACQUIRE(mutex); +#define SDL_mutexP(m) SDL_LockMutex(m) + +/** + * Try to lock a mutex without blocking. + * + * This works just like SDL_LockMutex(), but if the mutex is not available, + * this function returns `SDL_MUTEX_TIMEOUT` immediately. + * + * This technique is useful if you need exclusive access to a resource but + * don't want to wait for it, and will return to it to try again later. + * + * \param mutex the mutex to try to lock + * \returns 0, `SDL_MUTEX_TIMEDOUT`, or -1 on error; call SDL_GetError() for + * more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateMutex + * \sa SDL_DestroyMutex + * \sa SDL_LockMutex + * \sa SDL_UnlockMutex + */ +extern DECLSPEC int SDLCALL SDL_TryLockMutex(SDL_mutex * mutex) SDL_TRY_ACQUIRE(0, mutex); + +/** + * Unlock the mutex. + * + * It is legal for the owning thread to lock an already-locked mutex. It must + * unlock it the same number of times before it is actually made available for + * other threads in the system (this is known as a "recursive mutex"). + * + * It is an error to unlock a mutex that has not been locked by the current + * thread, and doing so results in undefined behavior. + * + * It is also an error to unlock a mutex that isn't locked at all. + * + * \param mutex the mutex to unlock. + * \returns 0, or -1 on error. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC int SDLCALL SDL_UnlockMutex(SDL_mutex * mutex) SDL_RELEASE(mutex); +#define SDL_mutexV(m) SDL_UnlockMutex(m) + +/** + * Destroy a mutex created with SDL_CreateMutex(). + * + * This function must be called on any mutex that is no longer needed. Failure + * to destroy a mutex will result in a system memory or resource leak. While + * it is safe to destroy a mutex that is _unlocked_, it is not safe to attempt + * to destroy a locked mutex, and may result in undefined behavior depending + * on the platform. + * + * \param mutex the mutex to destroy + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateMutex + * \sa SDL_LockMutex + * \sa SDL_TryLockMutex + * \sa SDL_UnlockMutex + */ +extern DECLSPEC void SDLCALL SDL_DestroyMutex(SDL_mutex * mutex); + +/* @} *//* Mutex functions */ + + +/** + * \name Semaphore functions + */ +/* @{ */ + +/* The SDL semaphore structure, defined in SDL_syssem.c */ +struct SDL_semaphore; +typedef struct SDL_semaphore SDL_sem; + +/** + * Create a semaphore. + * + * This function creates a new semaphore and initializes it with the value + * `initial_value`. Each wait operation on the semaphore will atomically + * decrement the semaphore value and potentially block if the semaphore value + * is 0. Each post operation will atomically increment the semaphore value and + * wake waiting threads and allow them to retry the wait operation. + * + * \param initial_value the starting value of the semaphore + * \returns a new semaphore or NULL on failure; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_DestroySemaphore + * \sa SDL_SemPost + * \sa SDL_SemTryWait + * \sa SDL_SemValue + * \sa SDL_SemWait + * \sa SDL_SemWaitTimeout + */ +extern DECLSPEC SDL_sem *SDLCALL SDL_CreateSemaphore(Uint32 initial_value); + +/** + * Destroy a semaphore. + * + * It is not safe to destroy a semaphore if there are threads currently + * waiting on it. + * + * \param sem the semaphore to destroy + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateSemaphore + * \sa SDL_SemPost + * \sa SDL_SemTryWait + * \sa SDL_SemValue + * \sa SDL_SemWait + * \sa SDL_SemWaitTimeout + */ +extern DECLSPEC void SDLCALL SDL_DestroySemaphore(SDL_sem * sem); + +/** + * Wait until a semaphore has a positive value and then decrements it. + * + * This function suspends the calling thread until either the semaphore + * pointed to by `sem` has a positive value or the call is interrupted by a + * signal or error. If the call is successful it will atomically decrement the + * semaphore value. + * + * This function is the equivalent of calling SDL_SemWaitTimeout() with a time + * length of `SDL_MUTEX_MAXWAIT`. + * + * \param sem the semaphore wait on + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateSemaphore + * \sa SDL_DestroySemaphore + * \sa SDL_SemPost + * \sa SDL_SemTryWait + * \sa SDL_SemValue + * \sa SDL_SemWait + * \sa SDL_SemWaitTimeout + */ +extern DECLSPEC int SDLCALL SDL_SemWait(SDL_sem * sem); + +/** + * See if a semaphore has a positive value and decrement it if it does. + * + * This function checks to see if the semaphore pointed to by `sem` has a + * positive value and atomically decrements the semaphore value if it does. If + * the semaphore doesn't have a positive value, the function immediately + * returns SDL_MUTEX_TIMEDOUT. + * + * \param sem the semaphore to wait on + * \returns 0 if the wait succeeds, `SDL_MUTEX_TIMEDOUT` if the wait would + * block, or a negative error code on failure; call SDL_GetError() + * for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateSemaphore + * \sa SDL_DestroySemaphore + * \sa SDL_SemPost + * \sa SDL_SemValue + * \sa SDL_SemWait + * \sa SDL_SemWaitTimeout + */ +extern DECLSPEC int SDLCALL SDL_SemTryWait(SDL_sem * sem); + +/** + * Wait until a semaphore has a positive value and then decrements it. + * + * This function suspends the calling thread until either the semaphore + * pointed to by `sem` has a positive value, the call is interrupted by a + * signal or error, or the specified time has elapsed. If the call is + * successful it will atomically decrement the semaphore value. + * + * \param sem the semaphore to wait on + * \param timeout the length of the timeout, in milliseconds + * \returns 0 if the wait succeeds, `SDL_MUTEX_TIMEDOUT` if the wait does not + * succeed in the allotted time, or a negative error code on failure; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateSemaphore + * \sa SDL_DestroySemaphore + * \sa SDL_SemPost + * \sa SDL_SemTryWait + * \sa SDL_SemValue + * \sa SDL_SemWait + */ +extern DECLSPEC int SDLCALL SDL_SemWaitTimeout(SDL_sem *sem, Uint32 timeout); + +/** + * Atomically increment a semaphore's value and wake waiting threads. + * + * \param sem the semaphore to increment + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateSemaphore + * \sa SDL_DestroySemaphore + * \sa SDL_SemTryWait + * \sa SDL_SemValue + * \sa SDL_SemWait + * \sa SDL_SemWaitTimeout + */ +extern DECLSPEC int SDLCALL SDL_SemPost(SDL_sem * sem); + +/** + * Get the current value of a semaphore. + * + * \param sem the semaphore to query + * \returns the current value of the semaphore. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateSemaphore + */ +extern DECLSPEC Uint32 SDLCALL SDL_SemValue(SDL_sem * sem); + +/* @} *//* Semaphore functions */ + + +/** + * \name Condition variable functions + */ +/* @{ */ + +/* The SDL condition variable structure, defined in SDL_syscond.c */ +struct SDL_cond; +typedef struct SDL_cond SDL_cond; + +/** + * Create a condition variable. + * + * \returns a new condition variable or NULL on failure; call SDL_GetError() + * for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CondBroadcast + * \sa SDL_CondSignal + * \sa SDL_CondWait + * \sa SDL_CondWaitTimeout + * \sa SDL_DestroyCond + */ +extern DECLSPEC SDL_cond *SDLCALL SDL_CreateCond(void); + +/** + * Destroy a condition variable. + * + * \param cond the condition variable to destroy + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CondBroadcast + * \sa SDL_CondSignal + * \sa SDL_CondWait + * \sa SDL_CondWaitTimeout + * \sa SDL_CreateCond + */ +extern DECLSPEC void SDLCALL SDL_DestroyCond(SDL_cond * cond); + +/** + * Restart one of the threads that are waiting on the condition variable. + * + * \param cond the condition variable to signal + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CondBroadcast + * \sa SDL_CondWait + * \sa SDL_CondWaitTimeout + * \sa SDL_CreateCond + * \sa SDL_DestroyCond + */ +extern DECLSPEC int SDLCALL SDL_CondSignal(SDL_cond * cond); + +/** + * Restart all threads that are waiting on the condition variable. + * + * \param cond the condition variable to signal + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CondSignal + * \sa SDL_CondWait + * \sa SDL_CondWaitTimeout + * \sa SDL_CreateCond + * \sa SDL_DestroyCond + */ +extern DECLSPEC int SDLCALL SDL_CondBroadcast(SDL_cond * cond); + +/** + * Wait until a condition variable is signaled. + * + * This function unlocks the specified `mutex` and waits for another thread to + * call SDL_CondSignal() or SDL_CondBroadcast() on the condition variable + * `cond`. Once the condition variable is signaled, the mutex is re-locked and + * the function returns. + * + * The mutex must be locked before calling this function. + * + * This function is the equivalent of calling SDL_CondWaitTimeout() with a + * time length of `SDL_MUTEX_MAXWAIT`. + * + * \param cond the condition variable to wait on + * \param mutex the mutex used to coordinate thread access + * \returns 0 when it is signaled or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CondBroadcast + * \sa SDL_CondSignal + * \sa SDL_CondWaitTimeout + * \sa SDL_CreateCond + * \sa SDL_DestroyCond + */ +extern DECLSPEC int SDLCALL SDL_CondWait(SDL_cond * cond, SDL_mutex * mutex); + +/** + * Wait until a condition variable is signaled or a certain time has passed. + * + * This function unlocks the specified `mutex` and waits for another thread to + * call SDL_CondSignal() or SDL_CondBroadcast() on the condition variable + * `cond`, or for the specified time to elapse. Once the condition variable is + * signaled or the time elapsed, the mutex is re-locked and the function + * returns. + * + * The mutex must be locked before calling this function. + * + * \param cond the condition variable to wait on + * \param mutex the mutex used to coordinate thread access + * \param ms the maximum time to wait, in milliseconds, or `SDL_MUTEX_MAXWAIT` + * to wait indefinitely + * \returns 0 if the condition variable is signaled, `SDL_MUTEX_TIMEDOUT` if + * the condition is not signaled in the allotted time, or a negative + * error code on failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CondBroadcast + * \sa SDL_CondSignal + * \sa SDL_CondWait + * \sa SDL_CreateCond + * \sa SDL_DestroyCond + */ +extern DECLSPEC int SDLCALL SDL_CondWaitTimeout(SDL_cond * cond, + SDL_mutex * mutex, Uint32 ms); + +/* @} *//* Condition variable functions */ + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_mutex_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_name.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_name.h new file mode 100644 index 00000000..5c3e07ab --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_name.h @@ -0,0 +1,33 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +#ifndef SDLname_h_ +#define SDLname_h_ + +#if defined(__STDC__) || defined(__cplusplus) +#define NeedFunctionPrototypes 1 +#endif + +#define SDL_NAME(X) SDL_##X + +#endif /* SDLname_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengl.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengl.h new file mode 100644 index 00000000..0ba89127 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengl.h @@ -0,0 +1,2132 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_opengl.h + * + * This is a simple file to encapsulate the OpenGL API headers. + */ + +/** + * \def NO_SDL_GLEXT + * + * Define this if you have your own version of glext.h and want to disable the + * version included in SDL_opengl.h. + */ + +#ifndef SDL_opengl_h_ +#define SDL_opengl_h_ + +#include "SDL_config.h" + +#ifndef __IPHONEOS__ /* No OpenGL on iOS. */ + +/* + * Mesa 3-D graphics library + * + * Copyright (C) 1999-2006 Brian Paul All Rights Reserved. + * Copyright (C) 2009 VMware, Inc. All Rights Reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a + * copy of this software and associated documentation files (the "Software"), + * to deal in the Software without restriction, including without limitation + * the rights to use, copy, modify, merge, publish, distribute, sublicense, + * and/or sell copies of the Software, and to permit persons to whom the + * Software is furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included + * in all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS + * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR + * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, + * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR + * OTHER DEALINGS IN THE SOFTWARE. + */ + + +#ifndef __gl_h_ +#define __gl_h_ + +#if defined(USE_MGL_NAMESPACE) +#include "gl_mangle.h" +#endif + + +/********************************************************************** + * Begin system-specific stuff. + */ + +#if defined(_WIN32) && !defined(__WIN32__) && !defined(__CYGWIN__) +#define __WIN32__ +#endif + +#if defined(__WIN32__) && !defined(__CYGWIN__) +# if (defined(_MSC_VER) || defined(__MINGW32__)) && defined(BUILD_GL32) /* tag specify we're building mesa as a DLL */ +# define GLAPI __declspec(dllexport) +# elif (defined(_MSC_VER) || defined(__MINGW32__)) && defined(_DLL) /* tag specifying we're building for DLL runtime support */ +# define GLAPI __declspec(dllimport) +# else /* for use with static link lib build of Win32 edition only */ +# define GLAPI extern +# endif /* _STATIC_MESA support */ +# if defined(__MINGW32__) && defined(GL_NO_STDCALL) || defined(UNDER_CE) /* The generated DLLs by MingW with STDCALL are not compatible with the ones done by Microsoft's compilers */ +# define GLAPIENTRY +# else +# define GLAPIENTRY __stdcall +# endif +#elif defined(__CYGWIN__) && defined(USE_OPENGL32) /* use native windows opengl32 */ +# define GLAPI extern +# define GLAPIENTRY __stdcall +#elif defined(__OS2__) || defined(__EMX__) /* native os/2 opengl */ +# define GLAPI extern +# define GLAPIENTRY _System +# define APIENTRY _System +# if defined(__GNUC__) && !defined(_System) +# define _System +# endif +#elif (defined(__GNUC__) && __GNUC__ >= 4) || (defined(__SUNPRO_C) && (__SUNPRO_C >= 0x590)) +# define GLAPI __attribute__((visibility("default"))) +# define GLAPIENTRY +#endif /* WIN32 && !CYGWIN */ + +/* + * WINDOWS: Include windows.h here to define APIENTRY. + * It is also useful when applications include this file by + * including only glut.h, since glut.h depends on windows.h. + * Applications needing to include windows.h with parms other + * than "WIN32_LEAN_AND_MEAN" may include windows.h before + * glut.h or gl.h. + */ +#if defined(_WIN32) && !defined(APIENTRY) && !defined(__CYGWIN__) +#ifndef WIN32_LEAN_AND_MEAN +#define WIN32_LEAN_AND_MEAN 1 +#endif +#ifndef NOMINMAX /* don't define min() and max(). */ +#define NOMINMAX +#endif +#include +#endif + +#ifndef GLAPI +#define GLAPI extern +#endif + +#ifndef GLAPIENTRY +#define GLAPIENTRY +#endif + +#ifndef APIENTRY +#define APIENTRY GLAPIENTRY +#endif + +/* "P" suffix to be used for a pointer to a function */ +#ifndef APIENTRYP +#define APIENTRYP APIENTRY * +#endif + +#ifndef GLAPIENTRYP +#define GLAPIENTRYP GLAPIENTRY * +#endif + +#if defined(PRAGMA_EXPORT_SUPPORTED) +#pragma export on +#endif + +/* + * End system-specific stuff. + **********************************************************************/ + + + +#ifdef __cplusplus +extern "C" { +#endif + + + +#define GL_VERSION_1_1 1 +#define GL_VERSION_1_2 1 +#define GL_VERSION_1_3 1 +#define GL_ARB_imaging 1 + + +/* + * Datatypes + */ +typedef unsigned int GLenum; +typedef unsigned char GLboolean; +typedef unsigned int GLbitfield; +typedef void GLvoid; +typedef signed char GLbyte; /* 1-byte signed */ +typedef short GLshort; /* 2-byte signed */ +typedef int GLint; /* 4-byte signed */ +typedef unsigned char GLubyte; /* 1-byte unsigned */ +typedef unsigned short GLushort; /* 2-byte unsigned */ +typedef unsigned int GLuint; /* 4-byte unsigned */ +typedef int GLsizei; /* 4-byte signed */ +typedef float GLfloat; /* single precision float */ +typedef float GLclampf; /* single precision float in [0,1] */ +typedef double GLdouble; /* double precision float */ +typedef double GLclampd; /* double precision float in [0,1] */ + + + +/* + * Constants + */ + +/* Boolean values */ +#define GL_FALSE 0 +#define GL_TRUE 1 + +/* Data types */ +#define GL_BYTE 0x1400 +#define GL_UNSIGNED_BYTE 0x1401 +#define GL_SHORT 0x1402 +#define GL_UNSIGNED_SHORT 0x1403 +#define GL_INT 0x1404 +#define GL_UNSIGNED_INT 0x1405 +#define GL_FLOAT 0x1406 +#define GL_2_BYTES 0x1407 +#define GL_3_BYTES 0x1408 +#define GL_4_BYTES 0x1409 +#define GL_DOUBLE 0x140A + +/* Primitives */ +#define GL_POINTS 0x0000 +#define GL_LINES 0x0001 +#define GL_LINE_LOOP 0x0002 +#define GL_LINE_STRIP 0x0003 +#define GL_TRIANGLES 0x0004 +#define GL_TRIANGLE_STRIP 0x0005 +#define GL_TRIANGLE_FAN 0x0006 +#define GL_QUADS 0x0007 +#define GL_QUAD_STRIP 0x0008 +#define GL_POLYGON 0x0009 + +/* Vertex Arrays */ +#define GL_VERTEX_ARRAY 0x8074 +#define GL_NORMAL_ARRAY 0x8075 +#define GL_COLOR_ARRAY 0x8076 +#define GL_INDEX_ARRAY 0x8077 +#define GL_TEXTURE_COORD_ARRAY 0x8078 +#define GL_EDGE_FLAG_ARRAY 0x8079 +#define GL_VERTEX_ARRAY_SIZE 0x807A +#define GL_VERTEX_ARRAY_TYPE 0x807B +#define GL_VERTEX_ARRAY_STRIDE 0x807C +#define GL_NORMAL_ARRAY_TYPE 0x807E +#define GL_NORMAL_ARRAY_STRIDE 0x807F +#define GL_COLOR_ARRAY_SIZE 0x8081 +#define GL_COLOR_ARRAY_TYPE 0x8082 +#define GL_COLOR_ARRAY_STRIDE 0x8083 +#define GL_INDEX_ARRAY_TYPE 0x8085 +#define GL_INDEX_ARRAY_STRIDE 0x8086 +#define GL_TEXTURE_COORD_ARRAY_SIZE 0x8088 +#define GL_TEXTURE_COORD_ARRAY_TYPE 0x8089 +#define GL_TEXTURE_COORD_ARRAY_STRIDE 0x808A +#define GL_EDGE_FLAG_ARRAY_STRIDE 0x808C +#define GL_VERTEX_ARRAY_POINTER 0x808E +#define GL_NORMAL_ARRAY_POINTER 0x808F +#define GL_COLOR_ARRAY_POINTER 0x8090 +#define GL_INDEX_ARRAY_POINTER 0x8091 +#define GL_TEXTURE_COORD_ARRAY_POINTER 0x8092 +#define GL_EDGE_FLAG_ARRAY_POINTER 0x8093 +#define GL_V2F 0x2A20 +#define GL_V3F 0x2A21 +#define GL_C4UB_V2F 0x2A22 +#define GL_C4UB_V3F 0x2A23 +#define GL_C3F_V3F 0x2A24 +#define GL_N3F_V3F 0x2A25 +#define GL_C4F_N3F_V3F 0x2A26 +#define GL_T2F_V3F 0x2A27 +#define GL_T4F_V4F 0x2A28 +#define GL_T2F_C4UB_V3F 0x2A29 +#define GL_T2F_C3F_V3F 0x2A2A +#define GL_T2F_N3F_V3F 0x2A2B +#define GL_T2F_C4F_N3F_V3F 0x2A2C +#define GL_T4F_C4F_N3F_V4F 0x2A2D + +/* Matrix Mode */ +#define GL_MATRIX_MODE 0x0BA0 +#define GL_MODELVIEW 0x1700 +#define GL_PROJECTION 0x1701 +#define GL_TEXTURE 0x1702 + +/* Points */ +#define GL_POINT_SMOOTH 0x0B10 +#define GL_POINT_SIZE 0x0B11 +#define GL_POINT_SIZE_GRANULARITY 0x0B13 +#define GL_POINT_SIZE_RANGE 0x0B12 + +/* Lines */ +#define GL_LINE_SMOOTH 0x0B20 +#define GL_LINE_STIPPLE 0x0B24 +#define GL_LINE_STIPPLE_PATTERN 0x0B25 +#define GL_LINE_STIPPLE_REPEAT 0x0B26 +#define GL_LINE_WIDTH 0x0B21 +#define GL_LINE_WIDTH_GRANULARITY 0x0B23 +#define GL_LINE_WIDTH_RANGE 0x0B22 + +/* Polygons */ +#define GL_POINT 0x1B00 +#define GL_LINE 0x1B01 +#define GL_FILL 0x1B02 +#define GL_CW 0x0900 +#define GL_CCW 0x0901 +#define GL_FRONT 0x0404 +#define GL_BACK 0x0405 +#define GL_POLYGON_MODE 0x0B40 +#define GL_POLYGON_SMOOTH 0x0B41 +#define GL_POLYGON_STIPPLE 0x0B42 +#define GL_EDGE_FLAG 0x0B43 +#define GL_CULL_FACE 0x0B44 +#define GL_CULL_FACE_MODE 0x0B45 +#define GL_FRONT_FACE 0x0B46 +#define GL_POLYGON_OFFSET_FACTOR 0x8038 +#define GL_POLYGON_OFFSET_UNITS 0x2A00 +#define GL_POLYGON_OFFSET_POINT 0x2A01 +#define GL_POLYGON_OFFSET_LINE 0x2A02 +#define GL_POLYGON_OFFSET_FILL 0x8037 + +/* Display Lists */ +#define GL_COMPILE 0x1300 +#define GL_COMPILE_AND_EXECUTE 0x1301 +#define GL_LIST_BASE 0x0B32 +#define GL_LIST_INDEX 0x0B33 +#define GL_LIST_MODE 0x0B30 + +/* Depth buffer */ +#define GL_NEVER 0x0200 +#define GL_LESS 0x0201 +#define GL_EQUAL 0x0202 +#define GL_LEQUAL 0x0203 +#define GL_GREATER 0x0204 +#define GL_NOTEQUAL 0x0205 +#define GL_GEQUAL 0x0206 +#define GL_ALWAYS 0x0207 +#define GL_DEPTH_TEST 0x0B71 +#define GL_DEPTH_BITS 0x0D56 +#define GL_DEPTH_CLEAR_VALUE 0x0B73 +#define GL_DEPTH_FUNC 0x0B74 +#define GL_DEPTH_RANGE 0x0B70 +#define GL_DEPTH_WRITEMASK 0x0B72 +#define GL_DEPTH_COMPONENT 0x1902 + +/* Lighting */ +#define GL_LIGHTING 0x0B50 +#define GL_LIGHT0 0x4000 +#define GL_LIGHT1 0x4001 +#define GL_LIGHT2 0x4002 +#define GL_LIGHT3 0x4003 +#define GL_LIGHT4 0x4004 +#define GL_LIGHT5 0x4005 +#define GL_LIGHT6 0x4006 +#define GL_LIGHT7 0x4007 +#define GL_SPOT_EXPONENT 0x1205 +#define GL_SPOT_CUTOFF 0x1206 +#define GL_CONSTANT_ATTENUATION 0x1207 +#define GL_LINEAR_ATTENUATION 0x1208 +#define GL_QUADRATIC_ATTENUATION 0x1209 +#define GL_AMBIENT 0x1200 +#define GL_DIFFUSE 0x1201 +#define GL_SPECULAR 0x1202 +#define GL_SHININESS 0x1601 +#define GL_EMISSION 0x1600 +#define GL_POSITION 0x1203 +#define GL_SPOT_DIRECTION 0x1204 +#define GL_AMBIENT_AND_DIFFUSE 0x1602 +#define GL_COLOR_INDEXES 0x1603 +#define GL_LIGHT_MODEL_TWO_SIDE 0x0B52 +#define GL_LIGHT_MODEL_LOCAL_VIEWER 0x0B51 +#define GL_LIGHT_MODEL_AMBIENT 0x0B53 +#define GL_FRONT_AND_BACK 0x0408 +#define GL_SHADE_MODEL 0x0B54 +#define GL_FLAT 0x1D00 +#define GL_SMOOTH 0x1D01 +#define GL_COLOR_MATERIAL 0x0B57 +#define GL_COLOR_MATERIAL_FACE 0x0B55 +#define GL_COLOR_MATERIAL_PARAMETER 0x0B56 +#define GL_NORMALIZE 0x0BA1 + +/* User clipping planes */ +#define GL_CLIP_PLANE0 0x3000 +#define GL_CLIP_PLANE1 0x3001 +#define GL_CLIP_PLANE2 0x3002 +#define GL_CLIP_PLANE3 0x3003 +#define GL_CLIP_PLANE4 0x3004 +#define GL_CLIP_PLANE5 0x3005 + +/* Accumulation buffer */ +#define GL_ACCUM_RED_BITS 0x0D58 +#define GL_ACCUM_GREEN_BITS 0x0D59 +#define GL_ACCUM_BLUE_BITS 0x0D5A +#define GL_ACCUM_ALPHA_BITS 0x0D5B +#define GL_ACCUM_CLEAR_VALUE 0x0B80 +#define GL_ACCUM 0x0100 +#define GL_ADD 0x0104 +#define GL_LOAD 0x0101 +#define GL_MULT 0x0103 +#define GL_RETURN 0x0102 + +/* Alpha testing */ +#define GL_ALPHA_TEST 0x0BC0 +#define GL_ALPHA_TEST_REF 0x0BC2 +#define GL_ALPHA_TEST_FUNC 0x0BC1 + +/* Blending */ +#define GL_BLEND 0x0BE2 +#define GL_BLEND_SRC 0x0BE1 +#define GL_BLEND_DST 0x0BE0 +#define GL_ZERO 0 +#define GL_ONE 1 +#define GL_SRC_COLOR 0x0300 +#define GL_ONE_MINUS_SRC_COLOR 0x0301 +#define GL_SRC_ALPHA 0x0302 +#define GL_ONE_MINUS_SRC_ALPHA 0x0303 +#define GL_DST_ALPHA 0x0304 +#define GL_ONE_MINUS_DST_ALPHA 0x0305 +#define GL_DST_COLOR 0x0306 +#define GL_ONE_MINUS_DST_COLOR 0x0307 +#define GL_SRC_ALPHA_SATURATE 0x0308 + +/* Render Mode */ +#define GL_FEEDBACK 0x1C01 +#define GL_RENDER 0x1C00 +#define GL_SELECT 0x1C02 + +/* Feedback */ +#define GL_2D 0x0600 +#define GL_3D 0x0601 +#define GL_3D_COLOR 0x0602 +#define GL_3D_COLOR_TEXTURE 0x0603 +#define GL_4D_COLOR_TEXTURE 0x0604 +#define GL_POINT_TOKEN 0x0701 +#define GL_LINE_TOKEN 0x0702 +#define GL_LINE_RESET_TOKEN 0x0707 +#define GL_POLYGON_TOKEN 0x0703 +#define GL_BITMAP_TOKEN 0x0704 +#define GL_DRAW_PIXEL_TOKEN 0x0705 +#define GL_COPY_PIXEL_TOKEN 0x0706 +#define GL_PASS_THROUGH_TOKEN 0x0700 +#define GL_FEEDBACK_BUFFER_POINTER 0x0DF0 +#define GL_FEEDBACK_BUFFER_SIZE 0x0DF1 +#define GL_FEEDBACK_BUFFER_TYPE 0x0DF2 + +/* Selection */ +#define GL_SELECTION_BUFFER_POINTER 0x0DF3 +#define GL_SELECTION_BUFFER_SIZE 0x0DF4 + +/* Fog */ +#define GL_FOG 0x0B60 +#define GL_FOG_MODE 0x0B65 +#define GL_FOG_DENSITY 0x0B62 +#define GL_FOG_COLOR 0x0B66 +#define GL_FOG_INDEX 0x0B61 +#define GL_FOG_START 0x0B63 +#define GL_FOG_END 0x0B64 +#define GL_LINEAR 0x2601 +#define GL_EXP 0x0800 +#define GL_EXP2 0x0801 + +/* Logic Ops */ +#define GL_LOGIC_OP 0x0BF1 +#define GL_INDEX_LOGIC_OP 0x0BF1 +#define GL_COLOR_LOGIC_OP 0x0BF2 +#define GL_LOGIC_OP_MODE 0x0BF0 +#define GL_CLEAR 0x1500 +#define GL_SET 0x150F +#define GL_COPY 0x1503 +#define GL_COPY_INVERTED 0x150C +#define GL_NOOP 0x1505 +#define GL_INVERT 0x150A +#define GL_AND 0x1501 +#define GL_NAND 0x150E +#define GL_OR 0x1507 +#define GL_NOR 0x1508 +#define GL_XOR 0x1506 +#define GL_EQUIV 0x1509 +#define GL_AND_REVERSE 0x1502 +#define GL_AND_INVERTED 0x1504 +#define GL_OR_REVERSE 0x150B +#define GL_OR_INVERTED 0x150D + +/* Stencil */ +#define GL_STENCIL_BITS 0x0D57 +#define GL_STENCIL_TEST 0x0B90 +#define GL_STENCIL_CLEAR_VALUE 0x0B91 +#define GL_STENCIL_FUNC 0x0B92 +#define GL_STENCIL_VALUE_MASK 0x0B93 +#define GL_STENCIL_FAIL 0x0B94 +#define GL_STENCIL_PASS_DEPTH_FAIL 0x0B95 +#define GL_STENCIL_PASS_DEPTH_PASS 0x0B96 +#define GL_STENCIL_REF 0x0B97 +#define GL_STENCIL_WRITEMASK 0x0B98 +#define GL_STENCIL_INDEX 0x1901 +#define GL_KEEP 0x1E00 +#define GL_REPLACE 0x1E01 +#define GL_INCR 0x1E02 +#define GL_DECR 0x1E03 + +/* Buffers, Pixel Drawing/Reading */ +#define GL_NONE 0 +#define GL_LEFT 0x0406 +#define GL_RIGHT 0x0407 +/*GL_FRONT 0x0404 */ +/*GL_BACK 0x0405 */ +/*GL_FRONT_AND_BACK 0x0408 */ +#define GL_FRONT_LEFT 0x0400 +#define GL_FRONT_RIGHT 0x0401 +#define GL_BACK_LEFT 0x0402 +#define GL_BACK_RIGHT 0x0403 +#define GL_AUX0 0x0409 +#define GL_AUX1 0x040A +#define GL_AUX2 0x040B +#define GL_AUX3 0x040C +#define GL_COLOR_INDEX 0x1900 +#define GL_RED 0x1903 +#define GL_GREEN 0x1904 +#define GL_BLUE 0x1905 +#define GL_ALPHA 0x1906 +#define GL_LUMINANCE 0x1909 +#define GL_LUMINANCE_ALPHA 0x190A +#define GL_ALPHA_BITS 0x0D55 +#define GL_RED_BITS 0x0D52 +#define GL_GREEN_BITS 0x0D53 +#define GL_BLUE_BITS 0x0D54 +#define GL_INDEX_BITS 0x0D51 +#define GL_SUBPIXEL_BITS 0x0D50 +#define GL_AUX_BUFFERS 0x0C00 +#define GL_READ_BUFFER 0x0C02 +#define GL_DRAW_BUFFER 0x0C01 +#define GL_DOUBLEBUFFER 0x0C32 +#define GL_STEREO 0x0C33 +#define GL_BITMAP 0x1A00 +#define GL_COLOR 0x1800 +#define GL_DEPTH 0x1801 +#define GL_STENCIL 0x1802 +#define GL_DITHER 0x0BD0 +#define GL_RGB 0x1907 +#define GL_RGBA 0x1908 + +/* Implementation limits */ +#define GL_MAX_LIST_NESTING 0x0B31 +#define GL_MAX_EVAL_ORDER 0x0D30 +#define GL_MAX_LIGHTS 0x0D31 +#define GL_MAX_CLIP_PLANES 0x0D32 +#define GL_MAX_TEXTURE_SIZE 0x0D33 +#define GL_MAX_PIXEL_MAP_TABLE 0x0D34 +#define GL_MAX_ATTRIB_STACK_DEPTH 0x0D35 +#define GL_MAX_MODELVIEW_STACK_DEPTH 0x0D36 +#define GL_MAX_NAME_STACK_DEPTH 0x0D37 +#define GL_MAX_PROJECTION_STACK_DEPTH 0x0D38 +#define GL_MAX_TEXTURE_STACK_DEPTH 0x0D39 +#define GL_MAX_VIEWPORT_DIMS 0x0D3A +#define GL_MAX_CLIENT_ATTRIB_STACK_DEPTH 0x0D3B + +/* Gets */ +#define GL_ATTRIB_STACK_DEPTH 0x0BB0 +#define GL_CLIENT_ATTRIB_STACK_DEPTH 0x0BB1 +#define GL_COLOR_CLEAR_VALUE 0x0C22 +#define GL_COLOR_WRITEMASK 0x0C23 +#define GL_CURRENT_INDEX 0x0B01 +#define GL_CURRENT_COLOR 0x0B00 +#define GL_CURRENT_NORMAL 0x0B02 +#define GL_CURRENT_RASTER_COLOR 0x0B04 +#define GL_CURRENT_RASTER_DISTANCE 0x0B09 +#define GL_CURRENT_RASTER_INDEX 0x0B05 +#define GL_CURRENT_RASTER_POSITION 0x0B07 +#define GL_CURRENT_RASTER_TEXTURE_COORDS 0x0B06 +#define GL_CURRENT_RASTER_POSITION_VALID 0x0B08 +#define GL_CURRENT_TEXTURE_COORDS 0x0B03 +#define GL_INDEX_CLEAR_VALUE 0x0C20 +#define GL_INDEX_MODE 0x0C30 +#define GL_INDEX_WRITEMASK 0x0C21 +#define GL_MODELVIEW_MATRIX 0x0BA6 +#define GL_MODELVIEW_STACK_DEPTH 0x0BA3 +#define GL_NAME_STACK_DEPTH 0x0D70 +#define GL_PROJECTION_MATRIX 0x0BA7 +#define GL_PROJECTION_STACK_DEPTH 0x0BA4 +#define GL_RENDER_MODE 0x0C40 +#define GL_RGBA_MODE 0x0C31 +#define GL_TEXTURE_MATRIX 0x0BA8 +#define GL_TEXTURE_STACK_DEPTH 0x0BA5 +#define GL_VIEWPORT 0x0BA2 + +/* Evaluators */ +#define GL_AUTO_NORMAL 0x0D80 +#define GL_MAP1_COLOR_4 0x0D90 +#define GL_MAP1_INDEX 0x0D91 +#define GL_MAP1_NORMAL 0x0D92 +#define GL_MAP1_TEXTURE_COORD_1 0x0D93 +#define GL_MAP1_TEXTURE_COORD_2 0x0D94 +#define GL_MAP1_TEXTURE_COORD_3 0x0D95 +#define GL_MAP1_TEXTURE_COORD_4 0x0D96 +#define GL_MAP1_VERTEX_3 0x0D97 +#define GL_MAP1_VERTEX_4 0x0D98 +#define GL_MAP2_COLOR_4 0x0DB0 +#define GL_MAP2_INDEX 0x0DB1 +#define GL_MAP2_NORMAL 0x0DB2 +#define GL_MAP2_TEXTURE_COORD_1 0x0DB3 +#define GL_MAP2_TEXTURE_COORD_2 0x0DB4 +#define GL_MAP2_TEXTURE_COORD_3 0x0DB5 +#define GL_MAP2_TEXTURE_COORD_4 0x0DB6 +#define GL_MAP2_VERTEX_3 0x0DB7 +#define GL_MAP2_VERTEX_4 0x0DB8 +#define GL_MAP1_GRID_DOMAIN 0x0DD0 +#define GL_MAP1_GRID_SEGMENTS 0x0DD1 +#define GL_MAP2_GRID_DOMAIN 0x0DD2 +#define GL_MAP2_GRID_SEGMENTS 0x0DD3 +#define GL_COEFF 0x0A00 +#define GL_ORDER 0x0A01 +#define GL_DOMAIN 0x0A02 + +/* Hints */ +#define GL_PERSPECTIVE_CORRECTION_HINT 0x0C50 +#define GL_POINT_SMOOTH_HINT 0x0C51 +#define GL_LINE_SMOOTH_HINT 0x0C52 +#define GL_POLYGON_SMOOTH_HINT 0x0C53 +#define GL_FOG_HINT 0x0C54 +#define GL_DONT_CARE 0x1100 +#define GL_FASTEST 0x1101 +#define GL_NICEST 0x1102 + +/* Scissor box */ +#define GL_SCISSOR_BOX 0x0C10 +#define GL_SCISSOR_TEST 0x0C11 + +/* Pixel Mode / Transfer */ +#define GL_MAP_COLOR 0x0D10 +#define GL_MAP_STENCIL 0x0D11 +#define GL_INDEX_SHIFT 0x0D12 +#define GL_INDEX_OFFSET 0x0D13 +#define GL_RED_SCALE 0x0D14 +#define GL_RED_BIAS 0x0D15 +#define GL_GREEN_SCALE 0x0D18 +#define GL_GREEN_BIAS 0x0D19 +#define GL_BLUE_SCALE 0x0D1A +#define GL_BLUE_BIAS 0x0D1B +#define GL_ALPHA_SCALE 0x0D1C +#define GL_ALPHA_BIAS 0x0D1D +#define GL_DEPTH_SCALE 0x0D1E +#define GL_DEPTH_BIAS 0x0D1F +#define GL_PIXEL_MAP_S_TO_S_SIZE 0x0CB1 +#define GL_PIXEL_MAP_I_TO_I_SIZE 0x0CB0 +#define GL_PIXEL_MAP_I_TO_R_SIZE 0x0CB2 +#define GL_PIXEL_MAP_I_TO_G_SIZE 0x0CB3 +#define GL_PIXEL_MAP_I_TO_B_SIZE 0x0CB4 +#define GL_PIXEL_MAP_I_TO_A_SIZE 0x0CB5 +#define GL_PIXEL_MAP_R_TO_R_SIZE 0x0CB6 +#define GL_PIXEL_MAP_G_TO_G_SIZE 0x0CB7 +#define GL_PIXEL_MAP_B_TO_B_SIZE 0x0CB8 +#define GL_PIXEL_MAP_A_TO_A_SIZE 0x0CB9 +#define GL_PIXEL_MAP_S_TO_S 0x0C71 +#define GL_PIXEL_MAP_I_TO_I 0x0C70 +#define GL_PIXEL_MAP_I_TO_R 0x0C72 +#define GL_PIXEL_MAP_I_TO_G 0x0C73 +#define GL_PIXEL_MAP_I_TO_B 0x0C74 +#define GL_PIXEL_MAP_I_TO_A 0x0C75 +#define GL_PIXEL_MAP_R_TO_R 0x0C76 +#define GL_PIXEL_MAP_G_TO_G 0x0C77 +#define GL_PIXEL_MAP_B_TO_B 0x0C78 +#define GL_PIXEL_MAP_A_TO_A 0x0C79 +#define GL_PACK_ALIGNMENT 0x0D05 +#define GL_PACK_LSB_FIRST 0x0D01 +#define GL_PACK_ROW_LENGTH 0x0D02 +#define GL_PACK_SKIP_PIXELS 0x0D04 +#define GL_PACK_SKIP_ROWS 0x0D03 +#define GL_PACK_SWAP_BYTES 0x0D00 +#define GL_UNPACK_ALIGNMENT 0x0CF5 +#define GL_UNPACK_LSB_FIRST 0x0CF1 +#define GL_UNPACK_ROW_LENGTH 0x0CF2 +#define GL_UNPACK_SKIP_PIXELS 0x0CF4 +#define GL_UNPACK_SKIP_ROWS 0x0CF3 +#define GL_UNPACK_SWAP_BYTES 0x0CF0 +#define GL_ZOOM_X 0x0D16 +#define GL_ZOOM_Y 0x0D17 + +/* Texture mapping */ +#define GL_TEXTURE_ENV 0x2300 +#define GL_TEXTURE_ENV_MODE 0x2200 +#define GL_TEXTURE_1D 0x0DE0 +#define GL_TEXTURE_2D 0x0DE1 +#define GL_TEXTURE_WRAP_S 0x2802 +#define GL_TEXTURE_WRAP_T 0x2803 +#define GL_TEXTURE_MAG_FILTER 0x2800 +#define GL_TEXTURE_MIN_FILTER 0x2801 +#define GL_TEXTURE_ENV_COLOR 0x2201 +#define GL_TEXTURE_GEN_S 0x0C60 +#define GL_TEXTURE_GEN_T 0x0C61 +#define GL_TEXTURE_GEN_R 0x0C62 +#define GL_TEXTURE_GEN_Q 0x0C63 +#define GL_TEXTURE_GEN_MODE 0x2500 +#define GL_TEXTURE_BORDER_COLOR 0x1004 +#define GL_TEXTURE_WIDTH 0x1000 +#define GL_TEXTURE_HEIGHT 0x1001 +#define GL_TEXTURE_BORDER 0x1005 +#define GL_TEXTURE_COMPONENTS 0x1003 +#define GL_TEXTURE_RED_SIZE 0x805C +#define GL_TEXTURE_GREEN_SIZE 0x805D +#define GL_TEXTURE_BLUE_SIZE 0x805E +#define GL_TEXTURE_ALPHA_SIZE 0x805F +#define GL_TEXTURE_LUMINANCE_SIZE 0x8060 +#define GL_TEXTURE_INTENSITY_SIZE 0x8061 +#define GL_NEAREST_MIPMAP_NEAREST 0x2700 +#define GL_NEAREST_MIPMAP_LINEAR 0x2702 +#define GL_LINEAR_MIPMAP_NEAREST 0x2701 +#define GL_LINEAR_MIPMAP_LINEAR 0x2703 +#define GL_OBJECT_LINEAR 0x2401 +#define GL_OBJECT_PLANE 0x2501 +#define GL_EYE_LINEAR 0x2400 +#define GL_EYE_PLANE 0x2502 +#define GL_SPHERE_MAP 0x2402 +#define GL_DECAL 0x2101 +#define GL_MODULATE 0x2100 +#define GL_NEAREST 0x2600 +#define GL_REPEAT 0x2901 +#define GL_CLAMP 0x2900 +#define GL_S 0x2000 +#define GL_T 0x2001 +#define GL_R 0x2002 +#define GL_Q 0x2003 + +/* Utility */ +#define GL_VENDOR 0x1F00 +#define GL_RENDERER 0x1F01 +#define GL_VERSION 0x1F02 +#define GL_EXTENSIONS 0x1F03 + +/* Errors */ +#define GL_NO_ERROR 0 +#define GL_INVALID_ENUM 0x0500 +#define GL_INVALID_VALUE 0x0501 +#define GL_INVALID_OPERATION 0x0502 +#define GL_STACK_OVERFLOW 0x0503 +#define GL_STACK_UNDERFLOW 0x0504 +#define GL_OUT_OF_MEMORY 0x0505 + +/* glPush/PopAttrib bits */ +#define GL_CURRENT_BIT 0x00000001 +#define GL_POINT_BIT 0x00000002 +#define GL_LINE_BIT 0x00000004 +#define GL_POLYGON_BIT 0x00000008 +#define GL_POLYGON_STIPPLE_BIT 0x00000010 +#define GL_PIXEL_MODE_BIT 0x00000020 +#define GL_LIGHTING_BIT 0x00000040 +#define GL_FOG_BIT 0x00000080 +#define GL_DEPTH_BUFFER_BIT 0x00000100 +#define GL_ACCUM_BUFFER_BIT 0x00000200 +#define GL_STENCIL_BUFFER_BIT 0x00000400 +#define GL_VIEWPORT_BIT 0x00000800 +#define GL_TRANSFORM_BIT 0x00001000 +#define GL_ENABLE_BIT 0x00002000 +#define GL_COLOR_BUFFER_BIT 0x00004000 +#define GL_HINT_BIT 0x00008000 +#define GL_EVAL_BIT 0x00010000 +#define GL_LIST_BIT 0x00020000 +#define GL_TEXTURE_BIT 0x00040000 +#define GL_SCISSOR_BIT 0x00080000 +#define GL_ALL_ATTRIB_BITS 0x000FFFFF + + +/* OpenGL 1.1 */ +#define GL_PROXY_TEXTURE_1D 0x8063 +#define GL_PROXY_TEXTURE_2D 0x8064 +#define GL_TEXTURE_PRIORITY 0x8066 +#define GL_TEXTURE_RESIDENT 0x8067 +#define GL_TEXTURE_BINDING_1D 0x8068 +#define GL_TEXTURE_BINDING_2D 0x8069 +#define GL_TEXTURE_INTERNAL_FORMAT 0x1003 +#define GL_ALPHA4 0x803B +#define GL_ALPHA8 0x803C +#define GL_ALPHA12 0x803D +#define GL_ALPHA16 0x803E +#define GL_LUMINANCE4 0x803F +#define GL_LUMINANCE8 0x8040 +#define GL_LUMINANCE12 0x8041 +#define GL_LUMINANCE16 0x8042 +#define GL_LUMINANCE4_ALPHA4 0x8043 +#define GL_LUMINANCE6_ALPHA2 0x8044 +#define GL_LUMINANCE8_ALPHA8 0x8045 +#define GL_LUMINANCE12_ALPHA4 0x8046 +#define GL_LUMINANCE12_ALPHA12 0x8047 +#define GL_LUMINANCE16_ALPHA16 0x8048 +#define GL_INTENSITY 0x8049 +#define GL_INTENSITY4 0x804A +#define GL_INTENSITY8 0x804B +#define GL_INTENSITY12 0x804C +#define GL_INTENSITY16 0x804D +#define GL_R3_G3_B2 0x2A10 +#define GL_RGB4 0x804F +#define GL_RGB5 0x8050 +#define GL_RGB8 0x8051 +#define GL_RGB10 0x8052 +#define GL_RGB12 0x8053 +#define GL_RGB16 0x8054 +#define GL_RGBA2 0x8055 +#define GL_RGBA4 0x8056 +#define GL_RGB5_A1 0x8057 +#define GL_RGBA8 0x8058 +#define GL_RGB10_A2 0x8059 +#define GL_RGBA12 0x805A +#define GL_RGBA16 0x805B +#define GL_CLIENT_PIXEL_STORE_BIT 0x00000001 +#define GL_CLIENT_VERTEX_ARRAY_BIT 0x00000002 +#define GL_ALL_CLIENT_ATTRIB_BITS 0xFFFFFFFF +#define GL_CLIENT_ALL_ATTRIB_BITS 0xFFFFFFFF + + + +/* + * Miscellaneous + */ + +GLAPI void GLAPIENTRY glClearIndex( GLfloat c ); + +GLAPI void GLAPIENTRY glClearColor( GLclampf red, GLclampf green, GLclampf blue, GLclampf alpha ); + +GLAPI void GLAPIENTRY glClear( GLbitfield mask ); + +GLAPI void GLAPIENTRY glIndexMask( GLuint mask ); + +GLAPI void GLAPIENTRY glColorMask( GLboolean red, GLboolean green, GLboolean blue, GLboolean alpha ); + +GLAPI void GLAPIENTRY glAlphaFunc( GLenum func, GLclampf ref ); + +GLAPI void GLAPIENTRY glBlendFunc( GLenum sfactor, GLenum dfactor ); + +GLAPI void GLAPIENTRY glLogicOp( GLenum opcode ); + +GLAPI void GLAPIENTRY glCullFace( GLenum mode ); + +GLAPI void GLAPIENTRY glFrontFace( GLenum mode ); + +GLAPI void GLAPIENTRY glPointSize( GLfloat size ); + +GLAPI void GLAPIENTRY glLineWidth( GLfloat width ); + +GLAPI void GLAPIENTRY glLineStipple( GLint factor, GLushort pattern ); + +GLAPI void GLAPIENTRY glPolygonMode( GLenum face, GLenum mode ); + +GLAPI void GLAPIENTRY glPolygonOffset( GLfloat factor, GLfloat units ); + +GLAPI void GLAPIENTRY glPolygonStipple( const GLubyte *mask ); + +GLAPI void GLAPIENTRY glGetPolygonStipple( GLubyte *mask ); + +GLAPI void GLAPIENTRY glEdgeFlag( GLboolean flag ); + +GLAPI void GLAPIENTRY glEdgeFlagv( const GLboolean *flag ); + +GLAPI void GLAPIENTRY glScissor( GLint x, GLint y, GLsizei width, GLsizei height); + +GLAPI void GLAPIENTRY glClipPlane( GLenum plane, const GLdouble *equation ); + +GLAPI void GLAPIENTRY glGetClipPlane( GLenum plane, GLdouble *equation ); + +GLAPI void GLAPIENTRY glDrawBuffer( GLenum mode ); + +GLAPI void GLAPIENTRY glReadBuffer( GLenum mode ); + +GLAPI void GLAPIENTRY glEnable( GLenum cap ); + +GLAPI void GLAPIENTRY glDisable( GLenum cap ); + +GLAPI GLboolean GLAPIENTRY glIsEnabled( GLenum cap ); + + +GLAPI void GLAPIENTRY glEnableClientState( GLenum cap ); /* 1.1 */ + +GLAPI void GLAPIENTRY glDisableClientState( GLenum cap ); /* 1.1 */ + + +GLAPI void GLAPIENTRY glGetBooleanv( GLenum pname, GLboolean *params ); + +GLAPI void GLAPIENTRY glGetDoublev( GLenum pname, GLdouble *params ); + +GLAPI void GLAPIENTRY glGetFloatv( GLenum pname, GLfloat *params ); + +GLAPI void GLAPIENTRY glGetIntegerv( GLenum pname, GLint *params ); + + +GLAPI void GLAPIENTRY glPushAttrib( GLbitfield mask ); + +GLAPI void GLAPIENTRY glPopAttrib( void ); + + +GLAPI void GLAPIENTRY glPushClientAttrib( GLbitfield mask ); /* 1.1 */ + +GLAPI void GLAPIENTRY glPopClientAttrib( void ); /* 1.1 */ + + +GLAPI GLint GLAPIENTRY glRenderMode( GLenum mode ); + +GLAPI GLenum GLAPIENTRY glGetError( void ); + +GLAPI const GLubyte * GLAPIENTRY glGetString( GLenum name ); + +GLAPI void GLAPIENTRY glFinish( void ); + +GLAPI void GLAPIENTRY glFlush( void ); + +GLAPI void GLAPIENTRY glHint( GLenum target, GLenum mode ); + + +/* + * Depth Buffer + */ + +GLAPI void GLAPIENTRY glClearDepth( GLclampd depth ); + +GLAPI void GLAPIENTRY glDepthFunc( GLenum func ); + +GLAPI void GLAPIENTRY glDepthMask( GLboolean flag ); + +GLAPI void GLAPIENTRY glDepthRange( GLclampd near_val, GLclampd far_val ); + + +/* + * Accumulation Buffer + */ + +GLAPI void GLAPIENTRY glClearAccum( GLfloat red, GLfloat green, GLfloat blue, GLfloat alpha ); + +GLAPI void GLAPIENTRY glAccum( GLenum op, GLfloat value ); + + +/* + * Transformation + */ + +GLAPI void GLAPIENTRY glMatrixMode( GLenum mode ); + +GLAPI void GLAPIENTRY glOrtho( GLdouble left, GLdouble right, + GLdouble bottom, GLdouble top, + GLdouble near_val, GLdouble far_val ); + +GLAPI void GLAPIENTRY glFrustum( GLdouble left, GLdouble right, + GLdouble bottom, GLdouble top, + GLdouble near_val, GLdouble far_val ); + +GLAPI void GLAPIENTRY glViewport( GLint x, GLint y, + GLsizei width, GLsizei height ); + +GLAPI void GLAPIENTRY glPushMatrix( void ); + +GLAPI void GLAPIENTRY glPopMatrix( void ); + +GLAPI void GLAPIENTRY glLoadIdentity( void ); + +GLAPI void GLAPIENTRY glLoadMatrixd( const GLdouble *m ); +GLAPI void GLAPIENTRY glLoadMatrixf( const GLfloat *m ); + +GLAPI void GLAPIENTRY glMultMatrixd( const GLdouble *m ); +GLAPI void GLAPIENTRY glMultMatrixf( const GLfloat *m ); + +GLAPI void GLAPIENTRY glRotated( GLdouble angle, + GLdouble x, GLdouble y, GLdouble z ); +GLAPI void GLAPIENTRY glRotatef( GLfloat angle, + GLfloat x, GLfloat y, GLfloat z ); + +GLAPI void GLAPIENTRY glScaled( GLdouble x, GLdouble y, GLdouble z ); +GLAPI void GLAPIENTRY glScalef( GLfloat x, GLfloat y, GLfloat z ); + +GLAPI void GLAPIENTRY glTranslated( GLdouble x, GLdouble y, GLdouble z ); +GLAPI void GLAPIENTRY glTranslatef( GLfloat x, GLfloat y, GLfloat z ); + + +/* + * Display Lists + */ + +GLAPI GLboolean GLAPIENTRY glIsList( GLuint list ); + +GLAPI void GLAPIENTRY glDeleteLists( GLuint list, GLsizei range ); + +GLAPI GLuint GLAPIENTRY glGenLists( GLsizei range ); + +GLAPI void GLAPIENTRY glNewList( GLuint list, GLenum mode ); + +GLAPI void GLAPIENTRY glEndList( void ); + +GLAPI void GLAPIENTRY glCallList( GLuint list ); + +GLAPI void GLAPIENTRY glCallLists( GLsizei n, GLenum type, + const GLvoid *lists ); + +GLAPI void GLAPIENTRY glListBase( GLuint base ); + + +/* + * Drawing Functions + */ + +GLAPI void GLAPIENTRY glBegin( GLenum mode ); + +GLAPI void GLAPIENTRY glEnd( void ); + + +GLAPI void GLAPIENTRY glVertex2d( GLdouble x, GLdouble y ); +GLAPI void GLAPIENTRY glVertex2f( GLfloat x, GLfloat y ); +GLAPI void GLAPIENTRY glVertex2i( GLint x, GLint y ); +GLAPI void GLAPIENTRY glVertex2s( GLshort x, GLshort y ); + +GLAPI void GLAPIENTRY glVertex3d( GLdouble x, GLdouble y, GLdouble z ); +GLAPI void GLAPIENTRY glVertex3f( GLfloat x, GLfloat y, GLfloat z ); +GLAPI void GLAPIENTRY glVertex3i( GLint x, GLint y, GLint z ); +GLAPI void GLAPIENTRY glVertex3s( GLshort x, GLshort y, GLshort z ); + +GLAPI void GLAPIENTRY glVertex4d( GLdouble x, GLdouble y, GLdouble z, GLdouble w ); +GLAPI void GLAPIENTRY glVertex4f( GLfloat x, GLfloat y, GLfloat z, GLfloat w ); +GLAPI void GLAPIENTRY glVertex4i( GLint x, GLint y, GLint z, GLint w ); +GLAPI void GLAPIENTRY glVertex4s( GLshort x, GLshort y, GLshort z, GLshort w ); + +GLAPI void GLAPIENTRY glVertex2dv( const GLdouble *v ); +GLAPI void GLAPIENTRY glVertex2fv( const GLfloat *v ); +GLAPI void GLAPIENTRY glVertex2iv( const GLint *v ); +GLAPI void GLAPIENTRY glVertex2sv( const GLshort *v ); + +GLAPI void GLAPIENTRY glVertex3dv( const GLdouble *v ); +GLAPI void GLAPIENTRY glVertex3fv( const GLfloat *v ); +GLAPI void GLAPIENTRY glVertex3iv( const GLint *v ); +GLAPI void GLAPIENTRY glVertex3sv( const GLshort *v ); + +GLAPI void GLAPIENTRY glVertex4dv( const GLdouble *v ); +GLAPI void GLAPIENTRY glVertex4fv( const GLfloat *v ); +GLAPI void GLAPIENTRY glVertex4iv( const GLint *v ); +GLAPI void GLAPIENTRY glVertex4sv( const GLshort *v ); + + +GLAPI void GLAPIENTRY glNormal3b( GLbyte nx, GLbyte ny, GLbyte nz ); +GLAPI void GLAPIENTRY glNormal3d( GLdouble nx, GLdouble ny, GLdouble nz ); +GLAPI void GLAPIENTRY glNormal3f( GLfloat nx, GLfloat ny, GLfloat nz ); +GLAPI void GLAPIENTRY glNormal3i( GLint nx, GLint ny, GLint nz ); +GLAPI void GLAPIENTRY glNormal3s( GLshort nx, GLshort ny, GLshort nz ); + +GLAPI void GLAPIENTRY glNormal3bv( const GLbyte *v ); +GLAPI void GLAPIENTRY glNormal3dv( const GLdouble *v ); +GLAPI void GLAPIENTRY glNormal3fv( const GLfloat *v ); +GLAPI void GLAPIENTRY glNormal3iv( const GLint *v ); +GLAPI void GLAPIENTRY glNormal3sv( const GLshort *v ); + + +GLAPI void GLAPIENTRY glIndexd( GLdouble c ); +GLAPI void GLAPIENTRY glIndexf( GLfloat c ); +GLAPI void GLAPIENTRY glIndexi( GLint c ); +GLAPI void GLAPIENTRY glIndexs( GLshort c ); +GLAPI void GLAPIENTRY glIndexub( GLubyte c ); /* 1.1 */ + +GLAPI void GLAPIENTRY glIndexdv( const GLdouble *c ); +GLAPI void GLAPIENTRY glIndexfv( const GLfloat *c ); +GLAPI void GLAPIENTRY glIndexiv( const GLint *c ); +GLAPI void GLAPIENTRY glIndexsv( const GLshort *c ); +GLAPI void GLAPIENTRY glIndexubv( const GLubyte *c ); /* 1.1 */ + +GLAPI void GLAPIENTRY glColor3b( GLbyte red, GLbyte green, GLbyte blue ); +GLAPI void GLAPIENTRY glColor3d( GLdouble red, GLdouble green, GLdouble blue ); +GLAPI void GLAPIENTRY glColor3f( GLfloat red, GLfloat green, GLfloat blue ); +GLAPI void GLAPIENTRY glColor3i( GLint red, GLint green, GLint blue ); +GLAPI void GLAPIENTRY glColor3s( GLshort red, GLshort green, GLshort blue ); +GLAPI void GLAPIENTRY glColor3ub( GLubyte red, GLubyte green, GLubyte blue ); +GLAPI void GLAPIENTRY glColor3ui( GLuint red, GLuint green, GLuint blue ); +GLAPI void GLAPIENTRY glColor3us( GLushort red, GLushort green, GLushort blue ); + +GLAPI void GLAPIENTRY glColor4b( GLbyte red, GLbyte green, + GLbyte blue, GLbyte alpha ); +GLAPI void GLAPIENTRY glColor4d( GLdouble red, GLdouble green, + GLdouble blue, GLdouble alpha ); +GLAPI void GLAPIENTRY glColor4f( GLfloat red, GLfloat green, + GLfloat blue, GLfloat alpha ); +GLAPI void GLAPIENTRY glColor4i( GLint red, GLint green, + GLint blue, GLint alpha ); +GLAPI void GLAPIENTRY glColor4s( GLshort red, GLshort green, + GLshort blue, GLshort alpha ); +GLAPI void GLAPIENTRY glColor4ub( GLubyte red, GLubyte green, + GLubyte blue, GLubyte alpha ); +GLAPI void GLAPIENTRY glColor4ui( GLuint red, GLuint green, + GLuint blue, GLuint alpha ); +GLAPI void GLAPIENTRY glColor4us( GLushort red, GLushort green, + GLushort blue, GLushort alpha ); + + +GLAPI void GLAPIENTRY glColor3bv( const GLbyte *v ); +GLAPI void GLAPIENTRY glColor3dv( const GLdouble *v ); +GLAPI void GLAPIENTRY glColor3fv( const GLfloat *v ); +GLAPI void GLAPIENTRY glColor3iv( const GLint *v ); +GLAPI void GLAPIENTRY glColor3sv( const GLshort *v ); +GLAPI void GLAPIENTRY glColor3ubv( const GLubyte *v ); +GLAPI void GLAPIENTRY glColor3uiv( const GLuint *v ); +GLAPI void GLAPIENTRY glColor3usv( const GLushort *v ); + +GLAPI void GLAPIENTRY glColor4bv( const GLbyte *v ); +GLAPI void GLAPIENTRY glColor4dv( const GLdouble *v ); +GLAPI void GLAPIENTRY glColor4fv( const GLfloat *v ); +GLAPI void GLAPIENTRY glColor4iv( const GLint *v ); +GLAPI void GLAPIENTRY glColor4sv( const GLshort *v ); +GLAPI void GLAPIENTRY glColor4ubv( const GLubyte *v ); +GLAPI void GLAPIENTRY glColor4uiv( const GLuint *v ); +GLAPI void GLAPIENTRY glColor4usv( const GLushort *v ); + + +GLAPI void GLAPIENTRY glTexCoord1d( GLdouble s ); +GLAPI void GLAPIENTRY glTexCoord1f( GLfloat s ); +GLAPI void GLAPIENTRY glTexCoord1i( GLint s ); +GLAPI void GLAPIENTRY glTexCoord1s( GLshort s ); + +GLAPI void GLAPIENTRY glTexCoord2d( GLdouble s, GLdouble t ); +GLAPI void GLAPIENTRY glTexCoord2f( GLfloat s, GLfloat t ); +GLAPI void GLAPIENTRY glTexCoord2i( GLint s, GLint t ); +GLAPI void GLAPIENTRY glTexCoord2s( GLshort s, GLshort t ); + +GLAPI void GLAPIENTRY glTexCoord3d( GLdouble s, GLdouble t, GLdouble r ); +GLAPI void GLAPIENTRY glTexCoord3f( GLfloat s, GLfloat t, GLfloat r ); +GLAPI void GLAPIENTRY glTexCoord3i( GLint s, GLint t, GLint r ); +GLAPI void GLAPIENTRY glTexCoord3s( GLshort s, GLshort t, GLshort r ); + +GLAPI void GLAPIENTRY glTexCoord4d( GLdouble s, GLdouble t, GLdouble r, GLdouble q ); +GLAPI void GLAPIENTRY glTexCoord4f( GLfloat s, GLfloat t, GLfloat r, GLfloat q ); +GLAPI void GLAPIENTRY glTexCoord4i( GLint s, GLint t, GLint r, GLint q ); +GLAPI void GLAPIENTRY glTexCoord4s( GLshort s, GLshort t, GLshort r, GLshort q ); + +GLAPI void GLAPIENTRY glTexCoord1dv( const GLdouble *v ); +GLAPI void GLAPIENTRY glTexCoord1fv( const GLfloat *v ); +GLAPI void GLAPIENTRY glTexCoord1iv( const GLint *v ); +GLAPI void GLAPIENTRY glTexCoord1sv( const GLshort *v ); + +GLAPI void GLAPIENTRY glTexCoord2dv( const GLdouble *v ); +GLAPI void GLAPIENTRY glTexCoord2fv( const GLfloat *v ); +GLAPI void GLAPIENTRY glTexCoord2iv( const GLint *v ); +GLAPI void GLAPIENTRY glTexCoord2sv( const GLshort *v ); + +GLAPI void GLAPIENTRY glTexCoord3dv( const GLdouble *v ); +GLAPI void GLAPIENTRY glTexCoord3fv( const GLfloat *v ); +GLAPI void GLAPIENTRY glTexCoord3iv( const GLint *v ); +GLAPI void GLAPIENTRY glTexCoord3sv( const GLshort *v ); + +GLAPI void GLAPIENTRY glTexCoord4dv( const GLdouble *v ); +GLAPI void GLAPIENTRY glTexCoord4fv( const GLfloat *v ); +GLAPI void GLAPIENTRY glTexCoord4iv( const GLint *v ); +GLAPI void GLAPIENTRY glTexCoord4sv( const GLshort *v ); + + +GLAPI void GLAPIENTRY glRasterPos2d( GLdouble x, GLdouble y ); +GLAPI void GLAPIENTRY glRasterPos2f( GLfloat x, GLfloat y ); +GLAPI void GLAPIENTRY glRasterPos2i( GLint x, GLint y ); +GLAPI void GLAPIENTRY glRasterPos2s( GLshort x, GLshort y ); + +GLAPI void GLAPIENTRY glRasterPos3d( GLdouble x, GLdouble y, GLdouble z ); +GLAPI void GLAPIENTRY glRasterPos3f( GLfloat x, GLfloat y, GLfloat z ); +GLAPI void GLAPIENTRY glRasterPos3i( GLint x, GLint y, GLint z ); +GLAPI void GLAPIENTRY glRasterPos3s( GLshort x, GLshort y, GLshort z ); + +GLAPI void GLAPIENTRY glRasterPos4d( GLdouble x, GLdouble y, GLdouble z, GLdouble w ); +GLAPI void GLAPIENTRY glRasterPos4f( GLfloat x, GLfloat y, GLfloat z, GLfloat w ); +GLAPI void GLAPIENTRY glRasterPos4i( GLint x, GLint y, GLint z, GLint w ); +GLAPI void GLAPIENTRY glRasterPos4s( GLshort x, GLshort y, GLshort z, GLshort w ); + +GLAPI void GLAPIENTRY glRasterPos2dv( const GLdouble *v ); +GLAPI void GLAPIENTRY glRasterPos2fv( const GLfloat *v ); +GLAPI void GLAPIENTRY glRasterPos2iv( const GLint *v ); +GLAPI void GLAPIENTRY glRasterPos2sv( const GLshort *v ); + +GLAPI void GLAPIENTRY glRasterPos3dv( const GLdouble *v ); +GLAPI void GLAPIENTRY glRasterPos3fv( const GLfloat *v ); +GLAPI void GLAPIENTRY glRasterPos3iv( const GLint *v ); +GLAPI void GLAPIENTRY glRasterPos3sv( const GLshort *v ); + +GLAPI void GLAPIENTRY glRasterPos4dv( const GLdouble *v ); +GLAPI void GLAPIENTRY glRasterPos4fv( const GLfloat *v ); +GLAPI void GLAPIENTRY glRasterPos4iv( const GLint *v ); +GLAPI void GLAPIENTRY glRasterPos4sv( const GLshort *v ); + + +GLAPI void GLAPIENTRY glRectd( GLdouble x1, GLdouble y1, GLdouble x2, GLdouble y2 ); +GLAPI void GLAPIENTRY glRectf( GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2 ); +GLAPI void GLAPIENTRY glRecti( GLint x1, GLint y1, GLint x2, GLint y2 ); +GLAPI void GLAPIENTRY glRects( GLshort x1, GLshort y1, GLshort x2, GLshort y2 ); + + +GLAPI void GLAPIENTRY glRectdv( const GLdouble *v1, const GLdouble *v2 ); +GLAPI void GLAPIENTRY glRectfv( const GLfloat *v1, const GLfloat *v2 ); +GLAPI void GLAPIENTRY glRectiv( const GLint *v1, const GLint *v2 ); +GLAPI void GLAPIENTRY glRectsv( const GLshort *v1, const GLshort *v2 ); + + +/* + * Vertex Arrays (1.1) + */ + +GLAPI void GLAPIENTRY glVertexPointer( GLint size, GLenum type, + GLsizei stride, const GLvoid *ptr ); + +GLAPI void GLAPIENTRY glNormalPointer( GLenum type, GLsizei stride, + const GLvoid *ptr ); + +GLAPI void GLAPIENTRY glColorPointer( GLint size, GLenum type, + GLsizei stride, const GLvoid *ptr ); + +GLAPI void GLAPIENTRY glIndexPointer( GLenum type, GLsizei stride, + const GLvoid *ptr ); + +GLAPI void GLAPIENTRY glTexCoordPointer( GLint size, GLenum type, + GLsizei stride, const GLvoid *ptr ); + +GLAPI void GLAPIENTRY glEdgeFlagPointer( GLsizei stride, const GLvoid *ptr ); + +GLAPI void GLAPIENTRY glGetPointerv( GLenum pname, GLvoid **params ); + +GLAPI void GLAPIENTRY glArrayElement( GLint i ); + +GLAPI void GLAPIENTRY glDrawArrays( GLenum mode, GLint first, GLsizei count ); + +GLAPI void GLAPIENTRY glDrawElements( GLenum mode, GLsizei count, + GLenum type, const GLvoid *indices ); + +GLAPI void GLAPIENTRY glInterleavedArrays( GLenum format, GLsizei stride, + const GLvoid *pointer ); + +/* + * Lighting + */ + +GLAPI void GLAPIENTRY glShadeModel( GLenum mode ); + +GLAPI void GLAPIENTRY glLightf( GLenum light, GLenum pname, GLfloat param ); +GLAPI void GLAPIENTRY glLighti( GLenum light, GLenum pname, GLint param ); +GLAPI void GLAPIENTRY glLightfv( GLenum light, GLenum pname, + const GLfloat *params ); +GLAPI void GLAPIENTRY glLightiv( GLenum light, GLenum pname, + const GLint *params ); + +GLAPI void GLAPIENTRY glGetLightfv( GLenum light, GLenum pname, + GLfloat *params ); +GLAPI void GLAPIENTRY glGetLightiv( GLenum light, GLenum pname, + GLint *params ); + +GLAPI void GLAPIENTRY glLightModelf( GLenum pname, GLfloat param ); +GLAPI void GLAPIENTRY glLightModeli( GLenum pname, GLint param ); +GLAPI void GLAPIENTRY glLightModelfv( GLenum pname, const GLfloat *params ); +GLAPI void GLAPIENTRY glLightModeliv( GLenum pname, const GLint *params ); + +GLAPI void GLAPIENTRY glMaterialf( GLenum face, GLenum pname, GLfloat param ); +GLAPI void GLAPIENTRY glMateriali( GLenum face, GLenum pname, GLint param ); +GLAPI void GLAPIENTRY glMaterialfv( GLenum face, GLenum pname, const GLfloat *params ); +GLAPI void GLAPIENTRY glMaterialiv( GLenum face, GLenum pname, const GLint *params ); + +GLAPI void GLAPIENTRY glGetMaterialfv( GLenum face, GLenum pname, GLfloat *params ); +GLAPI void GLAPIENTRY glGetMaterialiv( GLenum face, GLenum pname, GLint *params ); + +GLAPI void GLAPIENTRY glColorMaterial( GLenum face, GLenum mode ); + + +/* + * Raster functions + */ + +GLAPI void GLAPIENTRY glPixelZoom( GLfloat xfactor, GLfloat yfactor ); + +GLAPI void GLAPIENTRY glPixelStoref( GLenum pname, GLfloat param ); +GLAPI void GLAPIENTRY glPixelStorei( GLenum pname, GLint param ); + +GLAPI void GLAPIENTRY glPixelTransferf( GLenum pname, GLfloat param ); +GLAPI void GLAPIENTRY glPixelTransferi( GLenum pname, GLint param ); + +GLAPI void GLAPIENTRY glPixelMapfv( GLenum map, GLsizei mapsize, + const GLfloat *values ); +GLAPI void GLAPIENTRY glPixelMapuiv( GLenum map, GLsizei mapsize, + const GLuint *values ); +GLAPI void GLAPIENTRY glPixelMapusv( GLenum map, GLsizei mapsize, + const GLushort *values ); + +GLAPI void GLAPIENTRY glGetPixelMapfv( GLenum map, GLfloat *values ); +GLAPI void GLAPIENTRY glGetPixelMapuiv( GLenum map, GLuint *values ); +GLAPI void GLAPIENTRY glGetPixelMapusv( GLenum map, GLushort *values ); + +GLAPI void GLAPIENTRY glBitmap( GLsizei width, GLsizei height, + GLfloat xorig, GLfloat yorig, + GLfloat xmove, GLfloat ymove, + const GLubyte *bitmap ); + +GLAPI void GLAPIENTRY glReadPixels( GLint x, GLint y, + GLsizei width, GLsizei height, + GLenum format, GLenum type, + GLvoid *pixels ); + +GLAPI void GLAPIENTRY glDrawPixels( GLsizei width, GLsizei height, + GLenum format, GLenum type, + const GLvoid *pixels ); + +GLAPI void GLAPIENTRY glCopyPixels( GLint x, GLint y, + GLsizei width, GLsizei height, + GLenum type ); + +/* + * Stenciling + */ + +GLAPI void GLAPIENTRY glStencilFunc( GLenum func, GLint ref, GLuint mask ); + +GLAPI void GLAPIENTRY glStencilMask( GLuint mask ); + +GLAPI void GLAPIENTRY glStencilOp( GLenum fail, GLenum zfail, GLenum zpass ); + +GLAPI void GLAPIENTRY glClearStencil( GLint s ); + + + +/* + * Texture mapping + */ + +GLAPI void GLAPIENTRY glTexGend( GLenum coord, GLenum pname, GLdouble param ); +GLAPI void GLAPIENTRY glTexGenf( GLenum coord, GLenum pname, GLfloat param ); +GLAPI void GLAPIENTRY glTexGeni( GLenum coord, GLenum pname, GLint param ); + +GLAPI void GLAPIENTRY glTexGendv( GLenum coord, GLenum pname, const GLdouble *params ); +GLAPI void GLAPIENTRY glTexGenfv( GLenum coord, GLenum pname, const GLfloat *params ); +GLAPI void GLAPIENTRY glTexGeniv( GLenum coord, GLenum pname, const GLint *params ); + +GLAPI void GLAPIENTRY glGetTexGendv( GLenum coord, GLenum pname, GLdouble *params ); +GLAPI void GLAPIENTRY glGetTexGenfv( GLenum coord, GLenum pname, GLfloat *params ); +GLAPI void GLAPIENTRY glGetTexGeniv( GLenum coord, GLenum pname, GLint *params ); + + +GLAPI void GLAPIENTRY glTexEnvf( GLenum target, GLenum pname, GLfloat param ); +GLAPI void GLAPIENTRY glTexEnvi( GLenum target, GLenum pname, GLint param ); + +GLAPI void GLAPIENTRY glTexEnvfv( GLenum target, GLenum pname, const GLfloat *params ); +GLAPI void GLAPIENTRY glTexEnviv( GLenum target, GLenum pname, const GLint *params ); + +GLAPI void GLAPIENTRY glGetTexEnvfv( GLenum target, GLenum pname, GLfloat *params ); +GLAPI void GLAPIENTRY glGetTexEnviv( GLenum target, GLenum pname, GLint *params ); + + +GLAPI void GLAPIENTRY glTexParameterf( GLenum target, GLenum pname, GLfloat param ); +GLAPI void GLAPIENTRY glTexParameteri( GLenum target, GLenum pname, GLint param ); + +GLAPI void GLAPIENTRY glTexParameterfv( GLenum target, GLenum pname, + const GLfloat *params ); +GLAPI void GLAPIENTRY glTexParameteriv( GLenum target, GLenum pname, + const GLint *params ); + +GLAPI void GLAPIENTRY glGetTexParameterfv( GLenum target, + GLenum pname, GLfloat *params); +GLAPI void GLAPIENTRY glGetTexParameteriv( GLenum target, + GLenum pname, GLint *params ); + +GLAPI void GLAPIENTRY glGetTexLevelParameterfv( GLenum target, GLint level, + GLenum pname, GLfloat *params ); +GLAPI void GLAPIENTRY glGetTexLevelParameteriv( GLenum target, GLint level, + GLenum pname, GLint *params ); + + +GLAPI void GLAPIENTRY glTexImage1D( GLenum target, GLint level, + GLint internalFormat, + GLsizei width, GLint border, + GLenum format, GLenum type, + const GLvoid *pixels ); + +GLAPI void GLAPIENTRY glTexImage2D( GLenum target, GLint level, + GLint internalFormat, + GLsizei width, GLsizei height, + GLint border, GLenum format, GLenum type, + const GLvoid *pixels ); + +GLAPI void GLAPIENTRY glGetTexImage( GLenum target, GLint level, + GLenum format, GLenum type, + GLvoid *pixels ); + + +/* 1.1 functions */ + +GLAPI void GLAPIENTRY glGenTextures( GLsizei n, GLuint *textures ); + +GLAPI void GLAPIENTRY glDeleteTextures( GLsizei n, const GLuint *textures); + +GLAPI void GLAPIENTRY glBindTexture( GLenum target, GLuint texture ); + +GLAPI void GLAPIENTRY glPrioritizeTextures( GLsizei n, + const GLuint *textures, + const GLclampf *priorities ); + +GLAPI GLboolean GLAPIENTRY glAreTexturesResident( GLsizei n, + const GLuint *textures, + GLboolean *residences ); + +GLAPI GLboolean GLAPIENTRY glIsTexture( GLuint texture ); + + +GLAPI void GLAPIENTRY glTexSubImage1D( GLenum target, GLint level, + GLint xoffset, + GLsizei width, GLenum format, + GLenum type, const GLvoid *pixels ); + + +GLAPI void GLAPIENTRY glTexSubImage2D( GLenum target, GLint level, + GLint xoffset, GLint yoffset, + GLsizei width, GLsizei height, + GLenum format, GLenum type, + const GLvoid *pixels ); + + +GLAPI void GLAPIENTRY glCopyTexImage1D( GLenum target, GLint level, + GLenum internalformat, + GLint x, GLint y, + GLsizei width, GLint border ); + + +GLAPI void GLAPIENTRY glCopyTexImage2D( GLenum target, GLint level, + GLenum internalformat, + GLint x, GLint y, + GLsizei width, GLsizei height, + GLint border ); + + +GLAPI void GLAPIENTRY glCopyTexSubImage1D( GLenum target, GLint level, + GLint xoffset, GLint x, GLint y, + GLsizei width ); + + +GLAPI void GLAPIENTRY glCopyTexSubImage2D( GLenum target, GLint level, + GLint xoffset, GLint yoffset, + GLint x, GLint y, + GLsizei width, GLsizei height ); + + +/* + * Evaluators + */ + +GLAPI void GLAPIENTRY glMap1d( GLenum target, GLdouble u1, GLdouble u2, + GLint stride, + GLint order, const GLdouble *points ); +GLAPI void GLAPIENTRY glMap1f( GLenum target, GLfloat u1, GLfloat u2, + GLint stride, + GLint order, const GLfloat *points ); + +GLAPI void GLAPIENTRY glMap2d( GLenum target, + GLdouble u1, GLdouble u2, GLint ustride, GLint uorder, + GLdouble v1, GLdouble v2, GLint vstride, GLint vorder, + const GLdouble *points ); +GLAPI void GLAPIENTRY glMap2f( GLenum target, + GLfloat u1, GLfloat u2, GLint ustride, GLint uorder, + GLfloat v1, GLfloat v2, GLint vstride, GLint vorder, + const GLfloat *points ); + +GLAPI void GLAPIENTRY glGetMapdv( GLenum target, GLenum query, GLdouble *v ); +GLAPI void GLAPIENTRY glGetMapfv( GLenum target, GLenum query, GLfloat *v ); +GLAPI void GLAPIENTRY glGetMapiv( GLenum target, GLenum query, GLint *v ); + +GLAPI void GLAPIENTRY glEvalCoord1d( GLdouble u ); +GLAPI void GLAPIENTRY glEvalCoord1f( GLfloat u ); + +GLAPI void GLAPIENTRY glEvalCoord1dv( const GLdouble *u ); +GLAPI void GLAPIENTRY glEvalCoord1fv( const GLfloat *u ); + +GLAPI void GLAPIENTRY glEvalCoord2d( GLdouble u, GLdouble v ); +GLAPI void GLAPIENTRY glEvalCoord2f( GLfloat u, GLfloat v ); + +GLAPI void GLAPIENTRY glEvalCoord2dv( const GLdouble *u ); +GLAPI void GLAPIENTRY glEvalCoord2fv( const GLfloat *u ); + +GLAPI void GLAPIENTRY glMapGrid1d( GLint un, GLdouble u1, GLdouble u2 ); +GLAPI void GLAPIENTRY glMapGrid1f( GLint un, GLfloat u1, GLfloat u2 ); + +GLAPI void GLAPIENTRY glMapGrid2d( GLint un, GLdouble u1, GLdouble u2, + GLint vn, GLdouble v1, GLdouble v2 ); +GLAPI void GLAPIENTRY glMapGrid2f( GLint un, GLfloat u1, GLfloat u2, + GLint vn, GLfloat v1, GLfloat v2 ); + +GLAPI void GLAPIENTRY glEvalPoint1( GLint i ); + +GLAPI void GLAPIENTRY glEvalPoint2( GLint i, GLint j ); + +GLAPI void GLAPIENTRY glEvalMesh1( GLenum mode, GLint i1, GLint i2 ); + +GLAPI void GLAPIENTRY glEvalMesh2( GLenum mode, GLint i1, GLint i2, GLint j1, GLint j2 ); + + +/* + * Fog + */ + +GLAPI void GLAPIENTRY glFogf( GLenum pname, GLfloat param ); + +GLAPI void GLAPIENTRY glFogi( GLenum pname, GLint param ); + +GLAPI void GLAPIENTRY glFogfv( GLenum pname, const GLfloat *params ); + +GLAPI void GLAPIENTRY glFogiv( GLenum pname, const GLint *params ); + + +/* + * Selection and Feedback + */ + +GLAPI void GLAPIENTRY glFeedbackBuffer( GLsizei size, GLenum type, GLfloat *buffer ); + +GLAPI void GLAPIENTRY glPassThrough( GLfloat token ); + +GLAPI void GLAPIENTRY glSelectBuffer( GLsizei size, GLuint *buffer ); + +GLAPI void GLAPIENTRY glInitNames( void ); + +GLAPI void GLAPIENTRY glLoadName( GLuint name ); + +GLAPI void GLAPIENTRY glPushName( GLuint name ); + +GLAPI void GLAPIENTRY glPopName( void ); + + + +/* + * OpenGL 1.2 + */ + +#define GL_RESCALE_NORMAL 0x803A +#define GL_CLAMP_TO_EDGE 0x812F +#define GL_MAX_ELEMENTS_VERTICES 0x80E8 +#define GL_MAX_ELEMENTS_INDICES 0x80E9 +#define GL_BGR 0x80E0 +#define GL_BGRA 0x80E1 +#define GL_UNSIGNED_BYTE_3_3_2 0x8032 +#define GL_UNSIGNED_BYTE_2_3_3_REV 0x8362 +#define GL_UNSIGNED_SHORT_5_6_5 0x8363 +#define GL_UNSIGNED_SHORT_5_6_5_REV 0x8364 +#define GL_UNSIGNED_SHORT_4_4_4_4 0x8033 +#define GL_UNSIGNED_SHORT_4_4_4_4_REV 0x8365 +#define GL_UNSIGNED_SHORT_5_5_5_1 0x8034 +#define GL_UNSIGNED_SHORT_1_5_5_5_REV 0x8366 +#define GL_UNSIGNED_INT_8_8_8_8 0x8035 +#define GL_UNSIGNED_INT_8_8_8_8_REV 0x8367 +#define GL_UNSIGNED_INT_10_10_10_2 0x8036 +#define GL_UNSIGNED_INT_2_10_10_10_REV 0x8368 +#define GL_LIGHT_MODEL_COLOR_CONTROL 0x81F8 +#define GL_SINGLE_COLOR 0x81F9 +#define GL_SEPARATE_SPECULAR_COLOR 0x81FA +#define GL_TEXTURE_MIN_LOD 0x813A +#define GL_TEXTURE_MAX_LOD 0x813B +#define GL_TEXTURE_BASE_LEVEL 0x813C +#define GL_TEXTURE_MAX_LEVEL 0x813D +#define GL_SMOOTH_POINT_SIZE_RANGE 0x0B12 +#define GL_SMOOTH_POINT_SIZE_GRANULARITY 0x0B13 +#define GL_SMOOTH_LINE_WIDTH_RANGE 0x0B22 +#define GL_SMOOTH_LINE_WIDTH_GRANULARITY 0x0B23 +#define GL_ALIASED_POINT_SIZE_RANGE 0x846D +#define GL_ALIASED_LINE_WIDTH_RANGE 0x846E +#define GL_PACK_SKIP_IMAGES 0x806B +#define GL_PACK_IMAGE_HEIGHT 0x806C +#define GL_UNPACK_SKIP_IMAGES 0x806D +#define GL_UNPACK_IMAGE_HEIGHT 0x806E +#define GL_TEXTURE_3D 0x806F +#define GL_PROXY_TEXTURE_3D 0x8070 +#define GL_TEXTURE_DEPTH 0x8071 +#define GL_TEXTURE_WRAP_R 0x8072 +#define GL_MAX_3D_TEXTURE_SIZE 0x8073 +#define GL_TEXTURE_BINDING_3D 0x806A + +GLAPI void GLAPIENTRY glDrawRangeElements( GLenum mode, GLuint start, + GLuint end, GLsizei count, GLenum type, const GLvoid *indices ); + +GLAPI void GLAPIENTRY glTexImage3D( GLenum target, GLint level, + GLint internalFormat, + GLsizei width, GLsizei height, + GLsizei depth, GLint border, + GLenum format, GLenum type, + const GLvoid *pixels ); + +GLAPI void GLAPIENTRY glTexSubImage3D( GLenum target, GLint level, + GLint xoffset, GLint yoffset, + GLint zoffset, GLsizei width, + GLsizei height, GLsizei depth, + GLenum format, + GLenum type, const GLvoid *pixels); + +GLAPI void GLAPIENTRY glCopyTexSubImage3D( GLenum target, GLint level, + GLint xoffset, GLint yoffset, + GLint zoffset, GLint x, + GLint y, GLsizei width, + GLsizei height ); + +typedef void (APIENTRYP PFNGLDRAWRANGEELEMENTSPROC) (GLenum mode, GLuint start, GLuint end, GLsizei count, GLenum type, const GLvoid *indices); +typedef void (APIENTRYP PFNGLTEXIMAGE3DPROC) (GLenum target, GLint level, GLint internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLenum format, GLenum type, const GLvoid *pixels); +typedef void (APIENTRYP PFNGLTEXSUBIMAGE3DPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const GLvoid *pixels); +typedef void (APIENTRYP PFNGLCOPYTEXSUBIMAGE3DPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint x, GLint y, GLsizei width, GLsizei height); + + +/* + * GL_ARB_imaging + */ + +#define GL_CONSTANT_COLOR 0x8001 +#define GL_ONE_MINUS_CONSTANT_COLOR 0x8002 +#define GL_CONSTANT_ALPHA 0x8003 +#define GL_ONE_MINUS_CONSTANT_ALPHA 0x8004 +#define GL_COLOR_TABLE 0x80D0 +#define GL_POST_CONVOLUTION_COLOR_TABLE 0x80D1 +#define GL_POST_COLOR_MATRIX_COLOR_TABLE 0x80D2 +#define GL_PROXY_COLOR_TABLE 0x80D3 +#define GL_PROXY_POST_CONVOLUTION_COLOR_TABLE 0x80D4 +#define GL_PROXY_POST_COLOR_MATRIX_COLOR_TABLE 0x80D5 +#define GL_COLOR_TABLE_SCALE 0x80D6 +#define GL_COLOR_TABLE_BIAS 0x80D7 +#define GL_COLOR_TABLE_FORMAT 0x80D8 +#define GL_COLOR_TABLE_WIDTH 0x80D9 +#define GL_COLOR_TABLE_RED_SIZE 0x80DA +#define GL_COLOR_TABLE_GREEN_SIZE 0x80DB +#define GL_COLOR_TABLE_BLUE_SIZE 0x80DC +#define GL_COLOR_TABLE_ALPHA_SIZE 0x80DD +#define GL_COLOR_TABLE_LUMINANCE_SIZE 0x80DE +#define GL_COLOR_TABLE_INTENSITY_SIZE 0x80DF +#define GL_CONVOLUTION_1D 0x8010 +#define GL_CONVOLUTION_2D 0x8011 +#define GL_SEPARABLE_2D 0x8012 +#define GL_CONVOLUTION_BORDER_MODE 0x8013 +#define GL_CONVOLUTION_FILTER_SCALE 0x8014 +#define GL_CONVOLUTION_FILTER_BIAS 0x8015 +#define GL_REDUCE 0x8016 +#define GL_CONVOLUTION_FORMAT 0x8017 +#define GL_CONVOLUTION_WIDTH 0x8018 +#define GL_CONVOLUTION_HEIGHT 0x8019 +#define GL_MAX_CONVOLUTION_WIDTH 0x801A +#define GL_MAX_CONVOLUTION_HEIGHT 0x801B +#define GL_POST_CONVOLUTION_RED_SCALE 0x801C +#define GL_POST_CONVOLUTION_GREEN_SCALE 0x801D +#define GL_POST_CONVOLUTION_BLUE_SCALE 0x801E +#define GL_POST_CONVOLUTION_ALPHA_SCALE 0x801F +#define GL_POST_CONVOLUTION_RED_BIAS 0x8020 +#define GL_POST_CONVOLUTION_GREEN_BIAS 0x8021 +#define GL_POST_CONVOLUTION_BLUE_BIAS 0x8022 +#define GL_POST_CONVOLUTION_ALPHA_BIAS 0x8023 +#define GL_CONSTANT_BORDER 0x8151 +#define GL_REPLICATE_BORDER 0x8153 +#define GL_CONVOLUTION_BORDER_COLOR 0x8154 +#define GL_COLOR_MATRIX 0x80B1 +#define GL_COLOR_MATRIX_STACK_DEPTH 0x80B2 +#define GL_MAX_COLOR_MATRIX_STACK_DEPTH 0x80B3 +#define GL_POST_COLOR_MATRIX_RED_SCALE 0x80B4 +#define GL_POST_COLOR_MATRIX_GREEN_SCALE 0x80B5 +#define GL_POST_COLOR_MATRIX_BLUE_SCALE 0x80B6 +#define GL_POST_COLOR_MATRIX_ALPHA_SCALE 0x80B7 +#define GL_POST_COLOR_MATRIX_RED_BIAS 0x80B8 +#define GL_POST_COLOR_MATRIX_GREEN_BIAS 0x80B9 +#define GL_POST_COLOR_MATRIX_BLUE_BIAS 0x80BA +#define GL_POST_COLOR_MATRIX_ALPHA_BIAS 0x80BB +#define GL_HISTOGRAM 0x8024 +#define GL_PROXY_HISTOGRAM 0x8025 +#define GL_HISTOGRAM_WIDTH 0x8026 +#define GL_HISTOGRAM_FORMAT 0x8027 +#define GL_HISTOGRAM_RED_SIZE 0x8028 +#define GL_HISTOGRAM_GREEN_SIZE 0x8029 +#define GL_HISTOGRAM_BLUE_SIZE 0x802A +#define GL_HISTOGRAM_ALPHA_SIZE 0x802B +#define GL_HISTOGRAM_LUMINANCE_SIZE 0x802C +#define GL_HISTOGRAM_SINK 0x802D +#define GL_MINMAX 0x802E +#define GL_MINMAX_FORMAT 0x802F +#define GL_MINMAX_SINK 0x8030 +#define GL_TABLE_TOO_LARGE 0x8031 +#define GL_BLEND_EQUATION 0x8009 +#define GL_MIN 0x8007 +#define GL_MAX 0x8008 +#define GL_FUNC_ADD 0x8006 +#define GL_FUNC_SUBTRACT 0x800A +#define GL_FUNC_REVERSE_SUBTRACT 0x800B +#define GL_BLEND_COLOR 0x8005 + + +GLAPI void GLAPIENTRY glColorTable( GLenum target, GLenum internalformat, + GLsizei width, GLenum format, + GLenum type, const GLvoid *table ); + +GLAPI void GLAPIENTRY glColorSubTable( GLenum target, + GLsizei start, GLsizei count, + GLenum format, GLenum type, + const GLvoid *data ); + +GLAPI void GLAPIENTRY glColorTableParameteriv(GLenum target, GLenum pname, + const GLint *params); + +GLAPI void GLAPIENTRY glColorTableParameterfv(GLenum target, GLenum pname, + const GLfloat *params); + +GLAPI void GLAPIENTRY glCopyColorSubTable( GLenum target, GLsizei start, + GLint x, GLint y, GLsizei width ); + +GLAPI void GLAPIENTRY glCopyColorTable( GLenum target, GLenum internalformat, + GLint x, GLint y, GLsizei width ); + +GLAPI void GLAPIENTRY glGetColorTable( GLenum target, GLenum format, + GLenum type, GLvoid *table ); + +GLAPI void GLAPIENTRY glGetColorTableParameterfv( GLenum target, GLenum pname, + GLfloat *params ); + +GLAPI void GLAPIENTRY glGetColorTableParameteriv( GLenum target, GLenum pname, + GLint *params ); + +GLAPI void GLAPIENTRY glBlendEquation( GLenum mode ); + +GLAPI void GLAPIENTRY glBlendColor( GLclampf red, GLclampf green, + GLclampf blue, GLclampf alpha ); + +GLAPI void GLAPIENTRY glHistogram( GLenum target, GLsizei width, + GLenum internalformat, GLboolean sink ); + +GLAPI void GLAPIENTRY glResetHistogram( GLenum target ); + +GLAPI void GLAPIENTRY glGetHistogram( GLenum target, GLboolean reset, + GLenum format, GLenum type, + GLvoid *values ); + +GLAPI void GLAPIENTRY glGetHistogramParameterfv( GLenum target, GLenum pname, + GLfloat *params ); + +GLAPI void GLAPIENTRY glGetHistogramParameteriv( GLenum target, GLenum pname, + GLint *params ); + +GLAPI void GLAPIENTRY glMinmax( GLenum target, GLenum internalformat, + GLboolean sink ); + +GLAPI void GLAPIENTRY glResetMinmax( GLenum target ); + +GLAPI void GLAPIENTRY glGetMinmax( GLenum target, GLboolean reset, + GLenum format, GLenum types, + GLvoid *values ); + +GLAPI void GLAPIENTRY glGetMinmaxParameterfv( GLenum target, GLenum pname, + GLfloat *params ); + +GLAPI void GLAPIENTRY glGetMinmaxParameteriv( GLenum target, GLenum pname, + GLint *params ); + +GLAPI void GLAPIENTRY glConvolutionFilter1D( GLenum target, + GLenum internalformat, GLsizei width, GLenum format, GLenum type, + const GLvoid *image ); + +GLAPI void GLAPIENTRY glConvolutionFilter2D( GLenum target, + GLenum internalformat, GLsizei width, GLsizei height, GLenum format, + GLenum type, const GLvoid *image ); + +GLAPI void GLAPIENTRY glConvolutionParameterf( GLenum target, GLenum pname, + GLfloat params ); + +GLAPI void GLAPIENTRY glConvolutionParameterfv( GLenum target, GLenum pname, + const GLfloat *params ); + +GLAPI void GLAPIENTRY glConvolutionParameteri( GLenum target, GLenum pname, + GLint params ); + +GLAPI void GLAPIENTRY glConvolutionParameteriv( GLenum target, GLenum pname, + const GLint *params ); + +GLAPI void GLAPIENTRY glCopyConvolutionFilter1D( GLenum target, + GLenum internalformat, GLint x, GLint y, GLsizei width ); + +GLAPI void GLAPIENTRY glCopyConvolutionFilter2D( GLenum target, + GLenum internalformat, GLint x, GLint y, GLsizei width, + GLsizei height); + +GLAPI void GLAPIENTRY glGetConvolutionFilter( GLenum target, GLenum format, + GLenum type, GLvoid *image ); + +GLAPI void GLAPIENTRY glGetConvolutionParameterfv( GLenum target, GLenum pname, + GLfloat *params ); + +GLAPI void GLAPIENTRY glGetConvolutionParameteriv( GLenum target, GLenum pname, + GLint *params ); + +GLAPI void GLAPIENTRY glSeparableFilter2D( GLenum target, + GLenum internalformat, GLsizei width, GLsizei height, GLenum format, + GLenum type, const GLvoid *row, const GLvoid *column ); + +GLAPI void GLAPIENTRY glGetSeparableFilter( GLenum target, GLenum format, + GLenum type, GLvoid *row, GLvoid *column, GLvoid *span ); + + + + +/* + * OpenGL 1.3 + */ + +/* multitexture */ +#define GL_TEXTURE0 0x84C0 +#define GL_TEXTURE1 0x84C1 +#define GL_TEXTURE2 0x84C2 +#define GL_TEXTURE3 0x84C3 +#define GL_TEXTURE4 0x84C4 +#define GL_TEXTURE5 0x84C5 +#define GL_TEXTURE6 0x84C6 +#define GL_TEXTURE7 0x84C7 +#define GL_TEXTURE8 0x84C8 +#define GL_TEXTURE9 0x84C9 +#define GL_TEXTURE10 0x84CA +#define GL_TEXTURE11 0x84CB +#define GL_TEXTURE12 0x84CC +#define GL_TEXTURE13 0x84CD +#define GL_TEXTURE14 0x84CE +#define GL_TEXTURE15 0x84CF +#define GL_TEXTURE16 0x84D0 +#define GL_TEXTURE17 0x84D1 +#define GL_TEXTURE18 0x84D2 +#define GL_TEXTURE19 0x84D3 +#define GL_TEXTURE20 0x84D4 +#define GL_TEXTURE21 0x84D5 +#define GL_TEXTURE22 0x84D6 +#define GL_TEXTURE23 0x84D7 +#define GL_TEXTURE24 0x84D8 +#define GL_TEXTURE25 0x84D9 +#define GL_TEXTURE26 0x84DA +#define GL_TEXTURE27 0x84DB +#define GL_TEXTURE28 0x84DC +#define GL_TEXTURE29 0x84DD +#define GL_TEXTURE30 0x84DE +#define GL_TEXTURE31 0x84DF +#define GL_ACTIVE_TEXTURE 0x84E0 +#define GL_CLIENT_ACTIVE_TEXTURE 0x84E1 +#define GL_MAX_TEXTURE_UNITS 0x84E2 +/* texture_cube_map */ +#define GL_NORMAL_MAP 0x8511 +#define GL_REFLECTION_MAP 0x8512 +#define GL_TEXTURE_CUBE_MAP 0x8513 +#define GL_TEXTURE_BINDING_CUBE_MAP 0x8514 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_X 0x8515 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_X 0x8516 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Y 0x8517 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Y 0x8518 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Z 0x8519 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Z 0x851A +#define GL_PROXY_TEXTURE_CUBE_MAP 0x851B +#define GL_MAX_CUBE_MAP_TEXTURE_SIZE 0x851C +/* texture_compression */ +#define GL_COMPRESSED_ALPHA 0x84E9 +#define GL_COMPRESSED_LUMINANCE 0x84EA +#define GL_COMPRESSED_LUMINANCE_ALPHA 0x84EB +#define GL_COMPRESSED_INTENSITY 0x84EC +#define GL_COMPRESSED_RGB 0x84ED +#define GL_COMPRESSED_RGBA 0x84EE +#define GL_TEXTURE_COMPRESSION_HINT 0x84EF +#define GL_TEXTURE_COMPRESSED_IMAGE_SIZE 0x86A0 +#define GL_TEXTURE_COMPRESSED 0x86A1 +#define GL_NUM_COMPRESSED_TEXTURE_FORMATS 0x86A2 +#define GL_COMPRESSED_TEXTURE_FORMATS 0x86A3 +/* multisample */ +#define GL_MULTISAMPLE 0x809D +#define GL_SAMPLE_ALPHA_TO_COVERAGE 0x809E +#define GL_SAMPLE_ALPHA_TO_ONE 0x809F +#define GL_SAMPLE_COVERAGE 0x80A0 +#define GL_SAMPLE_BUFFERS 0x80A8 +#define GL_SAMPLES 0x80A9 +#define GL_SAMPLE_COVERAGE_VALUE 0x80AA +#define GL_SAMPLE_COVERAGE_INVERT 0x80AB +#define GL_MULTISAMPLE_BIT 0x20000000 +/* transpose_matrix */ +#define GL_TRANSPOSE_MODELVIEW_MATRIX 0x84E3 +#define GL_TRANSPOSE_PROJECTION_MATRIX 0x84E4 +#define GL_TRANSPOSE_TEXTURE_MATRIX 0x84E5 +#define GL_TRANSPOSE_COLOR_MATRIX 0x84E6 +/* texture_env_combine */ +#define GL_COMBINE 0x8570 +#define GL_COMBINE_RGB 0x8571 +#define GL_COMBINE_ALPHA 0x8572 +#define GL_SOURCE0_RGB 0x8580 +#define GL_SOURCE1_RGB 0x8581 +#define GL_SOURCE2_RGB 0x8582 +#define GL_SOURCE0_ALPHA 0x8588 +#define GL_SOURCE1_ALPHA 0x8589 +#define GL_SOURCE2_ALPHA 0x858A +#define GL_OPERAND0_RGB 0x8590 +#define GL_OPERAND1_RGB 0x8591 +#define GL_OPERAND2_RGB 0x8592 +#define GL_OPERAND0_ALPHA 0x8598 +#define GL_OPERAND1_ALPHA 0x8599 +#define GL_OPERAND2_ALPHA 0x859A +#define GL_RGB_SCALE 0x8573 +#define GL_ADD_SIGNED 0x8574 +#define GL_INTERPOLATE 0x8575 +#define GL_SUBTRACT 0x84E7 +#define GL_CONSTANT 0x8576 +#define GL_PRIMARY_COLOR 0x8577 +#define GL_PREVIOUS 0x8578 +/* texture_env_dot3 */ +#define GL_DOT3_RGB 0x86AE +#define GL_DOT3_RGBA 0x86AF +/* texture_border_clamp */ +#define GL_CLAMP_TO_BORDER 0x812D + +GLAPI void GLAPIENTRY glActiveTexture( GLenum texture ); + +GLAPI void GLAPIENTRY glClientActiveTexture( GLenum texture ); + +GLAPI void GLAPIENTRY glCompressedTexImage1D( GLenum target, GLint level, GLenum internalformat, GLsizei width, GLint border, GLsizei imageSize, const GLvoid *data ); + +GLAPI void GLAPIENTRY glCompressedTexImage2D( GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLsizei imageSize, const GLvoid *data ); + +GLAPI void GLAPIENTRY glCompressedTexImage3D( GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLsizei imageSize, const GLvoid *data ); + +GLAPI void GLAPIENTRY glCompressedTexSubImage1D( GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLsizei imageSize, const GLvoid *data ); + +GLAPI void GLAPIENTRY glCompressedTexSubImage2D( GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const GLvoid *data ); + +GLAPI void GLAPIENTRY glCompressedTexSubImage3D( GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const GLvoid *data ); + +GLAPI void GLAPIENTRY glGetCompressedTexImage( GLenum target, GLint lod, GLvoid *img ); + +GLAPI void GLAPIENTRY glMultiTexCoord1d( GLenum target, GLdouble s ); + +GLAPI void GLAPIENTRY glMultiTexCoord1dv( GLenum target, const GLdouble *v ); + +GLAPI void GLAPIENTRY glMultiTexCoord1f( GLenum target, GLfloat s ); + +GLAPI void GLAPIENTRY glMultiTexCoord1fv( GLenum target, const GLfloat *v ); + +GLAPI void GLAPIENTRY glMultiTexCoord1i( GLenum target, GLint s ); + +GLAPI void GLAPIENTRY glMultiTexCoord1iv( GLenum target, const GLint *v ); + +GLAPI void GLAPIENTRY glMultiTexCoord1s( GLenum target, GLshort s ); + +GLAPI void GLAPIENTRY glMultiTexCoord1sv( GLenum target, const GLshort *v ); + +GLAPI void GLAPIENTRY glMultiTexCoord2d( GLenum target, GLdouble s, GLdouble t ); + +GLAPI void GLAPIENTRY glMultiTexCoord2dv( GLenum target, const GLdouble *v ); + +GLAPI void GLAPIENTRY glMultiTexCoord2f( GLenum target, GLfloat s, GLfloat t ); + +GLAPI void GLAPIENTRY glMultiTexCoord2fv( GLenum target, const GLfloat *v ); + +GLAPI void GLAPIENTRY glMultiTexCoord2i( GLenum target, GLint s, GLint t ); + +GLAPI void GLAPIENTRY glMultiTexCoord2iv( GLenum target, const GLint *v ); + +GLAPI void GLAPIENTRY glMultiTexCoord2s( GLenum target, GLshort s, GLshort t ); + +GLAPI void GLAPIENTRY glMultiTexCoord2sv( GLenum target, const GLshort *v ); + +GLAPI void GLAPIENTRY glMultiTexCoord3d( GLenum target, GLdouble s, GLdouble t, GLdouble r ); + +GLAPI void GLAPIENTRY glMultiTexCoord3dv( GLenum target, const GLdouble *v ); + +GLAPI void GLAPIENTRY glMultiTexCoord3f( GLenum target, GLfloat s, GLfloat t, GLfloat r ); + +GLAPI void GLAPIENTRY glMultiTexCoord3fv( GLenum target, const GLfloat *v ); + +GLAPI void GLAPIENTRY glMultiTexCoord3i( GLenum target, GLint s, GLint t, GLint r ); + +GLAPI void GLAPIENTRY glMultiTexCoord3iv( GLenum target, const GLint *v ); + +GLAPI void GLAPIENTRY glMultiTexCoord3s( GLenum target, GLshort s, GLshort t, GLshort r ); + +GLAPI void GLAPIENTRY glMultiTexCoord3sv( GLenum target, const GLshort *v ); + +GLAPI void GLAPIENTRY glMultiTexCoord4d( GLenum target, GLdouble s, GLdouble t, GLdouble r, GLdouble q ); + +GLAPI void GLAPIENTRY glMultiTexCoord4dv( GLenum target, const GLdouble *v ); + +GLAPI void GLAPIENTRY glMultiTexCoord4f( GLenum target, GLfloat s, GLfloat t, GLfloat r, GLfloat q ); + +GLAPI void GLAPIENTRY glMultiTexCoord4fv( GLenum target, const GLfloat *v ); + +GLAPI void GLAPIENTRY glMultiTexCoord4i( GLenum target, GLint s, GLint t, GLint r, GLint q ); + +GLAPI void GLAPIENTRY glMultiTexCoord4iv( GLenum target, const GLint *v ); + +GLAPI void GLAPIENTRY glMultiTexCoord4s( GLenum target, GLshort s, GLshort t, GLshort r, GLshort q ); + +GLAPI void GLAPIENTRY glMultiTexCoord4sv( GLenum target, const GLshort *v ); + + +GLAPI void GLAPIENTRY glLoadTransposeMatrixd( const GLdouble m[16] ); + +GLAPI void GLAPIENTRY glLoadTransposeMatrixf( const GLfloat m[16] ); + +GLAPI void GLAPIENTRY glMultTransposeMatrixd( const GLdouble m[16] ); + +GLAPI void GLAPIENTRY glMultTransposeMatrixf( const GLfloat m[16] ); + +GLAPI void GLAPIENTRY glSampleCoverage( GLclampf value, GLboolean invert ); + + +typedef void (APIENTRYP PFNGLACTIVETEXTUREPROC) (GLenum texture); +typedef void (APIENTRYP PFNGLSAMPLECOVERAGEPROC) (GLclampf value, GLboolean invert); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXIMAGE3DPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXIMAGE2DPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXIMAGE1DPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLint border, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE3DPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE2DPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE1DPROC) (GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLGETCOMPRESSEDTEXIMAGEPROC) (GLenum target, GLint level, GLvoid *img); + + + +/* + * GL_ARB_multitexture (ARB extension 1 and OpenGL 1.2.1) + */ +#ifndef GL_ARB_multitexture +#define GL_ARB_multitexture 1 + +#define GL_TEXTURE0_ARB 0x84C0 +#define GL_TEXTURE1_ARB 0x84C1 +#define GL_TEXTURE2_ARB 0x84C2 +#define GL_TEXTURE3_ARB 0x84C3 +#define GL_TEXTURE4_ARB 0x84C4 +#define GL_TEXTURE5_ARB 0x84C5 +#define GL_TEXTURE6_ARB 0x84C6 +#define GL_TEXTURE7_ARB 0x84C7 +#define GL_TEXTURE8_ARB 0x84C8 +#define GL_TEXTURE9_ARB 0x84C9 +#define GL_TEXTURE10_ARB 0x84CA +#define GL_TEXTURE11_ARB 0x84CB +#define GL_TEXTURE12_ARB 0x84CC +#define GL_TEXTURE13_ARB 0x84CD +#define GL_TEXTURE14_ARB 0x84CE +#define GL_TEXTURE15_ARB 0x84CF +#define GL_TEXTURE16_ARB 0x84D0 +#define GL_TEXTURE17_ARB 0x84D1 +#define GL_TEXTURE18_ARB 0x84D2 +#define GL_TEXTURE19_ARB 0x84D3 +#define GL_TEXTURE20_ARB 0x84D4 +#define GL_TEXTURE21_ARB 0x84D5 +#define GL_TEXTURE22_ARB 0x84D6 +#define GL_TEXTURE23_ARB 0x84D7 +#define GL_TEXTURE24_ARB 0x84D8 +#define GL_TEXTURE25_ARB 0x84D9 +#define GL_TEXTURE26_ARB 0x84DA +#define GL_TEXTURE27_ARB 0x84DB +#define GL_TEXTURE28_ARB 0x84DC +#define GL_TEXTURE29_ARB 0x84DD +#define GL_TEXTURE30_ARB 0x84DE +#define GL_TEXTURE31_ARB 0x84DF +#define GL_ACTIVE_TEXTURE_ARB 0x84E0 +#define GL_CLIENT_ACTIVE_TEXTURE_ARB 0x84E1 +#define GL_MAX_TEXTURE_UNITS_ARB 0x84E2 + +GLAPI void GLAPIENTRY glActiveTextureARB(GLenum texture); +GLAPI void GLAPIENTRY glClientActiveTextureARB(GLenum texture); +GLAPI void GLAPIENTRY glMultiTexCoord1dARB(GLenum target, GLdouble s); +GLAPI void GLAPIENTRY glMultiTexCoord1dvARB(GLenum target, const GLdouble *v); +GLAPI void GLAPIENTRY glMultiTexCoord1fARB(GLenum target, GLfloat s); +GLAPI void GLAPIENTRY glMultiTexCoord1fvARB(GLenum target, const GLfloat *v); +GLAPI void GLAPIENTRY glMultiTexCoord1iARB(GLenum target, GLint s); +GLAPI void GLAPIENTRY glMultiTexCoord1ivARB(GLenum target, const GLint *v); +GLAPI void GLAPIENTRY glMultiTexCoord1sARB(GLenum target, GLshort s); +GLAPI void GLAPIENTRY glMultiTexCoord1svARB(GLenum target, const GLshort *v); +GLAPI void GLAPIENTRY glMultiTexCoord2dARB(GLenum target, GLdouble s, GLdouble t); +GLAPI void GLAPIENTRY glMultiTexCoord2dvARB(GLenum target, const GLdouble *v); +GLAPI void GLAPIENTRY glMultiTexCoord2fARB(GLenum target, GLfloat s, GLfloat t); +GLAPI void GLAPIENTRY glMultiTexCoord2fvARB(GLenum target, const GLfloat *v); +GLAPI void GLAPIENTRY glMultiTexCoord2iARB(GLenum target, GLint s, GLint t); +GLAPI void GLAPIENTRY glMultiTexCoord2ivARB(GLenum target, const GLint *v); +GLAPI void GLAPIENTRY glMultiTexCoord2sARB(GLenum target, GLshort s, GLshort t); +GLAPI void GLAPIENTRY glMultiTexCoord2svARB(GLenum target, const GLshort *v); +GLAPI void GLAPIENTRY glMultiTexCoord3dARB(GLenum target, GLdouble s, GLdouble t, GLdouble r); +GLAPI void GLAPIENTRY glMultiTexCoord3dvARB(GLenum target, const GLdouble *v); +GLAPI void GLAPIENTRY glMultiTexCoord3fARB(GLenum target, GLfloat s, GLfloat t, GLfloat r); +GLAPI void GLAPIENTRY glMultiTexCoord3fvARB(GLenum target, const GLfloat *v); +GLAPI void GLAPIENTRY glMultiTexCoord3iARB(GLenum target, GLint s, GLint t, GLint r); +GLAPI void GLAPIENTRY glMultiTexCoord3ivARB(GLenum target, const GLint *v); +GLAPI void GLAPIENTRY glMultiTexCoord3sARB(GLenum target, GLshort s, GLshort t, GLshort r); +GLAPI void GLAPIENTRY glMultiTexCoord3svARB(GLenum target, const GLshort *v); +GLAPI void GLAPIENTRY glMultiTexCoord4dARB(GLenum target, GLdouble s, GLdouble t, GLdouble r, GLdouble q); +GLAPI void GLAPIENTRY glMultiTexCoord4dvARB(GLenum target, const GLdouble *v); +GLAPI void GLAPIENTRY glMultiTexCoord4fARB(GLenum target, GLfloat s, GLfloat t, GLfloat r, GLfloat q); +GLAPI void GLAPIENTRY glMultiTexCoord4fvARB(GLenum target, const GLfloat *v); +GLAPI void GLAPIENTRY glMultiTexCoord4iARB(GLenum target, GLint s, GLint t, GLint r, GLint q); +GLAPI void GLAPIENTRY glMultiTexCoord4ivARB(GLenum target, const GLint *v); +GLAPI void GLAPIENTRY glMultiTexCoord4sARB(GLenum target, GLshort s, GLshort t, GLshort r, GLshort q); +GLAPI void GLAPIENTRY glMultiTexCoord4svARB(GLenum target, const GLshort *v); + +typedef void (APIENTRYP PFNGLACTIVETEXTUREARBPROC) (GLenum texture); +typedef void (APIENTRYP PFNGLCLIENTACTIVETEXTUREARBPROC) (GLenum texture); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1DARBPROC) (GLenum target, GLdouble s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1DVARBPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1FARBPROC) (GLenum target, GLfloat s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1FVARBPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1IARBPROC) (GLenum target, GLint s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1IVARBPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1SARBPROC) (GLenum target, GLshort s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1SVARBPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2DARBPROC) (GLenum target, GLdouble s, GLdouble t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2DVARBPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2FARBPROC) (GLenum target, GLfloat s, GLfloat t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2FVARBPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2IARBPROC) (GLenum target, GLint s, GLint t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2IVARBPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2SARBPROC) (GLenum target, GLshort s, GLshort t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2SVARBPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3DARBPROC) (GLenum target, GLdouble s, GLdouble t, GLdouble r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3DVARBPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3FARBPROC) (GLenum target, GLfloat s, GLfloat t, GLfloat r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3FVARBPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3IARBPROC) (GLenum target, GLint s, GLint t, GLint r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3IVARBPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3SARBPROC) (GLenum target, GLshort s, GLshort t, GLshort r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3SVARBPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4DARBPROC) (GLenum target, GLdouble s, GLdouble t, GLdouble r, GLdouble q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4DVARBPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4FARBPROC) (GLenum target, GLfloat s, GLfloat t, GLfloat r, GLfloat q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4FVARBPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4IARBPROC) (GLenum target, GLint s, GLint t, GLint r, GLint q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4IVARBPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4SARBPROC) (GLenum target, GLshort s, GLshort t, GLshort r, GLshort q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4SVARBPROC) (GLenum target, const GLshort *v); + +#endif /* GL_ARB_multitexture */ + + + +/* + * Define this token if you want "old-style" header file behaviour (extensions + * defined in gl.h). Otherwise, extensions will be included from glext.h. + */ +#if !defined(NO_SDL_GLEXT) && !defined(GL_GLEXT_LEGACY) +#include "SDL_opengl_glext.h" +#endif /* GL_GLEXT_LEGACY */ + + + +/********************************************************************** + * Begin system-specific stuff + */ +#if defined(PRAGMA_EXPORT_SUPPORTED) +#pragma export off +#endif + +/* + * End system-specific stuff + **********************************************************************/ + + +#ifdef __cplusplus +} +#endif + +#endif /* __gl_h_ */ + +#endif /* !__IPHONEOS__ */ + +#endif /* SDL_opengl_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengl_glext.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengl_glext.h new file mode 100644 index 00000000..ff6ad12c --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengl_glext.h @@ -0,0 +1,13213 @@ +/* SDL modified the include guard to be compatible with Mesa and Apple include guards: + * - Mesa uses: __gl_glext_h_ + * - Apple uses: __glext_h_ */ +#if !defined(__glext_h_) && !defined(__gl_glext_h_) +#define __glext_h_ 1 +#define __gl_glext_h_ 1 + +#ifdef __cplusplus +extern "C" { +#endif + +/* +** Copyright 2013-2020 The Khronos Group Inc. +** SPDX-License-Identifier: MIT +** +** This header is generated from the Khronos OpenGL / OpenGL ES XML +** API Registry. The current version of the Registry, generator scripts +** used to make the header, and the header can be found at +** https://github.com/KhronosGroup/OpenGL-Registry +*/ + +#if defined(_WIN32) && !defined(APIENTRY) && !defined(__CYGWIN__) && !defined(__SCITECH_SNAP__) +#ifndef WIN32_LEAN_AND_MEAN +#define WIN32_LEAN_AND_MEAN 1 +#endif +#include +#endif + +#ifndef APIENTRY +#define APIENTRY +#endif +#ifndef APIENTRYP +#define APIENTRYP APIENTRY * +#endif +#ifndef GLAPI +#define GLAPI extern +#endif + +#define GL_GLEXT_VERSION 20220530 + +/*#include */ +#ifndef __khrplatform_h_ +#define __khrplatform_h_ + +/* +** Copyright (c) 2008-2018 The Khronos Group Inc. +** +** Permission is hereby granted, free of charge, to any person obtaining a +** copy of this software and/or associated documentation files (the +** "Materials"), to deal in the Materials without restriction, including +** without limitation the rights to use, copy, modify, merge, publish, +** distribute, sublicense, and/or sell copies of the Materials, and to +** permit persons to whom the Materials are furnished to do so, subject to +** the following conditions: +** +** The above copyright notice and this permission notice shall be included +** in all copies or substantial portions of the Materials. +** +** THE MATERIALS ARE PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. +** IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY +** CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, +** TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE +** MATERIALS OR THE USE OR OTHER DEALINGS IN THE MATERIALS. +*/ + +/* Khronos platform-specific types and definitions. + * + * The master copy of khrplatform.h is maintained in the Khronos EGL + * Registry repository at https://github.com/KhronosGroup/EGL-Registry + * The last semantic modification to khrplatform.h was at commit ID: + * 67a3e0864c2d75ea5287b9f3d2eb74a745936692 + * + * Adopters may modify this file to suit their platform. Adopters are + * encouraged to submit platform specific modifications to the Khronos + * group so that they can be included in future versions of this file. + * Please submit changes by filing pull requests or issues on + * the EGL Registry repository linked above. + * + * + * See the Implementer's Guidelines for information about where this file + * should be located on your system and for more details of its use: + * http://www.khronos.org/registry/implementers_guide.pdf + * + * This file should be included as + * #include + * by Khronos client API header files that use its types and defines. + * + * The types in khrplatform.h should only be used to define API-specific types. + * + * Types defined in khrplatform.h: + * khronos_int8_t signed 8 bit + * khronos_uint8_t unsigned 8 bit + * khronos_int16_t signed 16 bit + * khronos_uint16_t unsigned 16 bit + * khronos_int32_t signed 32 bit + * khronos_uint32_t unsigned 32 bit + * khronos_int64_t signed 64 bit + * khronos_uint64_t unsigned 64 bit + * khronos_intptr_t signed same number of bits as a pointer + * khronos_uintptr_t unsigned same number of bits as a pointer + * khronos_ssize_t signed size + * khronos_usize_t unsigned size + * khronos_float_t signed 32 bit floating point + * khronos_time_ns_t unsigned 64 bit time in nanoseconds + * khronos_utime_nanoseconds_t unsigned time interval or absolute time in + * nanoseconds + * khronos_stime_nanoseconds_t signed time interval in nanoseconds + * khronos_boolean_enum_t enumerated boolean type. This should + * only be used as a base type when a client API's boolean type is + * an enum. Client APIs which use an integer or other type for + * booleans cannot use this as the base type for their boolean. + * + * Tokens defined in khrplatform.h: + * + * KHRONOS_FALSE, KHRONOS_TRUE Enumerated boolean false/true values. + * + * KHRONOS_SUPPORT_INT64 is 1 if 64 bit integers are supported; otherwise 0. + * KHRONOS_SUPPORT_FLOAT is 1 if floats are supported; otherwise 0. + * + * Calling convention macros defined in this file: + * KHRONOS_APICALL + * KHRONOS_APIENTRY + * KHRONOS_APIATTRIBUTES + * + * These may be used in function prototypes as: + * + * KHRONOS_APICALL void KHRONOS_APIENTRY funcname( + * int arg1, + * int arg2) KHRONOS_APIATTRIBUTES; + */ + +#if defined(__SCITECH_SNAP__) && !defined(KHRONOS_STATIC) +# define KHRONOS_STATIC 1 +#endif + +/*------------------------------------------------------------------------- + * Definition of KHRONOS_APICALL + *------------------------------------------------------------------------- + * This precedes the return type of the function in the function prototype. + */ +#if defined(KHRONOS_STATIC) + /* If the preprocessor constant KHRONOS_STATIC is defined, make the + * header compatible with static linking. */ +# define KHRONOS_APICALL +#elif defined(_WIN32) +# define KHRONOS_APICALL __declspec(dllimport) +#elif defined (__SYMBIAN32__) +# define KHRONOS_APICALL IMPORT_C +#elif defined(__ANDROID__) +# define KHRONOS_APICALL __attribute__((visibility("default"))) +#else +# define KHRONOS_APICALL +#endif + +/*------------------------------------------------------------------------- + * Definition of KHRONOS_APIENTRY + *------------------------------------------------------------------------- + * This follows the return type of the function and precedes the function + * name in the function prototype. + */ +#if defined(_WIN32) && !defined(_WIN32_WCE) && !defined(__SCITECH_SNAP__) + /* Win32 but not WinCE */ +# define KHRONOS_APIENTRY __stdcall +#else +# define KHRONOS_APIENTRY +#endif + +/*------------------------------------------------------------------------- + * Definition of KHRONOS_APIATTRIBUTES + *------------------------------------------------------------------------- + * This follows the closing parenthesis of the function prototype arguments. + */ +#if defined (__ARMCC_2__) +#define KHRONOS_APIATTRIBUTES __softfp +#else +#define KHRONOS_APIATTRIBUTES +#endif + +/*------------------------------------------------------------------------- + * basic type definitions + *-----------------------------------------------------------------------*/ +#if (defined(__STDC_VERSION__) && __STDC_VERSION__ >= 199901L) || defined(__GNUC__) || defined(__SCO__) || defined(__USLC__) + + +/* + * Using + */ +#include +typedef int32_t khronos_int32_t; +typedef uint32_t khronos_uint32_t; +typedef int64_t khronos_int64_t; +typedef uint64_t khronos_uint64_t; +#define KHRONOS_SUPPORT_INT64 1 +#define KHRONOS_SUPPORT_FLOAT 1 +/* + * To support platform where unsigned long cannot be used interchangeably with + * inptr_t (e.g. CHERI-extended ISAs), we can use the stdint.h intptr_t. + * Ideally, we could just use (u)intptr_t everywhere, but this could result in + * ABI breakage if khronos_uintptr_t is changed from unsigned long to + * unsigned long long or similar (this results in different C++ name mangling). + * To avoid changes for existing platforms, we restrict usage of intptr_t to + * platforms where the size of a pointer is larger than the size of long. + */ +#if defined(__SIZEOF_LONG__) && defined(__SIZEOF_POINTER__) +#if __SIZEOF_POINTER__ > __SIZEOF_LONG__ +#define KHRONOS_USE_INTPTR_T +#endif +#endif + +#elif defined(__VMS ) || defined(__sgi) + +/* + * Using + */ +#include +typedef int32_t khronos_int32_t; +typedef uint32_t khronos_uint32_t; +typedef int64_t khronos_int64_t; +typedef uint64_t khronos_uint64_t; +#define KHRONOS_SUPPORT_INT64 1 +#define KHRONOS_SUPPORT_FLOAT 1 + +#elif defined(_WIN32) && !defined(__SCITECH_SNAP__) + +/* + * Win32 + */ +typedef __int32 khronos_int32_t; +typedef unsigned __int32 khronos_uint32_t; +typedef __int64 khronos_int64_t; +typedef unsigned __int64 khronos_uint64_t; +#define KHRONOS_SUPPORT_INT64 1 +#define KHRONOS_SUPPORT_FLOAT 1 + +#elif defined(__sun__) || defined(__digital__) + +/* + * Sun or Digital + */ +typedef int khronos_int32_t; +typedef unsigned int khronos_uint32_t; +#if defined(__arch64__) || defined(_LP64) +typedef long int khronos_int64_t; +typedef unsigned long int khronos_uint64_t; +#else +typedef long long int khronos_int64_t; +typedef unsigned long long int khronos_uint64_t; +#endif /* __arch64__ */ +#define KHRONOS_SUPPORT_INT64 1 +#define KHRONOS_SUPPORT_FLOAT 1 + +#elif 0 + +/* + * Hypothetical platform with no float or int64 support + */ +typedef int khronos_int32_t; +typedef unsigned int khronos_uint32_t; +#define KHRONOS_SUPPORT_INT64 0 +#define KHRONOS_SUPPORT_FLOAT 0 + +#else + +/* + * Generic fallback + */ +#include +typedef int32_t khronos_int32_t; +typedef uint32_t khronos_uint32_t; +typedef int64_t khronos_int64_t; +typedef uint64_t khronos_uint64_t; +#define KHRONOS_SUPPORT_INT64 1 +#define KHRONOS_SUPPORT_FLOAT 1 + +#endif + + +/* + * Types that are (so far) the same on all platforms + */ +typedef signed char khronos_int8_t; +typedef unsigned char khronos_uint8_t; +typedef signed short int khronos_int16_t; +typedef unsigned short int khronos_uint16_t; + +/* + * Types that differ between LLP64 and LP64 architectures - in LLP64, + * pointers are 64 bits, but 'long' is still 32 bits. Win64 appears + * to be the only LLP64 architecture in current use. + */ +#ifdef KHRONOS_USE_INTPTR_T +typedef intptr_t khronos_intptr_t; +typedef uintptr_t khronos_uintptr_t; +#elif defined(_WIN64) +typedef signed long long int khronos_intptr_t; +typedef unsigned long long int khronos_uintptr_t; +#else +typedef signed long int khronos_intptr_t; +typedef unsigned long int khronos_uintptr_t; +#endif + +#if defined(_WIN64) +typedef signed long long int khronos_ssize_t; +typedef unsigned long long int khronos_usize_t; +#else +typedef signed long int khronos_ssize_t; +typedef unsigned long int khronos_usize_t; +#endif + +#if KHRONOS_SUPPORT_FLOAT +/* + * Float type + */ +typedef float khronos_float_t; +#endif + +#if KHRONOS_SUPPORT_INT64 +/* Time types + * + * These types can be used to represent a time interval in nanoseconds or + * an absolute Unadjusted System Time. Unadjusted System Time is the number + * of nanoseconds since some arbitrary system event (e.g. since the last + * time the system booted). The Unadjusted System Time is an unsigned + * 64 bit value that wraps back to 0 every 584 years. Time intervals + * may be either signed or unsigned. + */ +typedef khronos_uint64_t khronos_utime_nanoseconds_t; +typedef khronos_int64_t khronos_stime_nanoseconds_t; +#endif + +/* + * Dummy value used to pad enum types to 32 bits. + */ +#ifndef KHRONOS_MAX_ENUM +#define KHRONOS_MAX_ENUM 0x7FFFFFFF +#endif + +/* + * Enumerated boolean type + * + * Values other than zero should be considered to be true. Therefore + * comparisons should not be made against KHRONOS_TRUE. + */ +typedef enum { + KHRONOS_FALSE = 0, + KHRONOS_TRUE = 1, + KHRONOS_BOOLEAN_ENUM_FORCE_SIZE = KHRONOS_MAX_ENUM +} khronos_boolean_enum_t; + +#endif /* __khrplatform_h_ */ + +/* Generated C header for: + * API: gl + * Profile: compatibility + * Versions considered: .* + * Versions emitted: 1\.[2-9]|[234]\.[0-9] + * Default extensions included: gl + * Additional extensions included: _nomatch_^ + * Extensions removed: _nomatch_^ + */ + +#ifndef GL_VERSION_1_2 +#define GL_VERSION_1_2 1 +#define GL_UNSIGNED_BYTE_3_3_2 0x8032 +#define GL_UNSIGNED_SHORT_4_4_4_4 0x8033 +#define GL_UNSIGNED_SHORT_5_5_5_1 0x8034 +#define GL_UNSIGNED_INT_8_8_8_8 0x8035 +#define GL_UNSIGNED_INT_10_10_10_2 0x8036 +#define GL_TEXTURE_BINDING_3D 0x806A +#define GL_PACK_SKIP_IMAGES 0x806B +#define GL_PACK_IMAGE_HEIGHT 0x806C +#define GL_UNPACK_SKIP_IMAGES 0x806D +#define GL_UNPACK_IMAGE_HEIGHT 0x806E +#define GL_TEXTURE_3D 0x806F +#define GL_PROXY_TEXTURE_3D 0x8070 +#define GL_TEXTURE_DEPTH 0x8071 +#define GL_TEXTURE_WRAP_R 0x8072 +#define GL_MAX_3D_TEXTURE_SIZE 0x8073 +#define GL_UNSIGNED_BYTE_2_3_3_REV 0x8362 +#define GL_UNSIGNED_SHORT_5_6_5 0x8363 +#define GL_UNSIGNED_SHORT_5_6_5_REV 0x8364 +#define GL_UNSIGNED_SHORT_4_4_4_4_REV 0x8365 +#define GL_UNSIGNED_SHORT_1_5_5_5_REV 0x8366 +#define GL_UNSIGNED_INT_8_8_8_8_REV 0x8367 +#define GL_UNSIGNED_INT_2_10_10_10_REV 0x8368 +#define GL_BGR 0x80E0 +#define GL_BGRA 0x80E1 +#define GL_MAX_ELEMENTS_VERTICES 0x80E8 +#define GL_MAX_ELEMENTS_INDICES 0x80E9 +#define GL_CLAMP_TO_EDGE 0x812F +#define GL_TEXTURE_MIN_LOD 0x813A +#define GL_TEXTURE_MAX_LOD 0x813B +#define GL_TEXTURE_BASE_LEVEL 0x813C +#define GL_TEXTURE_MAX_LEVEL 0x813D +#define GL_SMOOTH_POINT_SIZE_RANGE 0x0B12 +#define GL_SMOOTH_POINT_SIZE_GRANULARITY 0x0B13 +#define GL_SMOOTH_LINE_WIDTH_RANGE 0x0B22 +#define GL_SMOOTH_LINE_WIDTH_GRANULARITY 0x0B23 +#define GL_ALIASED_LINE_WIDTH_RANGE 0x846E +#define GL_RESCALE_NORMAL 0x803A +#define GL_LIGHT_MODEL_COLOR_CONTROL 0x81F8 +#define GL_SINGLE_COLOR 0x81F9 +#define GL_SEPARATE_SPECULAR_COLOR 0x81FA +#define GL_ALIASED_POINT_SIZE_RANGE 0x846D +typedef void (APIENTRYP PFNGLDRAWRANGEELEMENTSPROC) (GLenum mode, GLuint start, GLuint end, GLsizei count, GLenum type, const void *indices); +typedef void (APIENTRYP PFNGLTEXIMAGE3DPROC) (GLenum target, GLint level, GLint internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLTEXSUBIMAGE3DPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLCOPYTEXSUBIMAGE3DPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint x, GLint y, GLsizei width, GLsizei height); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawRangeElements (GLenum mode, GLuint start, GLuint end, GLsizei count, GLenum type, const void *indices); +GLAPI void APIENTRY glTexImage3D (GLenum target, GLint level, GLint internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glTexSubImage3D (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glCopyTexSubImage3D (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint x, GLint y, GLsizei width, GLsizei height); +#endif +#endif /* GL_VERSION_1_2 */ + +#ifndef GL_VERSION_1_3 +#define GL_VERSION_1_3 1 +#define GL_TEXTURE0 0x84C0 +#define GL_TEXTURE1 0x84C1 +#define GL_TEXTURE2 0x84C2 +#define GL_TEXTURE3 0x84C3 +#define GL_TEXTURE4 0x84C4 +#define GL_TEXTURE5 0x84C5 +#define GL_TEXTURE6 0x84C6 +#define GL_TEXTURE7 0x84C7 +#define GL_TEXTURE8 0x84C8 +#define GL_TEXTURE9 0x84C9 +#define GL_TEXTURE10 0x84CA +#define GL_TEXTURE11 0x84CB +#define GL_TEXTURE12 0x84CC +#define GL_TEXTURE13 0x84CD +#define GL_TEXTURE14 0x84CE +#define GL_TEXTURE15 0x84CF +#define GL_TEXTURE16 0x84D0 +#define GL_TEXTURE17 0x84D1 +#define GL_TEXTURE18 0x84D2 +#define GL_TEXTURE19 0x84D3 +#define GL_TEXTURE20 0x84D4 +#define GL_TEXTURE21 0x84D5 +#define GL_TEXTURE22 0x84D6 +#define GL_TEXTURE23 0x84D7 +#define GL_TEXTURE24 0x84D8 +#define GL_TEXTURE25 0x84D9 +#define GL_TEXTURE26 0x84DA +#define GL_TEXTURE27 0x84DB +#define GL_TEXTURE28 0x84DC +#define GL_TEXTURE29 0x84DD +#define GL_TEXTURE30 0x84DE +#define GL_TEXTURE31 0x84DF +#define GL_ACTIVE_TEXTURE 0x84E0 +#define GL_MULTISAMPLE 0x809D +#define GL_SAMPLE_ALPHA_TO_COVERAGE 0x809E +#define GL_SAMPLE_ALPHA_TO_ONE 0x809F +#define GL_SAMPLE_COVERAGE 0x80A0 +#define GL_SAMPLE_BUFFERS 0x80A8 +#define GL_SAMPLES 0x80A9 +#define GL_SAMPLE_COVERAGE_VALUE 0x80AA +#define GL_SAMPLE_COVERAGE_INVERT 0x80AB +#define GL_TEXTURE_CUBE_MAP 0x8513 +#define GL_TEXTURE_BINDING_CUBE_MAP 0x8514 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_X 0x8515 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_X 0x8516 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Y 0x8517 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Y 0x8518 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Z 0x8519 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Z 0x851A +#define GL_PROXY_TEXTURE_CUBE_MAP 0x851B +#define GL_MAX_CUBE_MAP_TEXTURE_SIZE 0x851C +#define GL_COMPRESSED_RGB 0x84ED +#define GL_COMPRESSED_RGBA 0x84EE +#define GL_TEXTURE_COMPRESSION_HINT 0x84EF +#define GL_TEXTURE_COMPRESSED_IMAGE_SIZE 0x86A0 +#define GL_TEXTURE_COMPRESSED 0x86A1 +#define GL_NUM_COMPRESSED_TEXTURE_FORMATS 0x86A2 +#define GL_COMPRESSED_TEXTURE_FORMATS 0x86A3 +#define GL_CLAMP_TO_BORDER 0x812D +#define GL_CLIENT_ACTIVE_TEXTURE 0x84E1 +#define GL_MAX_TEXTURE_UNITS 0x84E2 +#define GL_TRANSPOSE_MODELVIEW_MATRIX 0x84E3 +#define GL_TRANSPOSE_PROJECTION_MATRIX 0x84E4 +#define GL_TRANSPOSE_TEXTURE_MATRIX 0x84E5 +#define GL_TRANSPOSE_COLOR_MATRIX 0x84E6 +#define GL_MULTISAMPLE_BIT 0x20000000 +#define GL_NORMAL_MAP 0x8511 +#define GL_REFLECTION_MAP 0x8512 +#define GL_COMPRESSED_ALPHA 0x84E9 +#define GL_COMPRESSED_LUMINANCE 0x84EA +#define GL_COMPRESSED_LUMINANCE_ALPHA 0x84EB +#define GL_COMPRESSED_INTENSITY 0x84EC +#define GL_COMBINE 0x8570 +#define GL_COMBINE_RGB 0x8571 +#define GL_COMBINE_ALPHA 0x8572 +#define GL_SOURCE0_RGB 0x8580 +#define GL_SOURCE1_RGB 0x8581 +#define GL_SOURCE2_RGB 0x8582 +#define GL_SOURCE0_ALPHA 0x8588 +#define GL_SOURCE1_ALPHA 0x8589 +#define GL_SOURCE2_ALPHA 0x858A +#define GL_OPERAND0_RGB 0x8590 +#define GL_OPERAND1_RGB 0x8591 +#define GL_OPERAND2_RGB 0x8592 +#define GL_OPERAND0_ALPHA 0x8598 +#define GL_OPERAND1_ALPHA 0x8599 +#define GL_OPERAND2_ALPHA 0x859A +#define GL_RGB_SCALE 0x8573 +#define GL_ADD_SIGNED 0x8574 +#define GL_INTERPOLATE 0x8575 +#define GL_SUBTRACT 0x84E7 +#define GL_CONSTANT 0x8576 +#define GL_PRIMARY_COLOR 0x8577 +#define GL_PREVIOUS 0x8578 +#define GL_DOT3_RGB 0x86AE +#define GL_DOT3_RGBA 0x86AF +typedef void (APIENTRYP PFNGLACTIVETEXTUREPROC) (GLenum texture); +typedef void (APIENTRYP PFNGLSAMPLECOVERAGEPROC) (GLfloat value, GLboolean invert); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXIMAGE3DPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLsizei imageSize, const void *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXIMAGE2DPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLsizei imageSize, const void *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXIMAGE1DPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLint border, GLsizei imageSize, const void *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE3DPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const void *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE2DPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const void *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE1DPROC) (GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLsizei imageSize, const void *data); +typedef void (APIENTRYP PFNGLGETCOMPRESSEDTEXIMAGEPROC) (GLenum target, GLint level, void *img); +typedef void (APIENTRYP PFNGLCLIENTACTIVETEXTUREPROC) (GLenum texture); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1DPROC) (GLenum target, GLdouble s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1DVPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1FPROC) (GLenum target, GLfloat s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1FVPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1IPROC) (GLenum target, GLint s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1IVPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1SPROC) (GLenum target, GLshort s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1SVPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2DPROC) (GLenum target, GLdouble s, GLdouble t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2DVPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2FPROC) (GLenum target, GLfloat s, GLfloat t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2FVPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2IPROC) (GLenum target, GLint s, GLint t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2IVPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2SPROC) (GLenum target, GLshort s, GLshort t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2SVPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3DPROC) (GLenum target, GLdouble s, GLdouble t, GLdouble r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3DVPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3FPROC) (GLenum target, GLfloat s, GLfloat t, GLfloat r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3FVPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3IPROC) (GLenum target, GLint s, GLint t, GLint r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3IVPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3SPROC) (GLenum target, GLshort s, GLshort t, GLshort r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3SVPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4DPROC) (GLenum target, GLdouble s, GLdouble t, GLdouble r, GLdouble q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4DVPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4FPROC) (GLenum target, GLfloat s, GLfloat t, GLfloat r, GLfloat q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4FVPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4IPROC) (GLenum target, GLint s, GLint t, GLint r, GLint q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4IVPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4SPROC) (GLenum target, GLshort s, GLshort t, GLshort r, GLshort q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4SVPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLLOADTRANSPOSEMATRIXFPROC) (const GLfloat *m); +typedef void (APIENTRYP PFNGLLOADTRANSPOSEMATRIXDPROC) (const GLdouble *m); +typedef void (APIENTRYP PFNGLMULTTRANSPOSEMATRIXFPROC) (const GLfloat *m); +typedef void (APIENTRYP PFNGLMULTTRANSPOSEMATRIXDPROC) (const GLdouble *m); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glActiveTexture (GLenum texture); +GLAPI void APIENTRY glSampleCoverage (GLfloat value, GLboolean invert); +GLAPI void APIENTRY glCompressedTexImage3D (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLsizei imageSize, const void *data); +GLAPI void APIENTRY glCompressedTexImage2D (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLsizei imageSize, const void *data); +GLAPI void APIENTRY glCompressedTexImage1D (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLint border, GLsizei imageSize, const void *data); +GLAPI void APIENTRY glCompressedTexSubImage3D (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const void *data); +GLAPI void APIENTRY glCompressedTexSubImage2D (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const void *data); +GLAPI void APIENTRY glCompressedTexSubImage1D (GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLsizei imageSize, const void *data); +GLAPI void APIENTRY glGetCompressedTexImage (GLenum target, GLint level, void *img); +GLAPI void APIENTRY glClientActiveTexture (GLenum texture); +GLAPI void APIENTRY glMultiTexCoord1d (GLenum target, GLdouble s); +GLAPI void APIENTRY glMultiTexCoord1dv (GLenum target, const GLdouble *v); +GLAPI void APIENTRY glMultiTexCoord1f (GLenum target, GLfloat s); +GLAPI void APIENTRY glMultiTexCoord1fv (GLenum target, const GLfloat *v); +GLAPI void APIENTRY glMultiTexCoord1i (GLenum target, GLint s); +GLAPI void APIENTRY glMultiTexCoord1iv (GLenum target, const GLint *v); +GLAPI void APIENTRY glMultiTexCoord1s (GLenum target, GLshort s); +GLAPI void APIENTRY glMultiTexCoord1sv (GLenum target, const GLshort *v); +GLAPI void APIENTRY glMultiTexCoord2d (GLenum target, GLdouble s, GLdouble t); +GLAPI void APIENTRY glMultiTexCoord2dv (GLenum target, const GLdouble *v); +GLAPI void APIENTRY glMultiTexCoord2f (GLenum target, GLfloat s, GLfloat t); +GLAPI void APIENTRY glMultiTexCoord2fv (GLenum target, const GLfloat *v); +GLAPI void APIENTRY glMultiTexCoord2i (GLenum target, GLint s, GLint t); +GLAPI void APIENTRY glMultiTexCoord2iv (GLenum target, const GLint *v); +GLAPI void APIENTRY glMultiTexCoord2s (GLenum target, GLshort s, GLshort t); +GLAPI void APIENTRY glMultiTexCoord2sv (GLenum target, const GLshort *v); +GLAPI void APIENTRY glMultiTexCoord3d (GLenum target, GLdouble s, GLdouble t, GLdouble r); +GLAPI void APIENTRY glMultiTexCoord3dv (GLenum target, const GLdouble *v); +GLAPI void APIENTRY glMultiTexCoord3f (GLenum target, GLfloat s, GLfloat t, GLfloat r); +GLAPI void APIENTRY glMultiTexCoord3fv (GLenum target, const GLfloat *v); +GLAPI void APIENTRY glMultiTexCoord3i (GLenum target, GLint s, GLint t, GLint r); +GLAPI void APIENTRY glMultiTexCoord3iv (GLenum target, const GLint *v); +GLAPI void APIENTRY glMultiTexCoord3s (GLenum target, GLshort s, GLshort t, GLshort r); +GLAPI void APIENTRY glMultiTexCoord3sv (GLenum target, const GLshort *v); +GLAPI void APIENTRY glMultiTexCoord4d (GLenum target, GLdouble s, GLdouble t, GLdouble r, GLdouble q); +GLAPI void APIENTRY glMultiTexCoord4dv (GLenum target, const GLdouble *v); +GLAPI void APIENTRY glMultiTexCoord4f (GLenum target, GLfloat s, GLfloat t, GLfloat r, GLfloat q); +GLAPI void APIENTRY glMultiTexCoord4fv (GLenum target, const GLfloat *v); +GLAPI void APIENTRY glMultiTexCoord4i (GLenum target, GLint s, GLint t, GLint r, GLint q); +GLAPI void APIENTRY glMultiTexCoord4iv (GLenum target, const GLint *v); +GLAPI void APIENTRY glMultiTexCoord4s (GLenum target, GLshort s, GLshort t, GLshort r, GLshort q); +GLAPI void APIENTRY glMultiTexCoord4sv (GLenum target, const GLshort *v); +GLAPI void APIENTRY glLoadTransposeMatrixf (const GLfloat *m); +GLAPI void APIENTRY glLoadTransposeMatrixd (const GLdouble *m); +GLAPI void APIENTRY glMultTransposeMatrixf (const GLfloat *m); +GLAPI void APIENTRY glMultTransposeMatrixd (const GLdouble *m); +#endif +#endif /* GL_VERSION_1_3 */ + +#ifndef GL_VERSION_1_4 +#define GL_VERSION_1_4 1 +#define GL_BLEND_DST_RGB 0x80C8 +#define GL_BLEND_SRC_RGB 0x80C9 +#define GL_BLEND_DST_ALPHA 0x80CA +#define GL_BLEND_SRC_ALPHA 0x80CB +#define GL_POINT_FADE_THRESHOLD_SIZE 0x8128 +#define GL_DEPTH_COMPONENT16 0x81A5 +#define GL_DEPTH_COMPONENT24 0x81A6 +#define GL_DEPTH_COMPONENT32 0x81A7 +#define GL_MIRRORED_REPEAT 0x8370 +#define GL_MAX_TEXTURE_LOD_BIAS 0x84FD +#define GL_TEXTURE_LOD_BIAS 0x8501 +#define GL_INCR_WRAP 0x8507 +#define GL_DECR_WRAP 0x8508 +#define GL_TEXTURE_DEPTH_SIZE 0x884A +#define GL_TEXTURE_COMPARE_MODE 0x884C +#define GL_TEXTURE_COMPARE_FUNC 0x884D +#define GL_POINT_SIZE_MIN 0x8126 +#define GL_POINT_SIZE_MAX 0x8127 +#define GL_POINT_DISTANCE_ATTENUATION 0x8129 +#define GL_GENERATE_MIPMAP 0x8191 +#define GL_GENERATE_MIPMAP_HINT 0x8192 +#define GL_FOG_COORDINATE_SOURCE 0x8450 +#define GL_FOG_COORDINATE 0x8451 +#define GL_FRAGMENT_DEPTH 0x8452 +#define GL_CURRENT_FOG_COORDINATE 0x8453 +#define GL_FOG_COORDINATE_ARRAY_TYPE 0x8454 +#define GL_FOG_COORDINATE_ARRAY_STRIDE 0x8455 +#define GL_FOG_COORDINATE_ARRAY_POINTER 0x8456 +#define GL_FOG_COORDINATE_ARRAY 0x8457 +#define GL_COLOR_SUM 0x8458 +#define GL_CURRENT_SECONDARY_COLOR 0x8459 +#define GL_SECONDARY_COLOR_ARRAY_SIZE 0x845A +#define GL_SECONDARY_COLOR_ARRAY_TYPE 0x845B +#define GL_SECONDARY_COLOR_ARRAY_STRIDE 0x845C +#define GL_SECONDARY_COLOR_ARRAY_POINTER 0x845D +#define GL_SECONDARY_COLOR_ARRAY 0x845E +#define GL_TEXTURE_FILTER_CONTROL 0x8500 +#define GL_DEPTH_TEXTURE_MODE 0x884B +#define GL_COMPARE_R_TO_TEXTURE 0x884E +#define GL_BLEND_COLOR 0x8005 +#define GL_BLEND_EQUATION 0x8009 +#define GL_CONSTANT_COLOR 0x8001 +#define GL_ONE_MINUS_CONSTANT_COLOR 0x8002 +#define GL_CONSTANT_ALPHA 0x8003 +#define GL_ONE_MINUS_CONSTANT_ALPHA 0x8004 +#define GL_FUNC_ADD 0x8006 +#define GL_FUNC_REVERSE_SUBTRACT 0x800B +#define GL_FUNC_SUBTRACT 0x800A +#define GL_MIN 0x8007 +#define GL_MAX 0x8008 +typedef void (APIENTRYP PFNGLBLENDFUNCSEPARATEPROC) (GLenum sfactorRGB, GLenum dfactorRGB, GLenum sfactorAlpha, GLenum dfactorAlpha); +typedef void (APIENTRYP PFNGLMULTIDRAWARRAYSPROC) (GLenum mode, const GLint *first, const GLsizei *count, GLsizei drawcount); +typedef void (APIENTRYP PFNGLMULTIDRAWELEMENTSPROC) (GLenum mode, const GLsizei *count, GLenum type, const void *const*indices, GLsizei drawcount); +typedef void (APIENTRYP PFNGLPOINTPARAMETERFPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLPOINTPARAMETERFVPROC) (GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLPOINTPARAMETERIPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLPOINTPARAMETERIVPROC) (GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLFOGCOORDFPROC) (GLfloat coord); +typedef void (APIENTRYP PFNGLFOGCOORDFVPROC) (const GLfloat *coord); +typedef void (APIENTRYP PFNGLFOGCOORDDPROC) (GLdouble coord); +typedef void (APIENTRYP PFNGLFOGCOORDDVPROC) (const GLdouble *coord); +typedef void (APIENTRYP PFNGLFOGCOORDPOINTERPROC) (GLenum type, GLsizei stride, const void *pointer); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3BPROC) (GLbyte red, GLbyte green, GLbyte blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3BVPROC) (const GLbyte *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3DPROC) (GLdouble red, GLdouble green, GLdouble blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3DVPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3FPROC) (GLfloat red, GLfloat green, GLfloat blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3FVPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3IPROC) (GLint red, GLint green, GLint blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3IVPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3SPROC) (GLshort red, GLshort green, GLshort blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3SVPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UBPROC) (GLubyte red, GLubyte green, GLubyte blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UBVPROC) (const GLubyte *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UIPROC) (GLuint red, GLuint green, GLuint blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UIVPROC) (const GLuint *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3USPROC) (GLushort red, GLushort green, GLushort blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3USVPROC) (const GLushort *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLORPOINTERPROC) (GLint size, GLenum type, GLsizei stride, const void *pointer); +typedef void (APIENTRYP PFNGLWINDOWPOS2DPROC) (GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLWINDOWPOS2DVPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2FPROC) (GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLWINDOWPOS2FVPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2IPROC) (GLint x, GLint y); +typedef void (APIENTRYP PFNGLWINDOWPOS2IVPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2SPROC) (GLshort x, GLshort y); +typedef void (APIENTRYP PFNGLWINDOWPOS2SVPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3DPROC) (GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLWINDOWPOS3DVPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3FPROC) (GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLWINDOWPOS3FVPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3IPROC) (GLint x, GLint y, GLint z); +typedef void (APIENTRYP PFNGLWINDOWPOS3IVPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3SPROC) (GLshort x, GLshort y, GLshort z); +typedef void (APIENTRYP PFNGLWINDOWPOS3SVPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLBLENDCOLORPROC) (GLfloat red, GLfloat green, GLfloat blue, GLfloat alpha); +typedef void (APIENTRYP PFNGLBLENDEQUATIONPROC) (GLenum mode); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendFuncSeparate (GLenum sfactorRGB, GLenum dfactorRGB, GLenum sfactorAlpha, GLenum dfactorAlpha); +GLAPI void APIENTRY glMultiDrawArrays (GLenum mode, const GLint *first, const GLsizei *count, GLsizei drawcount); +GLAPI void APIENTRY glMultiDrawElements (GLenum mode, const GLsizei *count, GLenum type, const void *const*indices, GLsizei drawcount); +GLAPI void APIENTRY glPointParameterf (GLenum pname, GLfloat param); +GLAPI void APIENTRY glPointParameterfv (GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glPointParameteri (GLenum pname, GLint param); +GLAPI void APIENTRY glPointParameteriv (GLenum pname, const GLint *params); +GLAPI void APIENTRY glFogCoordf (GLfloat coord); +GLAPI void APIENTRY glFogCoordfv (const GLfloat *coord); +GLAPI void APIENTRY glFogCoordd (GLdouble coord); +GLAPI void APIENTRY glFogCoorddv (const GLdouble *coord); +GLAPI void APIENTRY glFogCoordPointer (GLenum type, GLsizei stride, const void *pointer); +GLAPI void APIENTRY glSecondaryColor3b (GLbyte red, GLbyte green, GLbyte blue); +GLAPI void APIENTRY glSecondaryColor3bv (const GLbyte *v); +GLAPI void APIENTRY glSecondaryColor3d (GLdouble red, GLdouble green, GLdouble blue); +GLAPI void APIENTRY glSecondaryColor3dv (const GLdouble *v); +GLAPI void APIENTRY glSecondaryColor3f (GLfloat red, GLfloat green, GLfloat blue); +GLAPI void APIENTRY glSecondaryColor3fv (const GLfloat *v); +GLAPI void APIENTRY glSecondaryColor3i (GLint red, GLint green, GLint blue); +GLAPI void APIENTRY glSecondaryColor3iv (const GLint *v); +GLAPI void APIENTRY glSecondaryColor3s (GLshort red, GLshort green, GLshort blue); +GLAPI void APIENTRY glSecondaryColor3sv (const GLshort *v); +GLAPI void APIENTRY glSecondaryColor3ub (GLubyte red, GLubyte green, GLubyte blue); +GLAPI void APIENTRY glSecondaryColor3ubv (const GLubyte *v); +GLAPI void APIENTRY glSecondaryColor3ui (GLuint red, GLuint green, GLuint blue); +GLAPI void APIENTRY glSecondaryColor3uiv (const GLuint *v); +GLAPI void APIENTRY glSecondaryColor3us (GLushort red, GLushort green, GLushort blue); +GLAPI void APIENTRY glSecondaryColor3usv (const GLushort *v); +GLAPI void APIENTRY glSecondaryColorPointer (GLint size, GLenum type, GLsizei stride, const void *pointer); +GLAPI void APIENTRY glWindowPos2d (GLdouble x, GLdouble y); +GLAPI void APIENTRY glWindowPos2dv (const GLdouble *v); +GLAPI void APIENTRY glWindowPos2f (GLfloat x, GLfloat y); +GLAPI void APIENTRY glWindowPos2fv (const GLfloat *v); +GLAPI void APIENTRY glWindowPos2i (GLint x, GLint y); +GLAPI void APIENTRY glWindowPos2iv (const GLint *v); +GLAPI void APIENTRY glWindowPos2s (GLshort x, GLshort y); +GLAPI void APIENTRY glWindowPos2sv (const GLshort *v); +GLAPI void APIENTRY glWindowPos3d (GLdouble x, GLdouble y, GLdouble z); +GLAPI void APIENTRY glWindowPos3dv (const GLdouble *v); +GLAPI void APIENTRY glWindowPos3f (GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glWindowPos3fv (const GLfloat *v); +GLAPI void APIENTRY glWindowPos3i (GLint x, GLint y, GLint z); +GLAPI void APIENTRY glWindowPos3iv (const GLint *v); +GLAPI void APIENTRY glWindowPos3s (GLshort x, GLshort y, GLshort z); +GLAPI void APIENTRY glWindowPos3sv (const GLshort *v); +GLAPI void APIENTRY glBlendColor (GLfloat red, GLfloat green, GLfloat blue, GLfloat alpha); +GLAPI void APIENTRY glBlendEquation (GLenum mode); +#endif +#endif /* GL_VERSION_1_4 */ + +#ifndef GL_VERSION_1_5 +#define GL_VERSION_1_5 1 +typedef khronos_ssize_t GLsizeiptr; +typedef khronos_intptr_t GLintptr; +#define GL_BUFFER_SIZE 0x8764 +#define GL_BUFFER_USAGE 0x8765 +#define GL_QUERY_COUNTER_BITS 0x8864 +#define GL_CURRENT_QUERY 0x8865 +#define GL_QUERY_RESULT 0x8866 +#define GL_QUERY_RESULT_AVAILABLE 0x8867 +#define GL_ARRAY_BUFFER 0x8892 +#define GL_ELEMENT_ARRAY_BUFFER 0x8893 +#define GL_ARRAY_BUFFER_BINDING 0x8894 +#define GL_ELEMENT_ARRAY_BUFFER_BINDING 0x8895 +#define GL_VERTEX_ATTRIB_ARRAY_BUFFER_BINDING 0x889F +#define GL_READ_ONLY 0x88B8 +#define GL_WRITE_ONLY 0x88B9 +#define GL_READ_WRITE 0x88BA +#define GL_BUFFER_ACCESS 0x88BB +#define GL_BUFFER_MAPPED 0x88BC +#define GL_BUFFER_MAP_POINTER 0x88BD +#define GL_STREAM_DRAW 0x88E0 +#define GL_STREAM_READ 0x88E1 +#define GL_STREAM_COPY 0x88E2 +#define GL_STATIC_DRAW 0x88E4 +#define GL_STATIC_READ 0x88E5 +#define GL_STATIC_COPY 0x88E6 +#define GL_DYNAMIC_DRAW 0x88E8 +#define GL_DYNAMIC_READ 0x88E9 +#define GL_DYNAMIC_COPY 0x88EA +#define GL_SAMPLES_PASSED 0x8914 +#define GL_SRC1_ALPHA 0x8589 +#define GL_VERTEX_ARRAY_BUFFER_BINDING 0x8896 +#define GL_NORMAL_ARRAY_BUFFER_BINDING 0x8897 +#define GL_COLOR_ARRAY_BUFFER_BINDING 0x8898 +#define GL_INDEX_ARRAY_BUFFER_BINDING 0x8899 +#define GL_TEXTURE_COORD_ARRAY_BUFFER_BINDING 0x889A +#define GL_EDGE_FLAG_ARRAY_BUFFER_BINDING 0x889B +#define GL_SECONDARY_COLOR_ARRAY_BUFFER_BINDING 0x889C +#define GL_FOG_COORDINATE_ARRAY_BUFFER_BINDING 0x889D +#define GL_WEIGHT_ARRAY_BUFFER_BINDING 0x889E +#define GL_FOG_COORD_SRC 0x8450 +#define GL_FOG_COORD 0x8451 +#define GL_CURRENT_FOG_COORD 0x8453 +#define GL_FOG_COORD_ARRAY_TYPE 0x8454 +#define GL_FOG_COORD_ARRAY_STRIDE 0x8455 +#define GL_FOG_COORD_ARRAY_POINTER 0x8456 +#define GL_FOG_COORD_ARRAY 0x8457 +#define GL_FOG_COORD_ARRAY_BUFFER_BINDING 0x889D +#define GL_SRC0_RGB 0x8580 +#define GL_SRC1_RGB 0x8581 +#define GL_SRC2_RGB 0x8582 +#define GL_SRC0_ALPHA 0x8588 +#define GL_SRC2_ALPHA 0x858A +typedef void (APIENTRYP PFNGLGENQUERIESPROC) (GLsizei n, GLuint *ids); +typedef void (APIENTRYP PFNGLDELETEQUERIESPROC) (GLsizei n, const GLuint *ids); +typedef GLboolean (APIENTRYP PFNGLISQUERYPROC) (GLuint id); +typedef void (APIENTRYP PFNGLBEGINQUERYPROC) (GLenum target, GLuint id); +typedef void (APIENTRYP PFNGLENDQUERYPROC) (GLenum target); +typedef void (APIENTRYP PFNGLGETQUERYIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETQUERYOBJECTIVPROC) (GLuint id, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETQUERYOBJECTUIVPROC) (GLuint id, GLenum pname, GLuint *params); +typedef void (APIENTRYP PFNGLBINDBUFFERPROC) (GLenum target, GLuint buffer); +typedef void (APIENTRYP PFNGLDELETEBUFFERSPROC) (GLsizei n, const GLuint *buffers); +typedef void (APIENTRYP PFNGLGENBUFFERSPROC) (GLsizei n, GLuint *buffers); +typedef GLboolean (APIENTRYP PFNGLISBUFFERPROC) (GLuint buffer); +typedef void (APIENTRYP PFNGLBUFFERDATAPROC) (GLenum target, GLsizeiptr size, const void *data, GLenum usage); +typedef void (APIENTRYP PFNGLBUFFERSUBDATAPROC) (GLenum target, GLintptr offset, GLsizeiptr size, const void *data); +typedef void (APIENTRYP PFNGLGETBUFFERSUBDATAPROC) (GLenum target, GLintptr offset, GLsizeiptr size, void *data); +typedef void *(APIENTRYP PFNGLMAPBUFFERPROC) (GLenum target, GLenum access); +typedef GLboolean (APIENTRYP PFNGLUNMAPBUFFERPROC) (GLenum target); +typedef void (APIENTRYP PFNGLGETBUFFERPARAMETERIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETBUFFERPOINTERVPROC) (GLenum target, GLenum pname, void **params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGenQueries (GLsizei n, GLuint *ids); +GLAPI void APIENTRY glDeleteQueries (GLsizei n, const GLuint *ids); +GLAPI GLboolean APIENTRY glIsQuery (GLuint id); +GLAPI void APIENTRY glBeginQuery (GLenum target, GLuint id); +GLAPI void APIENTRY glEndQuery (GLenum target); +GLAPI void APIENTRY glGetQueryiv (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetQueryObjectiv (GLuint id, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetQueryObjectuiv (GLuint id, GLenum pname, GLuint *params); +GLAPI void APIENTRY glBindBuffer (GLenum target, GLuint buffer); +GLAPI void APIENTRY glDeleteBuffers (GLsizei n, const GLuint *buffers); +GLAPI void APIENTRY glGenBuffers (GLsizei n, GLuint *buffers); +GLAPI GLboolean APIENTRY glIsBuffer (GLuint buffer); +GLAPI void APIENTRY glBufferData (GLenum target, GLsizeiptr size, const void *data, GLenum usage); +GLAPI void APIENTRY glBufferSubData (GLenum target, GLintptr offset, GLsizeiptr size, const void *data); +GLAPI void APIENTRY glGetBufferSubData (GLenum target, GLintptr offset, GLsizeiptr size, void *data); +GLAPI void *APIENTRY glMapBuffer (GLenum target, GLenum access); +GLAPI GLboolean APIENTRY glUnmapBuffer (GLenum target); +GLAPI void APIENTRY glGetBufferParameteriv (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetBufferPointerv (GLenum target, GLenum pname, void **params); +#endif +#endif /* GL_VERSION_1_5 */ + +#ifndef GL_VERSION_2_0 +#define GL_VERSION_2_0 1 +typedef char GLchar; +#define GL_BLEND_EQUATION_RGB 0x8009 +#define GL_VERTEX_ATTRIB_ARRAY_ENABLED 0x8622 +#define GL_VERTEX_ATTRIB_ARRAY_SIZE 0x8623 +#define GL_VERTEX_ATTRIB_ARRAY_STRIDE 0x8624 +#define GL_VERTEX_ATTRIB_ARRAY_TYPE 0x8625 +#define GL_CURRENT_VERTEX_ATTRIB 0x8626 +#define GL_VERTEX_PROGRAM_POINT_SIZE 0x8642 +#define GL_VERTEX_ATTRIB_ARRAY_POINTER 0x8645 +#define GL_STENCIL_BACK_FUNC 0x8800 +#define GL_STENCIL_BACK_FAIL 0x8801 +#define GL_STENCIL_BACK_PASS_DEPTH_FAIL 0x8802 +#define GL_STENCIL_BACK_PASS_DEPTH_PASS 0x8803 +#define GL_MAX_DRAW_BUFFERS 0x8824 +#define GL_DRAW_BUFFER0 0x8825 +#define GL_DRAW_BUFFER1 0x8826 +#define GL_DRAW_BUFFER2 0x8827 +#define GL_DRAW_BUFFER3 0x8828 +#define GL_DRAW_BUFFER4 0x8829 +#define GL_DRAW_BUFFER5 0x882A +#define GL_DRAW_BUFFER6 0x882B +#define GL_DRAW_BUFFER7 0x882C +#define GL_DRAW_BUFFER8 0x882D +#define GL_DRAW_BUFFER9 0x882E +#define GL_DRAW_BUFFER10 0x882F +#define GL_DRAW_BUFFER11 0x8830 +#define GL_DRAW_BUFFER12 0x8831 +#define GL_DRAW_BUFFER13 0x8832 +#define GL_DRAW_BUFFER14 0x8833 +#define GL_DRAW_BUFFER15 0x8834 +#define GL_BLEND_EQUATION_ALPHA 0x883D +#define GL_MAX_VERTEX_ATTRIBS 0x8869 +#define GL_VERTEX_ATTRIB_ARRAY_NORMALIZED 0x886A +#define GL_MAX_TEXTURE_IMAGE_UNITS 0x8872 +#define GL_FRAGMENT_SHADER 0x8B30 +#define GL_VERTEX_SHADER 0x8B31 +#define GL_MAX_FRAGMENT_UNIFORM_COMPONENTS 0x8B49 +#define GL_MAX_VERTEX_UNIFORM_COMPONENTS 0x8B4A +#define GL_MAX_VARYING_FLOATS 0x8B4B +#define GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS 0x8B4C +#define GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS 0x8B4D +#define GL_SHADER_TYPE 0x8B4F +#define GL_FLOAT_VEC2 0x8B50 +#define GL_FLOAT_VEC3 0x8B51 +#define GL_FLOAT_VEC4 0x8B52 +#define GL_INT_VEC2 0x8B53 +#define GL_INT_VEC3 0x8B54 +#define GL_INT_VEC4 0x8B55 +#define GL_BOOL 0x8B56 +#define GL_BOOL_VEC2 0x8B57 +#define GL_BOOL_VEC3 0x8B58 +#define GL_BOOL_VEC4 0x8B59 +#define GL_FLOAT_MAT2 0x8B5A +#define GL_FLOAT_MAT3 0x8B5B +#define GL_FLOAT_MAT4 0x8B5C +#define GL_SAMPLER_1D 0x8B5D +#define GL_SAMPLER_2D 0x8B5E +#define GL_SAMPLER_3D 0x8B5F +#define GL_SAMPLER_CUBE 0x8B60 +#define GL_SAMPLER_1D_SHADOW 0x8B61 +#define GL_SAMPLER_2D_SHADOW 0x8B62 +#define GL_DELETE_STATUS 0x8B80 +#define GL_COMPILE_STATUS 0x8B81 +#define GL_LINK_STATUS 0x8B82 +#define GL_VALIDATE_STATUS 0x8B83 +#define GL_INFO_LOG_LENGTH 0x8B84 +#define GL_ATTACHED_SHADERS 0x8B85 +#define GL_ACTIVE_UNIFORMS 0x8B86 +#define GL_ACTIVE_UNIFORM_MAX_LENGTH 0x8B87 +#define GL_SHADER_SOURCE_LENGTH 0x8B88 +#define GL_ACTIVE_ATTRIBUTES 0x8B89 +#define GL_ACTIVE_ATTRIBUTE_MAX_LENGTH 0x8B8A +#define GL_FRAGMENT_SHADER_DERIVATIVE_HINT 0x8B8B +#define GL_SHADING_LANGUAGE_VERSION 0x8B8C +#define GL_CURRENT_PROGRAM 0x8B8D +#define GL_POINT_SPRITE_COORD_ORIGIN 0x8CA0 +#define GL_LOWER_LEFT 0x8CA1 +#define GL_UPPER_LEFT 0x8CA2 +#define GL_STENCIL_BACK_REF 0x8CA3 +#define GL_STENCIL_BACK_VALUE_MASK 0x8CA4 +#define GL_STENCIL_BACK_WRITEMASK 0x8CA5 +#define GL_VERTEX_PROGRAM_TWO_SIDE 0x8643 +#define GL_POINT_SPRITE 0x8861 +#define GL_COORD_REPLACE 0x8862 +#define GL_MAX_TEXTURE_COORDS 0x8871 +typedef void (APIENTRYP PFNGLBLENDEQUATIONSEPARATEPROC) (GLenum modeRGB, GLenum modeAlpha); +typedef void (APIENTRYP PFNGLDRAWBUFFERSPROC) (GLsizei n, const GLenum *bufs); +typedef void (APIENTRYP PFNGLSTENCILOPSEPARATEPROC) (GLenum face, GLenum sfail, GLenum dpfail, GLenum dppass); +typedef void (APIENTRYP PFNGLSTENCILFUNCSEPARATEPROC) (GLenum face, GLenum func, GLint ref, GLuint mask); +typedef void (APIENTRYP PFNGLSTENCILMASKSEPARATEPROC) (GLenum face, GLuint mask); +typedef void (APIENTRYP PFNGLATTACHSHADERPROC) (GLuint program, GLuint shader); +typedef void (APIENTRYP PFNGLBINDATTRIBLOCATIONPROC) (GLuint program, GLuint index, const GLchar *name); +typedef void (APIENTRYP PFNGLCOMPILESHADERPROC) (GLuint shader); +typedef GLuint (APIENTRYP PFNGLCREATEPROGRAMPROC) (void); +typedef GLuint (APIENTRYP PFNGLCREATESHADERPROC) (GLenum type); +typedef void (APIENTRYP PFNGLDELETEPROGRAMPROC) (GLuint program); +typedef void (APIENTRYP PFNGLDELETESHADERPROC) (GLuint shader); +typedef void (APIENTRYP PFNGLDETACHSHADERPROC) (GLuint program, GLuint shader); +typedef void (APIENTRYP PFNGLDISABLEVERTEXATTRIBARRAYPROC) (GLuint index); +typedef void (APIENTRYP PFNGLENABLEVERTEXATTRIBARRAYPROC) (GLuint index); +typedef void (APIENTRYP PFNGLGETACTIVEATTRIBPROC) (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLint *size, GLenum *type, GLchar *name); +typedef void (APIENTRYP PFNGLGETACTIVEUNIFORMPROC) (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLint *size, GLenum *type, GLchar *name); +typedef void (APIENTRYP PFNGLGETATTACHEDSHADERSPROC) (GLuint program, GLsizei maxCount, GLsizei *count, GLuint *shaders); +typedef GLint (APIENTRYP PFNGLGETATTRIBLOCATIONPROC) (GLuint program, const GLchar *name); +typedef void (APIENTRYP PFNGLGETPROGRAMIVPROC) (GLuint program, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETPROGRAMINFOLOGPROC) (GLuint program, GLsizei bufSize, GLsizei *length, GLchar *infoLog); +typedef void (APIENTRYP PFNGLGETSHADERIVPROC) (GLuint shader, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETSHADERINFOLOGPROC) (GLuint shader, GLsizei bufSize, GLsizei *length, GLchar *infoLog); +typedef void (APIENTRYP PFNGLGETSHADERSOURCEPROC) (GLuint shader, GLsizei bufSize, GLsizei *length, GLchar *source); +typedef GLint (APIENTRYP PFNGLGETUNIFORMLOCATIONPROC) (GLuint program, const GLchar *name); +typedef void (APIENTRYP PFNGLGETUNIFORMFVPROC) (GLuint program, GLint location, GLfloat *params); +typedef void (APIENTRYP PFNGLGETUNIFORMIVPROC) (GLuint program, GLint location, GLint *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBDVPROC) (GLuint index, GLenum pname, GLdouble *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBFVPROC) (GLuint index, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBIVPROC) (GLuint index, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBPOINTERVPROC) (GLuint index, GLenum pname, void **pointer); +typedef GLboolean (APIENTRYP PFNGLISPROGRAMPROC) (GLuint program); +typedef GLboolean (APIENTRYP PFNGLISSHADERPROC) (GLuint shader); +typedef void (APIENTRYP PFNGLLINKPROGRAMPROC) (GLuint program); +typedef void (APIENTRYP PFNGLSHADERSOURCEPROC) (GLuint shader, GLsizei count, const GLchar *const*string, const GLint *length); +typedef void (APIENTRYP PFNGLUSEPROGRAMPROC) (GLuint program); +typedef void (APIENTRYP PFNGLUNIFORM1FPROC) (GLint location, GLfloat v0); +typedef void (APIENTRYP PFNGLUNIFORM2FPROC) (GLint location, GLfloat v0, GLfloat v1); +typedef void (APIENTRYP PFNGLUNIFORM3FPROC) (GLint location, GLfloat v0, GLfloat v1, GLfloat v2); +typedef void (APIENTRYP PFNGLUNIFORM4FPROC) (GLint location, GLfloat v0, GLfloat v1, GLfloat v2, GLfloat v3); +typedef void (APIENTRYP PFNGLUNIFORM1IPROC) (GLint location, GLint v0); +typedef void (APIENTRYP PFNGLUNIFORM2IPROC) (GLint location, GLint v0, GLint v1); +typedef void (APIENTRYP PFNGLUNIFORM3IPROC) (GLint location, GLint v0, GLint v1, GLint v2); +typedef void (APIENTRYP PFNGLUNIFORM4IPROC) (GLint location, GLint v0, GLint v1, GLint v2, GLint v3); +typedef void (APIENTRYP PFNGLUNIFORM1FVPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM2FVPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM3FVPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM4FVPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM1IVPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORM2IVPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORM3IVPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORM4IVPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX2FVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX3FVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX4FVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLVALIDATEPROGRAMPROC) (GLuint program); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1DPROC) (GLuint index, GLdouble x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1DVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1FPROC) (GLuint index, GLfloat x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1FVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1SPROC) (GLuint index, GLshort x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1SVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2DPROC) (GLuint index, GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2DVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2FPROC) (GLuint index, GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2FVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2SPROC) (GLuint index, GLshort x, GLshort y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2SVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3DPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3DVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3FPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3FVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3SPROC) (GLuint index, GLshort x, GLshort y, GLshort z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3SVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NBVPROC) (GLuint index, const GLbyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NIVPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NSVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUBPROC) (GLuint index, GLubyte x, GLubyte y, GLubyte z, GLubyte w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUBVPROC) (GLuint index, const GLubyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUIVPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUSVPROC) (GLuint index, const GLushort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4BVPROC) (GLuint index, const GLbyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4DPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4DVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4FPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4FVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4IVPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4SPROC) (GLuint index, GLshort x, GLshort y, GLshort z, GLshort w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4SVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4UBVPROC) (GLuint index, const GLubyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4UIVPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4USVPROC) (GLuint index, const GLushort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBPOINTERPROC) (GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride, const void *pointer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendEquationSeparate (GLenum modeRGB, GLenum modeAlpha); +GLAPI void APIENTRY glDrawBuffers (GLsizei n, const GLenum *bufs); +GLAPI void APIENTRY glStencilOpSeparate (GLenum face, GLenum sfail, GLenum dpfail, GLenum dppass); +GLAPI void APIENTRY glStencilFuncSeparate (GLenum face, GLenum func, GLint ref, GLuint mask); +GLAPI void APIENTRY glStencilMaskSeparate (GLenum face, GLuint mask); +GLAPI void APIENTRY glAttachShader (GLuint program, GLuint shader); +GLAPI void APIENTRY glBindAttribLocation (GLuint program, GLuint index, const GLchar *name); +GLAPI void APIENTRY glCompileShader (GLuint shader); +GLAPI GLuint APIENTRY glCreateProgram (void); +GLAPI GLuint APIENTRY glCreateShader (GLenum type); +GLAPI void APIENTRY glDeleteProgram (GLuint program); +GLAPI void APIENTRY glDeleteShader (GLuint shader); +GLAPI void APIENTRY glDetachShader (GLuint program, GLuint shader); +GLAPI void APIENTRY glDisableVertexAttribArray (GLuint index); +GLAPI void APIENTRY glEnableVertexAttribArray (GLuint index); +GLAPI void APIENTRY glGetActiveAttrib (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLint *size, GLenum *type, GLchar *name); +GLAPI void APIENTRY glGetActiveUniform (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLint *size, GLenum *type, GLchar *name); +GLAPI void APIENTRY glGetAttachedShaders (GLuint program, GLsizei maxCount, GLsizei *count, GLuint *shaders); +GLAPI GLint APIENTRY glGetAttribLocation (GLuint program, const GLchar *name); +GLAPI void APIENTRY glGetProgramiv (GLuint program, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetProgramInfoLog (GLuint program, GLsizei bufSize, GLsizei *length, GLchar *infoLog); +GLAPI void APIENTRY glGetShaderiv (GLuint shader, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetShaderInfoLog (GLuint shader, GLsizei bufSize, GLsizei *length, GLchar *infoLog); +GLAPI void APIENTRY glGetShaderSource (GLuint shader, GLsizei bufSize, GLsizei *length, GLchar *source); +GLAPI GLint APIENTRY glGetUniformLocation (GLuint program, const GLchar *name); +GLAPI void APIENTRY glGetUniformfv (GLuint program, GLint location, GLfloat *params); +GLAPI void APIENTRY glGetUniformiv (GLuint program, GLint location, GLint *params); +GLAPI void APIENTRY glGetVertexAttribdv (GLuint index, GLenum pname, GLdouble *params); +GLAPI void APIENTRY glGetVertexAttribfv (GLuint index, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetVertexAttribiv (GLuint index, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetVertexAttribPointerv (GLuint index, GLenum pname, void **pointer); +GLAPI GLboolean APIENTRY glIsProgram (GLuint program); +GLAPI GLboolean APIENTRY glIsShader (GLuint shader); +GLAPI void APIENTRY glLinkProgram (GLuint program); +GLAPI void APIENTRY glShaderSource (GLuint shader, GLsizei count, const GLchar *const*string, const GLint *length); +GLAPI void APIENTRY glUseProgram (GLuint program); +GLAPI void APIENTRY glUniform1f (GLint location, GLfloat v0); +GLAPI void APIENTRY glUniform2f (GLint location, GLfloat v0, GLfloat v1); +GLAPI void APIENTRY glUniform3f (GLint location, GLfloat v0, GLfloat v1, GLfloat v2); +GLAPI void APIENTRY glUniform4f (GLint location, GLfloat v0, GLfloat v1, GLfloat v2, GLfloat v3); +GLAPI void APIENTRY glUniform1i (GLint location, GLint v0); +GLAPI void APIENTRY glUniform2i (GLint location, GLint v0, GLint v1); +GLAPI void APIENTRY glUniform3i (GLint location, GLint v0, GLint v1, GLint v2); +GLAPI void APIENTRY glUniform4i (GLint location, GLint v0, GLint v1, GLint v2, GLint v3); +GLAPI void APIENTRY glUniform1fv (GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glUniform2fv (GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glUniform3fv (GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glUniform4fv (GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glUniform1iv (GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glUniform2iv (GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glUniform3iv (GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glUniform4iv (GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glUniformMatrix2fv (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glUniformMatrix3fv (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glUniformMatrix4fv (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glValidateProgram (GLuint program); +GLAPI void APIENTRY glVertexAttrib1d (GLuint index, GLdouble x); +GLAPI void APIENTRY glVertexAttrib1dv (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttrib1f (GLuint index, GLfloat x); +GLAPI void APIENTRY glVertexAttrib1fv (GLuint index, const GLfloat *v); +GLAPI void APIENTRY glVertexAttrib1s (GLuint index, GLshort x); +GLAPI void APIENTRY glVertexAttrib1sv (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttrib2d (GLuint index, GLdouble x, GLdouble y); +GLAPI void APIENTRY glVertexAttrib2dv (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttrib2f (GLuint index, GLfloat x, GLfloat y); +GLAPI void APIENTRY glVertexAttrib2fv (GLuint index, const GLfloat *v); +GLAPI void APIENTRY glVertexAttrib2s (GLuint index, GLshort x, GLshort y); +GLAPI void APIENTRY glVertexAttrib2sv (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttrib3d (GLuint index, GLdouble x, GLdouble y, GLdouble z); +GLAPI void APIENTRY glVertexAttrib3dv (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttrib3f (GLuint index, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glVertexAttrib3fv (GLuint index, const GLfloat *v); +GLAPI void APIENTRY glVertexAttrib3s (GLuint index, GLshort x, GLshort y, GLshort z); +GLAPI void APIENTRY glVertexAttrib3sv (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttrib4Nbv (GLuint index, const GLbyte *v); +GLAPI void APIENTRY glVertexAttrib4Niv (GLuint index, const GLint *v); +GLAPI void APIENTRY glVertexAttrib4Nsv (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttrib4Nub (GLuint index, GLubyte x, GLubyte y, GLubyte z, GLubyte w); +GLAPI void APIENTRY glVertexAttrib4Nubv (GLuint index, const GLubyte *v); +GLAPI void APIENTRY glVertexAttrib4Nuiv (GLuint index, const GLuint *v); +GLAPI void APIENTRY glVertexAttrib4Nusv (GLuint index, const GLushort *v); +GLAPI void APIENTRY glVertexAttrib4bv (GLuint index, const GLbyte *v); +GLAPI void APIENTRY glVertexAttrib4d (GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +GLAPI void APIENTRY glVertexAttrib4dv (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttrib4f (GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +GLAPI void APIENTRY glVertexAttrib4fv (GLuint index, const GLfloat *v); +GLAPI void APIENTRY glVertexAttrib4iv (GLuint index, const GLint *v); +GLAPI void APIENTRY glVertexAttrib4s (GLuint index, GLshort x, GLshort y, GLshort z, GLshort w); +GLAPI void APIENTRY glVertexAttrib4sv (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttrib4ubv (GLuint index, const GLubyte *v); +GLAPI void APIENTRY glVertexAttrib4uiv (GLuint index, const GLuint *v); +GLAPI void APIENTRY glVertexAttrib4usv (GLuint index, const GLushort *v); +GLAPI void APIENTRY glVertexAttribPointer (GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride, const void *pointer); +#endif +#endif /* GL_VERSION_2_0 */ + +#ifndef GL_VERSION_2_1 +#define GL_VERSION_2_1 1 +#define GL_PIXEL_PACK_BUFFER 0x88EB +#define GL_PIXEL_UNPACK_BUFFER 0x88EC +#define GL_PIXEL_PACK_BUFFER_BINDING 0x88ED +#define GL_PIXEL_UNPACK_BUFFER_BINDING 0x88EF +#define GL_FLOAT_MAT2x3 0x8B65 +#define GL_FLOAT_MAT2x4 0x8B66 +#define GL_FLOAT_MAT3x2 0x8B67 +#define GL_FLOAT_MAT3x4 0x8B68 +#define GL_FLOAT_MAT4x2 0x8B69 +#define GL_FLOAT_MAT4x3 0x8B6A +#define GL_SRGB 0x8C40 +#define GL_SRGB8 0x8C41 +#define GL_SRGB_ALPHA 0x8C42 +#define GL_SRGB8_ALPHA8 0x8C43 +#define GL_COMPRESSED_SRGB 0x8C48 +#define GL_COMPRESSED_SRGB_ALPHA 0x8C49 +#define GL_CURRENT_RASTER_SECONDARY_COLOR 0x845F +#define GL_SLUMINANCE_ALPHA 0x8C44 +#define GL_SLUMINANCE8_ALPHA8 0x8C45 +#define GL_SLUMINANCE 0x8C46 +#define GL_SLUMINANCE8 0x8C47 +#define GL_COMPRESSED_SLUMINANCE 0x8C4A +#define GL_COMPRESSED_SLUMINANCE_ALPHA 0x8C4B +typedef void (APIENTRYP PFNGLUNIFORMMATRIX2X3FVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX3X2FVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX2X4FVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX4X2FVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX3X4FVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX4X3FVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glUniformMatrix2x3fv (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glUniformMatrix3x2fv (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glUniformMatrix2x4fv (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glUniformMatrix4x2fv (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glUniformMatrix3x4fv (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glUniformMatrix4x3fv (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +#endif +#endif /* GL_VERSION_2_1 */ + +#ifndef GL_VERSION_3_0 +#define GL_VERSION_3_0 1 +typedef khronos_uint16_t GLhalf; +#define GL_COMPARE_REF_TO_TEXTURE 0x884E +#define GL_CLIP_DISTANCE0 0x3000 +#define GL_CLIP_DISTANCE1 0x3001 +#define GL_CLIP_DISTANCE2 0x3002 +#define GL_CLIP_DISTANCE3 0x3003 +#define GL_CLIP_DISTANCE4 0x3004 +#define GL_CLIP_DISTANCE5 0x3005 +#define GL_CLIP_DISTANCE6 0x3006 +#define GL_CLIP_DISTANCE7 0x3007 +#define GL_MAX_CLIP_DISTANCES 0x0D32 +#define GL_MAJOR_VERSION 0x821B +#define GL_MINOR_VERSION 0x821C +#define GL_NUM_EXTENSIONS 0x821D +#define GL_CONTEXT_FLAGS 0x821E +#define GL_COMPRESSED_RED 0x8225 +#define GL_COMPRESSED_RG 0x8226 +#define GL_CONTEXT_FLAG_FORWARD_COMPATIBLE_BIT 0x00000001 +#define GL_RGBA32F 0x8814 +#define GL_RGB32F 0x8815 +#define GL_RGBA16F 0x881A +#define GL_RGB16F 0x881B +#define GL_VERTEX_ATTRIB_ARRAY_INTEGER 0x88FD +#define GL_MAX_ARRAY_TEXTURE_LAYERS 0x88FF +#define GL_MIN_PROGRAM_TEXEL_OFFSET 0x8904 +#define GL_MAX_PROGRAM_TEXEL_OFFSET 0x8905 +#define GL_CLAMP_READ_COLOR 0x891C +#define GL_FIXED_ONLY 0x891D +#define GL_MAX_VARYING_COMPONENTS 0x8B4B +#define GL_TEXTURE_1D_ARRAY 0x8C18 +#define GL_PROXY_TEXTURE_1D_ARRAY 0x8C19 +#define GL_TEXTURE_2D_ARRAY 0x8C1A +#define GL_PROXY_TEXTURE_2D_ARRAY 0x8C1B +#define GL_TEXTURE_BINDING_1D_ARRAY 0x8C1C +#define GL_TEXTURE_BINDING_2D_ARRAY 0x8C1D +#define GL_R11F_G11F_B10F 0x8C3A +#define GL_UNSIGNED_INT_10F_11F_11F_REV 0x8C3B +#define GL_RGB9_E5 0x8C3D +#define GL_UNSIGNED_INT_5_9_9_9_REV 0x8C3E +#define GL_TEXTURE_SHARED_SIZE 0x8C3F +#define GL_TRANSFORM_FEEDBACK_VARYING_MAX_LENGTH 0x8C76 +#define GL_TRANSFORM_FEEDBACK_BUFFER_MODE 0x8C7F +#define GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_COMPONENTS 0x8C80 +#define GL_TRANSFORM_FEEDBACK_VARYINGS 0x8C83 +#define GL_TRANSFORM_FEEDBACK_BUFFER_START 0x8C84 +#define GL_TRANSFORM_FEEDBACK_BUFFER_SIZE 0x8C85 +#define GL_PRIMITIVES_GENERATED 0x8C87 +#define GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN 0x8C88 +#define GL_RASTERIZER_DISCARD 0x8C89 +#define GL_MAX_TRANSFORM_FEEDBACK_INTERLEAVED_COMPONENTS 0x8C8A +#define GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_ATTRIBS 0x8C8B +#define GL_INTERLEAVED_ATTRIBS 0x8C8C +#define GL_SEPARATE_ATTRIBS 0x8C8D +#define GL_TRANSFORM_FEEDBACK_BUFFER 0x8C8E +#define GL_TRANSFORM_FEEDBACK_BUFFER_BINDING 0x8C8F +#define GL_RGBA32UI 0x8D70 +#define GL_RGB32UI 0x8D71 +#define GL_RGBA16UI 0x8D76 +#define GL_RGB16UI 0x8D77 +#define GL_RGBA8UI 0x8D7C +#define GL_RGB8UI 0x8D7D +#define GL_RGBA32I 0x8D82 +#define GL_RGB32I 0x8D83 +#define GL_RGBA16I 0x8D88 +#define GL_RGB16I 0x8D89 +#define GL_RGBA8I 0x8D8E +#define GL_RGB8I 0x8D8F +#define GL_RED_INTEGER 0x8D94 +#define GL_GREEN_INTEGER 0x8D95 +#define GL_BLUE_INTEGER 0x8D96 +#define GL_RGB_INTEGER 0x8D98 +#define GL_RGBA_INTEGER 0x8D99 +#define GL_BGR_INTEGER 0x8D9A +#define GL_BGRA_INTEGER 0x8D9B +#define GL_SAMPLER_1D_ARRAY 0x8DC0 +#define GL_SAMPLER_2D_ARRAY 0x8DC1 +#define GL_SAMPLER_1D_ARRAY_SHADOW 0x8DC3 +#define GL_SAMPLER_2D_ARRAY_SHADOW 0x8DC4 +#define GL_SAMPLER_CUBE_SHADOW 0x8DC5 +#define GL_UNSIGNED_INT_VEC2 0x8DC6 +#define GL_UNSIGNED_INT_VEC3 0x8DC7 +#define GL_UNSIGNED_INT_VEC4 0x8DC8 +#define GL_INT_SAMPLER_1D 0x8DC9 +#define GL_INT_SAMPLER_2D 0x8DCA +#define GL_INT_SAMPLER_3D 0x8DCB +#define GL_INT_SAMPLER_CUBE 0x8DCC +#define GL_INT_SAMPLER_1D_ARRAY 0x8DCE +#define GL_INT_SAMPLER_2D_ARRAY 0x8DCF +#define GL_UNSIGNED_INT_SAMPLER_1D 0x8DD1 +#define GL_UNSIGNED_INT_SAMPLER_2D 0x8DD2 +#define GL_UNSIGNED_INT_SAMPLER_3D 0x8DD3 +#define GL_UNSIGNED_INT_SAMPLER_CUBE 0x8DD4 +#define GL_UNSIGNED_INT_SAMPLER_1D_ARRAY 0x8DD6 +#define GL_UNSIGNED_INT_SAMPLER_2D_ARRAY 0x8DD7 +#define GL_QUERY_WAIT 0x8E13 +#define GL_QUERY_NO_WAIT 0x8E14 +#define GL_QUERY_BY_REGION_WAIT 0x8E15 +#define GL_QUERY_BY_REGION_NO_WAIT 0x8E16 +#define GL_BUFFER_ACCESS_FLAGS 0x911F +#define GL_BUFFER_MAP_LENGTH 0x9120 +#define GL_BUFFER_MAP_OFFSET 0x9121 +#define GL_DEPTH_COMPONENT32F 0x8CAC +#define GL_DEPTH32F_STENCIL8 0x8CAD +#define GL_FLOAT_32_UNSIGNED_INT_24_8_REV 0x8DAD +#define GL_INVALID_FRAMEBUFFER_OPERATION 0x0506 +#define GL_FRAMEBUFFER_ATTACHMENT_COLOR_ENCODING 0x8210 +#define GL_FRAMEBUFFER_ATTACHMENT_COMPONENT_TYPE 0x8211 +#define GL_FRAMEBUFFER_ATTACHMENT_RED_SIZE 0x8212 +#define GL_FRAMEBUFFER_ATTACHMENT_GREEN_SIZE 0x8213 +#define GL_FRAMEBUFFER_ATTACHMENT_BLUE_SIZE 0x8214 +#define GL_FRAMEBUFFER_ATTACHMENT_ALPHA_SIZE 0x8215 +#define GL_FRAMEBUFFER_ATTACHMENT_DEPTH_SIZE 0x8216 +#define GL_FRAMEBUFFER_ATTACHMENT_STENCIL_SIZE 0x8217 +#define GL_FRAMEBUFFER_DEFAULT 0x8218 +#define GL_FRAMEBUFFER_UNDEFINED 0x8219 +#define GL_DEPTH_STENCIL_ATTACHMENT 0x821A +#define GL_MAX_RENDERBUFFER_SIZE 0x84E8 +#define GL_DEPTH_STENCIL 0x84F9 +#define GL_UNSIGNED_INT_24_8 0x84FA +#define GL_DEPTH24_STENCIL8 0x88F0 +#define GL_TEXTURE_STENCIL_SIZE 0x88F1 +#define GL_TEXTURE_RED_TYPE 0x8C10 +#define GL_TEXTURE_GREEN_TYPE 0x8C11 +#define GL_TEXTURE_BLUE_TYPE 0x8C12 +#define GL_TEXTURE_ALPHA_TYPE 0x8C13 +#define GL_TEXTURE_DEPTH_TYPE 0x8C16 +#define GL_UNSIGNED_NORMALIZED 0x8C17 +#define GL_FRAMEBUFFER_BINDING 0x8CA6 +#define GL_DRAW_FRAMEBUFFER_BINDING 0x8CA6 +#define GL_RENDERBUFFER_BINDING 0x8CA7 +#define GL_READ_FRAMEBUFFER 0x8CA8 +#define GL_DRAW_FRAMEBUFFER 0x8CA9 +#define GL_READ_FRAMEBUFFER_BINDING 0x8CAA +#define GL_RENDERBUFFER_SAMPLES 0x8CAB +#define GL_FRAMEBUFFER_ATTACHMENT_OBJECT_TYPE 0x8CD0 +#define GL_FRAMEBUFFER_ATTACHMENT_OBJECT_NAME 0x8CD1 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_LEVEL 0x8CD2 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_CUBE_MAP_FACE 0x8CD3 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_LAYER 0x8CD4 +#define GL_FRAMEBUFFER_COMPLETE 0x8CD5 +#define GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT 0x8CD6 +#define GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT 0x8CD7 +#define GL_FRAMEBUFFER_INCOMPLETE_DRAW_BUFFER 0x8CDB +#define GL_FRAMEBUFFER_INCOMPLETE_READ_BUFFER 0x8CDC +#define GL_FRAMEBUFFER_UNSUPPORTED 0x8CDD +#define GL_MAX_COLOR_ATTACHMENTS 0x8CDF +#define GL_COLOR_ATTACHMENT0 0x8CE0 +#define GL_COLOR_ATTACHMENT1 0x8CE1 +#define GL_COLOR_ATTACHMENT2 0x8CE2 +#define GL_COLOR_ATTACHMENT3 0x8CE3 +#define GL_COLOR_ATTACHMENT4 0x8CE4 +#define GL_COLOR_ATTACHMENT5 0x8CE5 +#define GL_COLOR_ATTACHMENT6 0x8CE6 +#define GL_COLOR_ATTACHMENT7 0x8CE7 +#define GL_COLOR_ATTACHMENT8 0x8CE8 +#define GL_COLOR_ATTACHMENT9 0x8CE9 +#define GL_COLOR_ATTACHMENT10 0x8CEA +#define GL_COLOR_ATTACHMENT11 0x8CEB +#define GL_COLOR_ATTACHMENT12 0x8CEC +#define GL_COLOR_ATTACHMENT13 0x8CED +#define GL_COLOR_ATTACHMENT14 0x8CEE +#define GL_COLOR_ATTACHMENT15 0x8CEF +#define GL_COLOR_ATTACHMENT16 0x8CF0 +#define GL_COLOR_ATTACHMENT17 0x8CF1 +#define GL_COLOR_ATTACHMENT18 0x8CF2 +#define GL_COLOR_ATTACHMENT19 0x8CF3 +#define GL_COLOR_ATTACHMENT20 0x8CF4 +#define GL_COLOR_ATTACHMENT21 0x8CF5 +#define GL_COLOR_ATTACHMENT22 0x8CF6 +#define GL_COLOR_ATTACHMENT23 0x8CF7 +#define GL_COLOR_ATTACHMENT24 0x8CF8 +#define GL_COLOR_ATTACHMENT25 0x8CF9 +#define GL_COLOR_ATTACHMENT26 0x8CFA +#define GL_COLOR_ATTACHMENT27 0x8CFB +#define GL_COLOR_ATTACHMENT28 0x8CFC +#define GL_COLOR_ATTACHMENT29 0x8CFD +#define GL_COLOR_ATTACHMENT30 0x8CFE +#define GL_COLOR_ATTACHMENT31 0x8CFF +#define GL_DEPTH_ATTACHMENT 0x8D00 +#define GL_STENCIL_ATTACHMENT 0x8D20 +#define GL_FRAMEBUFFER 0x8D40 +#define GL_RENDERBUFFER 0x8D41 +#define GL_RENDERBUFFER_WIDTH 0x8D42 +#define GL_RENDERBUFFER_HEIGHT 0x8D43 +#define GL_RENDERBUFFER_INTERNAL_FORMAT 0x8D44 +#define GL_STENCIL_INDEX1 0x8D46 +#define GL_STENCIL_INDEX4 0x8D47 +#define GL_STENCIL_INDEX8 0x8D48 +#define GL_STENCIL_INDEX16 0x8D49 +#define GL_RENDERBUFFER_RED_SIZE 0x8D50 +#define GL_RENDERBUFFER_GREEN_SIZE 0x8D51 +#define GL_RENDERBUFFER_BLUE_SIZE 0x8D52 +#define GL_RENDERBUFFER_ALPHA_SIZE 0x8D53 +#define GL_RENDERBUFFER_DEPTH_SIZE 0x8D54 +#define GL_RENDERBUFFER_STENCIL_SIZE 0x8D55 +#define GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE 0x8D56 +#define GL_MAX_SAMPLES 0x8D57 +#define GL_INDEX 0x8222 +#define GL_TEXTURE_LUMINANCE_TYPE 0x8C14 +#define GL_TEXTURE_INTENSITY_TYPE 0x8C15 +#define GL_FRAMEBUFFER_SRGB 0x8DB9 +#define GL_HALF_FLOAT 0x140B +#define GL_MAP_READ_BIT 0x0001 +#define GL_MAP_WRITE_BIT 0x0002 +#define GL_MAP_INVALIDATE_RANGE_BIT 0x0004 +#define GL_MAP_INVALIDATE_BUFFER_BIT 0x0008 +#define GL_MAP_FLUSH_EXPLICIT_BIT 0x0010 +#define GL_MAP_UNSYNCHRONIZED_BIT 0x0020 +#define GL_COMPRESSED_RED_RGTC1 0x8DBB +#define GL_COMPRESSED_SIGNED_RED_RGTC1 0x8DBC +#define GL_COMPRESSED_RG_RGTC2 0x8DBD +#define GL_COMPRESSED_SIGNED_RG_RGTC2 0x8DBE +#define GL_RG 0x8227 +#define GL_RG_INTEGER 0x8228 +#define GL_R8 0x8229 +#define GL_R16 0x822A +#define GL_RG8 0x822B +#define GL_RG16 0x822C +#define GL_R16F 0x822D +#define GL_R32F 0x822E +#define GL_RG16F 0x822F +#define GL_RG32F 0x8230 +#define GL_R8I 0x8231 +#define GL_R8UI 0x8232 +#define GL_R16I 0x8233 +#define GL_R16UI 0x8234 +#define GL_R32I 0x8235 +#define GL_R32UI 0x8236 +#define GL_RG8I 0x8237 +#define GL_RG8UI 0x8238 +#define GL_RG16I 0x8239 +#define GL_RG16UI 0x823A +#define GL_RG32I 0x823B +#define GL_RG32UI 0x823C +#define GL_VERTEX_ARRAY_BINDING 0x85B5 +#define GL_CLAMP_VERTEX_COLOR 0x891A +#define GL_CLAMP_FRAGMENT_COLOR 0x891B +#define GL_ALPHA_INTEGER 0x8D97 +typedef void (APIENTRYP PFNGLCOLORMASKIPROC) (GLuint index, GLboolean r, GLboolean g, GLboolean b, GLboolean a); +typedef void (APIENTRYP PFNGLGETBOOLEANI_VPROC) (GLenum target, GLuint index, GLboolean *data); +typedef void (APIENTRYP PFNGLGETINTEGERI_VPROC) (GLenum target, GLuint index, GLint *data); +typedef void (APIENTRYP PFNGLENABLEIPROC) (GLenum target, GLuint index); +typedef void (APIENTRYP PFNGLDISABLEIPROC) (GLenum target, GLuint index); +typedef GLboolean (APIENTRYP PFNGLISENABLEDIPROC) (GLenum target, GLuint index); +typedef void (APIENTRYP PFNGLBEGINTRANSFORMFEEDBACKPROC) (GLenum primitiveMode); +typedef void (APIENTRYP PFNGLENDTRANSFORMFEEDBACKPROC) (void); +typedef void (APIENTRYP PFNGLBINDBUFFERRANGEPROC) (GLenum target, GLuint index, GLuint buffer, GLintptr offset, GLsizeiptr size); +typedef void (APIENTRYP PFNGLBINDBUFFERBASEPROC) (GLenum target, GLuint index, GLuint buffer); +typedef void (APIENTRYP PFNGLTRANSFORMFEEDBACKVARYINGSPROC) (GLuint program, GLsizei count, const GLchar *const*varyings, GLenum bufferMode); +typedef void (APIENTRYP PFNGLGETTRANSFORMFEEDBACKVARYINGPROC) (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLsizei *size, GLenum *type, GLchar *name); +typedef void (APIENTRYP PFNGLCLAMPCOLORPROC) (GLenum target, GLenum clamp); +typedef void (APIENTRYP PFNGLBEGINCONDITIONALRENDERPROC) (GLuint id, GLenum mode); +typedef void (APIENTRYP PFNGLENDCONDITIONALRENDERPROC) (void); +typedef void (APIENTRYP PFNGLVERTEXATTRIBIPOINTERPROC) (GLuint index, GLint size, GLenum type, GLsizei stride, const void *pointer); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBIIVPROC) (GLuint index, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBIUIVPROC) (GLuint index, GLenum pname, GLuint *params); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI1IPROC) (GLuint index, GLint x); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI2IPROC) (GLuint index, GLint x, GLint y); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI3IPROC) (GLuint index, GLint x, GLint y, GLint z); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4IPROC) (GLuint index, GLint x, GLint y, GLint z, GLint w); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI1UIPROC) (GLuint index, GLuint x); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI2UIPROC) (GLuint index, GLuint x, GLuint y); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI3UIPROC) (GLuint index, GLuint x, GLuint y, GLuint z); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4UIPROC) (GLuint index, GLuint x, GLuint y, GLuint z, GLuint w); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI1IVPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI2IVPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI3IVPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4IVPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI1UIVPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI2UIVPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI3UIVPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4UIVPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4BVPROC) (GLuint index, const GLbyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4SVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4UBVPROC) (GLuint index, const GLubyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4USVPROC) (GLuint index, const GLushort *v); +typedef void (APIENTRYP PFNGLGETUNIFORMUIVPROC) (GLuint program, GLint location, GLuint *params); +typedef void (APIENTRYP PFNGLBINDFRAGDATALOCATIONPROC) (GLuint program, GLuint color, const GLchar *name); +typedef GLint (APIENTRYP PFNGLGETFRAGDATALOCATIONPROC) (GLuint program, const GLchar *name); +typedef void (APIENTRYP PFNGLUNIFORM1UIPROC) (GLint location, GLuint v0); +typedef void (APIENTRYP PFNGLUNIFORM2UIPROC) (GLint location, GLuint v0, GLuint v1); +typedef void (APIENTRYP PFNGLUNIFORM3UIPROC) (GLint location, GLuint v0, GLuint v1, GLuint v2); +typedef void (APIENTRYP PFNGLUNIFORM4UIPROC) (GLint location, GLuint v0, GLuint v1, GLuint v2, GLuint v3); +typedef void (APIENTRYP PFNGLUNIFORM1UIVPROC) (GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLUNIFORM2UIVPROC) (GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLUNIFORM3UIVPROC) (GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLUNIFORM4UIVPROC) (GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLTEXPARAMETERIIVPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLTEXPARAMETERIUIVPROC) (GLenum target, GLenum pname, const GLuint *params); +typedef void (APIENTRYP PFNGLGETTEXPARAMETERIIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETTEXPARAMETERIUIVPROC) (GLenum target, GLenum pname, GLuint *params); +typedef void (APIENTRYP PFNGLCLEARBUFFERIVPROC) (GLenum buffer, GLint drawbuffer, const GLint *value); +typedef void (APIENTRYP PFNGLCLEARBUFFERUIVPROC) (GLenum buffer, GLint drawbuffer, const GLuint *value); +typedef void (APIENTRYP PFNGLCLEARBUFFERFVPROC) (GLenum buffer, GLint drawbuffer, const GLfloat *value); +typedef void (APIENTRYP PFNGLCLEARBUFFERFIPROC) (GLenum buffer, GLint drawbuffer, GLfloat depth, GLint stencil); +typedef const GLubyte *(APIENTRYP PFNGLGETSTRINGIPROC) (GLenum name, GLuint index); +typedef GLboolean (APIENTRYP PFNGLISRENDERBUFFERPROC) (GLuint renderbuffer); +typedef void (APIENTRYP PFNGLBINDRENDERBUFFERPROC) (GLenum target, GLuint renderbuffer); +typedef void (APIENTRYP PFNGLDELETERENDERBUFFERSPROC) (GLsizei n, const GLuint *renderbuffers); +typedef void (APIENTRYP PFNGLGENRENDERBUFFERSPROC) (GLsizei n, GLuint *renderbuffers); +typedef void (APIENTRYP PFNGLRENDERBUFFERSTORAGEPROC) (GLenum target, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLGETRENDERBUFFERPARAMETERIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef GLboolean (APIENTRYP PFNGLISFRAMEBUFFERPROC) (GLuint framebuffer); +typedef void (APIENTRYP PFNGLBINDFRAMEBUFFERPROC) (GLenum target, GLuint framebuffer); +typedef void (APIENTRYP PFNGLDELETEFRAMEBUFFERSPROC) (GLsizei n, const GLuint *framebuffers); +typedef void (APIENTRYP PFNGLGENFRAMEBUFFERSPROC) (GLsizei n, GLuint *framebuffers); +typedef GLenum (APIENTRYP PFNGLCHECKFRAMEBUFFERSTATUSPROC) (GLenum target); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTURE1DPROC) (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTURE2DPROC) (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTURE3DPROC) (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level, GLint zoffset); +typedef void (APIENTRYP PFNGLFRAMEBUFFERRENDERBUFFERPROC) (GLenum target, GLenum attachment, GLenum renderbuffertarget, GLuint renderbuffer); +typedef void (APIENTRYP PFNGLGETFRAMEBUFFERATTACHMENTPARAMETERIVPROC) (GLenum target, GLenum attachment, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGENERATEMIPMAPPROC) (GLenum target); +typedef void (APIENTRYP PFNGLBLITFRAMEBUFFERPROC) (GLint srcX0, GLint srcY0, GLint srcX1, GLint srcY1, GLint dstX0, GLint dstY0, GLint dstX1, GLint dstY1, GLbitfield mask, GLenum filter); +typedef void (APIENTRYP PFNGLRENDERBUFFERSTORAGEMULTISAMPLEPROC) (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTURELAYERPROC) (GLenum target, GLenum attachment, GLuint texture, GLint level, GLint layer); +typedef void *(APIENTRYP PFNGLMAPBUFFERRANGEPROC) (GLenum target, GLintptr offset, GLsizeiptr length, GLbitfield access); +typedef void (APIENTRYP PFNGLFLUSHMAPPEDBUFFERRANGEPROC) (GLenum target, GLintptr offset, GLsizeiptr length); +typedef void (APIENTRYP PFNGLBINDVERTEXARRAYPROC) (GLuint array); +typedef void (APIENTRYP PFNGLDELETEVERTEXARRAYSPROC) (GLsizei n, const GLuint *arrays); +typedef void (APIENTRYP PFNGLGENVERTEXARRAYSPROC) (GLsizei n, GLuint *arrays); +typedef GLboolean (APIENTRYP PFNGLISVERTEXARRAYPROC) (GLuint array); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glColorMaski (GLuint index, GLboolean r, GLboolean g, GLboolean b, GLboolean a); +GLAPI void APIENTRY glGetBooleani_v (GLenum target, GLuint index, GLboolean *data); +GLAPI void APIENTRY glGetIntegeri_v (GLenum target, GLuint index, GLint *data); +GLAPI void APIENTRY glEnablei (GLenum target, GLuint index); +GLAPI void APIENTRY glDisablei (GLenum target, GLuint index); +GLAPI GLboolean APIENTRY glIsEnabledi (GLenum target, GLuint index); +GLAPI void APIENTRY glBeginTransformFeedback (GLenum primitiveMode); +GLAPI void APIENTRY glEndTransformFeedback (void); +GLAPI void APIENTRY glBindBufferRange (GLenum target, GLuint index, GLuint buffer, GLintptr offset, GLsizeiptr size); +GLAPI void APIENTRY glBindBufferBase (GLenum target, GLuint index, GLuint buffer); +GLAPI void APIENTRY glTransformFeedbackVaryings (GLuint program, GLsizei count, const GLchar *const*varyings, GLenum bufferMode); +GLAPI void APIENTRY glGetTransformFeedbackVarying (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLsizei *size, GLenum *type, GLchar *name); +GLAPI void APIENTRY glClampColor (GLenum target, GLenum clamp); +GLAPI void APIENTRY glBeginConditionalRender (GLuint id, GLenum mode); +GLAPI void APIENTRY glEndConditionalRender (void); +GLAPI void APIENTRY glVertexAttribIPointer (GLuint index, GLint size, GLenum type, GLsizei stride, const void *pointer); +GLAPI void APIENTRY glGetVertexAttribIiv (GLuint index, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetVertexAttribIuiv (GLuint index, GLenum pname, GLuint *params); +GLAPI void APIENTRY glVertexAttribI1i (GLuint index, GLint x); +GLAPI void APIENTRY glVertexAttribI2i (GLuint index, GLint x, GLint y); +GLAPI void APIENTRY glVertexAttribI3i (GLuint index, GLint x, GLint y, GLint z); +GLAPI void APIENTRY glVertexAttribI4i (GLuint index, GLint x, GLint y, GLint z, GLint w); +GLAPI void APIENTRY glVertexAttribI1ui (GLuint index, GLuint x); +GLAPI void APIENTRY glVertexAttribI2ui (GLuint index, GLuint x, GLuint y); +GLAPI void APIENTRY glVertexAttribI3ui (GLuint index, GLuint x, GLuint y, GLuint z); +GLAPI void APIENTRY glVertexAttribI4ui (GLuint index, GLuint x, GLuint y, GLuint z, GLuint w); +GLAPI void APIENTRY glVertexAttribI1iv (GLuint index, const GLint *v); +GLAPI void APIENTRY glVertexAttribI2iv (GLuint index, const GLint *v); +GLAPI void APIENTRY glVertexAttribI3iv (GLuint index, const GLint *v); +GLAPI void APIENTRY glVertexAttribI4iv (GLuint index, const GLint *v); +GLAPI void APIENTRY glVertexAttribI1uiv (GLuint index, const GLuint *v); +GLAPI void APIENTRY glVertexAttribI2uiv (GLuint index, const GLuint *v); +GLAPI void APIENTRY glVertexAttribI3uiv (GLuint index, const GLuint *v); +GLAPI void APIENTRY glVertexAttribI4uiv (GLuint index, const GLuint *v); +GLAPI void APIENTRY glVertexAttribI4bv (GLuint index, const GLbyte *v); +GLAPI void APIENTRY glVertexAttribI4sv (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttribI4ubv (GLuint index, const GLubyte *v); +GLAPI void APIENTRY glVertexAttribI4usv (GLuint index, const GLushort *v); +GLAPI void APIENTRY glGetUniformuiv (GLuint program, GLint location, GLuint *params); +GLAPI void APIENTRY glBindFragDataLocation (GLuint program, GLuint color, const GLchar *name); +GLAPI GLint APIENTRY glGetFragDataLocation (GLuint program, const GLchar *name); +GLAPI void APIENTRY glUniform1ui (GLint location, GLuint v0); +GLAPI void APIENTRY glUniform2ui (GLint location, GLuint v0, GLuint v1); +GLAPI void APIENTRY glUniform3ui (GLint location, GLuint v0, GLuint v1, GLuint v2); +GLAPI void APIENTRY glUniform4ui (GLint location, GLuint v0, GLuint v1, GLuint v2, GLuint v3); +GLAPI void APIENTRY glUniform1uiv (GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glUniform2uiv (GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glUniform3uiv (GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glUniform4uiv (GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glTexParameterIiv (GLenum target, GLenum pname, const GLint *params); +GLAPI void APIENTRY glTexParameterIuiv (GLenum target, GLenum pname, const GLuint *params); +GLAPI void APIENTRY glGetTexParameterIiv (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetTexParameterIuiv (GLenum target, GLenum pname, GLuint *params); +GLAPI void APIENTRY glClearBufferiv (GLenum buffer, GLint drawbuffer, const GLint *value); +GLAPI void APIENTRY glClearBufferuiv (GLenum buffer, GLint drawbuffer, const GLuint *value); +GLAPI void APIENTRY glClearBufferfv (GLenum buffer, GLint drawbuffer, const GLfloat *value); +GLAPI void APIENTRY glClearBufferfi (GLenum buffer, GLint drawbuffer, GLfloat depth, GLint stencil); +GLAPI const GLubyte *APIENTRY glGetStringi (GLenum name, GLuint index); +GLAPI GLboolean APIENTRY glIsRenderbuffer (GLuint renderbuffer); +GLAPI void APIENTRY glBindRenderbuffer (GLenum target, GLuint renderbuffer); +GLAPI void APIENTRY glDeleteRenderbuffers (GLsizei n, const GLuint *renderbuffers); +GLAPI void APIENTRY glGenRenderbuffers (GLsizei n, GLuint *renderbuffers); +GLAPI void APIENTRY glRenderbufferStorage (GLenum target, GLenum internalformat, GLsizei width, GLsizei height); +GLAPI void APIENTRY glGetRenderbufferParameteriv (GLenum target, GLenum pname, GLint *params); +GLAPI GLboolean APIENTRY glIsFramebuffer (GLuint framebuffer); +GLAPI void APIENTRY glBindFramebuffer (GLenum target, GLuint framebuffer); +GLAPI void APIENTRY glDeleteFramebuffers (GLsizei n, const GLuint *framebuffers); +GLAPI void APIENTRY glGenFramebuffers (GLsizei n, GLuint *framebuffers); +GLAPI GLenum APIENTRY glCheckFramebufferStatus (GLenum target); +GLAPI void APIENTRY glFramebufferTexture1D (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +GLAPI void APIENTRY glFramebufferTexture2D (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +GLAPI void APIENTRY glFramebufferTexture3D (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level, GLint zoffset); +GLAPI void APIENTRY glFramebufferRenderbuffer (GLenum target, GLenum attachment, GLenum renderbuffertarget, GLuint renderbuffer); +GLAPI void APIENTRY glGetFramebufferAttachmentParameteriv (GLenum target, GLenum attachment, GLenum pname, GLint *params); +GLAPI void APIENTRY glGenerateMipmap (GLenum target); +GLAPI void APIENTRY glBlitFramebuffer (GLint srcX0, GLint srcY0, GLint srcX1, GLint srcY1, GLint dstX0, GLint dstY0, GLint dstX1, GLint dstY1, GLbitfield mask, GLenum filter); +GLAPI void APIENTRY glRenderbufferStorageMultisample (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +GLAPI void APIENTRY glFramebufferTextureLayer (GLenum target, GLenum attachment, GLuint texture, GLint level, GLint layer); +GLAPI void *APIENTRY glMapBufferRange (GLenum target, GLintptr offset, GLsizeiptr length, GLbitfield access); +GLAPI void APIENTRY glFlushMappedBufferRange (GLenum target, GLintptr offset, GLsizeiptr length); +GLAPI void APIENTRY glBindVertexArray (GLuint array); +GLAPI void APIENTRY glDeleteVertexArrays (GLsizei n, const GLuint *arrays); +GLAPI void APIENTRY glGenVertexArrays (GLsizei n, GLuint *arrays); +GLAPI GLboolean APIENTRY glIsVertexArray (GLuint array); +#endif +#endif /* GL_VERSION_3_0 */ + +#ifndef GL_VERSION_3_1 +#define GL_VERSION_3_1 1 +#define GL_SAMPLER_2D_RECT 0x8B63 +#define GL_SAMPLER_2D_RECT_SHADOW 0x8B64 +#define GL_SAMPLER_BUFFER 0x8DC2 +#define GL_INT_SAMPLER_2D_RECT 0x8DCD +#define GL_INT_SAMPLER_BUFFER 0x8DD0 +#define GL_UNSIGNED_INT_SAMPLER_2D_RECT 0x8DD5 +#define GL_UNSIGNED_INT_SAMPLER_BUFFER 0x8DD8 +#define GL_TEXTURE_BUFFER 0x8C2A +#define GL_MAX_TEXTURE_BUFFER_SIZE 0x8C2B +#define GL_TEXTURE_BINDING_BUFFER 0x8C2C +#define GL_TEXTURE_BUFFER_DATA_STORE_BINDING 0x8C2D +#define GL_TEXTURE_RECTANGLE 0x84F5 +#define GL_TEXTURE_BINDING_RECTANGLE 0x84F6 +#define GL_PROXY_TEXTURE_RECTANGLE 0x84F7 +#define GL_MAX_RECTANGLE_TEXTURE_SIZE 0x84F8 +#define GL_R8_SNORM 0x8F94 +#define GL_RG8_SNORM 0x8F95 +#define GL_RGB8_SNORM 0x8F96 +#define GL_RGBA8_SNORM 0x8F97 +#define GL_R16_SNORM 0x8F98 +#define GL_RG16_SNORM 0x8F99 +#define GL_RGB16_SNORM 0x8F9A +#define GL_RGBA16_SNORM 0x8F9B +#define GL_SIGNED_NORMALIZED 0x8F9C +#define GL_PRIMITIVE_RESTART 0x8F9D +#define GL_PRIMITIVE_RESTART_INDEX 0x8F9E +#define GL_COPY_READ_BUFFER 0x8F36 +#define GL_COPY_WRITE_BUFFER 0x8F37 +#define GL_UNIFORM_BUFFER 0x8A11 +#define GL_UNIFORM_BUFFER_BINDING 0x8A28 +#define GL_UNIFORM_BUFFER_START 0x8A29 +#define GL_UNIFORM_BUFFER_SIZE 0x8A2A +#define GL_MAX_VERTEX_UNIFORM_BLOCKS 0x8A2B +#define GL_MAX_GEOMETRY_UNIFORM_BLOCKS 0x8A2C +#define GL_MAX_FRAGMENT_UNIFORM_BLOCKS 0x8A2D +#define GL_MAX_COMBINED_UNIFORM_BLOCKS 0x8A2E +#define GL_MAX_UNIFORM_BUFFER_BINDINGS 0x8A2F +#define GL_MAX_UNIFORM_BLOCK_SIZE 0x8A30 +#define GL_MAX_COMBINED_VERTEX_UNIFORM_COMPONENTS 0x8A31 +#define GL_MAX_COMBINED_GEOMETRY_UNIFORM_COMPONENTS 0x8A32 +#define GL_MAX_COMBINED_FRAGMENT_UNIFORM_COMPONENTS 0x8A33 +#define GL_UNIFORM_BUFFER_OFFSET_ALIGNMENT 0x8A34 +#define GL_ACTIVE_UNIFORM_BLOCK_MAX_NAME_LENGTH 0x8A35 +#define GL_ACTIVE_UNIFORM_BLOCKS 0x8A36 +#define GL_UNIFORM_TYPE 0x8A37 +#define GL_UNIFORM_SIZE 0x8A38 +#define GL_UNIFORM_NAME_LENGTH 0x8A39 +#define GL_UNIFORM_BLOCK_INDEX 0x8A3A +#define GL_UNIFORM_OFFSET 0x8A3B +#define GL_UNIFORM_ARRAY_STRIDE 0x8A3C +#define GL_UNIFORM_MATRIX_STRIDE 0x8A3D +#define GL_UNIFORM_IS_ROW_MAJOR 0x8A3E +#define GL_UNIFORM_BLOCK_BINDING 0x8A3F +#define GL_UNIFORM_BLOCK_DATA_SIZE 0x8A40 +#define GL_UNIFORM_BLOCK_NAME_LENGTH 0x8A41 +#define GL_UNIFORM_BLOCK_ACTIVE_UNIFORMS 0x8A42 +#define GL_UNIFORM_BLOCK_ACTIVE_UNIFORM_INDICES 0x8A43 +#define GL_UNIFORM_BLOCK_REFERENCED_BY_VERTEX_SHADER 0x8A44 +#define GL_UNIFORM_BLOCK_REFERENCED_BY_GEOMETRY_SHADER 0x8A45 +#define GL_UNIFORM_BLOCK_REFERENCED_BY_FRAGMENT_SHADER 0x8A46 +#define GL_INVALID_INDEX 0xFFFFFFFFu +typedef void (APIENTRYP PFNGLDRAWARRAYSINSTANCEDPROC) (GLenum mode, GLint first, GLsizei count, GLsizei instancecount); +typedef void (APIENTRYP PFNGLDRAWELEMENTSINSTANCEDPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount); +typedef void (APIENTRYP PFNGLTEXBUFFERPROC) (GLenum target, GLenum internalformat, GLuint buffer); +typedef void (APIENTRYP PFNGLPRIMITIVERESTARTINDEXPROC) (GLuint index); +typedef void (APIENTRYP PFNGLCOPYBUFFERSUBDATAPROC) (GLenum readTarget, GLenum writeTarget, GLintptr readOffset, GLintptr writeOffset, GLsizeiptr size); +typedef void (APIENTRYP PFNGLGETUNIFORMINDICESPROC) (GLuint program, GLsizei uniformCount, const GLchar *const*uniformNames, GLuint *uniformIndices); +typedef void (APIENTRYP PFNGLGETACTIVEUNIFORMSIVPROC) (GLuint program, GLsizei uniformCount, const GLuint *uniformIndices, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETACTIVEUNIFORMNAMEPROC) (GLuint program, GLuint uniformIndex, GLsizei bufSize, GLsizei *length, GLchar *uniformName); +typedef GLuint (APIENTRYP PFNGLGETUNIFORMBLOCKINDEXPROC) (GLuint program, const GLchar *uniformBlockName); +typedef void (APIENTRYP PFNGLGETACTIVEUNIFORMBLOCKIVPROC) (GLuint program, GLuint uniformBlockIndex, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETACTIVEUNIFORMBLOCKNAMEPROC) (GLuint program, GLuint uniformBlockIndex, GLsizei bufSize, GLsizei *length, GLchar *uniformBlockName); +typedef void (APIENTRYP PFNGLUNIFORMBLOCKBINDINGPROC) (GLuint program, GLuint uniformBlockIndex, GLuint uniformBlockBinding); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawArraysInstanced (GLenum mode, GLint first, GLsizei count, GLsizei instancecount); +GLAPI void APIENTRY glDrawElementsInstanced (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount); +GLAPI void APIENTRY glTexBuffer (GLenum target, GLenum internalformat, GLuint buffer); +GLAPI void APIENTRY glPrimitiveRestartIndex (GLuint index); +GLAPI void APIENTRY glCopyBufferSubData (GLenum readTarget, GLenum writeTarget, GLintptr readOffset, GLintptr writeOffset, GLsizeiptr size); +GLAPI void APIENTRY glGetUniformIndices (GLuint program, GLsizei uniformCount, const GLchar *const*uniformNames, GLuint *uniformIndices); +GLAPI void APIENTRY glGetActiveUniformsiv (GLuint program, GLsizei uniformCount, const GLuint *uniformIndices, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetActiveUniformName (GLuint program, GLuint uniformIndex, GLsizei bufSize, GLsizei *length, GLchar *uniformName); +GLAPI GLuint APIENTRY glGetUniformBlockIndex (GLuint program, const GLchar *uniformBlockName); +GLAPI void APIENTRY glGetActiveUniformBlockiv (GLuint program, GLuint uniformBlockIndex, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetActiveUniformBlockName (GLuint program, GLuint uniformBlockIndex, GLsizei bufSize, GLsizei *length, GLchar *uniformBlockName); +GLAPI void APIENTRY glUniformBlockBinding (GLuint program, GLuint uniformBlockIndex, GLuint uniformBlockBinding); +#endif +#endif /* GL_VERSION_3_1 */ + +#ifndef GL_VERSION_3_2 +#define GL_VERSION_3_2 1 +typedef struct __GLsync *GLsync; +typedef khronos_uint64_t GLuint64; +typedef khronos_int64_t GLint64; +#define GL_CONTEXT_CORE_PROFILE_BIT 0x00000001 +#define GL_CONTEXT_COMPATIBILITY_PROFILE_BIT 0x00000002 +#define GL_LINES_ADJACENCY 0x000A +#define GL_LINE_STRIP_ADJACENCY 0x000B +#define GL_TRIANGLES_ADJACENCY 0x000C +#define GL_TRIANGLE_STRIP_ADJACENCY 0x000D +#define GL_PROGRAM_POINT_SIZE 0x8642 +#define GL_MAX_GEOMETRY_TEXTURE_IMAGE_UNITS 0x8C29 +#define GL_FRAMEBUFFER_ATTACHMENT_LAYERED 0x8DA7 +#define GL_FRAMEBUFFER_INCOMPLETE_LAYER_TARGETS 0x8DA8 +#define GL_GEOMETRY_SHADER 0x8DD9 +#define GL_GEOMETRY_VERTICES_OUT 0x8916 +#define GL_GEOMETRY_INPUT_TYPE 0x8917 +#define GL_GEOMETRY_OUTPUT_TYPE 0x8918 +#define GL_MAX_GEOMETRY_UNIFORM_COMPONENTS 0x8DDF +#define GL_MAX_GEOMETRY_OUTPUT_VERTICES 0x8DE0 +#define GL_MAX_GEOMETRY_TOTAL_OUTPUT_COMPONENTS 0x8DE1 +#define GL_MAX_VERTEX_OUTPUT_COMPONENTS 0x9122 +#define GL_MAX_GEOMETRY_INPUT_COMPONENTS 0x9123 +#define GL_MAX_GEOMETRY_OUTPUT_COMPONENTS 0x9124 +#define GL_MAX_FRAGMENT_INPUT_COMPONENTS 0x9125 +#define GL_CONTEXT_PROFILE_MASK 0x9126 +#define GL_DEPTH_CLAMP 0x864F +#define GL_QUADS_FOLLOW_PROVOKING_VERTEX_CONVENTION 0x8E4C +#define GL_FIRST_VERTEX_CONVENTION 0x8E4D +#define GL_LAST_VERTEX_CONVENTION 0x8E4E +#define GL_PROVOKING_VERTEX 0x8E4F +#define GL_TEXTURE_CUBE_MAP_SEAMLESS 0x884F +#define GL_MAX_SERVER_WAIT_TIMEOUT 0x9111 +#define GL_OBJECT_TYPE 0x9112 +#define GL_SYNC_CONDITION 0x9113 +#define GL_SYNC_STATUS 0x9114 +#define GL_SYNC_FLAGS 0x9115 +#define GL_SYNC_FENCE 0x9116 +#define GL_SYNC_GPU_COMMANDS_COMPLETE 0x9117 +#define GL_UNSIGNALED 0x9118 +#define GL_SIGNALED 0x9119 +#define GL_ALREADY_SIGNALED 0x911A +#define GL_TIMEOUT_EXPIRED 0x911B +#define GL_CONDITION_SATISFIED 0x911C +#define GL_WAIT_FAILED 0x911D +#define GL_TIMEOUT_IGNORED 0xFFFFFFFFFFFFFFFFull +#define GL_SYNC_FLUSH_COMMANDS_BIT 0x00000001 +#define GL_SAMPLE_POSITION 0x8E50 +#define GL_SAMPLE_MASK 0x8E51 +#define GL_SAMPLE_MASK_VALUE 0x8E52 +#define GL_MAX_SAMPLE_MASK_WORDS 0x8E59 +#define GL_TEXTURE_2D_MULTISAMPLE 0x9100 +#define GL_PROXY_TEXTURE_2D_MULTISAMPLE 0x9101 +#define GL_TEXTURE_2D_MULTISAMPLE_ARRAY 0x9102 +#define GL_PROXY_TEXTURE_2D_MULTISAMPLE_ARRAY 0x9103 +#define GL_TEXTURE_BINDING_2D_MULTISAMPLE 0x9104 +#define GL_TEXTURE_BINDING_2D_MULTISAMPLE_ARRAY 0x9105 +#define GL_TEXTURE_SAMPLES 0x9106 +#define GL_TEXTURE_FIXED_SAMPLE_LOCATIONS 0x9107 +#define GL_SAMPLER_2D_MULTISAMPLE 0x9108 +#define GL_INT_SAMPLER_2D_MULTISAMPLE 0x9109 +#define GL_UNSIGNED_INT_SAMPLER_2D_MULTISAMPLE 0x910A +#define GL_SAMPLER_2D_MULTISAMPLE_ARRAY 0x910B +#define GL_INT_SAMPLER_2D_MULTISAMPLE_ARRAY 0x910C +#define GL_UNSIGNED_INT_SAMPLER_2D_MULTISAMPLE_ARRAY 0x910D +#define GL_MAX_COLOR_TEXTURE_SAMPLES 0x910E +#define GL_MAX_DEPTH_TEXTURE_SAMPLES 0x910F +#define GL_MAX_INTEGER_SAMPLES 0x9110 +typedef void (APIENTRYP PFNGLDRAWELEMENTSBASEVERTEXPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLint basevertex); +typedef void (APIENTRYP PFNGLDRAWRANGEELEMENTSBASEVERTEXPROC) (GLenum mode, GLuint start, GLuint end, GLsizei count, GLenum type, const void *indices, GLint basevertex); +typedef void (APIENTRYP PFNGLDRAWELEMENTSINSTANCEDBASEVERTEXPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount, GLint basevertex); +typedef void (APIENTRYP PFNGLMULTIDRAWELEMENTSBASEVERTEXPROC) (GLenum mode, const GLsizei *count, GLenum type, const void *const*indices, GLsizei drawcount, const GLint *basevertex); +typedef void (APIENTRYP PFNGLPROVOKINGVERTEXPROC) (GLenum mode); +typedef GLsync (APIENTRYP PFNGLFENCESYNCPROC) (GLenum condition, GLbitfield flags); +typedef GLboolean (APIENTRYP PFNGLISSYNCPROC) (GLsync sync); +typedef void (APIENTRYP PFNGLDELETESYNCPROC) (GLsync sync); +typedef GLenum (APIENTRYP PFNGLCLIENTWAITSYNCPROC) (GLsync sync, GLbitfield flags, GLuint64 timeout); +typedef void (APIENTRYP PFNGLWAITSYNCPROC) (GLsync sync, GLbitfield flags, GLuint64 timeout); +typedef void (APIENTRYP PFNGLGETINTEGER64VPROC) (GLenum pname, GLint64 *data); +typedef void (APIENTRYP PFNGLGETSYNCIVPROC) (GLsync sync, GLenum pname, GLsizei count, GLsizei *length, GLint *values); +typedef void (APIENTRYP PFNGLGETINTEGER64I_VPROC) (GLenum target, GLuint index, GLint64 *data); +typedef void (APIENTRYP PFNGLGETBUFFERPARAMETERI64VPROC) (GLenum target, GLenum pname, GLint64 *params); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTUREPROC) (GLenum target, GLenum attachment, GLuint texture, GLint level); +typedef void (APIENTRYP PFNGLTEXIMAGE2DMULTISAMPLEPROC) (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLboolean fixedsamplelocations); +typedef void (APIENTRYP PFNGLTEXIMAGE3DMULTISAMPLEPROC) (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedsamplelocations); +typedef void (APIENTRYP PFNGLGETMULTISAMPLEFVPROC) (GLenum pname, GLuint index, GLfloat *val); +typedef void (APIENTRYP PFNGLSAMPLEMASKIPROC) (GLuint maskNumber, GLbitfield mask); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawElementsBaseVertex (GLenum mode, GLsizei count, GLenum type, const void *indices, GLint basevertex); +GLAPI void APIENTRY glDrawRangeElementsBaseVertex (GLenum mode, GLuint start, GLuint end, GLsizei count, GLenum type, const void *indices, GLint basevertex); +GLAPI void APIENTRY glDrawElementsInstancedBaseVertex (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount, GLint basevertex); +GLAPI void APIENTRY glMultiDrawElementsBaseVertex (GLenum mode, const GLsizei *count, GLenum type, const void *const*indices, GLsizei drawcount, const GLint *basevertex); +GLAPI void APIENTRY glProvokingVertex (GLenum mode); +GLAPI GLsync APIENTRY glFenceSync (GLenum condition, GLbitfield flags); +GLAPI GLboolean APIENTRY glIsSync (GLsync sync); +GLAPI void APIENTRY glDeleteSync (GLsync sync); +GLAPI GLenum APIENTRY glClientWaitSync (GLsync sync, GLbitfield flags, GLuint64 timeout); +GLAPI void APIENTRY glWaitSync (GLsync sync, GLbitfield flags, GLuint64 timeout); +GLAPI void APIENTRY glGetInteger64v (GLenum pname, GLint64 *data); +GLAPI void APIENTRY glGetSynciv (GLsync sync, GLenum pname, GLsizei count, GLsizei *length, GLint *values); +GLAPI void APIENTRY glGetInteger64i_v (GLenum target, GLuint index, GLint64 *data); +GLAPI void APIENTRY glGetBufferParameteri64v (GLenum target, GLenum pname, GLint64 *params); +GLAPI void APIENTRY glFramebufferTexture (GLenum target, GLenum attachment, GLuint texture, GLint level); +GLAPI void APIENTRY glTexImage2DMultisample (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLboolean fixedsamplelocations); +GLAPI void APIENTRY glTexImage3DMultisample (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedsamplelocations); +GLAPI void APIENTRY glGetMultisamplefv (GLenum pname, GLuint index, GLfloat *val); +GLAPI void APIENTRY glSampleMaski (GLuint maskNumber, GLbitfield mask); +#endif +#endif /* GL_VERSION_3_2 */ + +#ifndef GL_VERSION_3_3 +#define GL_VERSION_3_3 1 +#define GL_VERTEX_ATTRIB_ARRAY_DIVISOR 0x88FE +#define GL_SRC1_COLOR 0x88F9 +#define GL_ONE_MINUS_SRC1_COLOR 0x88FA +#define GL_ONE_MINUS_SRC1_ALPHA 0x88FB +#define GL_MAX_DUAL_SOURCE_DRAW_BUFFERS 0x88FC +#define GL_ANY_SAMPLES_PASSED 0x8C2F +#define GL_SAMPLER_BINDING 0x8919 +#define GL_RGB10_A2UI 0x906F +#define GL_TEXTURE_SWIZZLE_R 0x8E42 +#define GL_TEXTURE_SWIZZLE_G 0x8E43 +#define GL_TEXTURE_SWIZZLE_B 0x8E44 +#define GL_TEXTURE_SWIZZLE_A 0x8E45 +#define GL_TEXTURE_SWIZZLE_RGBA 0x8E46 +#define GL_TIME_ELAPSED 0x88BF +#define GL_TIMESTAMP 0x8E28 +#define GL_INT_2_10_10_10_REV 0x8D9F +typedef void (APIENTRYP PFNGLBINDFRAGDATALOCATIONINDEXEDPROC) (GLuint program, GLuint colorNumber, GLuint index, const GLchar *name); +typedef GLint (APIENTRYP PFNGLGETFRAGDATAINDEXPROC) (GLuint program, const GLchar *name); +typedef void (APIENTRYP PFNGLGENSAMPLERSPROC) (GLsizei count, GLuint *samplers); +typedef void (APIENTRYP PFNGLDELETESAMPLERSPROC) (GLsizei count, const GLuint *samplers); +typedef GLboolean (APIENTRYP PFNGLISSAMPLERPROC) (GLuint sampler); +typedef void (APIENTRYP PFNGLBINDSAMPLERPROC) (GLuint unit, GLuint sampler); +typedef void (APIENTRYP PFNGLSAMPLERPARAMETERIPROC) (GLuint sampler, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLSAMPLERPARAMETERIVPROC) (GLuint sampler, GLenum pname, const GLint *param); +typedef void (APIENTRYP PFNGLSAMPLERPARAMETERFPROC) (GLuint sampler, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLSAMPLERPARAMETERFVPROC) (GLuint sampler, GLenum pname, const GLfloat *param); +typedef void (APIENTRYP PFNGLSAMPLERPARAMETERIIVPROC) (GLuint sampler, GLenum pname, const GLint *param); +typedef void (APIENTRYP PFNGLSAMPLERPARAMETERIUIVPROC) (GLuint sampler, GLenum pname, const GLuint *param); +typedef void (APIENTRYP PFNGLGETSAMPLERPARAMETERIVPROC) (GLuint sampler, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETSAMPLERPARAMETERIIVPROC) (GLuint sampler, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETSAMPLERPARAMETERFVPROC) (GLuint sampler, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETSAMPLERPARAMETERIUIVPROC) (GLuint sampler, GLenum pname, GLuint *params); +typedef void (APIENTRYP PFNGLQUERYCOUNTERPROC) (GLuint id, GLenum target); +typedef void (APIENTRYP PFNGLGETQUERYOBJECTI64VPROC) (GLuint id, GLenum pname, GLint64 *params); +typedef void (APIENTRYP PFNGLGETQUERYOBJECTUI64VPROC) (GLuint id, GLenum pname, GLuint64 *params); +typedef void (APIENTRYP PFNGLVERTEXATTRIBDIVISORPROC) (GLuint index, GLuint divisor); +typedef void (APIENTRYP PFNGLVERTEXATTRIBP1UIPROC) (GLuint index, GLenum type, GLboolean normalized, GLuint value); +typedef void (APIENTRYP PFNGLVERTEXATTRIBP1UIVPROC) (GLuint index, GLenum type, GLboolean normalized, const GLuint *value); +typedef void (APIENTRYP PFNGLVERTEXATTRIBP2UIPROC) (GLuint index, GLenum type, GLboolean normalized, GLuint value); +typedef void (APIENTRYP PFNGLVERTEXATTRIBP2UIVPROC) (GLuint index, GLenum type, GLboolean normalized, const GLuint *value); +typedef void (APIENTRYP PFNGLVERTEXATTRIBP3UIPROC) (GLuint index, GLenum type, GLboolean normalized, GLuint value); +typedef void (APIENTRYP PFNGLVERTEXATTRIBP3UIVPROC) (GLuint index, GLenum type, GLboolean normalized, const GLuint *value); +typedef void (APIENTRYP PFNGLVERTEXATTRIBP4UIPROC) (GLuint index, GLenum type, GLboolean normalized, GLuint value); +typedef void (APIENTRYP PFNGLVERTEXATTRIBP4UIVPROC) (GLuint index, GLenum type, GLboolean normalized, const GLuint *value); +typedef void (APIENTRYP PFNGLVERTEXP2UIPROC) (GLenum type, GLuint value); +typedef void (APIENTRYP PFNGLVERTEXP2UIVPROC) (GLenum type, const GLuint *value); +typedef void (APIENTRYP PFNGLVERTEXP3UIPROC) (GLenum type, GLuint value); +typedef void (APIENTRYP PFNGLVERTEXP3UIVPROC) (GLenum type, const GLuint *value); +typedef void (APIENTRYP PFNGLVERTEXP4UIPROC) (GLenum type, GLuint value); +typedef void (APIENTRYP PFNGLVERTEXP4UIVPROC) (GLenum type, const GLuint *value); +typedef void (APIENTRYP PFNGLTEXCOORDP1UIPROC) (GLenum type, GLuint coords); +typedef void (APIENTRYP PFNGLTEXCOORDP1UIVPROC) (GLenum type, const GLuint *coords); +typedef void (APIENTRYP PFNGLTEXCOORDP2UIPROC) (GLenum type, GLuint coords); +typedef void (APIENTRYP PFNGLTEXCOORDP2UIVPROC) (GLenum type, const GLuint *coords); +typedef void (APIENTRYP PFNGLTEXCOORDP3UIPROC) (GLenum type, GLuint coords); +typedef void (APIENTRYP PFNGLTEXCOORDP3UIVPROC) (GLenum type, const GLuint *coords); +typedef void (APIENTRYP PFNGLTEXCOORDP4UIPROC) (GLenum type, GLuint coords); +typedef void (APIENTRYP PFNGLTEXCOORDP4UIVPROC) (GLenum type, const GLuint *coords); +typedef void (APIENTRYP PFNGLMULTITEXCOORDP1UIPROC) (GLenum texture, GLenum type, GLuint coords); +typedef void (APIENTRYP PFNGLMULTITEXCOORDP1UIVPROC) (GLenum texture, GLenum type, const GLuint *coords); +typedef void (APIENTRYP PFNGLMULTITEXCOORDP2UIPROC) (GLenum texture, GLenum type, GLuint coords); +typedef void (APIENTRYP PFNGLMULTITEXCOORDP2UIVPROC) (GLenum texture, GLenum type, const GLuint *coords); +typedef void (APIENTRYP PFNGLMULTITEXCOORDP3UIPROC) (GLenum texture, GLenum type, GLuint coords); +typedef void (APIENTRYP PFNGLMULTITEXCOORDP3UIVPROC) (GLenum texture, GLenum type, const GLuint *coords); +typedef void (APIENTRYP PFNGLMULTITEXCOORDP4UIPROC) (GLenum texture, GLenum type, GLuint coords); +typedef void (APIENTRYP PFNGLMULTITEXCOORDP4UIVPROC) (GLenum texture, GLenum type, const GLuint *coords); +typedef void (APIENTRYP PFNGLNORMALP3UIPROC) (GLenum type, GLuint coords); +typedef void (APIENTRYP PFNGLNORMALP3UIVPROC) (GLenum type, const GLuint *coords); +typedef void (APIENTRYP PFNGLCOLORP3UIPROC) (GLenum type, GLuint color); +typedef void (APIENTRYP PFNGLCOLORP3UIVPROC) (GLenum type, const GLuint *color); +typedef void (APIENTRYP PFNGLCOLORP4UIPROC) (GLenum type, GLuint color); +typedef void (APIENTRYP PFNGLCOLORP4UIVPROC) (GLenum type, const GLuint *color); +typedef void (APIENTRYP PFNGLSECONDARYCOLORP3UIPROC) (GLenum type, GLuint color); +typedef void (APIENTRYP PFNGLSECONDARYCOLORP3UIVPROC) (GLenum type, const GLuint *color); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBindFragDataLocationIndexed (GLuint program, GLuint colorNumber, GLuint index, const GLchar *name); +GLAPI GLint APIENTRY glGetFragDataIndex (GLuint program, const GLchar *name); +GLAPI void APIENTRY glGenSamplers (GLsizei count, GLuint *samplers); +GLAPI void APIENTRY glDeleteSamplers (GLsizei count, const GLuint *samplers); +GLAPI GLboolean APIENTRY glIsSampler (GLuint sampler); +GLAPI void APIENTRY glBindSampler (GLuint unit, GLuint sampler); +GLAPI void APIENTRY glSamplerParameteri (GLuint sampler, GLenum pname, GLint param); +GLAPI void APIENTRY glSamplerParameteriv (GLuint sampler, GLenum pname, const GLint *param); +GLAPI void APIENTRY glSamplerParameterf (GLuint sampler, GLenum pname, GLfloat param); +GLAPI void APIENTRY glSamplerParameterfv (GLuint sampler, GLenum pname, const GLfloat *param); +GLAPI void APIENTRY glSamplerParameterIiv (GLuint sampler, GLenum pname, const GLint *param); +GLAPI void APIENTRY glSamplerParameterIuiv (GLuint sampler, GLenum pname, const GLuint *param); +GLAPI void APIENTRY glGetSamplerParameteriv (GLuint sampler, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetSamplerParameterIiv (GLuint sampler, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetSamplerParameterfv (GLuint sampler, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetSamplerParameterIuiv (GLuint sampler, GLenum pname, GLuint *params); +GLAPI void APIENTRY glQueryCounter (GLuint id, GLenum target); +GLAPI void APIENTRY glGetQueryObjecti64v (GLuint id, GLenum pname, GLint64 *params); +GLAPI void APIENTRY glGetQueryObjectui64v (GLuint id, GLenum pname, GLuint64 *params); +GLAPI void APIENTRY glVertexAttribDivisor (GLuint index, GLuint divisor); +GLAPI void APIENTRY glVertexAttribP1ui (GLuint index, GLenum type, GLboolean normalized, GLuint value); +GLAPI void APIENTRY glVertexAttribP1uiv (GLuint index, GLenum type, GLboolean normalized, const GLuint *value); +GLAPI void APIENTRY glVertexAttribP2ui (GLuint index, GLenum type, GLboolean normalized, GLuint value); +GLAPI void APIENTRY glVertexAttribP2uiv (GLuint index, GLenum type, GLboolean normalized, const GLuint *value); +GLAPI void APIENTRY glVertexAttribP3ui (GLuint index, GLenum type, GLboolean normalized, GLuint value); +GLAPI void APIENTRY glVertexAttribP3uiv (GLuint index, GLenum type, GLboolean normalized, const GLuint *value); +GLAPI void APIENTRY glVertexAttribP4ui (GLuint index, GLenum type, GLboolean normalized, GLuint value); +GLAPI void APIENTRY glVertexAttribP4uiv (GLuint index, GLenum type, GLboolean normalized, const GLuint *value); +GLAPI void APIENTRY glVertexP2ui (GLenum type, GLuint value); +GLAPI void APIENTRY glVertexP2uiv (GLenum type, const GLuint *value); +GLAPI void APIENTRY glVertexP3ui (GLenum type, GLuint value); +GLAPI void APIENTRY glVertexP3uiv (GLenum type, const GLuint *value); +GLAPI void APIENTRY glVertexP4ui (GLenum type, GLuint value); +GLAPI void APIENTRY glVertexP4uiv (GLenum type, const GLuint *value); +GLAPI void APIENTRY glTexCoordP1ui (GLenum type, GLuint coords); +GLAPI void APIENTRY glTexCoordP1uiv (GLenum type, const GLuint *coords); +GLAPI void APIENTRY glTexCoordP2ui (GLenum type, GLuint coords); +GLAPI void APIENTRY glTexCoordP2uiv (GLenum type, const GLuint *coords); +GLAPI void APIENTRY glTexCoordP3ui (GLenum type, GLuint coords); +GLAPI void APIENTRY glTexCoordP3uiv (GLenum type, const GLuint *coords); +GLAPI void APIENTRY glTexCoordP4ui (GLenum type, GLuint coords); +GLAPI void APIENTRY glTexCoordP4uiv (GLenum type, const GLuint *coords); +GLAPI void APIENTRY glMultiTexCoordP1ui (GLenum texture, GLenum type, GLuint coords); +GLAPI void APIENTRY glMultiTexCoordP1uiv (GLenum texture, GLenum type, const GLuint *coords); +GLAPI void APIENTRY glMultiTexCoordP2ui (GLenum texture, GLenum type, GLuint coords); +GLAPI void APIENTRY glMultiTexCoordP2uiv (GLenum texture, GLenum type, const GLuint *coords); +GLAPI void APIENTRY glMultiTexCoordP3ui (GLenum texture, GLenum type, GLuint coords); +GLAPI void APIENTRY glMultiTexCoordP3uiv (GLenum texture, GLenum type, const GLuint *coords); +GLAPI void APIENTRY glMultiTexCoordP4ui (GLenum texture, GLenum type, GLuint coords); +GLAPI void APIENTRY glMultiTexCoordP4uiv (GLenum texture, GLenum type, const GLuint *coords); +GLAPI void APIENTRY glNormalP3ui (GLenum type, GLuint coords); +GLAPI void APIENTRY glNormalP3uiv (GLenum type, const GLuint *coords); +GLAPI void APIENTRY glColorP3ui (GLenum type, GLuint color); +GLAPI void APIENTRY glColorP3uiv (GLenum type, const GLuint *color); +GLAPI void APIENTRY glColorP4ui (GLenum type, GLuint color); +GLAPI void APIENTRY glColorP4uiv (GLenum type, const GLuint *color); +GLAPI void APIENTRY glSecondaryColorP3ui (GLenum type, GLuint color); +GLAPI void APIENTRY glSecondaryColorP3uiv (GLenum type, const GLuint *color); +#endif +#endif /* GL_VERSION_3_3 */ + +#ifndef GL_VERSION_4_0 +#define GL_VERSION_4_0 1 +#define GL_SAMPLE_SHADING 0x8C36 +#define GL_MIN_SAMPLE_SHADING_VALUE 0x8C37 +#define GL_MIN_PROGRAM_TEXTURE_GATHER_OFFSET 0x8E5E +#define GL_MAX_PROGRAM_TEXTURE_GATHER_OFFSET 0x8E5F +#define GL_TEXTURE_CUBE_MAP_ARRAY 0x9009 +#define GL_TEXTURE_BINDING_CUBE_MAP_ARRAY 0x900A +#define GL_PROXY_TEXTURE_CUBE_MAP_ARRAY 0x900B +#define GL_SAMPLER_CUBE_MAP_ARRAY 0x900C +#define GL_SAMPLER_CUBE_MAP_ARRAY_SHADOW 0x900D +#define GL_INT_SAMPLER_CUBE_MAP_ARRAY 0x900E +#define GL_UNSIGNED_INT_SAMPLER_CUBE_MAP_ARRAY 0x900F +#define GL_DRAW_INDIRECT_BUFFER 0x8F3F +#define GL_DRAW_INDIRECT_BUFFER_BINDING 0x8F43 +#define GL_GEOMETRY_SHADER_INVOCATIONS 0x887F +#define GL_MAX_GEOMETRY_SHADER_INVOCATIONS 0x8E5A +#define GL_MIN_FRAGMENT_INTERPOLATION_OFFSET 0x8E5B +#define GL_MAX_FRAGMENT_INTERPOLATION_OFFSET 0x8E5C +#define GL_FRAGMENT_INTERPOLATION_OFFSET_BITS 0x8E5D +#define GL_MAX_VERTEX_STREAMS 0x8E71 +#define GL_DOUBLE_VEC2 0x8FFC +#define GL_DOUBLE_VEC3 0x8FFD +#define GL_DOUBLE_VEC4 0x8FFE +#define GL_DOUBLE_MAT2 0x8F46 +#define GL_DOUBLE_MAT3 0x8F47 +#define GL_DOUBLE_MAT4 0x8F48 +#define GL_DOUBLE_MAT2x3 0x8F49 +#define GL_DOUBLE_MAT2x4 0x8F4A +#define GL_DOUBLE_MAT3x2 0x8F4B +#define GL_DOUBLE_MAT3x4 0x8F4C +#define GL_DOUBLE_MAT4x2 0x8F4D +#define GL_DOUBLE_MAT4x3 0x8F4E +#define GL_ACTIVE_SUBROUTINES 0x8DE5 +#define GL_ACTIVE_SUBROUTINE_UNIFORMS 0x8DE6 +#define GL_ACTIVE_SUBROUTINE_UNIFORM_LOCATIONS 0x8E47 +#define GL_ACTIVE_SUBROUTINE_MAX_LENGTH 0x8E48 +#define GL_ACTIVE_SUBROUTINE_UNIFORM_MAX_LENGTH 0x8E49 +#define GL_MAX_SUBROUTINES 0x8DE7 +#define GL_MAX_SUBROUTINE_UNIFORM_LOCATIONS 0x8DE8 +#define GL_NUM_COMPATIBLE_SUBROUTINES 0x8E4A +#define GL_COMPATIBLE_SUBROUTINES 0x8E4B +#define GL_PATCHES 0x000E +#define GL_PATCH_VERTICES 0x8E72 +#define GL_PATCH_DEFAULT_INNER_LEVEL 0x8E73 +#define GL_PATCH_DEFAULT_OUTER_LEVEL 0x8E74 +#define GL_TESS_CONTROL_OUTPUT_VERTICES 0x8E75 +#define GL_TESS_GEN_MODE 0x8E76 +#define GL_TESS_GEN_SPACING 0x8E77 +#define GL_TESS_GEN_VERTEX_ORDER 0x8E78 +#define GL_TESS_GEN_POINT_MODE 0x8E79 +#define GL_ISOLINES 0x8E7A +#define GL_FRACTIONAL_ODD 0x8E7B +#define GL_FRACTIONAL_EVEN 0x8E7C +#define GL_MAX_PATCH_VERTICES 0x8E7D +#define GL_MAX_TESS_GEN_LEVEL 0x8E7E +#define GL_MAX_TESS_CONTROL_UNIFORM_COMPONENTS 0x8E7F +#define GL_MAX_TESS_EVALUATION_UNIFORM_COMPONENTS 0x8E80 +#define GL_MAX_TESS_CONTROL_TEXTURE_IMAGE_UNITS 0x8E81 +#define GL_MAX_TESS_EVALUATION_TEXTURE_IMAGE_UNITS 0x8E82 +#define GL_MAX_TESS_CONTROL_OUTPUT_COMPONENTS 0x8E83 +#define GL_MAX_TESS_PATCH_COMPONENTS 0x8E84 +#define GL_MAX_TESS_CONTROL_TOTAL_OUTPUT_COMPONENTS 0x8E85 +#define GL_MAX_TESS_EVALUATION_OUTPUT_COMPONENTS 0x8E86 +#define GL_MAX_TESS_CONTROL_UNIFORM_BLOCKS 0x8E89 +#define GL_MAX_TESS_EVALUATION_UNIFORM_BLOCKS 0x8E8A +#define GL_MAX_TESS_CONTROL_INPUT_COMPONENTS 0x886C +#define GL_MAX_TESS_EVALUATION_INPUT_COMPONENTS 0x886D +#define GL_MAX_COMBINED_TESS_CONTROL_UNIFORM_COMPONENTS 0x8E1E +#define GL_MAX_COMBINED_TESS_EVALUATION_UNIFORM_COMPONENTS 0x8E1F +#define GL_UNIFORM_BLOCK_REFERENCED_BY_TESS_CONTROL_SHADER 0x84F0 +#define GL_UNIFORM_BLOCK_REFERENCED_BY_TESS_EVALUATION_SHADER 0x84F1 +#define GL_TESS_EVALUATION_SHADER 0x8E87 +#define GL_TESS_CONTROL_SHADER 0x8E88 +#define GL_TRANSFORM_FEEDBACK 0x8E22 +#define GL_TRANSFORM_FEEDBACK_BUFFER_PAUSED 0x8E23 +#define GL_TRANSFORM_FEEDBACK_BUFFER_ACTIVE 0x8E24 +#define GL_TRANSFORM_FEEDBACK_BINDING 0x8E25 +#define GL_MAX_TRANSFORM_FEEDBACK_BUFFERS 0x8E70 +typedef void (APIENTRYP PFNGLMINSAMPLESHADINGPROC) (GLfloat value); +typedef void (APIENTRYP PFNGLBLENDEQUATIONIPROC) (GLuint buf, GLenum mode); +typedef void (APIENTRYP PFNGLBLENDEQUATIONSEPARATEIPROC) (GLuint buf, GLenum modeRGB, GLenum modeAlpha); +typedef void (APIENTRYP PFNGLBLENDFUNCIPROC) (GLuint buf, GLenum src, GLenum dst); +typedef void (APIENTRYP PFNGLBLENDFUNCSEPARATEIPROC) (GLuint buf, GLenum srcRGB, GLenum dstRGB, GLenum srcAlpha, GLenum dstAlpha); +typedef void (APIENTRYP PFNGLDRAWARRAYSINDIRECTPROC) (GLenum mode, const void *indirect); +typedef void (APIENTRYP PFNGLDRAWELEMENTSINDIRECTPROC) (GLenum mode, GLenum type, const void *indirect); +typedef void (APIENTRYP PFNGLUNIFORM1DPROC) (GLint location, GLdouble x); +typedef void (APIENTRYP PFNGLUNIFORM2DPROC) (GLint location, GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLUNIFORM3DPROC) (GLint location, GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLUNIFORM4DPROC) (GLint location, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLUNIFORM1DVPROC) (GLint location, GLsizei count, const GLdouble *value); +typedef void (APIENTRYP PFNGLUNIFORM2DVPROC) (GLint location, GLsizei count, const GLdouble *value); +typedef void (APIENTRYP PFNGLUNIFORM3DVPROC) (GLint location, GLsizei count, const GLdouble *value); +typedef void (APIENTRYP PFNGLUNIFORM4DVPROC) (GLint location, GLsizei count, const GLdouble *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX2DVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX3DVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX4DVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX2X3DVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX2X4DVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX3X2DVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX3X4DVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX4X2DVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX4X3DVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLGETUNIFORMDVPROC) (GLuint program, GLint location, GLdouble *params); +typedef GLint (APIENTRYP PFNGLGETSUBROUTINEUNIFORMLOCATIONPROC) (GLuint program, GLenum shadertype, const GLchar *name); +typedef GLuint (APIENTRYP PFNGLGETSUBROUTINEINDEXPROC) (GLuint program, GLenum shadertype, const GLchar *name); +typedef void (APIENTRYP PFNGLGETACTIVESUBROUTINEUNIFORMIVPROC) (GLuint program, GLenum shadertype, GLuint index, GLenum pname, GLint *values); +typedef void (APIENTRYP PFNGLGETACTIVESUBROUTINEUNIFORMNAMEPROC) (GLuint program, GLenum shadertype, GLuint index, GLsizei bufSize, GLsizei *length, GLchar *name); +typedef void (APIENTRYP PFNGLGETACTIVESUBROUTINENAMEPROC) (GLuint program, GLenum shadertype, GLuint index, GLsizei bufSize, GLsizei *length, GLchar *name); +typedef void (APIENTRYP PFNGLUNIFORMSUBROUTINESUIVPROC) (GLenum shadertype, GLsizei count, const GLuint *indices); +typedef void (APIENTRYP PFNGLGETUNIFORMSUBROUTINEUIVPROC) (GLenum shadertype, GLint location, GLuint *params); +typedef void (APIENTRYP PFNGLGETPROGRAMSTAGEIVPROC) (GLuint program, GLenum shadertype, GLenum pname, GLint *values); +typedef void (APIENTRYP PFNGLPATCHPARAMETERIPROC) (GLenum pname, GLint value); +typedef void (APIENTRYP PFNGLPATCHPARAMETERFVPROC) (GLenum pname, const GLfloat *values); +typedef void (APIENTRYP PFNGLBINDTRANSFORMFEEDBACKPROC) (GLenum target, GLuint id); +typedef void (APIENTRYP PFNGLDELETETRANSFORMFEEDBACKSPROC) (GLsizei n, const GLuint *ids); +typedef void (APIENTRYP PFNGLGENTRANSFORMFEEDBACKSPROC) (GLsizei n, GLuint *ids); +typedef GLboolean (APIENTRYP PFNGLISTRANSFORMFEEDBACKPROC) (GLuint id); +typedef void (APIENTRYP PFNGLPAUSETRANSFORMFEEDBACKPROC) (void); +typedef void (APIENTRYP PFNGLRESUMETRANSFORMFEEDBACKPROC) (void); +typedef void (APIENTRYP PFNGLDRAWTRANSFORMFEEDBACKPROC) (GLenum mode, GLuint id); +typedef void (APIENTRYP PFNGLDRAWTRANSFORMFEEDBACKSTREAMPROC) (GLenum mode, GLuint id, GLuint stream); +typedef void (APIENTRYP PFNGLBEGINQUERYINDEXEDPROC) (GLenum target, GLuint index, GLuint id); +typedef void (APIENTRYP PFNGLENDQUERYINDEXEDPROC) (GLenum target, GLuint index); +typedef void (APIENTRYP PFNGLGETQUERYINDEXEDIVPROC) (GLenum target, GLuint index, GLenum pname, GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMinSampleShading (GLfloat value); +GLAPI void APIENTRY glBlendEquationi (GLuint buf, GLenum mode); +GLAPI void APIENTRY glBlendEquationSeparatei (GLuint buf, GLenum modeRGB, GLenum modeAlpha); +GLAPI void APIENTRY glBlendFunci (GLuint buf, GLenum src, GLenum dst); +GLAPI void APIENTRY glBlendFuncSeparatei (GLuint buf, GLenum srcRGB, GLenum dstRGB, GLenum srcAlpha, GLenum dstAlpha); +GLAPI void APIENTRY glDrawArraysIndirect (GLenum mode, const void *indirect); +GLAPI void APIENTRY glDrawElementsIndirect (GLenum mode, GLenum type, const void *indirect); +GLAPI void APIENTRY glUniform1d (GLint location, GLdouble x); +GLAPI void APIENTRY glUniform2d (GLint location, GLdouble x, GLdouble y); +GLAPI void APIENTRY glUniform3d (GLint location, GLdouble x, GLdouble y, GLdouble z); +GLAPI void APIENTRY glUniform4d (GLint location, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +GLAPI void APIENTRY glUniform1dv (GLint location, GLsizei count, const GLdouble *value); +GLAPI void APIENTRY glUniform2dv (GLint location, GLsizei count, const GLdouble *value); +GLAPI void APIENTRY glUniform3dv (GLint location, GLsizei count, const GLdouble *value); +GLAPI void APIENTRY glUniform4dv (GLint location, GLsizei count, const GLdouble *value); +GLAPI void APIENTRY glUniformMatrix2dv (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glUniformMatrix3dv (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glUniformMatrix4dv (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glUniformMatrix2x3dv (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glUniformMatrix2x4dv (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glUniformMatrix3x2dv (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glUniformMatrix3x4dv (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glUniformMatrix4x2dv (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glUniformMatrix4x3dv (GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glGetUniformdv (GLuint program, GLint location, GLdouble *params); +GLAPI GLint APIENTRY glGetSubroutineUniformLocation (GLuint program, GLenum shadertype, const GLchar *name); +GLAPI GLuint APIENTRY glGetSubroutineIndex (GLuint program, GLenum shadertype, const GLchar *name); +GLAPI void APIENTRY glGetActiveSubroutineUniformiv (GLuint program, GLenum shadertype, GLuint index, GLenum pname, GLint *values); +GLAPI void APIENTRY glGetActiveSubroutineUniformName (GLuint program, GLenum shadertype, GLuint index, GLsizei bufSize, GLsizei *length, GLchar *name); +GLAPI void APIENTRY glGetActiveSubroutineName (GLuint program, GLenum shadertype, GLuint index, GLsizei bufSize, GLsizei *length, GLchar *name); +GLAPI void APIENTRY glUniformSubroutinesuiv (GLenum shadertype, GLsizei count, const GLuint *indices); +GLAPI void APIENTRY glGetUniformSubroutineuiv (GLenum shadertype, GLint location, GLuint *params); +GLAPI void APIENTRY glGetProgramStageiv (GLuint program, GLenum shadertype, GLenum pname, GLint *values); +GLAPI void APIENTRY glPatchParameteri (GLenum pname, GLint value); +GLAPI void APIENTRY glPatchParameterfv (GLenum pname, const GLfloat *values); +GLAPI void APIENTRY glBindTransformFeedback (GLenum target, GLuint id); +GLAPI void APIENTRY glDeleteTransformFeedbacks (GLsizei n, const GLuint *ids); +GLAPI void APIENTRY glGenTransformFeedbacks (GLsizei n, GLuint *ids); +GLAPI GLboolean APIENTRY glIsTransformFeedback (GLuint id); +GLAPI void APIENTRY glPauseTransformFeedback (void); +GLAPI void APIENTRY glResumeTransformFeedback (void); +GLAPI void APIENTRY glDrawTransformFeedback (GLenum mode, GLuint id); +GLAPI void APIENTRY glDrawTransformFeedbackStream (GLenum mode, GLuint id, GLuint stream); +GLAPI void APIENTRY glBeginQueryIndexed (GLenum target, GLuint index, GLuint id); +GLAPI void APIENTRY glEndQueryIndexed (GLenum target, GLuint index); +GLAPI void APIENTRY glGetQueryIndexediv (GLenum target, GLuint index, GLenum pname, GLint *params); +#endif +#endif /* GL_VERSION_4_0 */ + +#ifndef GL_VERSION_4_1 +#define GL_VERSION_4_1 1 +#define GL_FIXED 0x140C +#define GL_IMPLEMENTATION_COLOR_READ_TYPE 0x8B9A +#define GL_IMPLEMENTATION_COLOR_READ_FORMAT 0x8B9B +#define GL_LOW_FLOAT 0x8DF0 +#define GL_MEDIUM_FLOAT 0x8DF1 +#define GL_HIGH_FLOAT 0x8DF2 +#define GL_LOW_INT 0x8DF3 +#define GL_MEDIUM_INT 0x8DF4 +#define GL_HIGH_INT 0x8DF5 +#define GL_SHADER_COMPILER 0x8DFA +#define GL_SHADER_BINARY_FORMATS 0x8DF8 +#define GL_NUM_SHADER_BINARY_FORMATS 0x8DF9 +#define GL_MAX_VERTEX_UNIFORM_VECTORS 0x8DFB +#define GL_MAX_VARYING_VECTORS 0x8DFC +#define GL_MAX_FRAGMENT_UNIFORM_VECTORS 0x8DFD +#define GL_RGB565 0x8D62 +#define GL_PROGRAM_BINARY_RETRIEVABLE_HINT 0x8257 +#define GL_PROGRAM_BINARY_LENGTH 0x8741 +#define GL_NUM_PROGRAM_BINARY_FORMATS 0x87FE +#define GL_PROGRAM_BINARY_FORMATS 0x87FF +#define GL_VERTEX_SHADER_BIT 0x00000001 +#define GL_FRAGMENT_SHADER_BIT 0x00000002 +#define GL_GEOMETRY_SHADER_BIT 0x00000004 +#define GL_TESS_CONTROL_SHADER_BIT 0x00000008 +#define GL_TESS_EVALUATION_SHADER_BIT 0x00000010 +#define GL_ALL_SHADER_BITS 0xFFFFFFFF +#define GL_PROGRAM_SEPARABLE 0x8258 +#define GL_ACTIVE_PROGRAM 0x8259 +#define GL_PROGRAM_PIPELINE_BINDING 0x825A +#define GL_MAX_VIEWPORTS 0x825B +#define GL_VIEWPORT_SUBPIXEL_BITS 0x825C +#define GL_VIEWPORT_BOUNDS_RANGE 0x825D +#define GL_LAYER_PROVOKING_VERTEX 0x825E +#define GL_VIEWPORT_INDEX_PROVOKING_VERTEX 0x825F +#define GL_UNDEFINED_VERTEX 0x8260 +typedef void (APIENTRYP PFNGLRELEASESHADERCOMPILERPROC) (void); +typedef void (APIENTRYP PFNGLSHADERBINARYPROC) (GLsizei count, const GLuint *shaders, GLenum binaryFormat, const void *binary, GLsizei length); +typedef void (APIENTRYP PFNGLGETSHADERPRECISIONFORMATPROC) (GLenum shadertype, GLenum precisiontype, GLint *range, GLint *precision); +typedef void (APIENTRYP PFNGLDEPTHRANGEFPROC) (GLfloat n, GLfloat f); +typedef void (APIENTRYP PFNGLCLEARDEPTHFPROC) (GLfloat d); +typedef void (APIENTRYP PFNGLGETPROGRAMBINARYPROC) (GLuint program, GLsizei bufSize, GLsizei *length, GLenum *binaryFormat, void *binary); +typedef void (APIENTRYP PFNGLPROGRAMBINARYPROC) (GLuint program, GLenum binaryFormat, const void *binary, GLsizei length); +typedef void (APIENTRYP PFNGLPROGRAMPARAMETERIPROC) (GLuint program, GLenum pname, GLint value); +typedef void (APIENTRYP PFNGLUSEPROGRAMSTAGESPROC) (GLuint pipeline, GLbitfield stages, GLuint program); +typedef void (APIENTRYP PFNGLACTIVESHADERPROGRAMPROC) (GLuint pipeline, GLuint program); +typedef GLuint (APIENTRYP PFNGLCREATESHADERPROGRAMVPROC) (GLenum type, GLsizei count, const GLchar *const*strings); +typedef void (APIENTRYP PFNGLBINDPROGRAMPIPELINEPROC) (GLuint pipeline); +typedef void (APIENTRYP PFNGLDELETEPROGRAMPIPELINESPROC) (GLsizei n, const GLuint *pipelines); +typedef void (APIENTRYP PFNGLGENPROGRAMPIPELINESPROC) (GLsizei n, GLuint *pipelines); +typedef GLboolean (APIENTRYP PFNGLISPROGRAMPIPELINEPROC) (GLuint pipeline); +typedef void (APIENTRYP PFNGLGETPROGRAMPIPELINEIVPROC) (GLuint pipeline, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1IPROC) (GLuint program, GLint location, GLint v0); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1IVPROC) (GLuint program, GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1FPROC) (GLuint program, GLint location, GLfloat v0); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1FVPROC) (GLuint program, GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1DPROC) (GLuint program, GLint location, GLdouble v0); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1DVPROC) (GLuint program, GLint location, GLsizei count, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1UIPROC) (GLuint program, GLint location, GLuint v0); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1UIVPROC) (GLuint program, GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2IPROC) (GLuint program, GLint location, GLint v0, GLint v1); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2IVPROC) (GLuint program, GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2FPROC) (GLuint program, GLint location, GLfloat v0, GLfloat v1); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2FVPROC) (GLuint program, GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2DPROC) (GLuint program, GLint location, GLdouble v0, GLdouble v1); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2DVPROC) (GLuint program, GLint location, GLsizei count, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2UIPROC) (GLuint program, GLint location, GLuint v0, GLuint v1); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2UIVPROC) (GLuint program, GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3IPROC) (GLuint program, GLint location, GLint v0, GLint v1, GLint v2); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3IVPROC) (GLuint program, GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3FPROC) (GLuint program, GLint location, GLfloat v0, GLfloat v1, GLfloat v2); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3FVPROC) (GLuint program, GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3DPROC) (GLuint program, GLint location, GLdouble v0, GLdouble v1, GLdouble v2); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3DVPROC) (GLuint program, GLint location, GLsizei count, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3UIPROC) (GLuint program, GLint location, GLuint v0, GLuint v1, GLuint v2); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3UIVPROC) (GLuint program, GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4IPROC) (GLuint program, GLint location, GLint v0, GLint v1, GLint v2, GLint v3); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4IVPROC) (GLuint program, GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4FPROC) (GLuint program, GLint location, GLfloat v0, GLfloat v1, GLfloat v2, GLfloat v3); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4FVPROC) (GLuint program, GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4DPROC) (GLuint program, GLint location, GLdouble v0, GLdouble v1, GLdouble v2, GLdouble v3); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4DVPROC) (GLuint program, GLint location, GLsizei count, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4UIPROC) (GLuint program, GLint location, GLuint v0, GLuint v1, GLuint v2, GLuint v3); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4UIVPROC) (GLuint program, GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX2FVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX3FVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX4FVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX2DVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX3DVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX4DVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX2X3FVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX3X2FVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX2X4FVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX4X2FVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX3X4FVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX4X3FVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX2X3DVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX3X2DVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX2X4DVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX4X2DVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX3X4DVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX4X3DVPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLVALIDATEPROGRAMPIPELINEPROC) (GLuint pipeline); +typedef void (APIENTRYP PFNGLGETPROGRAMPIPELINEINFOLOGPROC) (GLuint pipeline, GLsizei bufSize, GLsizei *length, GLchar *infoLog); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL1DPROC) (GLuint index, GLdouble x); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL2DPROC) (GLuint index, GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL3DPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL4DPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL1DVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL2DVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL3DVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL4DVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBLPOINTERPROC) (GLuint index, GLint size, GLenum type, GLsizei stride, const void *pointer); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBLDVPROC) (GLuint index, GLenum pname, GLdouble *params); +typedef void (APIENTRYP PFNGLVIEWPORTARRAYVPROC) (GLuint first, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLVIEWPORTINDEXEDFPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat w, GLfloat h); +typedef void (APIENTRYP PFNGLVIEWPORTINDEXEDFVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLSCISSORARRAYVPROC) (GLuint first, GLsizei count, const GLint *v); +typedef void (APIENTRYP PFNGLSCISSORINDEXEDPROC) (GLuint index, GLint left, GLint bottom, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLSCISSORINDEXEDVPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLDEPTHRANGEARRAYVPROC) (GLuint first, GLsizei count, const GLdouble *v); +typedef void (APIENTRYP PFNGLDEPTHRANGEINDEXEDPROC) (GLuint index, GLdouble n, GLdouble f); +typedef void (APIENTRYP PFNGLGETFLOATI_VPROC) (GLenum target, GLuint index, GLfloat *data); +typedef void (APIENTRYP PFNGLGETDOUBLEI_VPROC) (GLenum target, GLuint index, GLdouble *data); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glReleaseShaderCompiler (void); +GLAPI void APIENTRY glShaderBinary (GLsizei count, const GLuint *shaders, GLenum binaryFormat, const void *binary, GLsizei length); +GLAPI void APIENTRY glGetShaderPrecisionFormat (GLenum shadertype, GLenum precisiontype, GLint *range, GLint *precision); +GLAPI void APIENTRY glDepthRangef (GLfloat n, GLfloat f); +GLAPI void APIENTRY glClearDepthf (GLfloat d); +GLAPI void APIENTRY glGetProgramBinary (GLuint program, GLsizei bufSize, GLsizei *length, GLenum *binaryFormat, void *binary); +GLAPI void APIENTRY glProgramBinary (GLuint program, GLenum binaryFormat, const void *binary, GLsizei length); +GLAPI void APIENTRY glProgramParameteri (GLuint program, GLenum pname, GLint value); +GLAPI void APIENTRY glUseProgramStages (GLuint pipeline, GLbitfield stages, GLuint program); +GLAPI void APIENTRY glActiveShaderProgram (GLuint pipeline, GLuint program); +GLAPI GLuint APIENTRY glCreateShaderProgramv (GLenum type, GLsizei count, const GLchar *const*strings); +GLAPI void APIENTRY glBindProgramPipeline (GLuint pipeline); +GLAPI void APIENTRY glDeleteProgramPipelines (GLsizei n, const GLuint *pipelines); +GLAPI void APIENTRY glGenProgramPipelines (GLsizei n, GLuint *pipelines); +GLAPI GLboolean APIENTRY glIsProgramPipeline (GLuint pipeline); +GLAPI void APIENTRY glGetProgramPipelineiv (GLuint pipeline, GLenum pname, GLint *params); +GLAPI void APIENTRY glProgramUniform1i (GLuint program, GLint location, GLint v0); +GLAPI void APIENTRY glProgramUniform1iv (GLuint program, GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glProgramUniform1f (GLuint program, GLint location, GLfloat v0); +GLAPI void APIENTRY glProgramUniform1fv (GLuint program, GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glProgramUniform1d (GLuint program, GLint location, GLdouble v0); +GLAPI void APIENTRY glProgramUniform1dv (GLuint program, GLint location, GLsizei count, const GLdouble *value); +GLAPI void APIENTRY glProgramUniform1ui (GLuint program, GLint location, GLuint v0); +GLAPI void APIENTRY glProgramUniform1uiv (GLuint program, GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glProgramUniform2i (GLuint program, GLint location, GLint v0, GLint v1); +GLAPI void APIENTRY glProgramUniform2iv (GLuint program, GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glProgramUniform2f (GLuint program, GLint location, GLfloat v0, GLfloat v1); +GLAPI void APIENTRY glProgramUniform2fv (GLuint program, GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glProgramUniform2d (GLuint program, GLint location, GLdouble v0, GLdouble v1); +GLAPI void APIENTRY glProgramUniform2dv (GLuint program, GLint location, GLsizei count, const GLdouble *value); +GLAPI void APIENTRY glProgramUniform2ui (GLuint program, GLint location, GLuint v0, GLuint v1); +GLAPI void APIENTRY glProgramUniform2uiv (GLuint program, GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glProgramUniform3i (GLuint program, GLint location, GLint v0, GLint v1, GLint v2); +GLAPI void APIENTRY glProgramUniform3iv (GLuint program, GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glProgramUniform3f (GLuint program, GLint location, GLfloat v0, GLfloat v1, GLfloat v2); +GLAPI void APIENTRY glProgramUniform3fv (GLuint program, GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glProgramUniform3d (GLuint program, GLint location, GLdouble v0, GLdouble v1, GLdouble v2); +GLAPI void APIENTRY glProgramUniform3dv (GLuint program, GLint location, GLsizei count, const GLdouble *value); +GLAPI void APIENTRY glProgramUniform3ui (GLuint program, GLint location, GLuint v0, GLuint v1, GLuint v2); +GLAPI void APIENTRY glProgramUniform3uiv (GLuint program, GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glProgramUniform4i (GLuint program, GLint location, GLint v0, GLint v1, GLint v2, GLint v3); +GLAPI void APIENTRY glProgramUniform4iv (GLuint program, GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glProgramUniform4f (GLuint program, GLint location, GLfloat v0, GLfloat v1, GLfloat v2, GLfloat v3); +GLAPI void APIENTRY glProgramUniform4fv (GLuint program, GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glProgramUniform4d (GLuint program, GLint location, GLdouble v0, GLdouble v1, GLdouble v2, GLdouble v3); +GLAPI void APIENTRY glProgramUniform4dv (GLuint program, GLint location, GLsizei count, const GLdouble *value); +GLAPI void APIENTRY glProgramUniform4ui (GLuint program, GLint location, GLuint v0, GLuint v1, GLuint v2, GLuint v3); +GLAPI void APIENTRY glProgramUniform4uiv (GLuint program, GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glProgramUniformMatrix2fv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix3fv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix4fv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix2dv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix3dv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix4dv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix2x3fv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix3x2fv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix2x4fv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix4x2fv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix3x4fv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix4x3fv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix2x3dv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix3x2dv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix2x4dv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix4x2dv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix3x4dv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix4x3dv (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glValidateProgramPipeline (GLuint pipeline); +GLAPI void APIENTRY glGetProgramPipelineInfoLog (GLuint pipeline, GLsizei bufSize, GLsizei *length, GLchar *infoLog); +GLAPI void APIENTRY glVertexAttribL1d (GLuint index, GLdouble x); +GLAPI void APIENTRY glVertexAttribL2d (GLuint index, GLdouble x, GLdouble y); +GLAPI void APIENTRY glVertexAttribL3d (GLuint index, GLdouble x, GLdouble y, GLdouble z); +GLAPI void APIENTRY glVertexAttribL4d (GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +GLAPI void APIENTRY glVertexAttribL1dv (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttribL2dv (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttribL3dv (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttribL4dv (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttribLPointer (GLuint index, GLint size, GLenum type, GLsizei stride, const void *pointer); +GLAPI void APIENTRY glGetVertexAttribLdv (GLuint index, GLenum pname, GLdouble *params); +GLAPI void APIENTRY glViewportArrayv (GLuint first, GLsizei count, const GLfloat *v); +GLAPI void APIENTRY glViewportIndexedf (GLuint index, GLfloat x, GLfloat y, GLfloat w, GLfloat h); +GLAPI void APIENTRY glViewportIndexedfv (GLuint index, const GLfloat *v); +GLAPI void APIENTRY glScissorArrayv (GLuint first, GLsizei count, const GLint *v); +GLAPI void APIENTRY glScissorIndexed (GLuint index, GLint left, GLint bottom, GLsizei width, GLsizei height); +GLAPI void APIENTRY glScissorIndexedv (GLuint index, const GLint *v); +GLAPI void APIENTRY glDepthRangeArrayv (GLuint first, GLsizei count, const GLdouble *v); +GLAPI void APIENTRY glDepthRangeIndexed (GLuint index, GLdouble n, GLdouble f); +GLAPI void APIENTRY glGetFloati_v (GLenum target, GLuint index, GLfloat *data); +GLAPI void APIENTRY glGetDoublei_v (GLenum target, GLuint index, GLdouble *data); +#endif +#endif /* GL_VERSION_4_1 */ + +#ifndef GL_VERSION_4_2 +#define GL_VERSION_4_2 1 +#define GL_COPY_READ_BUFFER_BINDING 0x8F36 +#define GL_COPY_WRITE_BUFFER_BINDING 0x8F37 +#define GL_TRANSFORM_FEEDBACK_ACTIVE 0x8E24 +#define GL_TRANSFORM_FEEDBACK_PAUSED 0x8E23 +#define GL_UNPACK_COMPRESSED_BLOCK_WIDTH 0x9127 +#define GL_UNPACK_COMPRESSED_BLOCK_HEIGHT 0x9128 +#define GL_UNPACK_COMPRESSED_BLOCK_DEPTH 0x9129 +#define GL_UNPACK_COMPRESSED_BLOCK_SIZE 0x912A +#define GL_PACK_COMPRESSED_BLOCK_WIDTH 0x912B +#define GL_PACK_COMPRESSED_BLOCK_HEIGHT 0x912C +#define GL_PACK_COMPRESSED_BLOCK_DEPTH 0x912D +#define GL_PACK_COMPRESSED_BLOCK_SIZE 0x912E +#define GL_NUM_SAMPLE_COUNTS 0x9380 +#define GL_MIN_MAP_BUFFER_ALIGNMENT 0x90BC +#define GL_ATOMIC_COUNTER_BUFFER 0x92C0 +#define GL_ATOMIC_COUNTER_BUFFER_BINDING 0x92C1 +#define GL_ATOMIC_COUNTER_BUFFER_START 0x92C2 +#define GL_ATOMIC_COUNTER_BUFFER_SIZE 0x92C3 +#define GL_ATOMIC_COUNTER_BUFFER_DATA_SIZE 0x92C4 +#define GL_ATOMIC_COUNTER_BUFFER_ACTIVE_ATOMIC_COUNTERS 0x92C5 +#define GL_ATOMIC_COUNTER_BUFFER_ACTIVE_ATOMIC_COUNTER_INDICES 0x92C6 +#define GL_ATOMIC_COUNTER_BUFFER_REFERENCED_BY_VERTEX_SHADER 0x92C7 +#define GL_ATOMIC_COUNTER_BUFFER_REFERENCED_BY_TESS_CONTROL_SHADER 0x92C8 +#define GL_ATOMIC_COUNTER_BUFFER_REFERENCED_BY_TESS_EVALUATION_SHADER 0x92C9 +#define GL_ATOMIC_COUNTER_BUFFER_REFERENCED_BY_GEOMETRY_SHADER 0x92CA +#define GL_ATOMIC_COUNTER_BUFFER_REFERENCED_BY_FRAGMENT_SHADER 0x92CB +#define GL_MAX_VERTEX_ATOMIC_COUNTER_BUFFERS 0x92CC +#define GL_MAX_TESS_CONTROL_ATOMIC_COUNTER_BUFFERS 0x92CD +#define GL_MAX_TESS_EVALUATION_ATOMIC_COUNTER_BUFFERS 0x92CE +#define GL_MAX_GEOMETRY_ATOMIC_COUNTER_BUFFERS 0x92CF +#define GL_MAX_FRAGMENT_ATOMIC_COUNTER_BUFFERS 0x92D0 +#define GL_MAX_COMBINED_ATOMIC_COUNTER_BUFFERS 0x92D1 +#define GL_MAX_VERTEX_ATOMIC_COUNTERS 0x92D2 +#define GL_MAX_TESS_CONTROL_ATOMIC_COUNTERS 0x92D3 +#define GL_MAX_TESS_EVALUATION_ATOMIC_COUNTERS 0x92D4 +#define GL_MAX_GEOMETRY_ATOMIC_COUNTERS 0x92D5 +#define GL_MAX_FRAGMENT_ATOMIC_COUNTERS 0x92D6 +#define GL_MAX_COMBINED_ATOMIC_COUNTERS 0x92D7 +#define GL_MAX_ATOMIC_COUNTER_BUFFER_SIZE 0x92D8 +#define GL_MAX_ATOMIC_COUNTER_BUFFER_BINDINGS 0x92DC +#define GL_ACTIVE_ATOMIC_COUNTER_BUFFERS 0x92D9 +#define GL_UNIFORM_ATOMIC_COUNTER_BUFFER_INDEX 0x92DA +#define GL_UNSIGNED_INT_ATOMIC_COUNTER 0x92DB +#define GL_VERTEX_ATTRIB_ARRAY_BARRIER_BIT 0x00000001 +#define GL_ELEMENT_ARRAY_BARRIER_BIT 0x00000002 +#define GL_UNIFORM_BARRIER_BIT 0x00000004 +#define GL_TEXTURE_FETCH_BARRIER_BIT 0x00000008 +#define GL_SHADER_IMAGE_ACCESS_BARRIER_BIT 0x00000020 +#define GL_COMMAND_BARRIER_BIT 0x00000040 +#define GL_PIXEL_BUFFER_BARRIER_BIT 0x00000080 +#define GL_TEXTURE_UPDATE_BARRIER_BIT 0x00000100 +#define GL_BUFFER_UPDATE_BARRIER_BIT 0x00000200 +#define GL_FRAMEBUFFER_BARRIER_BIT 0x00000400 +#define GL_TRANSFORM_FEEDBACK_BARRIER_BIT 0x00000800 +#define GL_ATOMIC_COUNTER_BARRIER_BIT 0x00001000 +#define GL_ALL_BARRIER_BITS 0xFFFFFFFF +#define GL_MAX_IMAGE_UNITS 0x8F38 +#define GL_MAX_COMBINED_IMAGE_UNITS_AND_FRAGMENT_OUTPUTS 0x8F39 +#define GL_IMAGE_BINDING_NAME 0x8F3A +#define GL_IMAGE_BINDING_LEVEL 0x8F3B +#define GL_IMAGE_BINDING_LAYERED 0x8F3C +#define GL_IMAGE_BINDING_LAYER 0x8F3D +#define GL_IMAGE_BINDING_ACCESS 0x8F3E +#define GL_IMAGE_1D 0x904C +#define GL_IMAGE_2D 0x904D +#define GL_IMAGE_3D 0x904E +#define GL_IMAGE_2D_RECT 0x904F +#define GL_IMAGE_CUBE 0x9050 +#define GL_IMAGE_BUFFER 0x9051 +#define GL_IMAGE_1D_ARRAY 0x9052 +#define GL_IMAGE_2D_ARRAY 0x9053 +#define GL_IMAGE_CUBE_MAP_ARRAY 0x9054 +#define GL_IMAGE_2D_MULTISAMPLE 0x9055 +#define GL_IMAGE_2D_MULTISAMPLE_ARRAY 0x9056 +#define GL_INT_IMAGE_1D 0x9057 +#define GL_INT_IMAGE_2D 0x9058 +#define GL_INT_IMAGE_3D 0x9059 +#define GL_INT_IMAGE_2D_RECT 0x905A +#define GL_INT_IMAGE_CUBE 0x905B +#define GL_INT_IMAGE_BUFFER 0x905C +#define GL_INT_IMAGE_1D_ARRAY 0x905D +#define GL_INT_IMAGE_2D_ARRAY 0x905E +#define GL_INT_IMAGE_CUBE_MAP_ARRAY 0x905F +#define GL_INT_IMAGE_2D_MULTISAMPLE 0x9060 +#define GL_INT_IMAGE_2D_MULTISAMPLE_ARRAY 0x9061 +#define GL_UNSIGNED_INT_IMAGE_1D 0x9062 +#define GL_UNSIGNED_INT_IMAGE_2D 0x9063 +#define GL_UNSIGNED_INT_IMAGE_3D 0x9064 +#define GL_UNSIGNED_INT_IMAGE_2D_RECT 0x9065 +#define GL_UNSIGNED_INT_IMAGE_CUBE 0x9066 +#define GL_UNSIGNED_INT_IMAGE_BUFFER 0x9067 +#define GL_UNSIGNED_INT_IMAGE_1D_ARRAY 0x9068 +#define GL_UNSIGNED_INT_IMAGE_2D_ARRAY 0x9069 +#define GL_UNSIGNED_INT_IMAGE_CUBE_MAP_ARRAY 0x906A +#define GL_UNSIGNED_INT_IMAGE_2D_MULTISAMPLE 0x906B +#define GL_UNSIGNED_INT_IMAGE_2D_MULTISAMPLE_ARRAY 0x906C +#define GL_MAX_IMAGE_SAMPLES 0x906D +#define GL_IMAGE_BINDING_FORMAT 0x906E +#define GL_IMAGE_FORMAT_COMPATIBILITY_TYPE 0x90C7 +#define GL_IMAGE_FORMAT_COMPATIBILITY_BY_SIZE 0x90C8 +#define GL_IMAGE_FORMAT_COMPATIBILITY_BY_CLASS 0x90C9 +#define GL_MAX_VERTEX_IMAGE_UNIFORMS 0x90CA +#define GL_MAX_TESS_CONTROL_IMAGE_UNIFORMS 0x90CB +#define GL_MAX_TESS_EVALUATION_IMAGE_UNIFORMS 0x90CC +#define GL_MAX_GEOMETRY_IMAGE_UNIFORMS 0x90CD +#define GL_MAX_FRAGMENT_IMAGE_UNIFORMS 0x90CE +#define GL_MAX_COMBINED_IMAGE_UNIFORMS 0x90CF +#define GL_COMPRESSED_RGBA_BPTC_UNORM 0x8E8C +#define GL_COMPRESSED_SRGB_ALPHA_BPTC_UNORM 0x8E8D +#define GL_COMPRESSED_RGB_BPTC_SIGNED_FLOAT 0x8E8E +#define GL_COMPRESSED_RGB_BPTC_UNSIGNED_FLOAT 0x8E8F +#define GL_TEXTURE_IMMUTABLE_FORMAT 0x912F +typedef void (APIENTRYP PFNGLDRAWARRAYSINSTANCEDBASEINSTANCEPROC) (GLenum mode, GLint first, GLsizei count, GLsizei instancecount, GLuint baseinstance); +typedef void (APIENTRYP PFNGLDRAWELEMENTSINSTANCEDBASEINSTANCEPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount, GLuint baseinstance); +typedef void (APIENTRYP PFNGLDRAWELEMENTSINSTANCEDBASEVERTEXBASEINSTANCEPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount, GLint basevertex, GLuint baseinstance); +typedef void (APIENTRYP PFNGLGETINTERNALFORMATIVPROC) (GLenum target, GLenum internalformat, GLenum pname, GLsizei count, GLint *params); +typedef void (APIENTRYP PFNGLGETACTIVEATOMICCOUNTERBUFFERIVPROC) (GLuint program, GLuint bufferIndex, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLBINDIMAGETEXTUREPROC) (GLuint unit, GLuint texture, GLint level, GLboolean layered, GLint layer, GLenum access, GLenum format); +typedef void (APIENTRYP PFNGLMEMORYBARRIERPROC) (GLbitfield barriers); +typedef void (APIENTRYP PFNGLTEXSTORAGE1DPROC) (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width); +typedef void (APIENTRYP PFNGLTEXSTORAGE2DPROC) (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLTEXSTORAGE3DPROC) (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth); +typedef void (APIENTRYP PFNGLDRAWTRANSFORMFEEDBACKINSTANCEDPROC) (GLenum mode, GLuint id, GLsizei instancecount); +typedef void (APIENTRYP PFNGLDRAWTRANSFORMFEEDBACKSTREAMINSTANCEDPROC) (GLenum mode, GLuint id, GLuint stream, GLsizei instancecount); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawArraysInstancedBaseInstance (GLenum mode, GLint first, GLsizei count, GLsizei instancecount, GLuint baseinstance); +GLAPI void APIENTRY glDrawElementsInstancedBaseInstance (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount, GLuint baseinstance); +GLAPI void APIENTRY glDrawElementsInstancedBaseVertexBaseInstance (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount, GLint basevertex, GLuint baseinstance); +GLAPI void APIENTRY glGetInternalformativ (GLenum target, GLenum internalformat, GLenum pname, GLsizei count, GLint *params); +GLAPI void APIENTRY glGetActiveAtomicCounterBufferiv (GLuint program, GLuint bufferIndex, GLenum pname, GLint *params); +GLAPI void APIENTRY glBindImageTexture (GLuint unit, GLuint texture, GLint level, GLboolean layered, GLint layer, GLenum access, GLenum format); +GLAPI void APIENTRY glMemoryBarrier (GLbitfield barriers); +GLAPI void APIENTRY glTexStorage1D (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width); +GLAPI void APIENTRY glTexStorage2D (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height); +GLAPI void APIENTRY glTexStorage3D (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth); +GLAPI void APIENTRY glDrawTransformFeedbackInstanced (GLenum mode, GLuint id, GLsizei instancecount); +GLAPI void APIENTRY glDrawTransformFeedbackStreamInstanced (GLenum mode, GLuint id, GLuint stream, GLsizei instancecount); +#endif +#endif /* GL_VERSION_4_2 */ + +#ifndef GL_VERSION_4_3 +#define GL_VERSION_4_3 1 +typedef void (APIENTRY *GLDEBUGPROC)(GLenum source,GLenum type,GLuint id,GLenum severity,GLsizei length,const GLchar *message,const void *userParam); +#define GL_NUM_SHADING_LANGUAGE_VERSIONS 0x82E9 +#define GL_VERTEX_ATTRIB_ARRAY_LONG 0x874E +#define GL_COMPRESSED_RGB8_ETC2 0x9274 +#define GL_COMPRESSED_SRGB8_ETC2 0x9275 +#define GL_COMPRESSED_RGB8_PUNCHTHROUGH_ALPHA1_ETC2 0x9276 +#define GL_COMPRESSED_SRGB8_PUNCHTHROUGH_ALPHA1_ETC2 0x9277 +#define GL_COMPRESSED_RGBA8_ETC2_EAC 0x9278 +#define GL_COMPRESSED_SRGB8_ALPHA8_ETC2_EAC 0x9279 +#define GL_COMPRESSED_R11_EAC 0x9270 +#define GL_COMPRESSED_SIGNED_R11_EAC 0x9271 +#define GL_COMPRESSED_RG11_EAC 0x9272 +#define GL_COMPRESSED_SIGNED_RG11_EAC 0x9273 +#define GL_PRIMITIVE_RESTART_FIXED_INDEX 0x8D69 +#define GL_ANY_SAMPLES_PASSED_CONSERVATIVE 0x8D6A +#define GL_MAX_ELEMENT_INDEX 0x8D6B +#define GL_COMPUTE_SHADER 0x91B9 +#define GL_MAX_COMPUTE_UNIFORM_BLOCKS 0x91BB +#define GL_MAX_COMPUTE_TEXTURE_IMAGE_UNITS 0x91BC +#define GL_MAX_COMPUTE_IMAGE_UNIFORMS 0x91BD +#define GL_MAX_COMPUTE_SHARED_MEMORY_SIZE 0x8262 +#define GL_MAX_COMPUTE_UNIFORM_COMPONENTS 0x8263 +#define GL_MAX_COMPUTE_ATOMIC_COUNTER_BUFFERS 0x8264 +#define GL_MAX_COMPUTE_ATOMIC_COUNTERS 0x8265 +#define GL_MAX_COMBINED_COMPUTE_UNIFORM_COMPONENTS 0x8266 +#define GL_MAX_COMPUTE_WORK_GROUP_INVOCATIONS 0x90EB +#define GL_MAX_COMPUTE_WORK_GROUP_COUNT 0x91BE +#define GL_MAX_COMPUTE_WORK_GROUP_SIZE 0x91BF +#define GL_COMPUTE_WORK_GROUP_SIZE 0x8267 +#define GL_UNIFORM_BLOCK_REFERENCED_BY_COMPUTE_SHADER 0x90EC +#define GL_ATOMIC_COUNTER_BUFFER_REFERENCED_BY_COMPUTE_SHADER 0x90ED +#define GL_DISPATCH_INDIRECT_BUFFER 0x90EE +#define GL_DISPATCH_INDIRECT_BUFFER_BINDING 0x90EF +#define GL_COMPUTE_SHADER_BIT 0x00000020 +#define GL_DEBUG_OUTPUT_SYNCHRONOUS 0x8242 +#define GL_DEBUG_NEXT_LOGGED_MESSAGE_LENGTH 0x8243 +#define GL_DEBUG_CALLBACK_FUNCTION 0x8244 +#define GL_DEBUG_CALLBACK_USER_PARAM 0x8245 +#define GL_DEBUG_SOURCE_API 0x8246 +#define GL_DEBUG_SOURCE_WINDOW_SYSTEM 0x8247 +#define GL_DEBUG_SOURCE_SHADER_COMPILER 0x8248 +#define GL_DEBUG_SOURCE_THIRD_PARTY 0x8249 +#define GL_DEBUG_SOURCE_APPLICATION 0x824A +#define GL_DEBUG_SOURCE_OTHER 0x824B +#define GL_DEBUG_TYPE_ERROR 0x824C +#define GL_DEBUG_TYPE_DEPRECATED_BEHAVIOR 0x824D +#define GL_DEBUG_TYPE_UNDEFINED_BEHAVIOR 0x824E +#define GL_DEBUG_TYPE_PORTABILITY 0x824F +#define GL_DEBUG_TYPE_PERFORMANCE 0x8250 +#define GL_DEBUG_TYPE_OTHER 0x8251 +#define GL_MAX_DEBUG_MESSAGE_LENGTH 0x9143 +#define GL_MAX_DEBUG_LOGGED_MESSAGES 0x9144 +#define GL_DEBUG_LOGGED_MESSAGES 0x9145 +#define GL_DEBUG_SEVERITY_HIGH 0x9146 +#define GL_DEBUG_SEVERITY_MEDIUM 0x9147 +#define GL_DEBUG_SEVERITY_LOW 0x9148 +#define GL_DEBUG_TYPE_MARKER 0x8268 +#define GL_DEBUG_TYPE_PUSH_GROUP 0x8269 +#define GL_DEBUG_TYPE_POP_GROUP 0x826A +#define GL_DEBUG_SEVERITY_NOTIFICATION 0x826B +#define GL_MAX_DEBUG_GROUP_STACK_DEPTH 0x826C +#define GL_DEBUG_GROUP_STACK_DEPTH 0x826D +#define GL_BUFFER 0x82E0 +#define GL_SHADER 0x82E1 +#define GL_PROGRAM 0x82E2 +#define GL_QUERY 0x82E3 +#define GL_PROGRAM_PIPELINE 0x82E4 +#define GL_SAMPLER 0x82E6 +#define GL_MAX_LABEL_LENGTH 0x82E8 +#define GL_DEBUG_OUTPUT 0x92E0 +#define GL_CONTEXT_FLAG_DEBUG_BIT 0x00000002 +#define GL_MAX_UNIFORM_LOCATIONS 0x826E +#define GL_FRAMEBUFFER_DEFAULT_WIDTH 0x9310 +#define GL_FRAMEBUFFER_DEFAULT_HEIGHT 0x9311 +#define GL_FRAMEBUFFER_DEFAULT_LAYERS 0x9312 +#define GL_FRAMEBUFFER_DEFAULT_SAMPLES 0x9313 +#define GL_FRAMEBUFFER_DEFAULT_FIXED_SAMPLE_LOCATIONS 0x9314 +#define GL_MAX_FRAMEBUFFER_WIDTH 0x9315 +#define GL_MAX_FRAMEBUFFER_HEIGHT 0x9316 +#define GL_MAX_FRAMEBUFFER_LAYERS 0x9317 +#define GL_MAX_FRAMEBUFFER_SAMPLES 0x9318 +#define GL_INTERNALFORMAT_SUPPORTED 0x826F +#define GL_INTERNALFORMAT_PREFERRED 0x8270 +#define GL_INTERNALFORMAT_RED_SIZE 0x8271 +#define GL_INTERNALFORMAT_GREEN_SIZE 0x8272 +#define GL_INTERNALFORMAT_BLUE_SIZE 0x8273 +#define GL_INTERNALFORMAT_ALPHA_SIZE 0x8274 +#define GL_INTERNALFORMAT_DEPTH_SIZE 0x8275 +#define GL_INTERNALFORMAT_STENCIL_SIZE 0x8276 +#define GL_INTERNALFORMAT_SHARED_SIZE 0x8277 +#define GL_INTERNALFORMAT_RED_TYPE 0x8278 +#define GL_INTERNALFORMAT_GREEN_TYPE 0x8279 +#define GL_INTERNALFORMAT_BLUE_TYPE 0x827A +#define GL_INTERNALFORMAT_ALPHA_TYPE 0x827B +#define GL_INTERNALFORMAT_DEPTH_TYPE 0x827C +#define GL_INTERNALFORMAT_STENCIL_TYPE 0x827D +#define GL_MAX_WIDTH 0x827E +#define GL_MAX_HEIGHT 0x827F +#define GL_MAX_DEPTH 0x8280 +#define GL_MAX_LAYERS 0x8281 +#define GL_MAX_COMBINED_DIMENSIONS 0x8282 +#define GL_COLOR_COMPONENTS 0x8283 +#define GL_DEPTH_COMPONENTS 0x8284 +#define GL_STENCIL_COMPONENTS 0x8285 +#define GL_COLOR_RENDERABLE 0x8286 +#define GL_DEPTH_RENDERABLE 0x8287 +#define GL_STENCIL_RENDERABLE 0x8288 +#define GL_FRAMEBUFFER_RENDERABLE 0x8289 +#define GL_FRAMEBUFFER_RENDERABLE_LAYERED 0x828A +#define GL_FRAMEBUFFER_BLEND 0x828B +#define GL_READ_PIXELS 0x828C +#define GL_READ_PIXELS_FORMAT 0x828D +#define GL_READ_PIXELS_TYPE 0x828E +#define GL_TEXTURE_IMAGE_FORMAT 0x828F +#define GL_TEXTURE_IMAGE_TYPE 0x8290 +#define GL_GET_TEXTURE_IMAGE_FORMAT 0x8291 +#define GL_GET_TEXTURE_IMAGE_TYPE 0x8292 +#define GL_MIPMAP 0x8293 +#define GL_MANUAL_GENERATE_MIPMAP 0x8294 +#define GL_AUTO_GENERATE_MIPMAP 0x8295 +#define GL_COLOR_ENCODING 0x8296 +#define GL_SRGB_READ 0x8297 +#define GL_SRGB_WRITE 0x8298 +#define GL_FILTER 0x829A +#define GL_VERTEX_TEXTURE 0x829B +#define GL_TESS_CONTROL_TEXTURE 0x829C +#define GL_TESS_EVALUATION_TEXTURE 0x829D +#define GL_GEOMETRY_TEXTURE 0x829E +#define GL_FRAGMENT_TEXTURE 0x829F +#define GL_COMPUTE_TEXTURE 0x82A0 +#define GL_TEXTURE_SHADOW 0x82A1 +#define GL_TEXTURE_GATHER 0x82A2 +#define GL_TEXTURE_GATHER_SHADOW 0x82A3 +#define GL_SHADER_IMAGE_LOAD 0x82A4 +#define GL_SHADER_IMAGE_STORE 0x82A5 +#define GL_SHADER_IMAGE_ATOMIC 0x82A6 +#define GL_IMAGE_TEXEL_SIZE 0x82A7 +#define GL_IMAGE_COMPATIBILITY_CLASS 0x82A8 +#define GL_IMAGE_PIXEL_FORMAT 0x82A9 +#define GL_IMAGE_PIXEL_TYPE 0x82AA +#define GL_SIMULTANEOUS_TEXTURE_AND_DEPTH_TEST 0x82AC +#define GL_SIMULTANEOUS_TEXTURE_AND_STENCIL_TEST 0x82AD +#define GL_SIMULTANEOUS_TEXTURE_AND_DEPTH_WRITE 0x82AE +#define GL_SIMULTANEOUS_TEXTURE_AND_STENCIL_WRITE 0x82AF +#define GL_TEXTURE_COMPRESSED_BLOCK_WIDTH 0x82B1 +#define GL_TEXTURE_COMPRESSED_BLOCK_HEIGHT 0x82B2 +#define GL_TEXTURE_COMPRESSED_BLOCK_SIZE 0x82B3 +#define GL_CLEAR_BUFFER 0x82B4 +#define GL_TEXTURE_VIEW 0x82B5 +#define GL_VIEW_COMPATIBILITY_CLASS 0x82B6 +#define GL_FULL_SUPPORT 0x82B7 +#define GL_CAVEAT_SUPPORT 0x82B8 +#define GL_IMAGE_CLASS_4_X_32 0x82B9 +#define GL_IMAGE_CLASS_2_X_32 0x82BA +#define GL_IMAGE_CLASS_1_X_32 0x82BB +#define GL_IMAGE_CLASS_4_X_16 0x82BC +#define GL_IMAGE_CLASS_2_X_16 0x82BD +#define GL_IMAGE_CLASS_1_X_16 0x82BE +#define GL_IMAGE_CLASS_4_X_8 0x82BF +#define GL_IMAGE_CLASS_2_X_8 0x82C0 +#define GL_IMAGE_CLASS_1_X_8 0x82C1 +#define GL_IMAGE_CLASS_11_11_10 0x82C2 +#define GL_IMAGE_CLASS_10_10_10_2 0x82C3 +#define GL_VIEW_CLASS_128_BITS 0x82C4 +#define GL_VIEW_CLASS_96_BITS 0x82C5 +#define GL_VIEW_CLASS_64_BITS 0x82C6 +#define GL_VIEW_CLASS_48_BITS 0x82C7 +#define GL_VIEW_CLASS_32_BITS 0x82C8 +#define GL_VIEW_CLASS_24_BITS 0x82C9 +#define GL_VIEW_CLASS_16_BITS 0x82CA +#define GL_VIEW_CLASS_8_BITS 0x82CB +#define GL_VIEW_CLASS_S3TC_DXT1_RGB 0x82CC +#define GL_VIEW_CLASS_S3TC_DXT1_RGBA 0x82CD +#define GL_VIEW_CLASS_S3TC_DXT3_RGBA 0x82CE +#define GL_VIEW_CLASS_S3TC_DXT5_RGBA 0x82CF +#define GL_VIEW_CLASS_RGTC1_RED 0x82D0 +#define GL_VIEW_CLASS_RGTC2_RG 0x82D1 +#define GL_VIEW_CLASS_BPTC_UNORM 0x82D2 +#define GL_VIEW_CLASS_BPTC_FLOAT 0x82D3 +#define GL_UNIFORM 0x92E1 +#define GL_UNIFORM_BLOCK 0x92E2 +#define GL_PROGRAM_INPUT 0x92E3 +#define GL_PROGRAM_OUTPUT 0x92E4 +#define GL_BUFFER_VARIABLE 0x92E5 +#define GL_SHADER_STORAGE_BLOCK 0x92E6 +#define GL_VERTEX_SUBROUTINE 0x92E8 +#define GL_TESS_CONTROL_SUBROUTINE 0x92E9 +#define GL_TESS_EVALUATION_SUBROUTINE 0x92EA +#define GL_GEOMETRY_SUBROUTINE 0x92EB +#define GL_FRAGMENT_SUBROUTINE 0x92EC +#define GL_COMPUTE_SUBROUTINE 0x92ED +#define GL_VERTEX_SUBROUTINE_UNIFORM 0x92EE +#define GL_TESS_CONTROL_SUBROUTINE_UNIFORM 0x92EF +#define GL_TESS_EVALUATION_SUBROUTINE_UNIFORM 0x92F0 +#define GL_GEOMETRY_SUBROUTINE_UNIFORM 0x92F1 +#define GL_FRAGMENT_SUBROUTINE_UNIFORM 0x92F2 +#define GL_COMPUTE_SUBROUTINE_UNIFORM 0x92F3 +#define GL_TRANSFORM_FEEDBACK_VARYING 0x92F4 +#define GL_ACTIVE_RESOURCES 0x92F5 +#define GL_MAX_NAME_LENGTH 0x92F6 +#define GL_MAX_NUM_ACTIVE_VARIABLES 0x92F7 +#define GL_MAX_NUM_COMPATIBLE_SUBROUTINES 0x92F8 +#define GL_NAME_LENGTH 0x92F9 +#define GL_TYPE 0x92FA +#define GL_ARRAY_SIZE 0x92FB +#define GL_OFFSET 0x92FC +#define GL_BLOCK_INDEX 0x92FD +#define GL_ARRAY_STRIDE 0x92FE +#define GL_MATRIX_STRIDE 0x92FF +#define GL_IS_ROW_MAJOR 0x9300 +#define GL_ATOMIC_COUNTER_BUFFER_INDEX 0x9301 +#define GL_BUFFER_BINDING 0x9302 +#define GL_BUFFER_DATA_SIZE 0x9303 +#define GL_NUM_ACTIVE_VARIABLES 0x9304 +#define GL_ACTIVE_VARIABLES 0x9305 +#define GL_REFERENCED_BY_VERTEX_SHADER 0x9306 +#define GL_REFERENCED_BY_TESS_CONTROL_SHADER 0x9307 +#define GL_REFERENCED_BY_TESS_EVALUATION_SHADER 0x9308 +#define GL_REFERENCED_BY_GEOMETRY_SHADER 0x9309 +#define GL_REFERENCED_BY_FRAGMENT_SHADER 0x930A +#define GL_REFERENCED_BY_COMPUTE_SHADER 0x930B +#define GL_TOP_LEVEL_ARRAY_SIZE 0x930C +#define GL_TOP_LEVEL_ARRAY_STRIDE 0x930D +#define GL_LOCATION 0x930E +#define GL_LOCATION_INDEX 0x930F +#define GL_IS_PER_PATCH 0x92E7 +#define GL_SHADER_STORAGE_BUFFER 0x90D2 +#define GL_SHADER_STORAGE_BUFFER_BINDING 0x90D3 +#define GL_SHADER_STORAGE_BUFFER_START 0x90D4 +#define GL_SHADER_STORAGE_BUFFER_SIZE 0x90D5 +#define GL_MAX_VERTEX_SHADER_STORAGE_BLOCKS 0x90D6 +#define GL_MAX_GEOMETRY_SHADER_STORAGE_BLOCKS 0x90D7 +#define GL_MAX_TESS_CONTROL_SHADER_STORAGE_BLOCKS 0x90D8 +#define GL_MAX_TESS_EVALUATION_SHADER_STORAGE_BLOCKS 0x90D9 +#define GL_MAX_FRAGMENT_SHADER_STORAGE_BLOCKS 0x90DA +#define GL_MAX_COMPUTE_SHADER_STORAGE_BLOCKS 0x90DB +#define GL_MAX_COMBINED_SHADER_STORAGE_BLOCKS 0x90DC +#define GL_MAX_SHADER_STORAGE_BUFFER_BINDINGS 0x90DD +#define GL_MAX_SHADER_STORAGE_BLOCK_SIZE 0x90DE +#define GL_SHADER_STORAGE_BUFFER_OFFSET_ALIGNMENT 0x90DF +#define GL_SHADER_STORAGE_BARRIER_BIT 0x00002000 +#define GL_MAX_COMBINED_SHADER_OUTPUT_RESOURCES 0x8F39 +#define GL_DEPTH_STENCIL_TEXTURE_MODE 0x90EA +#define GL_TEXTURE_BUFFER_OFFSET 0x919D +#define GL_TEXTURE_BUFFER_SIZE 0x919E +#define GL_TEXTURE_BUFFER_OFFSET_ALIGNMENT 0x919F +#define GL_TEXTURE_VIEW_MIN_LEVEL 0x82DB +#define GL_TEXTURE_VIEW_NUM_LEVELS 0x82DC +#define GL_TEXTURE_VIEW_MIN_LAYER 0x82DD +#define GL_TEXTURE_VIEW_NUM_LAYERS 0x82DE +#define GL_TEXTURE_IMMUTABLE_LEVELS 0x82DF +#define GL_VERTEX_ATTRIB_BINDING 0x82D4 +#define GL_VERTEX_ATTRIB_RELATIVE_OFFSET 0x82D5 +#define GL_VERTEX_BINDING_DIVISOR 0x82D6 +#define GL_VERTEX_BINDING_OFFSET 0x82D7 +#define GL_VERTEX_BINDING_STRIDE 0x82D8 +#define GL_MAX_VERTEX_ATTRIB_RELATIVE_OFFSET 0x82D9 +#define GL_MAX_VERTEX_ATTRIB_BINDINGS 0x82DA +#define GL_VERTEX_BINDING_BUFFER 0x8F4F +#define GL_DISPLAY_LIST 0x82E7 +typedef void (APIENTRYP PFNGLCLEARBUFFERDATAPROC) (GLenum target, GLenum internalformat, GLenum format, GLenum type, const void *data); +typedef void (APIENTRYP PFNGLCLEARBUFFERSUBDATAPROC) (GLenum target, GLenum internalformat, GLintptr offset, GLsizeiptr size, GLenum format, GLenum type, const void *data); +typedef void (APIENTRYP PFNGLDISPATCHCOMPUTEPROC) (GLuint num_groups_x, GLuint num_groups_y, GLuint num_groups_z); +typedef void (APIENTRYP PFNGLDISPATCHCOMPUTEINDIRECTPROC) (GLintptr indirect); +typedef void (APIENTRYP PFNGLCOPYIMAGESUBDATAPROC) (GLuint srcName, GLenum srcTarget, GLint srcLevel, GLint srcX, GLint srcY, GLint srcZ, GLuint dstName, GLenum dstTarget, GLint dstLevel, GLint dstX, GLint dstY, GLint dstZ, GLsizei srcWidth, GLsizei srcHeight, GLsizei srcDepth); +typedef void (APIENTRYP PFNGLFRAMEBUFFERPARAMETERIPROC) (GLenum target, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLGETFRAMEBUFFERPARAMETERIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETINTERNALFORMATI64VPROC) (GLenum target, GLenum internalformat, GLenum pname, GLsizei count, GLint64 *params); +typedef void (APIENTRYP PFNGLINVALIDATETEXSUBIMAGEPROC) (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth); +typedef void (APIENTRYP PFNGLINVALIDATETEXIMAGEPROC) (GLuint texture, GLint level); +typedef void (APIENTRYP PFNGLINVALIDATEBUFFERSUBDATAPROC) (GLuint buffer, GLintptr offset, GLsizeiptr length); +typedef void (APIENTRYP PFNGLINVALIDATEBUFFERDATAPROC) (GLuint buffer); +typedef void (APIENTRYP PFNGLINVALIDATEFRAMEBUFFERPROC) (GLenum target, GLsizei numAttachments, const GLenum *attachments); +typedef void (APIENTRYP PFNGLINVALIDATESUBFRAMEBUFFERPROC) (GLenum target, GLsizei numAttachments, const GLenum *attachments, GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLMULTIDRAWARRAYSINDIRECTPROC) (GLenum mode, const void *indirect, GLsizei drawcount, GLsizei stride); +typedef void (APIENTRYP PFNGLMULTIDRAWELEMENTSINDIRECTPROC) (GLenum mode, GLenum type, const void *indirect, GLsizei drawcount, GLsizei stride); +typedef void (APIENTRYP PFNGLGETPROGRAMINTERFACEIVPROC) (GLuint program, GLenum programInterface, GLenum pname, GLint *params); +typedef GLuint (APIENTRYP PFNGLGETPROGRAMRESOURCEINDEXPROC) (GLuint program, GLenum programInterface, const GLchar *name); +typedef void (APIENTRYP PFNGLGETPROGRAMRESOURCENAMEPROC) (GLuint program, GLenum programInterface, GLuint index, GLsizei bufSize, GLsizei *length, GLchar *name); +typedef void (APIENTRYP PFNGLGETPROGRAMRESOURCEIVPROC) (GLuint program, GLenum programInterface, GLuint index, GLsizei propCount, const GLenum *props, GLsizei count, GLsizei *length, GLint *params); +typedef GLint (APIENTRYP PFNGLGETPROGRAMRESOURCELOCATIONPROC) (GLuint program, GLenum programInterface, const GLchar *name); +typedef GLint (APIENTRYP PFNGLGETPROGRAMRESOURCELOCATIONINDEXPROC) (GLuint program, GLenum programInterface, const GLchar *name); +typedef void (APIENTRYP PFNGLSHADERSTORAGEBLOCKBINDINGPROC) (GLuint program, GLuint storageBlockIndex, GLuint storageBlockBinding); +typedef void (APIENTRYP PFNGLTEXBUFFERRANGEPROC) (GLenum target, GLenum internalformat, GLuint buffer, GLintptr offset, GLsizeiptr size); +typedef void (APIENTRYP PFNGLTEXSTORAGE2DMULTISAMPLEPROC) (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLboolean fixedsamplelocations); +typedef void (APIENTRYP PFNGLTEXSTORAGE3DMULTISAMPLEPROC) (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedsamplelocations); +typedef void (APIENTRYP PFNGLTEXTUREVIEWPROC) (GLuint texture, GLenum target, GLuint origtexture, GLenum internalformat, GLuint minlevel, GLuint numlevels, GLuint minlayer, GLuint numlayers); +typedef void (APIENTRYP PFNGLBINDVERTEXBUFFERPROC) (GLuint bindingindex, GLuint buffer, GLintptr offset, GLsizei stride); +typedef void (APIENTRYP PFNGLVERTEXATTRIBFORMATPROC) (GLuint attribindex, GLint size, GLenum type, GLboolean normalized, GLuint relativeoffset); +typedef void (APIENTRYP PFNGLVERTEXATTRIBIFORMATPROC) (GLuint attribindex, GLint size, GLenum type, GLuint relativeoffset); +typedef void (APIENTRYP PFNGLVERTEXATTRIBLFORMATPROC) (GLuint attribindex, GLint size, GLenum type, GLuint relativeoffset); +typedef void (APIENTRYP PFNGLVERTEXATTRIBBINDINGPROC) (GLuint attribindex, GLuint bindingindex); +typedef void (APIENTRYP PFNGLVERTEXBINDINGDIVISORPROC) (GLuint bindingindex, GLuint divisor); +typedef void (APIENTRYP PFNGLDEBUGMESSAGECONTROLPROC) (GLenum source, GLenum type, GLenum severity, GLsizei count, const GLuint *ids, GLboolean enabled); +typedef void (APIENTRYP PFNGLDEBUGMESSAGEINSERTPROC) (GLenum source, GLenum type, GLuint id, GLenum severity, GLsizei length, const GLchar *buf); +typedef void (APIENTRYP PFNGLDEBUGMESSAGECALLBACKPROC) (GLDEBUGPROC callback, const void *userParam); +typedef GLuint (APIENTRYP PFNGLGETDEBUGMESSAGELOGPROC) (GLuint count, GLsizei bufSize, GLenum *sources, GLenum *types, GLuint *ids, GLenum *severities, GLsizei *lengths, GLchar *messageLog); +typedef void (APIENTRYP PFNGLPUSHDEBUGGROUPPROC) (GLenum source, GLuint id, GLsizei length, const GLchar *message); +typedef void (APIENTRYP PFNGLPOPDEBUGGROUPPROC) (void); +typedef void (APIENTRYP PFNGLOBJECTLABELPROC) (GLenum identifier, GLuint name, GLsizei length, const GLchar *label); +typedef void (APIENTRYP PFNGLGETOBJECTLABELPROC) (GLenum identifier, GLuint name, GLsizei bufSize, GLsizei *length, GLchar *label); +typedef void (APIENTRYP PFNGLOBJECTPTRLABELPROC) (const void *ptr, GLsizei length, const GLchar *label); +typedef void (APIENTRYP PFNGLGETOBJECTPTRLABELPROC) (const void *ptr, GLsizei bufSize, GLsizei *length, GLchar *label); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glClearBufferData (GLenum target, GLenum internalformat, GLenum format, GLenum type, const void *data); +GLAPI void APIENTRY glClearBufferSubData (GLenum target, GLenum internalformat, GLintptr offset, GLsizeiptr size, GLenum format, GLenum type, const void *data); +GLAPI void APIENTRY glDispatchCompute (GLuint num_groups_x, GLuint num_groups_y, GLuint num_groups_z); +GLAPI void APIENTRY glDispatchComputeIndirect (GLintptr indirect); +GLAPI void APIENTRY glCopyImageSubData (GLuint srcName, GLenum srcTarget, GLint srcLevel, GLint srcX, GLint srcY, GLint srcZ, GLuint dstName, GLenum dstTarget, GLint dstLevel, GLint dstX, GLint dstY, GLint dstZ, GLsizei srcWidth, GLsizei srcHeight, GLsizei srcDepth); +GLAPI void APIENTRY glFramebufferParameteri (GLenum target, GLenum pname, GLint param); +GLAPI void APIENTRY glGetFramebufferParameteriv (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetInternalformati64v (GLenum target, GLenum internalformat, GLenum pname, GLsizei count, GLint64 *params); +GLAPI void APIENTRY glInvalidateTexSubImage (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth); +GLAPI void APIENTRY glInvalidateTexImage (GLuint texture, GLint level); +GLAPI void APIENTRY glInvalidateBufferSubData (GLuint buffer, GLintptr offset, GLsizeiptr length); +GLAPI void APIENTRY glInvalidateBufferData (GLuint buffer); +GLAPI void APIENTRY glInvalidateFramebuffer (GLenum target, GLsizei numAttachments, const GLenum *attachments); +GLAPI void APIENTRY glInvalidateSubFramebuffer (GLenum target, GLsizei numAttachments, const GLenum *attachments, GLint x, GLint y, GLsizei width, GLsizei height); +GLAPI void APIENTRY glMultiDrawArraysIndirect (GLenum mode, const void *indirect, GLsizei drawcount, GLsizei stride); +GLAPI void APIENTRY glMultiDrawElementsIndirect (GLenum mode, GLenum type, const void *indirect, GLsizei drawcount, GLsizei stride); +GLAPI void APIENTRY glGetProgramInterfaceiv (GLuint program, GLenum programInterface, GLenum pname, GLint *params); +GLAPI GLuint APIENTRY glGetProgramResourceIndex (GLuint program, GLenum programInterface, const GLchar *name); +GLAPI void APIENTRY glGetProgramResourceName (GLuint program, GLenum programInterface, GLuint index, GLsizei bufSize, GLsizei *length, GLchar *name); +GLAPI void APIENTRY glGetProgramResourceiv (GLuint program, GLenum programInterface, GLuint index, GLsizei propCount, const GLenum *props, GLsizei count, GLsizei *length, GLint *params); +GLAPI GLint APIENTRY glGetProgramResourceLocation (GLuint program, GLenum programInterface, const GLchar *name); +GLAPI GLint APIENTRY glGetProgramResourceLocationIndex (GLuint program, GLenum programInterface, const GLchar *name); +GLAPI void APIENTRY glShaderStorageBlockBinding (GLuint program, GLuint storageBlockIndex, GLuint storageBlockBinding); +GLAPI void APIENTRY glTexBufferRange (GLenum target, GLenum internalformat, GLuint buffer, GLintptr offset, GLsizeiptr size); +GLAPI void APIENTRY glTexStorage2DMultisample (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLboolean fixedsamplelocations); +GLAPI void APIENTRY glTexStorage3DMultisample (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedsamplelocations); +GLAPI void APIENTRY glTextureView (GLuint texture, GLenum target, GLuint origtexture, GLenum internalformat, GLuint minlevel, GLuint numlevels, GLuint minlayer, GLuint numlayers); +GLAPI void APIENTRY glBindVertexBuffer (GLuint bindingindex, GLuint buffer, GLintptr offset, GLsizei stride); +GLAPI void APIENTRY glVertexAttribFormat (GLuint attribindex, GLint size, GLenum type, GLboolean normalized, GLuint relativeoffset); +GLAPI void APIENTRY glVertexAttribIFormat (GLuint attribindex, GLint size, GLenum type, GLuint relativeoffset); +GLAPI void APIENTRY glVertexAttribLFormat (GLuint attribindex, GLint size, GLenum type, GLuint relativeoffset); +GLAPI void APIENTRY glVertexAttribBinding (GLuint attribindex, GLuint bindingindex); +GLAPI void APIENTRY glVertexBindingDivisor (GLuint bindingindex, GLuint divisor); +GLAPI void APIENTRY glDebugMessageControl (GLenum source, GLenum type, GLenum severity, GLsizei count, const GLuint *ids, GLboolean enabled); +GLAPI void APIENTRY glDebugMessageInsert (GLenum source, GLenum type, GLuint id, GLenum severity, GLsizei length, const GLchar *buf); +GLAPI void APIENTRY glDebugMessageCallback (GLDEBUGPROC callback, const void *userParam); +GLAPI GLuint APIENTRY glGetDebugMessageLog (GLuint count, GLsizei bufSize, GLenum *sources, GLenum *types, GLuint *ids, GLenum *severities, GLsizei *lengths, GLchar *messageLog); +GLAPI void APIENTRY glPushDebugGroup (GLenum source, GLuint id, GLsizei length, const GLchar *message); +GLAPI void APIENTRY glPopDebugGroup (void); +GLAPI void APIENTRY glObjectLabel (GLenum identifier, GLuint name, GLsizei length, const GLchar *label); +GLAPI void APIENTRY glGetObjectLabel (GLenum identifier, GLuint name, GLsizei bufSize, GLsizei *length, GLchar *label); +GLAPI void APIENTRY glObjectPtrLabel (const void *ptr, GLsizei length, const GLchar *label); +GLAPI void APIENTRY glGetObjectPtrLabel (const void *ptr, GLsizei bufSize, GLsizei *length, GLchar *label); +#endif +#endif /* GL_VERSION_4_3 */ + +#ifndef GL_VERSION_4_4 +#define GL_VERSION_4_4 1 +#define GL_MAX_VERTEX_ATTRIB_STRIDE 0x82E5 +#define GL_PRIMITIVE_RESTART_FOR_PATCHES_SUPPORTED 0x8221 +#define GL_TEXTURE_BUFFER_BINDING 0x8C2A +#define GL_MAP_PERSISTENT_BIT 0x0040 +#define GL_MAP_COHERENT_BIT 0x0080 +#define GL_DYNAMIC_STORAGE_BIT 0x0100 +#define GL_CLIENT_STORAGE_BIT 0x0200 +#define GL_CLIENT_MAPPED_BUFFER_BARRIER_BIT 0x00004000 +#define GL_BUFFER_IMMUTABLE_STORAGE 0x821F +#define GL_BUFFER_STORAGE_FLAGS 0x8220 +#define GL_CLEAR_TEXTURE 0x9365 +#define GL_LOCATION_COMPONENT 0x934A +#define GL_TRANSFORM_FEEDBACK_BUFFER_INDEX 0x934B +#define GL_TRANSFORM_FEEDBACK_BUFFER_STRIDE 0x934C +#define GL_QUERY_BUFFER 0x9192 +#define GL_QUERY_BUFFER_BARRIER_BIT 0x00008000 +#define GL_QUERY_BUFFER_BINDING 0x9193 +#define GL_QUERY_RESULT_NO_WAIT 0x9194 +#define GL_MIRROR_CLAMP_TO_EDGE 0x8743 +typedef void (APIENTRYP PFNGLBUFFERSTORAGEPROC) (GLenum target, GLsizeiptr size, const void *data, GLbitfield flags); +typedef void (APIENTRYP PFNGLCLEARTEXIMAGEPROC) (GLuint texture, GLint level, GLenum format, GLenum type, const void *data); +typedef void (APIENTRYP PFNGLCLEARTEXSUBIMAGEPROC) (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *data); +typedef void (APIENTRYP PFNGLBINDBUFFERSBASEPROC) (GLenum target, GLuint first, GLsizei count, const GLuint *buffers); +typedef void (APIENTRYP PFNGLBINDBUFFERSRANGEPROC) (GLenum target, GLuint first, GLsizei count, const GLuint *buffers, const GLintptr *offsets, const GLsizeiptr *sizes); +typedef void (APIENTRYP PFNGLBINDTEXTURESPROC) (GLuint first, GLsizei count, const GLuint *textures); +typedef void (APIENTRYP PFNGLBINDSAMPLERSPROC) (GLuint first, GLsizei count, const GLuint *samplers); +typedef void (APIENTRYP PFNGLBINDIMAGETEXTURESPROC) (GLuint first, GLsizei count, const GLuint *textures); +typedef void (APIENTRYP PFNGLBINDVERTEXBUFFERSPROC) (GLuint first, GLsizei count, const GLuint *buffers, const GLintptr *offsets, const GLsizei *strides); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBufferStorage (GLenum target, GLsizeiptr size, const void *data, GLbitfield flags); +GLAPI void APIENTRY glClearTexImage (GLuint texture, GLint level, GLenum format, GLenum type, const void *data); +GLAPI void APIENTRY glClearTexSubImage (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *data); +GLAPI void APIENTRY glBindBuffersBase (GLenum target, GLuint first, GLsizei count, const GLuint *buffers); +GLAPI void APIENTRY glBindBuffersRange (GLenum target, GLuint first, GLsizei count, const GLuint *buffers, const GLintptr *offsets, const GLsizeiptr *sizes); +GLAPI void APIENTRY glBindTextures (GLuint first, GLsizei count, const GLuint *textures); +GLAPI void APIENTRY glBindSamplers (GLuint first, GLsizei count, const GLuint *samplers); +GLAPI void APIENTRY glBindImageTextures (GLuint first, GLsizei count, const GLuint *textures); +GLAPI void APIENTRY glBindVertexBuffers (GLuint first, GLsizei count, const GLuint *buffers, const GLintptr *offsets, const GLsizei *strides); +#endif +#endif /* GL_VERSION_4_4 */ + +#ifndef GL_VERSION_4_5 +#define GL_VERSION_4_5 1 +#define GL_CONTEXT_LOST 0x0507 +#define GL_NEGATIVE_ONE_TO_ONE 0x935E +#define GL_ZERO_TO_ONE 0x935F +#define GL_CLIP_ORIGIN 0x935C +#define GL_CLIP_DEPTH_MODE 0x935D +#define GL_QUERY_WAIT_INVERTED 0x8E17 +#define GL_QUERY_NO_WAIT_INVERTED 0x8E18 +#define GL_QUERY_BY_REGION_WAIT_INVERTED 0x8E19 +#define GL_QUERY_BY_REGION_NO_WAIT_INVERTED 0x8E1A +#define GL_MAX_CULL_DISTANCES 0x82F9 +#define GL_MAX_COMBINED_CLIP_AND_CULL_DISTANCES 0x82FA +#define GL_TEXTURE_TARGET 0x1006 +#define GL_QUERY_TARGET 0x82EA +#define GL_GUILTY_CONTEXT_RESET 0x8253 +#define GL_INNOCENT_CONTEXT_RESET 0x8254 +#define GL_UNKNOWN_CONTEXT_RESET 0x8255 +#define GL_RESET_NOTIFICATION_STRATEGY 0x8256 +#define GL_LOSE_CONTEXT_ON_RESET 0x8252 +#define GL_NO_RESET_NOTIFICATION 0x8261 +#define GL_CONTEXT_FLAG_ROBUST_ACCESS_BIT 0x00000004 +#define GL_COLOR_TABLE 0x80D0 +#define GL_POST_CONVOLUTION_COLOR_TABLE 0x80D1 +#define GL_POST_COLOR_MATRIX_COLOR_TABLE 0x80D2 +#define GL_PROXY_COLOR_TABLE 0x80D3 +#define GL_PROXY_POST_CONVOLUTION_COLOR_TABLE 0x80D4 +#define GL_PROXY_POST_COLOR_MATRIX_COLOR_TABLE 0x80D5 +#define GL_CONVOLUTION_1D 0x8010 +#define GL_CONVOLUTION_2D 0x8011 +#define GL_SEPARABLE_2D 0x8012 +#define GL_HISTOGRAM 0x8024 +#define GL_PROXY_HISTOGRAM 0x8025 +#define GL_MINMAX 0x802E +#define GL_CONTEXT_RELEASE_BEHAVIOR 0x82FB +#define GL_CONTEXT_RELEASE_BEHAVIOR_FLUSH 0x82FC +typedef void (APIENTRYP PFNGLCLIPCONTROLPROC) (GLenum origin, GLenum depth); +typedef void (APIENTRYP PFNGLCREATETRANSFORMFEEDBACKSPROC) (GLsizei n, GLuint *ids); +typedef void (APIENTRYP PFNGLTRANSFORMFEEDBACKBUFFERBASEPROC) (GLuint xfb, GLuint index, GLuint buffer); +typedef void (APIENTRYP PFNGLTRANSFORMFEEDBACKBUFFERRANGEPROC) (GLuint xfb, GLuint index, GLuint buffer, GLintptr offset, GLsizeiptr size); +typedef void (APIENTRYP PFNGLGETTRANSFORMFEEDBACKIVPROC) (GLuint xfb, GLenum pname, GLint *param); +typedef void (APIENTRYP PFNGLGETTRANSFORMFEEDBACKI_VPROC) (GLuint xfb, GLenum pname, GLuint index, GLint *param); +typedef void (APIENTRYP PFNGLGETTRANSFORMFEEDBACKI64_VPROC) (GLuint xfb, GLenum pname, GLuint index, GLint64 *param); +typedef void (APIENTRYP PFNGLCREATEBUFFERSPROC) (GLsizei n, GLuint *buffers); +typedef void (APIENTRYP PFNGLNAMEDBUFFERSTORAGEPROC) (GLuint buffer, GLsizeiptr size, const void *data, GLbitfield flags); +typedef void (APIENTRYP PFNGLNAMEDBUFFERDATAPROC) (GLuint buffer, GLsizeiptr size, const void *data, GLenum usage); +typedef void (APIENTRYP PFNGLNAMEDBUFFERSUBDATAPROC) (GLuint buffer, GLintptr offset, GLsizeiptr size, const void *data); +typedef void (APIENTRYP PFNGLCOPYNAMEDBUFFERSUBDATAPROC) (GLuint readBuffer, GLuint writeBuffer, GLintptr readOffset, GLintptr writeOffset, GLsizeiptr size); +typedef void (APIENTRYP PFNGLCLEARNAMEDBUFFERDATAPROC) (GLuint buffer, GLenum internalformat, GLenum format, GLenum type, const void *data); +typedef void (APIENTRYP PFNGLCLEARNAMEDBUFFERSUBDATAPROC) (GLuint buffer, GLenum internalformat, GLintptr offset, GLsizeiptr size, GLenum format, GLenum type, const void *data); +typedef void *(APIENTRYP PFNGLMAPNAMEDBUFFERPROC) (GLuint buffer, GLenum access); +typedef void *(APIENTRYP PFNGLMAPNAMEDBUFFERRANGEPROC) (GLuint buffer, GLintptr offset, GLsizeiptr length, GLbitfield access); +typedef GLboolean (APIENTRYP PFNGLUNMAPNAMEDBUFFERPROC) (GLuint buffer); +typedef void (APIENTRYP PFNGLFLUSHMAPPEDNAMEDBUFFERRANGEPROC) (GLuint buffer, GLintptr offset, GLsizeiptr length); +typedef void (APIENTRYP PFNGLGETNAMEDBUFFERPARAMETERIVPROC) (GLuint buffer, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETNAMEDBUFFERPARAMETERI64VPROC) (GLuint buffer, GLenum pname, GLint64 *params); +typedef void (APIENTRYP PFNGLGETNAMEDBUFFERPOINTERVPROC) (GLuint buffer, GLenum pname, void **params); +typedef void (APIENTRYP PFNGLGETNAMEDBUFFERSUBDATAPROC) (GLuint buffer, GLintptr offset, GLsizeiptr size, void *data); +typedef void (APIENTRYP PFNGLCREATEFRAMEBUFFERSPROC) (GLsizei n, GLuint *framebuffers); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERRENDERBUFFERPROC) (GLuint framebuffer, GLenum attachment, GLenum renderbuffertarget, GLuint renderbuffer); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERPARAMETERIPROC) (GLuint framebuffer, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERTEXTUREPROC) (GLuint framebuffer, GLenum attachment, GLuint texture, GLint level); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERTEXTURELAYERPROC) (GLuint framebuffer, GLenum attachment, GLuint texture, GLint level, GLint layer); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERDRAWBUFFERPROC) (GLuint framebuffer, GLenum buf); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERDRAWBUFFERSPROC) (GLuint framebuffer, GLsizei n, const GLenum *bufs); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERREADBUFFERPROC) (GLuint framebuffer, GLenum src); +typedef void (APIENTRYP PFNGLINVALIDATENAMEDFRAMEBUFFERDATAPROC) (GLuint framebuffer, GLsizei numAttachments, const GLenum *attachments); +typedef void (APIENTRYP PFNGLINVALIDATENAMEDFRAMEBUFFERSUBDATAPROC) (GLuint framebuffer, GLsizei numAttachments, const GLenum *attachments, GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLCLEARNAMEDFRAMEBUFFERIVPROC) (GLuint framebuffer, GLenum buffer, GLint drawbuffer, const GLint *value); +typedef void (APIENTRYP PFNGLCLEARNAMEDFRAMEBUFFERUIVPROC) (GLuint framebuffer, GLenum buffer, GLint drawbuffer, const GLuint *value); +typedef void (APIENTRYP PFNGLCLEARNAMEDFRAMEBUFFERFVPROC) (GLuint framebuffer, GLenum buffer, GLint drawbuffer, const GLfloat *value); +typedef void (APIENTRYP PFNGLCLEARNAMEDFRAMEBUFFERFIPROC) (GLuint framebuffer, GLenum buffer, GLint drawbuffer, GLfloat depth, GLint stencil); +typedef void (APIENTRYP PFNGLBLITNAMEDFRAMEBUFFERPROC) (GLuint readFramebuffer, GLuint drawFramebuffer, GLint srcX0, GLint srcY0, GLint srcX1, GLint srcY1, GLint dstX0, GLint dstY0, GLint dstX1, GLint dstY1, GLbitfield mask, GLenum filter); +typedef GLenum (APIENTRYP PFNGLCHECKNAMEDFRAMEBUFFERSTATUSPROC) (GLuint framebuffer, GLenum target); +typedef void (APIENTRYP PFNGLGETNAMEDFRAMEBUFFERPARAMETERIVPROC) (GLuint framebuffer, GLenum pname, GLint *param); +typedef void (APIENTRYP PFNGLGETNAMEDFRAMEBUFFERATTACHMENTPARAMETERIVPROC) (GLuint framebuffer, GLenum attachment, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLCREATERENDERBUFFERSPROC) (GLsizei n, GLuint *renderbuffers); +typedef void (APIENTRYP PFNGLNAMEDRENDERBUFFERSTORAGEPROC) (GLuint renderbuffer, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLNAMEDRENDERBUFFERSTORAGEMULTISAMPLEPROC) (GLuint renderbuffer, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLGETNAMEDRENDERBUFFERPARAMETERIVPROC) (GLuint renderbuffer, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLCREATETEXTURESPROC) (GLenum target, GLsizei n, GLuint *textures); +typedef void (APIENTRYP PFNGLTEXTUREBUFFERPROC) (GLuint texture, GLenum internalformat, GLuint buffer); +typedef void (APIENTRYP PFNGLTEXTUREBUFFERRANGEPROC) (GLuint texture, GLenum internalformat, GLuint buffer, GLintptr offset, GLsizeiptr size); +typedef void (APIENTRYP PFNGLTEXTURESTORAGE1DPROC) (GLuint texture, GLsizei levels, GLenum internalformat, GLsizei width); +typedef void (APIENTRYP PFNGLTEXTURESTORAGE2DPROC) (GLuint texture, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLTEXTURESTORAGE3DPROC) (GLuint texture, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth); +typedef void (APIENTRYP PFNGLTEXTURESTORAGE2DMULTISAMPLEPROC) (GLuint texture, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLboolean fixedsamplelocations); +typedef void (APIENTRYP PFNGLTEXTURESTORAGE3DMULTISAMPLEPROC) (GLuint texture, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedsamplelocations); +typedef void (APIENTRYP PFNGLTEXTURESUBIMAGE1DPROC) (GLuint texture, GLint level, GLint xoffset, GLsizei width, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLTEXTURESUBIMAGE2DPROC) (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLTEXTURESUBIMAGE3DPROC) (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXTURESUBIMAGE1DPROC) (GLuint texture, GLint level, GLint xoffset, GLsizei width, GLenum format, GLsizei imageSize, const void *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXTURESUBIMAGE2DPROC) (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const void *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXTURESUBIMAGE3DPROC) (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const void *data); +typedef void (APIENTRYP PFNGLCOPYTEXTURESUBIMAGE1DPROC) (GLuint texture, GLint level, GLint xoffset, GLint x, GLint y, GLsizei width); +typedef void (APIENTRYP PFNGLCOPYTEXTURESUBIMAGE2DPROC) (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLCOPYTEXTURESUBIMAGE3DPROC) (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLTEXTUREPARAMETERFPROC) (GLuint texture, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLTEXTUREPARAMETERFVPROC) (GLuint texture, GLenum pname, const GLfloat *param); +typedef void (APIENTRYP PFNGLTEXTUREPARAMETERIPROC) (GLuint texture, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLTEXTUREPARAMETERIIVPROC) (GLuint texture, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLTEXTUREPARAMETERIUIVPROC) (GLuint texture, GLenum pname, const GLuint *params); +typedef void (APIENTRYP PFNGLTEXTUREPARAMETERIVPROC) (GLuint texture, GLenum pname, const GLint *param); +typedef void (APIENTRYP PFNGLGENERATETEXTUREMIPMAPPROC) (GLuint texture); +typedef void (APIENTRYP PFNGLBINDTEXTUREUNITPROC) (GLuint unit, GLuint texture); +typedef void (APIENTRYP PFNGLGETTEXTUREIMAGEPROC) (GLuint texture, GLint level, GLenum format, GLenum type, GLsizei bufSize, void *pixels); +typedef void (APIENTRYP PFNGLGETCOMPRESSEDTEXTUREIMAGEPROC) (GLuint texture, GLint level, GLsizei bufSize, void *pixels); +typedef void (APIENTRYP PFNGLGETTEXTURELEVELPARAMETERFVPROC) (GLuint texture, GLint level, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETTEXTURELEVELPARAMETERIVPROC) (GLuint texture, GLint level, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETTEXTUREPARAMETERFVPROC) (GLuint texture, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETTEXTUREPARAMETERIIVPROC) (GLuint texture, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETTEXTUREPARAMETERIUIVPROC) (GLuint texture, GLenum pname, GLuint *params); +typedef void (APIENTRYP PFNGLGETTEXTUREPARAMETERIVPROC) (GLuint texture, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLCREATEVERTEXARRAYSPROC) (GLsizei n, GLuint *arrays); +typedef void (APIENTRYP PFNGLDISABLEVERTEXARRAYATTRIBPROC) (GLuint vaobj, GLuint index); +typedef void (APIENTRYP PFNGLENABLEVERTEXARRAYATTRIBPROC) (GLuint vaobj, GLuint index); +typedef void (APIENTRYP PFNGLVERTEXARRAYELEMENTBUFFERPROC) (GLuint vaobj, GLuint buffer); +typedef void (APIENTRYP PFNGLVERTEXARRAYVERTEXBUFFERPROC) (GLuint vaobj, GLuint bindingindex, GLuint buffer, GLintptr offset, GLsizei stride); +typedef void (APIENTRYP PFNGLVERTEXARRAYVERTEXBUFFERSPROC) (GLuint vaobj, GLuint first, GLsizei count, const GLuint *buffers, const GLintptr *offsets, const GLsizei *strides); +typedef void (APIENTRYP PFNGLVERTEXARRAYATTRIBBINDINGPROC) (GLuint vaobj, GLuint attribindex, GLuint bindingindex); +typedef void (APIENTRYP PFNGLVERTEXARRAYATTRIBFORMATPROC) (GLuint vaobj, GLuint attribindex, GLint size, GLenum type, GLboolean normalized, GLuint relativeoffset); +typedef void (APIENTRYP PFNGLVERTEXARRAYATTRIBIFORMATPROC) (GLuint vaobj, GLuint attribindex, GLint size, GLenum type, GLuint relativeoffset); +typedef void (APIENTRYP PFNGLVERTEXARRAYATTRIBLFORMATPROC) (GLuint vaobj, GLuint attribindex, GLint size, GLenum type, GLuint relativeoffset); +typedef void (APIENTRYP PFNGLVERTEXARRAYBINDINGDIVISORPROC) (GLuint vaobj, GLuint bindingindex, GLuint divisor); +typedef void (APIENTRYP PFNGLGETVERTEXARRAYIVPROC) (GLuint vaobj, GLenum pname, GLint *param); +typedef void (APIENTRYP PFNGLGETVERTEXARRAYINDEXEDIVPROC) (GLuint vaobj, GLuint index, GLenum pname, GLint *param); +typedef void (APIENTRYP PFNGLGETVERTEXARRAYINDEXED64IVPROC) (GLuint vaobj, GLuint index, GLenum pname, GLint64 *param); +typedef void (APIENTRYP PFNGLCREATESAMPLERSPROC) (GLsizei n, GLuint *samplers); +typedef void (APIENTRYP PFNGLCREATEPROGRAMPIPELINESPROC) (GLsizei n, GLuint *pipelines); +typedef void (APIENTRYP PFNGLCREATEQUERIESPROC) (GLenum target, GLsizei n, GLuint *ids); +typedef void (APIENTRYP PFNGLGETQUERYBUFFEROBJECTI64VPROC) (GLuint id, GLuint buffer, GLenum pname, GLintptr offset); +typedef void (APIENTRYP PFNGLGETQUERYBUFFEROBJECTIVPROC) (GLuint id, GLuint buffer, GLenum pname, GLintptr offset); +typedef void (APIENTRYP PFNGLGETQUERYBUFFEROBJECTUI64VPROC) (GLuint id, GLuint buffer, GLenum pname, GLintptr offset); +typedef void (APIENTRYP PFNGLGETQUERYBUFFEROBJECTUIVPROC) (GLuint id, GLuint buffer, GLenum pname, GLintptr offset); +typedef void (APIENTRYP PFNGLMEMORYBARRIERBYREGIONPROC) (GLbitfield barriers); +typedef void (APIENTRYP PFNGLGETTEXTURESUBIMAGEPROC) (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, GLsizei bufSize, void *pixels); +typedef void (APIENTRYP PFNGLGETCOMPRESSEDTEXTURESUBIMAGEPROC) (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLsizei bufSize, void *pixels); +typedef GLenum (APIENTRYP PFNGLGETGRAPHICSRESETSTATUSPROC) (void); +typedef void (APIENTRYP PFNGLGETNCOMPRESSEDTEXIMAGEPROC) (GLenum target, GLint lod, GLsizei bufSize, void *pixels); +typedef void (APIENTRYP PFNGLGETNTEXIMAGEPROC) (GLenum target, GLint level, GLenum format, GLenum type, GLsizei bufSize, void *pixels); +typedef void (APIENTRYP PFNGLGETNUNIFORMDVPROC) (GLuint program, GLint location, GLsizei bufSize, GLdouble *params); +typedef void (APIENTRYP PFNGLGETNUNIFORMFVPROC) (GLuint program, GLint location, GLsizei bufSize, GLfloat *params); +typedef void (APIENTRYP PFNGLGETNUNIFORMIVPROC) (GLuint program, GLint location, GLsizei bufSize, GLint *params); +typedef void (APIENTRYP PFNGLGETNUNIFORMUIVPROC) (GLuint program, GLint location, GLsizei bufSize, GLuint *params); +typedef void (APIENTRYP PFNGLREADNPIXELSPROC) (GLint x, GLint y, GLsizei width, GLsizei height, GLenum format, GLenum type, GLsizei bufSize, void *data); +typedef void (APIENTRYP PFNGLGETNMAPDVPROC) (GLenum target, GLenum query, GLsizei bufSize, GLdouble *v); +typedef void (APIENTRYP PFNGLGETNMAPFVPROC) (GLenum target, GLenum query, GLsizei bufSize, GLfloat *v); +typedef void (APIENTRYP PFNGLGETNMAPIVPROC) (GLenum target, GLenum query, GLsizei bufSize, GLint *v); +typedef void (APIENTRYP PFNGLGETNPIXELMAPFVPROC) (GLenum map, GLsizei bufSize, GLfloat *values); +typedef void (APIENTRYP PFNGLGETNPIXELMAPUIVPROC) (GLenum map, GLsizei bufSize, GLuint *values); +typedef void (APIENTRYP PFNGLGETNPIXELMAPUSVPROC) (GLenum map, GLsizei bufSize, GLushort *values); +typedef void (APIENTRYP PFNGLGETNPOLYGONSTIPPLEPROC) (GLsizei bufSize, GLubyte *pattern); +typedef void (APIENTRYP PFNGLGETNCOLORTABLEPROC) (GLenum target, GLenum format, GLenum type, GLsizei bufSize, void *table); +typedef void (APIENTRYP PFNGLGETNCONVOLUTIONFILTERPROC) (GLenum target, GLenum format, GLenum type, GLsizei bufSize, void *image); +typedef void (APIENTRYP PFNGLGETNSEPARABLEFILTERPROC) (GLenum target, GLenum format, GLenum type, GLsizei rowBufSize, void *row, GLsizei columnBufSize, void *column, void *span); +typedef void (APIENTRYP PFNGLGETNHISTOGRAMPROC) (GLenum target, GLboolean reset, GLenum format, GLenum type, GLsizei bufSize, void *values); +typedef void (APIENTRYP PFNGLGETNMINMAXPROC) (GLenum target, GLboolean reset, GLenum format, GLenum type, GLsizei bufSize, void *values); +typedef void (APIENTRYP PFNGLTEXTUREBARRIERPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glClipControl (GLenum origin, GLenum depth); +GLAPI void APIENTRY glCreateTransformFeedbacks (GLsizei n, GLuint *ids); +GLAPI void APIENTRY glTransformFeedbackBufferBase (GLuint xfb, GLuint index, GLuint buffer); +GLAPI void APIENTRY glTransformFeedbackBufferRange (GLuint xfb, GLuint index, GLuint buffer, GLintptr offset, GLsizeiptr size); +GLAPI void APIENTRY glGetTransformFeedbackiv (GLuint xfb, GLenum pname, GLint *param); +GLAPI void APIENTRY glGetTransformFeedbacki_v (GLuint xfb, GLenum pname, GLuint index, GLint *param); +GLAPI void APIENTRY glGetTransformFeedbacki64_v (GLuint xfb, GLenum pname, GLuint index, GLint64 *param); +GLAPI void APIENTRY glCreateBuffers (GLsizei n, GLuint *buffers); +GLAPI void APIENTRY glNamedBufferStorage (GLuint buffer, GLsizeiptr size, const void *data, GLbitfield flags); +GLAPI void APIENTRY glNamedBufferData (GLuint buffer, GLsizeiptr size, const void *data, GLenum usage); +GLAPI void APIENTRY glNamedBufferSubData (GLuint buffer, GLintptr offset, GLsizeiptr size, const void *data); +GLAPI void APIENTRY glCopyNamedBufferSubData (GLuint readBuffer, GLuint writeBuffer, GLintptr readOffset, GLintptr writeOffset, GLsizeiptr size); +GLAPI void APIENTRY glClearNamedBufferData (GLuint buffer, GLenum internalformat, GLenum format, GLenum type, const void *data); +GLAPI void APIENTRY glClearNamedBufferSubData (GLuint buffer, GLenum internalformat, GLintptr offset, GLsizeiptr size, GLenum format, GLenum type, const void *data); +GLAPI void *APIENTRY glMapNamedBuffer (GLuint buffer, GLenum access); +GLAPI void *APIENTRY glMapNamedBufferRange (GLuint buffer, GLintptr offset, GLsizeiptr length, GLbitfield access); +GLAPI GLboolean APIENTRY glUnmapNamedBuffer (GLuint buffer); +GLAPI void APIENTRY glFlushMappedNamedBufferRange (GLuint buffer, GLintptr offset, GLsizeiptr length); +GLAPI void APIENTRY glGetNamedBufferParameteriv (GLuint buffer, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetNamedBufferParameteri64v (GLuint buffer, GLenum pname, GLint64 *params); +GLAPI void APIENTRY glGetNamedBufferPointerv (GLuint buffer, GLenum pname, void **params); +GLAPI void APIENTRY glGetNamedBufferSubData (GLuint buffer, GLintptr offset, GLsizeiptr size, void *data); +GLAPI void APIENTRY glCreateFramebuffers (GLsizei n, GLuint *framebuffers); +GLAPI void APIENTRY glNamedFramebufferRenderbuffer (GLuint framebuffer, GLenum attachment, GLenum renderbuffertarget, GLuint renderbuffer); +GLAPI void APIENTRY glNamedFramebufferParameteri (GLuint framebuffer, GLenum pname, GLint param); +GLAPI void APIENTRY glNamedFramebufferTexture (GLuint framebuffer, GLenum attachment, GLuint texture, GLint level); +GLAPI void APIENTRY glNamedFramebufferTextureLayer (GLuint framebuffer, GLenum attachment, GLuint texture, GLint level, GLint layer); +GLAPI void APIENTRY glNamedFramebufferDrawBuffer (GLuint framebuffer, GLenum buf); +GLAPI void APIENTRY glNamedFramebufferDrawBuffers (GLuint framebuffer, GLsizei n, const GLenum *bufs); +GLAPI void APIENTRY glNamedFramebufferReadBuffer (GLuint framebuffer, GLenum src); +GLAPI void APIENTRY glInvalidateNamedFramebufferData (GLuint framebuffer, GLsizei numAttachments, const GLenum *attachments); +GLAPI void APIENTRY glInvalidateNamedFramebufferSubData (GLuint framebuffer, GLsizei numAttachments, const GLenum *attachments, GLint x, GLint y, GLsizei width, GLsizei height); +GLAPI void APIENTRY glClearNamedFramebufferiv (GLuint framebuffer, GLenum buffer, GLint drawbuffer, const GLint *value); +GLAPI void APIENTRY glClearNamedFramebufferuiv (GLuint framebuffer, GLenum buffer, GLint drawbuffer, const GLuint *value); +GLAPI void APIENTRY glClearNamedFramebufferfv (GLuint framebuffer, GLenum buffer, GLint drawbuffer, const GLfloat *value); +GLAPI void APIENTRY glClearNamedFramebufferfi (GLuint framebuffer, GLenum buffer, GLint drawbuffer, GLfloat depth, GLint stencil); +GLAPI void APIENTRY glBlitNamedFramebuffer (GLuint readFramebuffer, GLuint drawFramebuffer, GLint srcX0, GLint srcY0, GLint srcX1, GLint srcY1, GLint dstX0, GLint dstY0, GLint dstX1, GLint dstY1, GLbitfield mask, GLenum filter); +GLAPI GLenum APIENTRY glCheckNamedFramebufferStatus (GLuint framebuffer, GLenum target); +GLAPI void APIENTRY glGetNamedFramebufferParameteriv (GLuint framebuffer, GLenum pname, GLint *param); +GLAPI void APIENTRY glGetNamedFramebufferAttachmentParameteriv (GLuint framebuffer, GLenum attachment, GLenum pname, GLint *params); +GLAPI void APIENTRY glCreateRenderbuffers (GLsizei n, GLuint *renderbuffers); +GLAPI void APIENTRY glNamedRenderbufferStorage (GLuint renderbuffer, GLenum internalformat, GLsizei width, GLsizei height); +GLAPI void APIENTRY glNamedRenderbufferStorageMultisample (GLuint renderbuffer, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +GLAPI void APIENTRY glGetNamedRenderbufferParameteriv (GLuint renderbuffer, GLenum pname, GLint *params); +GLAPI void APIENTRY glCreateTextures (GLenum target, GLsizei n, GLuint *textures); +GLAPI void APIENTRY glTextureBuffer (GLuint texture, GLenum internalformat, GLuint buffer); +GLAPI void APIENTRY glTextureBufferRange (GLuint texture, GLenum internalformat, GLuint buffer, GLintptr offset, GLsizeiptr size); +GLAPI void APIENTRY glTextureStorage1D (GLuint texture, GLsizei levels, GLenum internalformat, GLsizei width); +GLAPI void APIENTRY glTextureStorage2D (GLuint texture, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height); +GLAPI void APIENTRY glTextureStorage3D (GLuint texture, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth); +GLAPI void APIENTRY glTextureStorage2DMultisample (GLuint texture, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLboolean fixedsamplelocations); +GLAPI void APIENTRY glTextureStorage3DMultisample (GLuint texture, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedsamplelocations); +GLAPI void APIENTRY glTextureSubImage1D (GLuint texture, GLint level, GLint xoffset, GLsizei width, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glTextureSubImage2D (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glTextureSubImage3D (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glCompressedTextureSubImage1D (GLuint texture, GLint level, GLint xoffset, GLsizei width, GLenum format, GLsizei imageSize, const void *data); +GLAPI void APIENTRY glCompressedTextureSubImage2D (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const void *data); +GLAPI void APIENTRY glCompressedTextureSubImage3D (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const void *data); +GLAPI void APIENTRY glCopyTextureSubImage1D (GLuint texture, GLint level, GLint xoffset, GLint x, GLint y, GLsizei width); +GLAPI void APIENTRY glCopyTextureSubImage2D (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint x, GLint y, GLsizei width, GLsizei height); +GLAPI void APIENTRY glCopyTextureSubImage3D (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint x, GLint y, GLsizei width, GLsizei height); +GLAPI void APIENTRY glTextureParameterf (GLuint texture, GLenum pname, GLfloat param); +GLAPI void APIENTRY glTextureParameterfv (GLuint texture, GLenum pname, const GLfloat *param); +GLAPI void APIENTRY glTextureParameteri (GLuint texture, GLenum pname, GLint param); +GLAPI void APIENTRY glTextureParameterIiv (GLuint texture, GLenum pname, const GLint *params); +GLAPI void APIENTRY glTextureParameterIuiv (GLuint texture, GLenum pname, const GLuint *params); +GLAPI void APIENTRY glTextureParameteriv (GLuint texture, GLenum pname, const GLint *param); +GLAPI void APIENTRY glGenerateTextureMipmap (GLuint texture); +GLAPI void APIENTRY glBindTextureUnit (GLuint unit, GLuint texture); +GLAPI void APIENTRY glGetTextureImage (GLuint texture, GLint level, GLenum format, GLenum type, GLsizei bufSize, void *pixels); +GLAPI void APIENTRY glGetCompressedTextureImage (GLuint texture, GLint level, GLsizei bufSize, void *pixels); +GLAPI void APIENTRY glGetTextureLevelParameterfv (GLuint texture, GLint level, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetTextureLevelParameteriv (GLuint texture, GLint level, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetTextureParameterfv (GLuint texture, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetTextureParameterIiv (GLuint texture, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetTextureParameterIuiv (GLuint texture, GLenum pname, GLuint *params); +GLAPI void APIENTRY glGetTextureParameteriv (GLuint texture, GLenum pname, GLint *params); +GLAPI void APIENTRY glCreateVertexArrays (GLsizei n, GLuint *arrays); +GLAPI void APIENTRY glDisableVertexArrayAttrib (GLuint vaobj, GLuint index); +GLAPI void APIENTRY glEnableVertexArrayAttrib (GLuint vaobj, GLuint index); +GLAPI void APIENTRY glVertexArrayElementBuffer (GLuint vaobj, GLuint buffer); +GLAPI void APIENTRY glVertexArrayVertexBuffer (GLuint vaobj, GLuint bindingindex, GLuint buffer, GLintptr offset, GLsizei stride); +GLAPI void APIENTRY glVertexArrayVertexBuffers (GLuint vaobj, GLuint first, GLsizei count, const GLuint *buffers, const GLintptr *offsets, const GLsizei *strides); +GLAPI void APIENTRY glVertexArrayAttribBinding (GLuint vaobj, GLuint attribindex, GLuint bindingindex); +GLAPI void APIENTRY glVertexArrayAttribFormat (GLuint vaobj, GLuint attribindex, GLint size, GLenum type, GLboolean normalized, GLuint relativeoffset); +GLAPI void APIENTRY glVertexArrayAttribIFormat (GLuint vaobj, GLuint attribindex, GLint size, GLenum type, GLuint relativeoffset); +GLAPI void APIENTRY glVertexArrayAttribLFormat (GLuint vaobj, GLuint attribindex, GLint size, GLenum type, GLuint relativeoffset); +GLAPI void APIENTRY glVertexArrayBindingDivisor (GLuint vaobj, GLuint bindingindex, GLuint divisor); +GLAPI void APIENTRY glGetVertexArrayiv (GLuint vaobj, GLenum pname, GLint *param); +GLAPI void APIENTRY glGetVertexArrayIndexediv (GLuint vaobj, GLuint index, GLenum pname, GLint *param); +GLAPI void APIENTRY glGetVertexArrayIndexed64iv (GLuint vaobj, GLuint index, GLenum pname, GLint64 *param); +GLAPI void APIENTRY glCreateSamplers (GLsizei n, GLuint *samplers); +GLAPI void APIENTRY glCreateProgramPipelines (GLsizei n, GLuint *pipelines); +GLAPI void APIENTRY glCreateQueries (GLenum target, GLsizei n, GLuint *ids); +GLAPI void APIENTRY glGetQueryBufferObjecti64v (GLuint id, GLuint buffer, GLenum pname, GLintptr offset); +GLAPI void APIENTRY glGetQueryBufferObjectiv (GLuint id, GLuint buffer, GLenum pname, GLintptr offset); +GLAPI void APIENTRY glGetQueryBufferObjectui64v (GLuint id, GLuint buffer, GLenum pname, GLintptr offset); +GLAPI void APIENTRY glGetQueryBufferObjectuiv (GLuint id, GLuint buffer, GLenum pname, GLintptr offset); +GLAPI void APIENTRY glMemoryBarrierByRegion (GLbitfield barriers); +GLAPI void APIENTRY glGetTextureSubImage (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, GLsizei bufSize, void *pixels); +GLAPI void APIENTRY glGetCompressedTextureSubImage (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLsizei bufSize, void *pixels); +GLAPI GLenum APIENTRY glGetGraphicsResetStatus (void); +GLAPI void APIENTRY glGetnCompressedTexImage (GLenum target, GLint lod, GLsizei bufSize, void *pixels); +GLAPI void APIENTRY glGetnTexImage (GLenum target, GLint level, GLenum format, GLenum type, GLsizei bufSize, void *pixels); +GLAPI void APIENTRY glGetnUniformdv (GLuint program, GLint location, GLsizei bufSize, GLdouble *params); +GLAPI void APIENTRY glGetnUniformfv (GLuint program, GLint location, GLsizei bufSize, GLfloat *params); +GLAPI void APIENTRY glGetnUniformiv (GLuint program, GLint location, GLsizei bufSize, GLint *params); +GLAPI void APIENTRY glGetnUniformuiv (GLuint program, GLint location, GLsizei bufSize, GLuint *params); +GLAPI void APIENTRY glReadnPixels (GLint x, GLint y, GLsizei width, GLsizei height, GLenum format, GLenum type, GLsizei bufSize, void *data); +GLAPI void APIENTRY glGetnMapdv (GLenum target, GLenum query, GLsizei bufSize, GLdouble *v); +GLAPI void APIENTRY glGetnMapfv (GLenum target, GLenum query, GLsizei bufSize, GLfloat *v); +GLAPI void APIENTRY glGetnMapiv (GLenum target, GLenum query, GLsizei bufSize, GLint *v); +GLAPI void APIENTRY glGetnPixelMapfv (GLenum map, GLsizei bufSize, GLfloat *values); +GLAPI void APIENTRY glGetnPixelMapuiv (GLenum map, GLsizei bufSize, GLuint *values); +GLAPI void APIENTRY glGetnPixelMapusv (GLenum map, GLsizei bufSize, GLushort *values); +GLAPI void APIENTRY glGetnPolygonStipple (GLsizei bufSize, GLubyte *pattern); +GLAPI void APIENTRY glGetnColorTable (GLenum target, GLenum format, GLenum type, GLsizei bufSize, void *table); +GLAPI void APIENTRY glGetnConvolutionFilter (GLenum target, GLenum format, GLenum type, GLsizei bufSize, void *image); +GLAPI void APIENTRY glGetnSeparableFilter (GLenum target, GLenum format, GLenum type, GLsizei rowBufSize, void *row, GLsizei columnBufSize, void *column, void *span); +GLAPI void APIENTRY glGetnHistogram (GLenum target, GLboolean reset, GLenum format, GLenum type, GLsizei bufSize, void *values); +GLAPI void APIENTRY glGetnMinmax (GLenum target, GLboolean reset, GLenum format, GLenum type, GLsizei bufSize, void *values); +GLAPI void APIENTRY glTextureBarrier (void); +#endif +#endif /* GL_VERSION_4_5 */ + +#ifndef GL_VERSION_4_6 +#define GL_VERSION_4_6 1 +#define GL_SHADER_BINARY_FORMAT_SPIR_V 0x9551 +#define GL_SPIR_V_BINARY 0x9552 +#define GL_PARAMETER_BUFFER 0x80EE +#define GL_PARAMETER_BUFFER_BINDING 0x80EF +#define GL_CONTEXT_FLAG_NO_ERROR_BIT 0x00000008 +#define GL_VERTICES_SUBMITTED 0x82EE +#define GL_PRIMITIVES_SUBMITTED 0x82EF +#define GL_VERTEX_SHADER_INVOCATIONS 0x82F0 +#define GL_TESS_CONTROL_SHADER_PATCHES 0x82F1 +#define GL_TESS_EVALUATION_SHADER_INVOCATIONS 0x82F2 +#define GL_GEOMETRY_SHADER_PRIMITIVES_EMITTED 0x82F3 +#define GL_FRAGMENT_SHADER_INVOCATIONS 0x82F4 +#define GL_COMPUTE_SHADER_INVOCATIONS 0x82F5 +#define GL_CLIPPING_INPUT_PRIMITIVES 0x82F6 +#define GL_CLIPPING_OUTPUT_PRIMITIVES 0x82F7 +#define GL_POLYGON_OFFSET_CLAMP 0x8E1B +#define GL_SPIR_V_EXTENSIONS 0x9553 +#define GL_NUM_SPIR_V_EXTENSIONS 0x9554 +#define GL_TEXTURE_MAX_ANISOTROPY 0x84FE +#define GL_MAX_TEXTURE_MAX_ANISOTROPY 0x84FF +#define GL_TRANSFORM_FEEDBACK_OVERFLOW 0x82EC +#define GL_TRANSFORM_FEEDBACK_STREAM_OVERFLOW 0x82ED +typedef void (APIENTRYP PFNGLSPECIALIZESHADERPROC) (GLuint shader, const GLchar *pEntryPoint, GLuint numSpecializationConstants, const GLuint *pConstantIndex, const GLuint *pConstantValue); +typedef void (APIENTRYP PFNGLMULTIDRAWARRAYSINDIRECTCOUNTPROC) (GLenum mode, const void *indirect, GLintptr drawcount, GLsizei maxdrawcount, GLsizei stride); +typedef void (APIENTRYP PFNGLMULTIDRAWELEMENTSINDIRECTCOUNTPROC) (GLenum mode, GLenum type, const void *indirect, GLintptr drawcount, GLsizei maxdrawcount, GLsizei stride); +typedef void (APIENTRYP PFNGLPOLYGONOFFSETCLAMPPROC) (GLfloat factor, GLfloat units, GLfloat clamp); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSpecializeShader (GLuint shader, const GLchar *pEntryPoint, GLuint numSpecializationConstants, const GLuint *pConstantIndex, const GLuint *pConstantValue); +GLAPI void APIENTRY glMultiDrawArraysIndirectCount (GLenum mode, const void *indirect, GLintptr drawcount, GLsizei maxdrawcount, GLsizei stride); +GLAPI void APIENTRY glMultiDrawElementsIndirectCount (GLenum mode, GLenum type, const void *indirect, GLintptr drawcount, GLsizei maxdrawcount, GLsizei stride); +GLAPI void APIENTRY glPolygonOffsetClamp (GLfloat factor, GLfloat units, GLfloat clamp); +#endif +#endif /* GL_VERSION_4_6 */ + +#ifndef GL_ARB_ES2_compatibility +#define GL_ARB_ES2_compatibility 1 +#endif /* GL_ARB_ES2_compatibility */ + +#ifndef GL_ARB_ES3_1_compatibility +#define GL_ARB_ES3_1_compatibility 1 +#endif /* GL_ARB_ES3_1_compatibility */ + +#ifndef GL_ARB_ES3_2_compatibility +#define GL_ARB_ES3_2_compatibility 1 +#define GL_PRIMITIVE_BOUNDING_BOX_ARB 0x92BE +#define GL_MULTISAMPLE_LINE_WIDTH_RANGE_ARB 0x9381 +#define GL_MULTISAMPLE_LINE_WIDTH_GRANULARITY_ARB 0x9382 +typedef void (APIENTRYP PFNGLPRIMITIVEBOUNDINGBOXARBPROC) (GLfloat minX, GLfloat minY, GLfloat minZ, GLfloat minW, GLfloat maxX, GLfloat maxY, GLfloat maxZ, GLfloat maxW); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPrimitiveBoundingBoxARB (GLfloat minX, GLfloat minY, GLfloat minZ, GLfloat minW, GLfloat maxX, GLfloat maxY, GLfloat maxZ, GLfloat maxW); +#endif +#endif /* GL_ARB_ES3_2_compatibility */ + +#ifndef GL_ARB_ES3_compatibility +#define GL_ARB_ES3_compatibility 1 +#endif /* GL_ARB_ES3_compatibility */ + +#ifndef GL_ARB_arrays_of_arrays +#define GL_ARB_arrays_of_arrays 1 +#endif /* GL_ARB_arrays_of_arrays */ + +#ifndef GL_ARB_base_instance +#define GL_ARB_base_instance 1 +#endif /* GL_ARB_base_instance */ + +#ifndef GL_ARB_bindless_texture +#define GL_ARB_bindless_texture 1 +typedef khronos_uint64_t GLuint64EXT; +#define GL_UNSIGNED_INT64_ARB 0x140F +typedef GLuint64 (APIENTRYP PFNGLGETTEXTUREHANDLEARBPROC) (GLuint texture); +typedef GLuint64 (APIENTRYP PFNGLGETTEXTURESAMPLERHANDLEARBPROC) (GLuint texture, GLuint sampler); +typedef void (APIENTRYP PFNGLMAKETEXTUREHANDLERESIDENTARBPROC) (GLuint64 handle); +typedef void (APIENTRYP PFNGLMAKETEXTUREHANDLENONRESIDENTARBPROC) (GLuint64 handle); +typedef GLuint64 (APIENTRYP PFNGLGETIMAGEHANDLEARBPROC) (GLuint texture, GLint level, GLboolean layered, GLint layer, GLenum format); +typedef void (APIENTRYP PFNGLMAKEIMAGEHANDLERESIDENTARBPROC) (GLuint64 handle, GLenum access); +typedef void (APIENTRYP PFNGLMAKEIMAGEHANDLENONRESIDENTARBPROC) (GLuint64 handle); +typedef void (APIENTRYP PFNGLUNIFORMHANDLEUI64ARBPROC) (GLint location, GLuint64 value); +typedef void (APIENTRYP PFNGLUNIFORMHANDLEUI64VARBPROC) (GLint location, GLsizei count, const GLuint64 *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMHANDLEUI64ARBPROC) (GLuint program, GLint location, GLuint64 value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMHANDLEUI64VARBPROC) (GLuint program, GLint location, GLsizei count, const GLuint64 *values); +typedef GLboolean (APIENTRYP PFNGLISTEXTUREHANDLERESIDENTARBPROC) (GLuint64 handle); +typedef GLboolean (APIENTRYP PFNGLISIMAGEHANDLERESIDENTARBPROC) (GLuint64 handle); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL1UI64ARBPROC) (GLuint index, GLuint64EXT x); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL1UI64VARBPROC) (GLuint index, const GLuint64EXT *v); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBLUI64VARBPROC) (GLuint index, GLenum pname, GLuint64EXT *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLuint64 APIENTRY glGetTextureHandleARB (GLuint texture); +GLAPI GLuint64 APIENTRY glGetTextureSamplerHandleARB (GLuint texture, GLuint sampler); +GLAPI void APIENTRY glMakeTextureHandleResidentARB (GLuint64 handle); +GLAPI void APIENTRY glMakeTextureHandleNonResidentARB (GLuint64 handle); +GLAPI GLuint64 APIENTRY glGetImageHandleARB (GLuint texture, GLint level, GLboolean layered, GLint layer, GLenum format); +GLAPI void APIENTRY glMakeImageHandleResidentARB (GLuint64 handle, GLenum access); +GLAPI void APIENTRY glMakeImageHandleNonResidentARB (GLuint64 handle); +GLAPI void APIENTRY glUniformHandleui64ARB (GLint location, GLuint64 value); +GLAPI void APIENTRY glUniformHandleui64vARB (GLint location, GLsizei count, const GLuint64 *value); +GLAPI void APIENTRY glProgramUniformHandleui64ARB (GLuint program, GLint location, GLuint64 value); +GLAPI void APIENTRY glProgramUniformHandleui64vARB (GLuint program, GLint location, GLsizei count, const GLuint64 *values); +GLAPI GLboolean APIENTRY glIsTextureHandleResidentARB (GLuint64 handle); +GLAPI GLboolean APIENTRY glIsImageHandleResidentARB (GLuint64 handle); +GLAPI void APIENTRY glVertexAttribL1ui64ARB (GLuint index, GLuint64EXT x); +GLAPI void APIENTRY glVertexAttribL1ui64vARB (GLuint index, const GLuint64EXT *v); +GLAPI void APIENTRY glGetVertexAttribLui64vARB (GLuint index, GLenum pname, GLuint64EXT *params); +#endif +#endif /* GL_ARB_bindless_texture */ + +#ifndef GL_ARB_blend_func_extended +#define GL_ARB_blend_func_extended 1 +#endif /* GL_ARB_blend_func_extended */ + +#ifndef GL_ARB_buffer_storage +#define GL_ARB_buffer_storage 1 +#endif /* GL_ARB_buffer_storage */ + +#ifndef GL_ARB_cl_event +#define GL_ARB_cl_event 1 +struct _cl_context; +struct _cl_event; +#define GL_SYNC_CL_EVENT_ARB 0x8240 +#define GL_SYNC_CL_EVENT_COMPLETE_ARB 0x8241 +typedef GLsync (APIENTRYP PFNGLCREATESYNCFROMCLEVENTARBPROC) (struct _cl_context *context, struct _cl_event *event, GLbitfield flags); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLsync APIENTRY glCreateSyncFromCLeventARB (struct _cl_context *context, struct _cl_event *event, GLbitfield flags); +#endif +#endif /* GL_ARB_cl_event */ + +#ifndef GL_ARB_clear_buffer_object +#define GL_ARB_clear_buffer_object 1 +#endif /* GL_ARB_clear_buffer_object */ + +#ifndef GL_ARB_clear_texture +#define GL_ARB_clear_texture 1 +#endif /* GL_ARB_clear_texture */ + +#ifndef GL_ARB_clip_control +#define GL_ARB_clip_control 1 +#endif /* GL_ARB_clip_control */ + +#ifndef GL_ARB_color_buffer_float +#define GL_ARB_color_buffer_float 1 +#define GL_RGBA_FLOAT_MODE_ARB 0x8820 +#define GL_CLAMP_VERTEX_COLOR_ARB 0x891A +#define GL_CLAMP_FRAGMENT_COLOR_ARB 0x891B +#define GL_CLAMP_READ_COLOR_ARB 0x891C +#define GL_FIXED_ONLY_ARB 0x891D +typedef void (APIENTRYP PFNGLCLAMPCOLORARBPROC) (GLenum target, GLenum clamp); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glClampColorARB (GLenum target, GLenum clamp); +#endif +#endif /* GL_ARB_color_buffer_float */ + +#ifndef GL_ARB_compatibility +#define GL_ARB_compatibility 1 +#endif /* GL_ARB_compatibility */ + +#ifndef GL_ARB_compressed_texture_pixel_storage +#define GL_ARB_compressed_texture_pixel_storage 1 +#endif /* GL_ARB_compressed_texture_pixel_storage */ + +#ifndef GL_ARB_compute_shader +#define GL_ARB_compute_shader 1 +#endif /* GL_ARB_compute_shader */ + +#ifndef GL_ARB_compute_variable_group_size +#define GL_ARB_compute_variable_group_size 1 +#define GL_MAX_COMPUTE_VARIABLE_GROUP_INVOCATIONS_ARB 0x9344 +#define GL_MAX_COMPUTE_FIXED_GROUP_INVOCATIONS_ARB 0x90EB +#define GL_MAX_COMPUTE_VARIABLE_GROUP_SIZE_ARB 0x9345 +#define GL_MAX_COMPUTE_FIXED_GROUP_SIZE_ARB 0x91BF +typedef void (APIENTRYP PFNGLDISPATCHCOMPUTEGROUPSIZEARBPROC) (GLuint num_groups_x, GLuint num_groups_y, GLuint num_groups_z, GLuint group_size_x, GLuint group_size_y, GLuint group_size_z); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDispatchComputeGroupSizeARB (GLuint num_groups_x, GLuint num_groups_y, GLuint num_groups_z, GLuint group_size_x, GLuint group_size_y, GLuint group_size_z); +#endif +#endif /* GL_ARB_compute_variable_group_size */ + +#ifndef GL_ARB_conditional_render_inverted +#define GL_ARB_conditional_render_inverted 1 +#endif /* GL_ARB_conditional_render_inverted */ + +#ifndef GL_ARB_conservative_depth +#define GL_ARB_conservative_depth 1 +#endif /* GL_ARB_conservative_depth */ + +#ifndef GL_ARB_copy_buffer +#define GL_ARB_copy_buffer 1 +#endif /* GL_ARB_copy_buffer */ + +#ifndef GL_ARB_copy_image +#define GL_ARB_copy_image 1 +#endif /* GL_ARB_copy_image */ + +#ifndef GL_ARB_cull_distance +#define GL_ARB_cull_distance 1 +#endif /* GL_ARB_cull_distance */ + +#ifndef GL_ARB_debug_output +#define GL_ARB_debug_output 1 +typedef void (APIENTRY *GLDEBUGPROCARB)(GLenum source,GLenum type,GLuint id,GLenum severity,GLsizei length,const GLchar *message,const void *userParam); +#define GL_DEBUG_OUTPUT_SYNCHRONOUS_ARB 0x8242 +#define GL_DEBUG_NEXT_LOGGED_MESSAGE_LENGTH_ARB 0x8243 +#define GL_DEBUG_CALLBACK_FUNCTION_ARB 0x8244 +#define GL_DEBUG_CALLBACK_USER_PARAM_ARB 0x8245 +#define GL_DEBUG_SOURCE_API_ARB 0x8246 +#define GL_DEBUG_SOURCE_WINDOW_SYSTEM_ARB 0x8247 +#define GL_DEBUG_SOURCE_SHADER_COMPILER_ARB 0x8248 +#define GL_DEBUG_SOURCE_THIRD_PARTY_ARB 0x8249 +#define GL_DEBUG_SOURCE_APPLICATION_ARB 0x824A +#define GL_DEBUG_SOURCE_OTHER_ARB 0x824B +#define GL_DEBUG_TYPE_ERROR_ARB 0x824C +#define GL_DEBUG_TYPE_DEPRECATED_BEHAVIOR_ARB 0x824D +#define GL_DEBUG_TYPE_UNDEFINED_BEHAVIOR_ARB 0x824E +#define GL_DEBUG_TYPE_PORTABILITY_ARB 0x824F +#define GL_DEBUG_TYPE_PERFORMANCE_ARB 0x8250 +#define GL_DEBUG_TYPE_OTHER_ARB 0x8251 +#define GL_MAX_DEBUG_MESSAGE_LENGTH_ARB 0x9143 +#define GL_MAX_DEBUG_LOGGED_MESSAGES_ARB 0x9144 +#define GL_DEBUG_LOGGED_MESSAGES_ARB 0x9145 +#define GL_DEBUG_SEVERITY_HIGH_ARB 0x9146 +#define GL_DEBUG_SEVERITY_MEDIUM_ARB 0x9147 +#define GL_DEBUG_SEVERITY_LOW_ARB 0x9148 +typedef void (APIENTRYP PFNGLDEBUGMESSAGECONTROLARBPROC) (GLenum source, GLenum type, GLenum severity, GLsizei count, const GLuint *ids, GLboolean enabled); +typedef void (APIENTRYP PFNGLDEBUGMESSAGEINSERTARBPROC) (GLenum source, GLenum type, GLuint id, GLenum severity, GLsizei length, const GLchar *buf); +typedef void (APIENTRYP PFNGLDEBUGMESSAGECALLBACKARBPROC) (GLDEBUGPROCARB callback, const void *userParam); +typedef GLuint (APIENTRYP PFNGLGETDEBUGMESSAGELOGARBPROC) (GLuint count, GLsizei bufSize, GLenum *sources, GLenum *types, GLuint *ids, GLenum *severities, GLsizei *lengths, GLchar *messageLog); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDebugMessageControlARB (GLenum source, GLenum type, GLenum severity, GLsizei count, const GLuint *ids, GLboolean enabled); +GLAPI void APIENTRY glDebugMessageInsertARB (GLenum source, GLenum type, GLuint id, GLenum severity, GLsizei length, const GLchar *buf); +GLAPI void APIENTRY glDebugMessageCallbackARB (GLDEBUGPROCARB callback, const void *userParam); +GLAPI GLuint APIENTRY glGetDebugMessageLogARB (GLuint count, GLsizei bufSize, GLenum *sources, GLenum *types, GLuint *ids, GLenum *severities, GLsizei *lengths, GLchar *messageLog); +#endif +#endif /* GL_ARB_debug_output */ + +#ifndef GL_ARB_depth_buffer_float +#define GL_ARB_depth_buffer_float 1 +#endif /* GL_ARB_depth_buffer_float */ + +#ifndef GL_ARB_depth_clamp +#define GL_ARB_depth_clamp 1 +#endif /* GL_ARB_depth_clamp */ + +#ifndef GL_ARB_depth_texture +#define GL_ARB_depth_texture 1 +#define GL_DEPTH_COMPONENT16_ARB 0x81A5 +#define GL_DEPTH_COMPONENT24_ARB 0x81A6 +#define GL_DEPTH_COMPONENT32_ARB 0x81A7 +#define GL_TEXTURE_DEPTH_SIZE_ARB 0x884A +#define GL_DEPTH_TEXTURE_MODE_ARB 0x884B +#endif /* GL_ARB_depth_texture */ + +#ifndef GL_ARB_derivative_control +#define GL_ARB_derivative_control 1 +#endif /* GL_ARB_derivative_control */ + +#ifndef GL_ARB_direct_state_access +#define GL_ARB_direct_state_access 1 +#endif /* GL_ARB_direct_state_access */ + +#ifndef GL_ARB_draw_buffers +#define GL_ARB_draw_buffers 1 +#define GL_MAX_DRAW_BUFFERS_ARB 0x8824 +#define GL_DRAW_BUFFER0_ARB 0x8825 +#define GL_DRAW_BUFFER1_ARB 0x8826 +#define GL_DRAW_BUFFER2_ARB 0x8827 +#define GL_DRAW_BUFFER3_ARB 0x8828 +#define GL_DRAW_BUFFER4_ARB 0x8829 +#define GL_DRAW_BUFFER5_ARB 0x882A +#define GL_DRAW_BUFFER6_ARB 0x882B +#define GL_DRAW_BUFFER7_ARB 0x882C +#define GL_DRAW_BUFFER8_ARB 0x882D +#define GL_DRAW_BUFFER9_ARB 0x882E +#define GL_DRAW_BUFFER10_ARB 0x882F +#define GL_DRAW_BUFFER11_ARB 0x8830 +#define GL_DRAW_BUFFER12_ARB 0x8831 +#define GL_DRAW_BUFFER13_ARB 0x8832 +#define GL_DRAW_BUFFER14_ARB 0x8833 +#define GL_DRAW_BUFFER15_ARB 0x8834 +typedef void (APIENTRYP PFNGLDRAWBUFFERSARBPROC) (GLsizei n, const GLenum *bufs); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawBuffersARB (GLsizei n, const GLenum *bufs); +#endif +#endif /* GL_ARB_draw_buffers */ + +#ifndef GL_ARB_draw_buffers_blend +#define GL_ARB_draw_buffers_blend 1 +typedef void (APIENTRYP PFNGLBLENDEQUATIONIARBPROC) (GLuint buf, GLenum mode); +typedef void (APIENTRYP PFNGLBLENDEQUATIONSEPARATEIARBPROC) (GLuint buf, GLenum modeRGB, GLenum modeAlpha); +typedef void (APIENTRYP PFNGLBLENDFUNCIARBPROC) (GLuint buf, GLenum src, GLenum dst); +typedef void (APIENTRYP PFNGLBLENDFUNCSEPARATEIARBPROC) (GLuint buf, GLenum srcRGB, GLenum dstRGB, GLenum srcAlpha, GLenum dstAlpha); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendEquationiARB (GLuint buf, GLenum mode); +GLAPI void APIENTRY glBlendEquationSeparateiARB (GLuint buf, GLenum modeRGB, GLenum modeAlpha); +GLAPI void APIENTRY glBlendFunciARB (GLuint buf, GLenum src, GLenum dst); +GLAPI void APIENTRY glBlendFuncSeparateiARB (GLuint buf, GLenum srcRGB, GLenum dstRGB, GLenum srcAlpha, GLenum dstAlpha); +#endif +#endif /* GL_ARB_draw_buffers_blend */ + +#ifndef GL_ARB_draw_elements_base_vertex +#define GL_ARB_draw_elements_base_vertex 1 +#endif /* GL_ARB_draw_elements_base_vertex */ + +#ifndef GL_ARB_draw_indirect +#define GL_ARB_draw_indirect 1 +#endif /* GL_ARB_draw_indirect */ + +#ifndef GL_ARB_draw_instanced +#define GL_ARB_draw_instanced 1 +typedef void (APIENTRYP PFNGLDRAWARRAYSINSTANCEDARBPROC) (GLenum mode, GLint first, GLsizei count, GLsizei primcount); +typedef void (APIENTRYP PFNGLDRAWELEMENTSINSTANCEDARBPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei primcount); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawArraysInstancedARB (GLenum mode, GLint first, GLsizei count, GLsizei primcount); +GLAPI void APIENTRY glDrawElementsInstancedARB (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei primcount); +#endif +#endif /* GL_ARB_draw_instanced */ + +#ifndef GL_ARB_enhanced_layouts +#define GL_ARB_enhanced_layouts 1 +#endif /* GL_ARB_enhanced_layouts */ + +#ifndef GL_ARB_explicit_attrib_location +#define GL_ARB_explicit_attrib_location 1 +#endif /* GL_ARB_explicit_attrib_location */ + +#ifndef GL_ARB_explicit_uniform_location +#define GL_ARB_explicit_uniform_location 1 +#endif /* GL_ARB_explicit_uniform_location */ + +#ifndef GL_ARB_fragment_coord_conventions +#define GL_ARB_fragment_coord_conventions 1 +#endif /* GL_ARB_fragment_coord_conventions */ + +#ifndef GL_ARB_fragment_layer_viewport +#define GL_ARB_fragment_layer_viewport 1 +#endif /* GL_ARB_fragment_layer_viewport */ + +#ifndef GL_ARB_fragment_program +#define GL_ARB_fragment_program 1 +#define GL_FRAGMENT_PROGRAM_ARB 0x8804 +#define GL_PROGRAM_FORMAT_ASCII_ARB 0x8875 +#define GL_PROGRAM_LENGTH_ARB 0x8627 +#define GL_PROGRAM_FORMAT_ARB 0x8876 +#define GL_PROGRAM_BINDING_ARB 0x8677 +#define GL_PROGRAM_INSTRUCTIONS_ARB 0x88A0 +#define GL_MAX_PROGRAM_INSTRUCTIONS_ARB 0x88A1 +#define GL_PROGRAM_NATIVE_INSTRUCTIONS_ARB 0x88A2 +#define GL_MAX_PROGRAM_NATIVE_INSTRUCTIONS_ARB 0x88A3 +#define GL_PROGRAM_TEMPORARIES_ARB 0x88A4 +#define GL_MAX_PROGRAM_TEMPORARIES_ARB 0x88A5 +#define GL_PROGRAM_NATIVE_TEMPORARIES_ARB 0x88A6 +#define GL_MAX_PROGRAM_NATIVE_TEMPORARIES_ARB 0x88A7 +#define GL_PROGRAM_PARAMETERS_ARB 0x88A8 +#define GL_MAX_PROGRAM_PARAMETERS_ARB 0x88A9 +#define GL_PROGRAM_NATIVE_PARAMETERS_ARB 0x88AA +#define GL_MAX_PROGRAM_NATIVE_PARAMETERS_ARB 0x88AB +#define GL_PROGRAM_ATTRIBS_ARB 0x88AC +#define GL_MAX_PROGRAM_ATTRIBS_ARB 0x88AD +#define GL_PROGRAM_NATIVE_ATTRIBS_ARB 0x88AE +#define GL_MAX_PROGRAM_NATIVE_ATTRIBS_ARB 0x88AF +#define GL_MAX_PROGRAM_LOCAL_PARAMETERS_ARB 0x88B4 +#define GL_MAX_PROGRAM_ENV_PARAMETERS_ARB 0x88B5 +#define GL_PROGRAM_UNDER_NATIVE_LIMITS_ARB 0x88B6 +#define GL_PROGRAM_ALU_INSTRUCTIONS_ARB 0x8805 +#define GL_PROGRAM_TEX_INSTRUCTIONS_ARB 0x8806 +#define GL_PROGRAM_TEX_INDIRECTIONS_ARB 0x8807 +#define GL_PROGRAM_NATIVE_ALU_INSTRUCTIONS_ARB 0x8808 +#define GL_PROGRAM_NATIVE_TEX_INSTRUCTIONS_ARB 0x8809 +#define GL_PROGRAM_NATIVE_TEX_INDIRECTIONS_ARB 0x880A +#define GL_MAX_PROGRAM_ALU_INSTRUCTIONS_ARB 0x880B +#define GL_MAX_PROGRAM_TEX_INSTRUCTIONS_ARB 0x880C +#define GL_MAX_PROGRAM_TEX_INDIRECTIONS_ARB 0x880D +#define GL_MAX_PROGRAM_NATIVE_ALU_INSTRUCTIONS_ARB 0x880E +#define GL_MAX_PROGRAM_NATIVE_TEX_INSTRUCTIONS_ARB 0x880F +#define GL_MAX_PROGRAM_NATIVE_TEX_INDIRECTIONS_ARB 0x8810 +#define GL_PROGRAM_STRING_ARB 0x8628 +#define GL_PROGRAM_ERROR_POSITION_ARB 0x864B +#define GL_CURRENT_MATRIX_ARB 0x8641 +#define GL_TRANSPOSE_CURRENT_MATRIX_ARB 0x88B7 +#define GL_CURRENT_MATRIX_STACK_DEPTH_ARB 0x8640 +#define GL_MAX_PROGRAM_MATRICES_ARB 0x862F +#define GL_MAX_PROGRAM_MATRIX_STACK_DEPTH_ARB 0x862E +#define GL_MAX_TEXTURE_COORDS_ARB 0x8871 +#define GL_MAX_TEXTURE_IMAGE_UNITS_ARB 0x8872 +#define GL_PROGRAM_ERROR_STRING_ARB 0x8874 +#define GL_MATRIX0_ARB 0x88C0 +#define GL_MATRIX1_ARB 0x88C1 +#define GL_MATRIX2_ARB 0x88C2 +#define GL_MATRIX3_ARB 0x88C3 +#define GL_MATRIX4_ARB 0x88C4 +#define GL_MATRIX5_ARB 0x88C5 +#define GL_MATRIX6_ARB 0x88C6 +#define GL_MATRIX7_ARB 0x88C7 +#define GL_MATRIX8_ARB 0x88C8 +#define GL_MATRIX9_ARB 0x88C9 +#define GL_MATRIX10_ARB 0x88CA +#define GL_MATRIX11_ARB 0x88CB +#define GL_MATRIX12_ARB 0x88CC +#define GL_MATRIX13_ARB 0x88CD +#define GL_MATRIX14_ARB 0x88CE +#define GL_MATRIX15_ARB 0x88CF +#define GL_MATRIX16_ARB 0x88D0 +#define GL_MATRIX17_ARB 0x88D1 +#define GL_MATRIX18_ARB 0x88D2 +#define GL_MATRIX19_ARB 0x88D3 +#define GL_MATRIX20_ARB 0x88D4 +#define GL_MATRIX21_ARB 0x88D5 +#define GL_MATRIX22_ARB 0x88D6 +#define GL_MATRIX23_ARB 0x88D7 +#define GL_MATRIX24_ARB 0x88D8 +#define GL_MATRIX25_ARB 0x88D9 +#define GL_MATRIX26_ARB 0x88DA +#define GL_MATRIX27_ARB 0x88DB +#define GL_MATRIX28_ARB 0x88DC +#define GL_MATRIX29_ARB 0x88DD +#define GL_MATRIX30_ARB 0x88DE +#define GL_MATRIX31_ARB 0x88DF +typedef void (APIENTRYP PFNGLPROGRAMSTRINGARBPROC) (GLenum target, GLenum format, GLsizei len, const void *string); +typedef void (APIENTRYP PFNGLBINDPROGRAMARBPROC) (GLenum target, GLuint program); +typedef void (APIENTRYP PFNGLDELETEPROGRAMSARBPROC) (GLsizei n, const GLuint *programs); +typedef void (APIENTRYP PFNGLGENPROGRAMSARBPROC) (GLsizei n, GLuint *programs); +typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETER4DARBPROC) (GLenum target, GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETER4DVARBPROC) (GLenum target, GLuint index, const GLdouble *params); +typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETER4FARBPROC) (GLenum target, GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETER4FVARBPROC) (GLenum target, GLuint index, const GLfloat *params); +typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETER4DARBPROC) (GLenum target, GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETER4DVARBPROC) (GLenum target, GLuint index, const GLdouble *params); +typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETER4FARBPROC) (GLenum target, GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETER4FVARBPROC) (GLenum target, GLuint index, const GLfloat *params); +typedef void (APIENTRYP PFNGLGETPROGRAMENVPARAMETERDVARBPROC) (GLenum target, GLuint index, GLdouble *params); +typedef void (APIENTRYP PFNGLGETPROGRAMENVPARAMETERFVARBPROC) (GLenum target, GLuint index, GLfloat *params); +typedef void (APIENTRYP PFNGLGETPROGRAMLOCALPARAMETERDVARBPROC) (GLenum target, GLuint index, GLdouble *params); +typedef void (APIENTRYP PFNGLGETPROGRAMLOCALPARAMETERFVARBPROC) (GLenum target, GLuint index, GLfloat *params); +typedef void (APIENTRYP PFNGLGETPROGRAMIVARBPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETPROGRAMSTRINGARBPROC) (GLenum target, GLenum pname, void *string); +typedef GLboolean (APIENTRYP PFNGLISPROGRAMARBPROC) (GLuint program); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glProgramStringARB (GLenum target, GLenum format, GLsizei len, const void *string); +GLAPI void APIENTRY glBindProgramARB (GLenum target, GLuint program); +GLAPI void APIENTRY glDeleteProgramsARB (GLsizei n, const GLuint *programs); +GLAPI void APIENTRY glGenProgramsARB (GLsizei n, GLuint *programs); +GLAPI void APIENTRY glProgramEnvParameter4dARB (GLenum target, GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +GLAPI void APIENTRY glProgramEnvParameter4dvARB (GLenum target, GLuint index, const GLdouble *params); +GLAPI void APIENTRY glProgramEnvParameter4fARB (GLenum target, GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +GLAPI void APIENTRY glProgramEnvParameter4fvARB (GLenum target, GLuint index, const GLfloat *params); +GLAPI void APIENTRY glProgramLocalParameter4dARB (GLenum target, GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +GLAPI void APIENTRY glProgramLocalParameter4dvARB (GLenum target, GLuint index, const GLdouble *params); +GLAPI void APIENTRY glProgramLocalParameter4fARB (GLenum target, GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +GLAPI void APIENTRY glProgramLocalParameter4fvARB (GLenum target, GLuint index, const GLfloat *params); +GLAPI void APIENTRY glGetProgramEnvParameterdvARB (GLenum target, GLuint index, GLdouble *params); +GLAPI void APIENTRY glGetProgramEnvParameterfvARB (GLenum target, GLuint index, GLfloat *params); +GLAPI void APIENTRY glGetProgramLocalParameterdvARB (GLenum target, GLuint index, GLdouble *params); +GLAPI void APIENTRY glGetProgramLocalParameterfvARB (GLenum target, GLuint index, GLfloat *params); +GLAPI void APIENTRY glGetProgramivARB (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetProgramStringARB (GLenum target, GLenum pname, void *string); +GLAPI GLboolean APIENTRY glIsProgramARB (GLuint program); +#endif +#endif /* GL_ARB_fragment_program */ + +#ifndef GL_ARB_fragment_program_shadow +#define GL_ARB_fragment_program_shadow 1 +#endif /* GL_ARB_fragment_program_shadow */ + +#ifndef GL_ARB_fragment_shader +#define GL_ARB_fragment_shader 1 +#define GL_FRAGMENT_SHADER_ARB 0x8B30 +#define GL_MAX_FRAGMENT_UNIFORM_COMPONENTS_ARB 0x8B49 +#define GL_FRAGMENT_SHADER_DERIVATIVE_HINT_ARB 0x8B8B +#endif /* GL_ARB_fragment_shader */ + +#ifndef GL_ARB_fragment_shader_interlock +#define GL_ARB_fragment_shader_interlock 1 +#endif /* GL_ARB_fragment_shader_interlock */ + +#ifndef GL_ARB_framebuffer_no_attachments +#define GL_ARB_framebuffer_no_attachments 1 +#endif /* GL_ARB_framebuffer_no_attachments */ + +#ifndef GL_ARB_framebuffer_object +#define GL_ARB_framebuffer_object 1 +#endif /* GL_ARB_framebuffer_object */ + +#ifndef GL_ARB_framebuffer_sRGB +#define GL_ARB_framebuffer_sRGB 1 +#endif /* GL_ARB_framebuffer_sRGB */ + +#ifndef GL_ARB_geometry_shader4 +#define GL_ARB_geometry_shader4 1 +#define GL_LINES_ADJACENCY_ARB 0x000A +#define GL_LINE_STRIP_ADJACENCY_ARB 0x000B +#define GL_TRIANGLES_ADJACENCY_ARB 0x000C +#define GL_TRIANGLE_STRIP_ADJACENCY_ARB 0x000D +#define GL_PROGRAM_POINT_SIZE_ARB 0x8642 +#define GL_MAX_GEOMETRY_TEXTURE_IMAGE_UNITS_ARB 0x8C29 +#define GL_FRAMEBUFFER_ATTACHMENT_LAYERED_ARB 0x8DA7 +#define GL_FRAMEBUFFER_INCOMPLETE_LAYER_TARGETS_ARB 0x8DA8 +#define GL_FRAMEBUFFER_INCOMPLETE_LAYER_COUNT_ARB 0x8DA9 +#define GL_GEOMETRY_SHADER_ARB 0x8DD9 +#define GL_GEOMETRY_VERTICES_OUT_ARB 0x8DDA +#define GL_GEOMETRY_INPUT_TYPE_ARB 0x8DDB +#define GL_GEOMETRY_OUTPUT_TYPE_ARB 0x8DDC +#define GL_MAX_GEOMETRY_VARYING_COMPONENTS_ARB 0x8DDD +#define GL_MAX_VERTEX_VARYING_COMPONENTS_ARB 0x8DDE +#define GL_MAX_GEOMETRY_UNIFORM_COMPONENTS_ARB 0x8DDF +#define GL_MAX_GEOMETRY_OUTPUT_VERTICES_ARB 0x8DE0 +#define GL_MAX_GEOMETRY_TOTAL_OUTPUT_COMPONENTS_ARB 0x8DE1 +typedef void (APIENTRYP PFNGLPROGRAMPARAMETERIARBPROC) (GLuint program, GLenum pname, GLint value); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTUREARBPROC) (GLenum target, GLenum attachment, GLuint texture, GLint level); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTURELAYERARBPROC) (GLenum target, GLenum attachment, GLuint texture, GLint level, GLint layer); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTUREFACEARBPROC) (GLenum target, GLenum attachment, GLuint texture, GLint level, GLenum face); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glProgramParameteriARB (GLuint program, GLenum pname, GLint value); +GLAPI void APIENTRY glFramebufferTextureARB (GLenum target, GLenum attachment, GLuint texture, GLint level); +GLAPI void APIENTRY glFramebufferTextureLayerARB (GLenum target, GLenum attachment, GLuint texture, GLint level, GLint layer); +GLAPI void APIENTRY glFramebufferTextureFaceARB (GLenum target, GLenum attachment, GLuint texture, GLint level, GLenum face); +#endif +#endif /* GL_ARB_geometry_shader4 */ + +#ifndef GL_ARB_get_program_binary +#define GL_ARB_get_program_binary 1 +#endif /* GL_ARB_get_program_binary */ + +#ifndef GL_ARB_get_texture_sub_image +#define GL_ARB_get_texture_sub_image 1 +#endif /* GL_ARB_get_texture_sub_image */ + +#ifndef GL_ARB_gl_spirv +#define GL_ARB_gl_spirv 1 +#define GL_SHADER_BINARY_FORMAT_SPIR_V_ARB 0x9551 +#define GL_SPIR_V_BINARY_ARB 0x9552 +typedef void (APIENTRYP PFNGLSPECIALIZESHADERARBPROC) (GLuint shader, const GLchar *pEntryPoint, GLuint numSpecializationConstants, const GLuint *pConstantIndex, const GLuint *pConstantValue); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSpecializeShaderARB (GLuint shader, const GLchar *pEntryPoint, GLuint numSpecializationConstants, const GLuint *pConstantIndex, const GLuint *pConstantValue); +#endif +#endif /* GL_ARB_gl_spirv */ + +#ifndef GL_ARB_gpu_shader5 +#define GL_ARB_gpu_shader5 1 +#endif /* GL_ARB_gpu_shader5 */ + +#ifndef GL_ARB_gpu_shader_fp64 +#define GL_ARB_gpu_shader_fp64 1 +#endif /* GL_ARB_gpu_shader_fp64 */ + +#ifndef GL_ARB_gpu_shader_int64 +#define GL_ARB_gpu_shader_int64 1 +#define GL_INT64_ARB 0x140E +#define GL_INT64_VEC2_ARB 0x8FE9 +#define GL_INT64_VEC3_ARB 0x8FEA +#define GL_INT64_VEC4_ARB 0x8FEB +#define GL_UNSIGNED_INT64_VEC2_ARB 0x8FF5 +#define GL_UNSIGNED_INT64_VEC3_ARB 0x8FF6 +#define GL_UNSIGNED_INT64_VEC4_ARB 0x8FF7 +typedef void (APIENTRYP PFNGLUNIFORM1I64ARBPROC) (GLint location, GLint64 x); +typedef void (APIENTRYP PFNGLUNIFORM2I64ARBPROC) (GLint location, GLint64 x, GLint64 y); +typedef void (APIENTRYP PFNGLUNIFORM3I64ARBPROC) (GLint location, GLint64 x, GLint64 y, GLint64 z); +typedef void (APIENTRYP PFNGLUNIFORM4I64ARBPROC) (GLint location, GLint64 x, GLint64 y, GLint64 z, GLint64 w); +typedef void (APIENTRYP PFNGLUNIFORM1I64VARBPROC) (GLint location, GLsizei count, const GLint64 *value); +typedef void (APIENTRYP PFNGLUNIFORM2I64VARBPROC) (GLint location, GLsizei count, const GLint64 *value); +typedef void (APIENTRYP PFNGLUNIFORM3I64VARBPROC) (GLint location, GLsizei count, const GLint64 *value); +typedef void (APIENTRYP PFNGLUNIFORM4I64VARBPROC) (GLint location, GLsizei count, const GLint64 *value); +typedef void (APIENTRYP PFNGLUNIFORM1UI64ARBPROC) (GLint location, GLuint64 x); +typedef void (APIENTRYP PFNGLUNIFORM2UI64ARBPROC) (GLint location, GLuint64 x, GLuint64 y); +typedef void (APIENTRYP PFNGLUNIFORM3UI64ARBPROC) (GLint location, GLuint64 x, GLuint64 y, GLuint64 z); +typedef void (APIENTRYP PFNGLUNIFORM4UI64ARBPROC) (GLint location, GLuint64 x, GLuint64 y, GLuint64 z, GLuint64 w); +typedef void (APIENTRYP PFNGLUNIFORM1UI64VARBPROC) (GLint location, GLsizei count, const GLuint64 *value); +typedef void (APIENTRYP PFNGLUNIFORM2UI64VARBPROC) (GLint location, GLsizei count, const GLuint64 *value); +typedef void (APIENTRYP PFNGLUNIFORM3UI64VARBPROC) (GLint location, GLsizei count, const GLuint64 *value); +typedef void (APIENTRYP PFNGLUNIFORM4UI64VARBPROC) (GLint location, GLsizei count, const GLuint64 *value); +typedef void (APIENTRYP PFNGLGETUNIFORMI64VARBPROC) (GLuint program, GLint location, GLint64 *params); +typedef void (APIENTRYP PFNGLGETUNIFORMUI64VARBPROC) (GLuint program, GLint location, GLuint64 *params); +typedef void (APIENTRYP PFNGLGETNUNIFORMI64VARBPROC) (GLuint program, GLint location, GLsizei bufSize, GLint64 *params); +typedef void (APIENTRYP PFNGLGETNUNIFORMUI64VARBPROC) (GLuint program, GLint location, GLsizei bufSize, GLuint64 *params); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1I64ARBPROC) (GLuint program, GLint location, GLint64 x); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2I64ARBPROC) (GLuint program, GLint location, GLint64 x, GLint64 y); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3I64ARBPROC) (GLuint program, GLint location, GLint64 x, GLint64 y, GLint64 z); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4I64ARBPROC) (GLuint program, GLint location, GLint64 x, GLint64 y, GLint64 z, GLint64 w); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1I64VARBPROC) (GLuint program, GLint location, GLsizei count, const GLint64 *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2I64VARBPROC) (GLuint program, GLint location, GLsizei count, const GLint64 *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3I64VARBPROC) (GLuint program, GLint location, GLsizei count, const GLint64 *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4I64VARBPROC) (GLuint program, GLint location, GLsizei count, const GLint64 *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1UI64ARBPROC) (GLuint program, GLint location, GLuint64 x); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2UI64ARBPROC) (GLuint program, GLint location, GLuint64 x, GLuint64 y); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3UI64ARBPROC) (GLuint program, GLint location, GLuint64 x, GLuint64 y, GLuint64 z); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4UI64ARBPROC) (GLuint program, GLint location, GLuint64 x, GLuint64 y, GLuint64 z, GLuint64 w); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1UI64VARBPROC) (GLuint program, GLint location, GLsizei count, const GLuint64 *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2UI64VARBPROC) (GLuint program, GLint location, GLsizei count, const GLuint64 *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3UI64VARBPROC) (GLuint program, GLint location, GLsizei count, const GLuint64 *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4UI64VARBPROC) (GLuint program, GLint location, GLsizei count, const GLuint64 *value); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glUniform1i64ARB (GLint location, GLint64 x); +GLAPI void APIENTRY glUniform2i64ARB (GLint location, GLint64 x, GLint64 y); +GLAPI void APIENTRY glUniform3i64ARB (GLint location, GLint64 x, GLint64 y, GLint64 z); +GLAPI void APIENTRY glUniform4i64ARB (GLint location, GLint64 x, GLint64 y, GLint64 z, GLint64 w); +GLAPI void APIENTRY glUniform1i64vARB (GLint location, GLsizei count, const GLint64 *value); +GLAPI void APIENTRY glUniform2i64vARB (GLint location, GLsizei count, const GLint64 *value); +GLAPI void APIENTRY glUniform3i64vARB (GLint location, GLsizei count, const GLint64 *value); +GLAPI void APIENTRY glUniform4i64vARB (GLint location, GLsizei count, const GLint64 *value); +GLAPI void APIENTRY glUniform1ui64ARB (GLint location, GLuint64 x); +GLAPI void APIENTRY glUniform2ui64ARB (GLint location, GLuint64 x, GLuint64 y); +GLAPI void APIENTRY glUniform3ui64ARB (GLint location, GLuint64 x, GLuint64 y, GLuint64 z); +GLAPI void APIENTRY glUniform4ui64ARB (GLint location, GLuint64 x, GLuint64 y, GLuint64 z, GLuint64 w); +GLAPI void APIENTRY glUniform1ui64vARB (GLint location, GLsizei count, const GLuint64 *value); +GLAPI void APIENTRY glUniform2ui64vARB (GLint location, GLsizei count, const GLuint64 *value); +GLAPI void APIENTRY glUniform3ui64vARB (GLint location, GLsizei count, const GLuint64 *value); +GLAPI void APIENTRY glUniform4ui64vARB (GLint location, GLsizei count, const GLuint64 *value); +GLAPI void APIENTRY glGetUniformi64vARB (GLuint program, GLint location, GLint64 *params); +GLAPI void APIENTRY glGetUniformui64vARB (GLuint program, GLint location, GLuint64 *params); +GLAPI void APIENTRY glGetnUniformi64vARB (GLuint program, GLint location, GLsizei bufSize, GLint64 *params); +GLAPI void APIENTRY glGetnUniformui64vARB (GLuint program, GLint location, GLsizei bufSize, GLuint64 *params); +GLAPI void APIENTRY glProgramUniform1i64ARB (GLuint program, GLint location, GLint64 x); +GLAPI void APIENTRY glProgramUniform2i64ARB (GLuint program, GLint location, GLint64 x, GLint64 y); +GLAPI void APIENTRY glProgramUniform3i64ARB (GLuint program, GLint location, GLint64 x, GLint64 y, GLint64 z); +GLAPI void APIENTRY glProgramUniform4i64ARB (GLuint program, GLint location, GLint64 x, GLint64 y, GLint64 z, GLint64 w); +GLAPI void APIENTRY glProgramUniform1i64vARB (GLuint program, GLint location, GLsizei count, const GLint64 *value); +GLAPI void APIENTRY glProgramUniform2i64vARB (GLuint program, GLint location, GLsizei count, const GLint64 *value); +GLAPI void APIENTRY glProgramUniform3i64vARB (GLuint program, GLint location, GLsizei count, const GLint64 *value); +GLAPI void APIENTRY glProgramUniform4i64vARB (GLuint program, GLint location, GLsizei count, const GLint64 *value); +GLAPI void APIENTRY glProgramUniform1ui64ARB (GLuint program, GLint location, GLuint64 x); +GLAPI void APIENTRY glProgramUniform2ui64ARB (GLuint program, GLint location, GLuint64 x, GLuint64 y); +GLAPI void APIENTRY glProgramUniform3ui64ARB (GLuint program, GLint location, GLuint64 x, GLuint64 y, GLuint64 z); +GLAPI void APIENTRY glProgramUniform4ui64ARB (GLuint program, GLint location, GLuint64 x, GLuint64 y, GLuint64 z, GLuint64 w); +GLAPI void APIENTRY glProgramUniform1ui64vARB (GLuint program, GLint location, GLsizei count, const GLuint64 *value); +GLAPI void APIENTRY glProgramUniform2ui64vARB (GLuint program, GLint location, GLsizei count, const GLuint64 *value); +GLAPI void APIENTRY glProgramUniform3ui64vARB (GLuint program, GLint location, GLsizei count, const GLuint64 *value); +GLAPI void APIENTRY glProgramUniform4ui64vARB (GLuint program, GLint location, GLsizei count, const GLuint64 *value); +#endif +#endif /* GL_ARB_gpu_shader_int64 */ + +#ifndef GL_ARB_half_float_pixel +#define GL_ARB_half_float_pixel 1 +typedef khronos_uint16_t GLhalfARB; +#define GL_HALF_FLOAT_ARB 0x140B +#endif /* GL_ARB_half_float_pixel */ + +#ifndef GL_ARB_half_float_vertex +#define GL_ARB_half_float_vertex 1 +#endif /* GL_ARB_half_float_vertex */ + +#ifndef GL_ARB_imaging +#define GL_ARB_imaging 1 +#define GL_CONVOLUTION_BORDER_MODE 0x8013 +#define GL_CONVOLUTION_FILTER_SCALE 0x8014 +#define GL_CONVOLUTION_FILTER_BIAS 0x8015 +#define GL_REDUCE 0x8016 +#define GL_CONVOLUTION_FORMAT 0x8017 +#define GL_CONVOLUTION_WIDTH 0x8018 +#define GL_CONVOLUTION_HEIGHT 0x8019 +#define GL_MAX_CONVOLUTION_WIDTH 0x801A +#define GL_MAX_CONVOLUTION_HEIGHT 0x801B +#define GL_POST_CONVOLUTION_RED_SCALE 0x801C +#define GL_POST_CONVOLUTION_GREEN_SCALE 0x801D +#define GL_POST_CONVOLUTION_BLUE_SCALE 0x801E +#define GL_POST_CONVOLUTION_ALPHA_SCALE 0x801F +#define GL_POST_CONVOLUTION_RED_BIAS 0x8020 +#define GL_POST_CONVOLUTION_GREEN_BIAS 0x8021 +#define GL_POST_CONVOLUTION_BLUE_BIAS 0x8022 +#define GL_POST_CONVOLUTION_ALPHA_BIAS 0x8023 +#define GL_HISTOGRAM_WIDTH 0x8026 +#define GL_HISTOGRAM_FORMAT 0x8027 +#define GL_HISTOGRAM_RED_SIZE 0x8028 +#define GL_HISTOGRAM_GREEN_SIZE 0x8029 +#define GL_HISTOGRAM_BLUE_SIZE 0x802A +#define GL_HISTOGRAM_ALPHA_SIZE 0x802B +#define GL_HISTOGRAM_LUMINANCE_SIZE 0x802C +#define GL_HISTOGRAM_SINK 0x802D +#define GL_MINMAX_FORMAT 0x802F +#define GL_MINMAX_SINK 0x8030 +#define GL_TABLE_TOO_LARGE 0x8031 +#define GL_COLOR_MATRIX 0x80B1 +#define GL_COLOR_MATRIX_STACK_DEPTH 0x80B2 +#define GL_MAX_COLOR_MATRIX_STACK_DEPTH 0x80B3 +#define GL_POST_COLOR_MATRIX_RED_SCALE 0x80B4 +#define GL_POST_COLOR_MATRIX_GREEN_SCALE 0x80B5 +#define GL_POST_COLOR_MATRIX_BLUE_SCALE 0x80B6 +#define GL_POST_COLOR_MATRIX_ALPHA_SCALE 0x80B7 +#define GL_POST_COLOR_MATRIX_RED_BIAS 0x80B8 +#define GL_POST_COLOR_MATRIX_GREEN_BIAS 0x80B9 +#define GL_POST_COLOR_MATRIX_BLUE_BIAS 0x80BA +#define GL_POST_COLOR_MATRIX_ALPHA_BIAS 0x80BB +#define GL_COLOR_TABLE_SCALE 0x80D6 +#define GL_COLOR_TABLE_BIAS 0x80D7 +#define GL_COLOR_TABLE_FORMAT 0x80D8 +#define GL_COLOR_TABLE_WIDTH 0x80D9 +#define GL_COLOR_TABLE_RED_SIZE 0x80DA +#define GL_COLOR_TABLE_GREEN_SIZE 0x80DB +#define GL_COLOR_TABLE_BLUE_SIZE 0x80DC +#define GL_COLOR_TABLE_ALPHA_SIZE 0x80DD +#define GL_COLOR_TABLE_LUMINANCE_SIZE 0x80DE +#define GL_COLOR_TABLE_INTENSITY_SIZE 0x80DF +#define GL_CONSTANT_BORDER 0x8151 +#define GL_REPLICATE_BORDER 0x8153 +#define GL_CONVOLUTION_BORDER_COLOR 0x8154 +typedef void (APIENTRYP PFNGLCOLORTABLEPROC) (GLenum target, GLenum internalformat, GLsizei width, GLenum format, GLenum type, const void *table); +typedef void (APIENTRYP PFNGLCOLORTABLEPARAMETERFVPROC) (GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLCOLORTABLEPARAMETERIVPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLCOPYCOLORTABLEPROC) (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width); +typedef void (APIENTRYP PFNGLGETCOLORTABLEPROC) (GLenum target, GLenum format, GLenum type, void *table); +typedef void (APIENTRYP PFNGLGETCOLORTABLEPARAMETERFVPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETCOLORTABLEPARAMETERIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLCOLORSUBTABLEPROC) (GLenum target, GLsizei start, GLsizei count, GLenum format, GLenum type, const void *data); +typedef void (APIENTRYP PFNGLCOPYCOLORSUBTABLEPROC) (GLenum target, GLsizei start, GLint x, GLint y, GLsizei width); +typedef void (APIENTRYP PFNGLCONVOLUTIONFILTER1DPROC) (GLenum target, GLenum internalformat, GLsizei width, GLenum format, GLenum type, const void *image); +typedef void (APIENTRYP PFNGLCONVOLUTIONFILTER2DPROC) (GLenum target, GLenum internalformat, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *image); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERFPROC) (GLenum target, GLenum pname, GLfloat params); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERFVPROC) (GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERIPROC) (GLenum target, GLenum pname, GLint params); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERIVPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLCOPYCONVOLUTIONFILTER1DPROC) (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width); +typedef void (APIENTRYP PFNGLCOPYCONVOLUTIONFILTER2DPROC) (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLGETCONVOLUTIONFILTERPROC) (GLenum target, GLenum format, GLenum type, void *image); +typedef void (APIENTRYP PFNGLGETCONVOLUTIONPARAMETERFVPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETCONVOLUTIONPARAMETERIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETSEPARABLEFILTERPROC) (GLenum target, GLenum format, GLenum type, void *row, void *column, void *span); +typedef void (APIENTRYP PFNGLSEPARABLEFILTER2DPROC) (GLenum target, GLenum internalformat, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *row, const void *column); +typedef void (APIENTRYP PFNGLGETHISTOGRAMPROC) (GLenum target, GLboolean reset, GLenum format, GLenum type, void *values); +typedef void (APIENTRYP PFNGLGETHISTOGRAMPARAMETERFVPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETHISTOGRAMPARAMETERIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETMINMAXPROC) (GLenum target, GLboolean reset, GLenum format, GLenum type, void *values); +typedef void (APIENTRYP PFNGLGETMINMAXPARAMETERFVPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETMINMAXPARAMETERIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLHISTOGRAMPROC) (GLenum target, GLsizei width, GLenum internalformat, GLboolean sink); +typedef void (APIENTRYP PFNGLMINMAXPROC) (GLenum target, GLenum internalformat, GLboolean sink); +typedef void (APIENTRYP PFNGLRESETHISTOGRAMPROC) (GLenum target); +typedef void (APIENTRYP PFNGLRESETMINMAXPROC) (GLenum target); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glColorTable (GLenum target, GLenum internalformat, GLsizei width, GLenum format, GLenum type, const void *table); +GLAPI void APIENTRY glColorTableParameterfv (GLenum target, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glColorTableParameteriv (GLenum target, GLenum pname, const GLint *params); +GLAPI void APIENTRY glCopyColorTable (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width); +GLAPI void APIENTRY glGetColorTable (GLenum target, GLenum format, GLenum type, void *table); +GLAPI void APIENTRY glGetColorTableParameterfv (GLenum target, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetColorTableParameteriv (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glColorSubTable (GLenum target, GLsizei start, GLsizei count, GLenum format, GLenum type, const void *data); +GLAPI void APIENTRY glCopyColorSubTable (GLenum target, GLsizei start, GLint x, GLint y, GLsizei width); +GLAPI void APIENTRY glConvolutionFilter1D (GLenum target, GLenum internalformat, GLsizei width, GLenum format, GLenum type, const void *image); +GLAPI void APIENTRY glConvolutionFilter2D (GLenum target, GLenum internalformat, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *image); +GLAPI void APIENTRY glConvolutionParameterf (GLenum target, GLenum pname, GLfloat params); +GLAPI void APIENTRY glConvolutionParameterfv (GLenum target, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glConvolutionParameteri (GLenum target, GLenum pname, GLint params); +GLAPI void APIENTRY glConvolutionParameteriv (GLenum target, GLenum pname, const GLint *params); +GLAPI void APIENTRY glCopyConvolutionFilter1D (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width); +GLAPI void APIENTRY glCopyConvolutionFilter2D (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height); +GLAPI void APIENTRY glGetConvolutionFilter (GLenum target, GLenum format, GLenum type, void *image); +GLAPI void APIENTRY glGetConvolutionParameterfv (GLenum target, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetConvolutionParameteriv (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetSeparableFilter (GLenum target, GLenum format, GLenum type, void *row, void *column, void *span); +GLAPI void APIENTRY glSeparableFilter2D (GLenum target, GLenum internalformat, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *row, const void *column); +GLAPI void APIENTRY glGetHistogram (GLenum target, GLboolean reset, GLenum format, GLenum type, void *values); +GLAPI void APIENTRY glGetHistogramParameterfv (GLenum target, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetHistogramParameteriv (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetMinmax (GLenum target, GLboolean reset, GLenum format, GLenum type, void *values); +GLAPI void APIENTRY glGetMinmaxParameterfv (GLenum target, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetMinmaxParameteriv (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glHistogram (GLenum target, GLsizei width, GLenum internalformat, GLboolean sink); +GLAPI void APIENTRY glMinmax (GLenum target, GLenum internalformat, GLboolean sink); +GLAPI void APIENTRY glResetHistogram (GLenum target); +GLAPI void APIENTRY glResetMinmax (GLenum target); +#endif +#endif /* GL_ARB_imaging */ + +#ifndef GL_ARB_indirect_parameters +#define GL_ARB_indirect_parameters 1 +#define GL_PARAMETER_BUFFER_ARB 0x80EE +#define GL_PARAMETER_BUFFER_BINDING_ARB 0x80EF +typedef void (APIENTRYP PFNGLMULTIDRAWARRAYSINDIRECTCOUNTARBPROC) (GLenum mode, const void *indirect, GLintptr drawcount, GLsizei maxdrawcount, GLsizei stride); +typedef void (APIENTRYP PFNGLMULTIDRAWELEMENTSINDIRECTCOUNTARBPROC) (GLenum mode, GLenum type, const void *indirect, GLintptr drawcount, GLsizei maxdrawcount, GLsizei stride); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMultiDrawArraysIndirectCountARB (GLenum mode, const void *indirect, GLintptr drawcount, GLsizei maxdrawcount, GLsizei stride); +GLAPI void APIENTRY glMultiDrawElementsIndirectCountARB (GLenum mode, GLenum type, const void *indirect, GLintptr drawcount, GLsizei maxdrawcount, GLsizei stride); +#endif +#endif /* GL_ARB_indirect_parameters */ + +#ifndef GL_ARB_instanced_arrays +#define GL_ARB_instanced_arrays 1 +#define GL_VERTEX_ATTRIB_ARRAY_DIVISOR_ARB 0x88FE +typedef void (APIENTRYP PFNGLVERTEXATTRIBDIVISORARBPROC) (GLuint index, GLuint divisor); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexAttribDivisorARB (GLuint index, GLuint divisor); +#endif +#endif /* GL_ARB_instanced_arrays */ + +#ifndef GL_ARB_internalformat_query +#define GL_ARB_internalformat_query 1 +#endif /* GL_ARB_internalformat_query */ + +#ifndef GL_ARB_internalformat_query2 +#define GL_ARB_internalformat_query2 1 +#define GL_SRGB_DECODE_ARB 0x8299 +#define GL_VIEW_CLASS_EAC_R11 0x9383 +#define GL_VIEW_CLASS_EAC_RG11 0x9384 +#define GL_VIEW_CLASS_ETC2_RGB 0x9385 +#define GL_VIEW_CLASS_ETC2_RGBA 0x9386 +#define GL_VIEW_CLASS_ETC2_EAC_RGBA 0x9387 +#define GL_VIEW_CLASS_ASTC_4x4_RGBA 0x9388 +#define GL_VIEW_CLASS_ASTC_5x4_RGBA 0x9389 +#define GL_VIEW_CLASS_ASTC_5x5_RGBA 0x938A +#define GL_VIEW_CLASS_ASTC_6x5_RGBA 0x938B +#define GL_VIEW_CLASS_ASTC_6x6_RGBA 0x938C +#define GL_VIEW_CLASS_ASTC_8x5_RGBA 0x938D +#define GL_VIEW_CLASS_ASTC_8x6_RGBA 0x938E +#define GL_VIEW_CLASS_ASTC_8x8_RGBA 0x938F +#define GL_VIEW_CLASS_ASTC_10x5_RGBA 0x9390 +#define GL_VIEW_CLASS_ASTC_10x6_RGBA 0x9391 +#define GL_VIEW_CLASS_ASTC_10x8_RGBA 0x9392 +#define GL_VIEW_CLASS_ASTC_10x10_RGBA 0x9393 +#define GL_VIEW_CLASS_ASTC_12x10_RGBA 0x9394 +#define GL_VIEW_CLASS_ASTC_12x12_RGBA 0x9395 +#endif /* GL_ARB_internalformat_query2 */ + +#ifndef GL_ARB_invalidate_subdata +#define GL_ARB_invalidate_subdata 1 +#endif /* GL_ARB_invalidate_subdata */ + +#ifndef GL_ARB_map_buffer_alignment +#define GL_ARB_map_buffer_alignment 1 +#endif /* GL_ARB_map_buffer_alignment */ + +#ifndef GL_ARB_map_buffer_range +#define GL_ARB_map_buffer_range 1 +#endif /* GL_ARB_map_buffer_range */ + +#ifndef GL_ARB_matrix_palette +#define GL_ARB_matrix_palette 1 +#define GL_MATRIX_PALETTE_ARB 0x8840 +#define GL_MAX_MATRIX_PALETTE_STACK_DEPTH_ARB 0x8841 +#define GL_MAX_PALETTE_MATRICES_ARB 0x8842 +#define GL_CURRENT_PALETTE_MATRIX_ARB 0x8843 +#define GL_MATRIX_INDEX_ARRAY_ARB 0x8844 +#define GL_CURRENT_MATRIX_INDEX_ARB 0x8845 +#define GL_MATRIX_INDEX_ARRAY_SIZE_ARB 0x8846 +#define GL_MATRIX_INDEX_ARRAY_TYPE_ARB 0x8847 +#define GL_MATRIX_INDEX_ARRAY_STRIDE_ARB 0x8848 +#define GL_MATRIX_INDEX_ARRAY_POINTER_ARB 0x8849 +typedef void (APIENTRYP PFNGLCURRENTPALETTEMATRIXARBPROC) (GLint index); +typedef void (APIENTRYP PFNGLMATRIXINDEXUBVARBPROC) (GLint size, const GLubyte *indices); +typedef void (APIENTRYP PFNGLMATRIXINDEXUSVARBPROC) (GLint size, const GLushort *indices); +typedef void (APIENTRYP PFNGLMATRIXINDEXUIVARBPROC) (GLint size, const GLuint *indices); +typedef void (APIENTRYP PFNGLMATRIXINDEXPOINTERARBPROC) (GLint size, GLenum type, GLsizei stride, const void *pointer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCurrentPaletteMatrixARB (GLint index); +GLAPI void APIENTRY glMatrixIndexubvARB (GLint size, const GLubyte *indices); +GLAPI void APIENTRY glMatrixIndexusvARB (GLint size, const GLushort *indices); +GLAPI void APIENTRY glMatrixIndexuivARB (GLint size, const GLuint *indices); +GLAPI void APIENTRY glMatrixIndexPointerARB (GLint size, GLenum type, GLsizei stride, const void *pointer); +#endif +#endif /* GL_ARB_matrix_palette */ + +#ifndef GL_ARB_multi_bind +#define GL_ARB_multi_bind 1 +#endif /* GL_ARB_multi_bind */ + +#ifndef GL_ARB_multi_draw_indirect +#define GL_ARB_multi_draw_indirect 1 +#endif /* GL_ARB_multi_draw_indirect */ + +#ifndef GL_ARB_multisample +#define GL_ARB_multisample 1 +#define GL_MULTISAMPLE_ARB 0x809D +#define GL_SAMPLE_ALPHA_TO_COVERAGE_ARB 0x809E +#define GL_SAMPLE_ALPHA_TO_ONE_ARB 0x809F +#define GL_SAMPLE_COVERAGE_ARB 0x80A0 +#define GL_SAMPLE_BUFFERS_ARB 0x80A8 +#define GL_SAMPLES_ARB 0x80A9 +#define GL_SAMPLE_COVERAGE_VALUE_ARB 0x80AA +#define GL_SAMPLE_COVERAGE_INVERT_ARB 0x80AB +#define GL_MULTISAMPLE_BIT_ARB 0x20000000 +typedef void (APIENTRYP PFNGLSAMPLECOVERAGEARBPROC) (GLfloat value, GLboolean invert); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSampleCoverageARB (GLfloat value, GLboolean invert); +#endif +#endif /* GL_ARB_multisample */ + +#ifndef GL_ARB_multitexture +#define GL_ARB_multitexture 1 +#define GL_TEXTURE0_ARB 0x84C0 +#define GL_TEXTURE1_ARB 0x84C1 +#define GL_TEXTURE2_ARB 0x84C2 +#define GL_TEXTURE3_ARB 0x84C3 +#define GL_TEXTURE4_ARB 0x84C4 +#define GL_TEXTURE5_ARB 0x84C5 +#define GL_TEXTURE6_ARB 0x84C6 +#define GL_TEXTURE7_ARB 0x84C7 +#define GL_TEXTURE8_ARB 0x84C8 +#define GL_TEXTURE9_ARB 0x84C9 +#define GL_TEXTURE10_ARB 0x84CA +#define GL_TEXTURE11_ARB 0x84CB +#define GL_TEXTURE12_ARB 0x84CC +#define GL_TEXTURE13_ARB 0x84CD +#define GL_TEXTURE14_ARB 0x84CE +#define GL_TEXTURE15_ARB 0x84CF +#define GL_TEXTURE16_ARB 0x84D0 +#define GL_TEXTURE17_ARB 0x84D1 +#define GL_TEXTURE18_ARB 0x84D2 +#define GL_TEXTURE19_ARB 0x84D3 +#define GL_TEXTURE20_ARB 0x84D4 +#define GL_TEXTURE21_ARB 0x84D5 +#define GL_TEXTURE22_ARB 0x84D6 +#define GL_TEXTURE23_ARB 0x84D7 +#define GL_TEXTURE24_ARB 0x84D8 +#define GL_TEXTURE25_ARB 0x84D9 +#define GL_TEXTURE26_ARB 0x84DA +#define GL_TEXTURE27_ARB 0x84DB +#define GL_TEXTURE28_ARB 0x84DC +#define GL_TEXTURE29_ARB 0x84DD +#define GL_TEXTURE30_ARB 0x84DE +#define GL_TEXTURE31_ARB 0x84DF +#define GL_ACTIVE_TEXTURE_ARB 0x84E0 +#define GL_CLIENT_ACTIVE_TEXTURE_ARB 0x84E1 +#define GL_MAX_TEXTURE_UNITS_ARB 0x84E2 +typedef void (APIENTRYP PFNGLACTIVETEXTUREARBPROC) (GLenum texture); +typedef void (APIENTRYP PFNGLCLIENTACTIVETEXTUREARBPROC) (GLenum texture); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1DARBPROC) (GLenum target, GLdouble s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1DVARBPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1FARBPROC) (GLenum target, GLfloat s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1FVARBPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1IARBPROC) (GLenum target, GLint s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1IVARBPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1SARBPROC) (GLenum target, GLshort s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1SVARBPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2DARBPROC) (GLenum target, GLdouble s, GLdouble t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2DVARBPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2FARBPROC) (GLenum target, GLfloat s, GLfloat t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2FVARBPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2IARBPROC) (GLenum target, GLint s, GLint t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2IVARBPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2SARBPROC) (GLenum target, GLshort s, GLshort t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2SVARBPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3DARBPROC) (GLenum target, GLdouble s, GLdouble t, GLdouble r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3DVARBPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3FARBPROC) (GLenum target, GLfloat s, GLfloat t, GLfloat r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3FVARBPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3IARBPROC) (GLenum target, GLint s, GLint t, GLint r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3IVARBPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3SARBPROC) (GLenum target, GLshort s, GLshort t, GLshort r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3SVARBPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4DARBPROC) (GLenum target, GLdouble s, GLdouble t, GLdouble r, GLdouble q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4DVARBPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4FARBPROC) (GLenum target, GLfloat s, GLfloat t, GLfloat r, GLfloat q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4FVARBPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4IARBPROC) (GLenum target, GLint s, GLint t, GLint r, GLint q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4IVARBPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4SARBPROC) (GLenum target, GLshort s, GLshort t, GLshort r, GLshort q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4SVARBPROC) (GLenum target, const GLshort *v); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glActiveTextureARB (GLenum texture); +GLAPI void APIENTRY glClientActiveTextureARB (GLenum texture); +GLAPI void APIENTRY glMultiTexCoord1dARB (GLenum target, GLdouble s); +GLAPI void APIENTRY glMultiTexCoord1dvARB (GLenum target, const GLdouble *v); +GLAPI void APIENTRY glMultiTexCoord1fARB (GLenum target, GLfloat s); +GLAPI void APIENTRY glMultiTexCoord1fvARB (GLenum target, const GLfloat *v); +GLAPI void APIENTRY glMultiTexCoord1iARB (GLenum target, GLint s); +GLAPI void APIENTRY glMultiTexCoord1ivARB (GLenum target, const GLint *v); +GLAPI void APIENTRY glMultiTexCoord1sARB (GLenum target, GLshort s); +GLAPI void APIENTRY glMultiTexCoord1svARB (GLenum target, const GLshort *v); +GLAPI void APIENTRY glMultiTexCoord2dARB (GLenum target, GLdouble s, GLdouble t); +GLAPI void APIENTRY glMultiTexCoord2dvARB (GLenum target, const GLdouble *v); +GLAPI void APIENTRY glMultiTexCoord2fARB (GLenum target, GLfloat s, GLfloat t); +GLAPI void APIENTRY glMultiTexCoord2fvARB (GLenum target, const GLfloat *v); +GLAPI void APIENTRY glMultiTexCoord2iARB (GLenum target, GLint s, GLint t); +GLAPI void APIENTRY glMultiTexCoord2ivARB (GLenum target, const GLint *v); +GLAPI void APIENTRY glMultiTexCoord2sARB (GLenum target, GLshort s, GLshort t); +GLAPI void APIENTRY glMultiTexCoord2svARB (GLenum target, const GLshort *v); +GLAPI void APIENTRY glMultiTexCoord3dARB (GLenum target, GLdouble s, GLdouble t, GLdouble r); +GLAPI void APIENTRY glMultiTexCoord3dvARB (GLenum target, const GLdouble *v); +GLAPI void APIENTRY glMultiTexCoord3fARB (GLenum target, GLfloat s, GLfloat t, GLfloat r); +GLAPI void APIENTRY glMultiTexCoord3fvARB (GLenum target, const GLfloat *v); +GLAPI void APIENTRY glMultiTexCoord3iARB (GLenum target, GLint s, GLint t, GLint r); +GLAPI void APIENTRY glMultiTexCoord3ivARB (GLenum target, const GLint *v); +GLAPI void APIENTRY glMultiTexCoord3sARB (GLenum target, GLshort s, GLshort t, GLshort r); +GLAPI void APIENTRY glMultiTexCoord3svARB (GLenum target, const GLshort *v); +GLAPI void APIENTRY glMultiTexCoord4dARB (GLenum target, GLdouble s, GLdouble t, GLdouble r, GLdouble q); +GLAPI void APIENTRY glMultiTexCoord4dvARB (GLenum target, const GLdouble *v); +GLAPI void APIENTRY glMultiTexCoord4fARB (GLenum target, GLfloat s, GLfloat t, GLfloat r, GLfloat q); +GLAPI void APIENTRY glMultiTexCoord4fvARB (GLenum target, const GLfloat *v); +GLAPI void APIENTRY glMultiTexCoord4iARB (GLenum target, GLint s, GLint t, GLint r, GLint q); +GLAPI void APIENTRY glMultiTexCoord4ivARB (GLenum target, const GLint *v); +GLAPI void APIENTRY glMultiTexCoord4sARB (GLenum target, GLshort s, GLshort t, GLshort r, GLshort q); +GLAPI void APIENTRY glMultiTexCoord4svARB (GLenum target, const GLshort *v); +#endif +#endif /* GL_ARB_multitexture */ + +#ifndef GL_ARB_occlusion_query +#define GL_ARB_occlusion_query 1 +#define GL_QUERY_COUNTER_BITS_ARB 0x8864 +#define GL_CURRENT_QUERY_ARB 0x8865 +#define GL_QUERY_RESULT_ARB 0x8866 +#define GL_QUERY_RESULT_AVAILABLE_ARB 0x8867 +#define GL_SAMPLES_PASSED_ARB 0x8914 +typedef void (APIENTRYP PFNGLGENQUERIESARBPROC) (GLsizei n, GLuint *ids); +typedef void (APIENTRYP PFNGLDELETEQUERIESARBPROC) (GLsizei n, const GLuint *ids); +typedef GLboolean (APIENTRYP PFNGLISQUERYARBPROC) (GLuint id); +typedef void (APIENTRYP PFNGLBEGINQUERYARBPROC) (GLenum target, GLuint id); +typedef void (APIENTRYP PFNGLENDQUERYARBPROC) (GLenum target); +typedef void (APIENTRYP PFNGLGETQUERYIVARBPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETQUERYOBJECTIVARBPROC) (GLuint id, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETQUERYOBJECTUIVARBPROC) (GLuint id, GLenum pname, GLuint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGenQueriesARB (GLsizei n, GLuint *ids); +GLAPI void APIENTRY glDeleteQueriesARB (GLsizei n, const GLuint *ids); +GLAPI GLboolean APIENTRY glIsQueryARB (GLuint id); +GLAPI void APIENTRY glBeginQueryARB (GLenum target, GLuint id); +GLAPI void APIENTRY glEndQueryARB (GLenum target); +GLAPI void APIENTRY glGetQueryivARB (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetQueryObjectivARB (GLuint id, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetQueryObjectuivARB (GLuint id, GLenum pname, GLuint *params); +#endif +#endif /* GL_ARB_occlusion_query */ + +#ifndef GL_ARB_occlusion_query2 +#define GL_ARB_occlusion_query2 1 +#endif /* GL_ARB_occlusion_query2 */ + +#ifndef GL_ARB_parallel_shader_compile +#define GL_ARB_parallel_shader_compile 1 +#define GL_MAX_SHADER_COMPILER_THREADS_ARB 0x91B0 +#define GL_COMPLETION_STATUS_ARB 0x91B1 +typedef void (APIENTRYP PFNGLMAXSHADERCOMPILERTHREADSARBPROC) (GLuint count); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMaxShaderCompilerThreadsARB (GLuint count); +#endif +#endif /* GL_ARB_parallel_shader_compile */ + +#ifndef GL_ARB_pipeline_statistics_query +#define GL_ARB_pipeline_statistics_query 1 +#define GL_VERTICES_SUBMITTED_ARB 0x82EE +#define GL_PRIMITIVES_SUBMITTED_ARB 0x82EF +#define GL_VERTEX_SHADER_INVOCATIONS_ARB 0x82F0 +#define GL_TESS_CONTROL_SHADER_PATCHES_ARB 0x82F1 +#define GL_TESS_EVALUATION_SHADER_INVOCATIONS_ARB 0x82F2 +#define GL_GEOMETRY_SHADER_PRIMITIVES_EMITTED_ARB 0x82F3 +#define GL_FRAGMENT_SHADER_INVOCATIONS_ARB 0x82F4 +#define GL_COMPUTE_SHADER_INVOCATIONS_ARB 0x82F5 +#define GL_CLIPPING_INPUT_PRIMITIVES_ARB 0x82F6 +#define GL_CLIPPING_OUTPUT_PRIMITIVES_ARB 0x82F7 +#endif /* GL_ARB_pipeline_statistics_query */ + +#ifndef GL_ARB_pixel_buffer_object +#define GL_ARB_pixel_buffer_object 1 +#define GL_PIXEL_PACK_BUFFER_ARB 0x88EB +#define GL_PIXEL_UNPACK_BUFFER_ARB 0x88EC +#define GL_PIXEL_PACK_BUFFER_BINDING_ARB 0x88ED +#define GL_PIXEL_UNPACK_BUFFER_BINDING_ARB 0x88EF +#endif /* GL_ARB_pixel_buffer_object */ + +#ifndef GL_ARB_point_parameters +#define GL_ARB_point_parameters 1 +#define GL_POINT_SIZE_MIN_ARB 0x8126 +#define GL_POINT_SIZE_MAX_ARB 0x8127 +#define GL_POINT_FADE_THRESHOLD_SIZE_ARB 0x8128 +#define GL_POINT_DISTANCE_ATTENUATION_ARB 0x8129 +typedef void (APIENTRYP PFNGLPOINTPARAMETERFARBPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLPOINTPARAMETERFVARBPROC) (GLenum pname, const GLfloat *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPointParameterfARB (GLenum pname, GLfloat param); +GLAPI void APIENTRY glPointParameterfvARB (GLenum pname, const GLfloat *params); +#endif +#endif /* GL_ARB_point_parameters */ + +#ifndef GL_ARB_point_sprite +#define GL_ARB_point_sprite 1 +#define GL_POINT_SPRITE_ARB 0x8861 +#define GL_COORD_REPLACE_ARB 0x8862 +#endif /* GL_ARB_point_sprite */ + +#ifndef GL_ARB_polygon_offset_clamp +#define GL_ARB_polygon_offset_clamp 1 +#endif /* GL_ARB_polygon_offset_clamp */ + +#ifndef GL_ARB_post_depth_coverage +#define GL_ARB_post_depth_coverage 1 +#endif /* GL_ARB_post_depth_coverage */ + +#ifndef GL_ARB_program_interface_query +#define GL_ARB_program_interface_query 1 +#endif /* GL_ARB_program_interface_query */ + +#ifndef GL_ARB_provoking_vertex +#define GL_ARB_provoking_vertex 1 +#endif /* GL_ARB_provoking_vertex */ + +#ifndef GL_ARB_query_buffer_object +#define GL_ARB_query_buffer_object 1 +#endif /* GL_ARB_query_buffer_object */ + +#ifndef GL_ARB_robust_buffer_access_behavior +#define GL_ARB_robust_buffer_access_behavior 1 +#endif /* GL_ARB_robust_buffer_access_behavior */ + +#ifndef GL_ARB_robustness +#define GL_ARB_robustness 1 +#define GL_CONTEXT_FLAG_ROBUST_ACCESS_BIT_ARB 0x00000004 +#define GL_LOSE_CONTEXT_ON_RESET_ARB 0x8252 +#define GL_GUILTY_CONTEXT_RESET_ARB 0x8253 +#define GL_INNOCENT_CONTEXT_RESET_ARB 0x8254 +#define GL_UNKNOWN_CONTEXT_RESET_ARB 0x8255 +#define GL_RESET_NOTIFICATION_STRATEGY_ARB 0x8256 +#define GL_NO_RESET_NOTIFICATION_ARB 0x8261 +typedef GLenum (APIENTRYP PFNGLGETGRAPHICSRESETSTATUSARBPROC) (void); +typedef void (APIENTRYP PFNGLGETNTEXIMAGEARBPROC) (GLenum target, GLint level, GLenum format, GLenum type, GLsizei bufSize, void *img); +typedef void (APIENTRYP PFNGLREADNPIXELSARBPROC) (GLint x, GLint y, GLsizei width, GLsizei height, GLenum format, GLenum type, GLsizei bufSize, void *data); +typedef void (APIENTRYP PFNGLGETNCOMPRESSEDTEXIMAGEARBPROC) (GLenum target, GLint lod, GLsizei bufSize, void *img); +typedef void (APIENTRYP PFNGLGETNUNIFORMFVARBPROC) (GLuint program, GLint location, GLsizei bufSize, GLfloat *params); +typedef void (APIENTRYP PFNGLGETNUNIFORMIVARBPROC) (GLuint program, GLint location, GLsizei bufSize, GLint *params); +typedef void (APIENTRYP PFNGLGETNUNIFORMUIVARBPROC) (GLuint program, GLint location, GLsizei bufSize, GLuint *params); +typedef void (APIENTRYP PFNGLGETNUNIFORMDVARBPROC) (GLuint program, GLint location, GLsizei bufSize, GLdouble *params); +typedef void (APIENTRYP PFNGLGETNMAPDVARBPROC) (GLenum target, GLenum query, GLsizei bufSize, GLdouble *v); +typedef void (APIENTRYP PFNGLGETNMAPFVARBPROC) (GLenum target, GLenum query, GLsizei bufSize, GLfloat *v); +typedef void (APIENTRYP PFNGLGETNMAPIVARBPROC) (GLenum target, GLenum query, GLsizei bufSize, GLint *v); +typedef void (APIENTRYP PFNGLGETNPIXELMAPFVARBPROC) (GLenum map, GLsizei bufSize, GLfloat *values); +typedef void (APIENTRYP PFNGLGETNPIXELMAPUIVARBPROC) (GLenum map, GLsizei bufSize, GLuint *values); +typedef void (APIENTRYP PFNGLGETNPIXELMAPUSVARBPROC) (GLenum map, GLsizei bufSize, GLushort *values); +typedef void (APIENTRYP PFNGLGETNPOLYGONSTIPPLEARBPROC) (GLsizei bufSize, GLubyte *pattern); +typedef void (APIENTRYP PFNGLGETNCOLORTABLEARBPROC) (GLenum target, GLenum format, GLenum type, GLsizei bufSize, void *table); +typedef void (APIENTRYP PFNGLGETNCONVOLUTIONFILTERARBPROC) (GLenum target, GLenum format, GLenum type, GLsizei bufSize, void *image); +typedef void (APIENTRYP PFNGLGETNSEPARABLEFILTERARBPROC) (GLenum target, GLenum format, GLenum type, GLsizei rowBufSize, void *row, GLsizei columnBufSize, void *column, void *span); +typedef void (APIENTRYP PFNGLGETNHISTOGRAMARBPROC) (GLenum target, GLboolean reset, GLenum format, GLenum type, GLsizei bufSize, void *values); +typedef void (APIENTRYP PFNGLGETNMINMAXARBPROC) (GLenum target, GLboolean reset, GLenum format, GLenum type, GLsizei bufSize, void *values); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLenum APIENTRY glGetGraphicsResetStatusARB (void); +GLAPI void APIENTRY glGetnTexImageARB (GLenum target, GLint level, GLenum format, GLenum type, GLsizei bufSize, void *img); +GLAPI void APIENTRY glReadnPixelsARB (GLint x, GLint y, GLsizei width, GLsizei height, GLenum format, GLenum type, GLsizei bufSize, void *data); +GLAPI void APIENTRY glGetnCompressedTexImageARB (GLenum target, GLint lod, GLsizei bufSize, void *img); +GLAPI void APIENTRY glGetnUniformfvARB (GLuint program, GLint location, GLsizei bufSize, GLfloat *params); +GLAPI void APIENTRY glGetnUniformivARB (GLuint program, GLint location, GLsizei bufSize, GLint *params); +GLAPI void APIENTRY glGetnUniformuivARB (GLuint program, GLint location, GLsizei bufSize, GLuint *params); +GLAPI void APIENTRY glGetnUniformdvARB (GLuint program, GLint location, GLsizei bufSize, GLdouble *params); +GLAPI void APIENTRY glGetnMapdvARB (GLenum target, GLenum query, GLsizei bufSize, GLdouble *v); +GLAPI void APIENTRY glGetnMapfvARB (GLenum target, GLenum query, GLsizei bufSize, GLfloat *v); +GLAPI void APIENTRY glGetnMapivARB (GLenum target, GLenum query, GLsizei bufSize, GLint *v); +GLAPI void APIENTRY glGetnPixelMapfvARB (GLenum map, GLsizei bufSize, GLfloat *values); +GLAPI void APIENTRY glGetnPixelMapuivARB (GLenum map, GLsizei bufSize, GLuint *values); +GLAPI void APIENTRY glGetnPixelMapusvARB (GLenum map, GLsizei bufSize, GLushort *values); +GLAPI void APIENTRY glGetnPolygonStippleARB (GLsizei bufSize, GLubyte *pattern); +GLAPI void APIENTRY glGetnColorTableARB (GLenum target, GLenum format, GLenum type, GLsizei bufSize, void *table); +GLAPI void APIENTRY glGetnConvolutionFilterARB (GLenum target, GLenum format, GLenum type, GLsizei bufSize, void *image); +GLAPI void APIENTRY glGetnSeparableFilterARB (GLenum target, GLenum format, GLenum type, GLsizei rowBufSize, void *row, GLsizei columnBufSize, void *column, void *span); +GLAPI void APIENTRY glGetnHistogramARB (GLenum target, GLboolean reset, GLenum format, GLenum type, GLsizei bufSize, void *values); +GLAPI void APIENTRY glGetnMinmaxARB (GLenum target, GLboolean reset, GLenum format, GLenum type, GLsizei bufSize, void *values); +#endif +#endif /* GL_ARB_robustness */ + +#ifndef GL_ARB_robustness_isolation +#define GL_ARB_robustness_isolation 1 +#endif /* GL_ARB_robustness_isolation */ + +#ifndef GL_ARB_sample_locations +#define GL_ARB_sample_locations 1 +#define GL_SAMPLE_LOCATION_SUBPIXEL_BITS_ARB 0x933D +#define GL_SAMPLE_LOCATION_PIXEL_GRID_WIDTH_ARB 0x933E +#define GL_SAMPLE_LOCATION_PIXEL_GRID_HEIGHT_ARB 0x933F +#define GL_PROGRAMMABLE_SAMPLE_LOCATION_TABLE_SIZE_ARB 0x9340 +#define GL_SAMPLE_LOCATION_ARB 0x8E50 +#define GL_PROGRAMMABLE_SAMPLE_LOCATION_ARB 0x9341 +#define GL_FRAMEBUFFER_PROGRAMMABLE_SAMPLE_LOCATIONS_ARB 0x9342 +#define GL_FRAMEBUFFER_SAMPLE_LOCATION_PIXEL_GRID_ARB 0x9343 +typedef void (APIENTRYP PFNGLFRAMEBUFFERSAMPLELOCATIONSFVARBPROC) (GLenum target, GLuint start, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERSAMPLELOCATIONSFVARBPROC) (GLuint framebuffer, GLuint start, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLEVALUATEDEPTHVALUESARBPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFramebufferSampleLocationsfvARB (GLenum target, GLuint start, GLsizei count, const GLfloat *v); +GLAPI void APIENTRY glNamedFramebufferSampleLocationsfvARB (GLuint framebuffer, GLuint start, GLsizei count, const GLfloat *v); +GLAPI void APIENTRY glEvaluateDepthValuesARB (void); +#endif +#endif /* GL_ARB_sample_locations */ + +#ifndef GL_ARB_sample_shading +#define GL_ARB_sample_shading 1 +#define GL_SAMPLE_SHADING_ARB 0x8C36 +#define GL_MIN_SAMPLE_SHADING_VALUE_ARB 0x8C37 +typedef void (APIENTRYP PFNGLMINSAMPLESHADINGARBPROC) (GLfloat value); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMinSampleShadingARB (GLfloat value); +#endif +#endif /* GL_ARB_sample_shading */ + +#ifndef GL_ARB_sampler_objects +#define GL_ARB_sampler_objects 1 +#endif /* GL_ARB_sampler_objects */ + +#ifndef GL_ARB_seamless_cube_map +#define GL_ARB_seamless_cube_map 1 +#endif /* GL_ARB_seamless_cube_map */ + +#ifndef GL_ARB_seamless_cubemap_per_texture +#define GL_ARB_seamless_cubemap_per_texture 1 +#endif /* GL_ARB_seamless_cubemap_per_texture */ + +#ifndef GL_ARB_separate_shader_objects +#define GL_ARB_separate_shader_objects 1 +#endif /* GL_ARB_separate_shader_objects */ + +#ifndef GL_ARB_shader_atomic_counter_ops +#define GL_ARB_shader_atomic_counter_ops 1 +#endif /* GL_ARB_shader_atomic_counter_ops */ + +#ifndef GL_ARB_shader_atomic_counters +#define GL_ARB_shader_atomic_counters 1 +#endif /* GL_ARB_shader_atomic_counters */ + +#ifndef GL_ARB_shader_ballot +#define GL_ARB_shader_ballot 1 +#endif /* GL_ARB_shader_ballot */ + +#ifndef GL_ARB_shader_bit_encoding +#define GL_ARB_shader_bit_encoding 1 +#endif /* GL_ARB_shader_bit_encoding */ + +#ifndef GL_ARB_shader_clock +#define GL_ARB_shader_clock 1 +#endif /* GL_ARB_shader_clock */ + +#ifndef GL_ARB_shader_draw_parameters +#define GL_ARB_shader_draw_parameters 1 +#endif /* GL_ARB_shader_draw_parameters */ + +#ifndef GL_ARB_shader_group_vote +#define GL_ARB_shader_group_vote 1 +#endif /* GL_ARB_shader_group_vote */ + +#ifndef GL_ARB_shader_image_load_store +#define GL_ARB_shader_image_load_store 1 +#endif /* GL_ARB_shader_image_load_store */ + +#ifndef GL_ARB_shader_image_size +#define GL_ARB_shader_image_size 1 +#endif /* GL_ARB_shader_image_size */ + +#ifndef GL_ARB_shader_objects +#define GL_ARB_shader_objects 1 +#ifdef __APPLE__ +typedef void *GLhandleARB; +#else +typedef unsigned int GLhandleARB; +#endif +typedef char GLcharARB; +#define GL_PROGRAM_OBJECT_ARB 0x8B40 +#define GL_SHADER_OBJECT_ARB 0x8B48 +#define GL_OBJECT_TYPE_ARB 0x8B4E +#define GL_OBJECT_SUBTYPE_ARB 0x8B4F +#define GL_FLOAT_VEC2_ARB 0x8B50 +#define GL_FLOAT_VEC3_ARB 0x8B51 +#define GL_FLOAT_VEC4_ARB 0x8B52 +#define GL_INT_VEC2_ARB 0x8B53 +#define GL_INT_VEC3_ARB 0x8B54 +#define GL_INT_VEC4_ARB 0x8B55 +#define GL_BOOL_ARB 0x8B56 +#define GL_BOOL_VEC2_ARB 0x8B57 +#define GL_BOOL_VEC3_ARB 0x8B58 +#define GL_BOOL_VEC4_ARB 0x8B59 +#define GL_FLOAT_MAT2_ARB 0x8B5A +#define GL_FLOAT_MAT3_ARB 0x8B5B +#define GL_FLOAT_MAT4_ARB 0x8B5C +#define GL_SAMPLER_1D_ARB 0x8B5D +#define GL_SAMPLER_2D_ARB 0x8B5E +#define GL_SAMPLER_3D_ARB 0x8B5F +#define GL_SAMPLER_CUBE_ARB 0x8B60 +#define GL_SAMPLER_1D_SHADOW_ARB 0x8B61 +#define GL_SAMPLER_2D_SHADOW_ARB 0x8B62 +#define GL_SAMPLER_2D_RECT_ARB 0x8B63 +#define GL_SAMPLER_2D_RECT_SHADOW_ARB 0x8B64 +#define GL_OBJECT_DELETE_STATUS_ARB 0x8B80 +#define GL_OBJECT_COMPILE_STATUS_ARB 0x8B81 +#define GL_OBJECT_LINK_STATUS_ARB 0x8B82 +#define GL_OBJECT_VALIDATE_STATUS_ARB 0x8B83 +#define GL_OBJECT_INFO_LOG_LENGTH_ARB 0x8B84 +#define GL_OBJECT_ATTACHED_OBJECTS_ARB 0x8B85 +#define GL_OBJECT_ACTIVE_UNIFORMS_ARB 0x8B86 +#define GL_OBJECT_ACTIVE_UNIFORM_MAX_LENGTH_ARB 0x8B87 +#define GL_OBJECT_SHADER_SOURCE_LENGTH_ARB 0x8B88 +typedef void (APIENTRYP PFNGLDELETEOBJECTARBPROC) (GLhandleARB obj); +typedef GLhandleARB (APIENTRYP PFNGLGETHANDLEARBPROC) (GLenum pname); +typedef void (APIENTRYP PFNGLDETACHOBJECTARBPROC) (GLhandleARB containerObj, GLhandleARB attachedObj); +typedef GLhandleARB (APIENTRYP PFNGLCREATESHADEROBJECTARBPROC) (GLenum shaderType); +typedef void (APIENTRYP PFNGLSHADERSOURCEARBPROC) (GLhandleARB shaderObj, GLsizei count, const GLcharARB **string, const GLint *length); +typedef void (APIENTRYP PFNGLCOMPILESHADERARBPROC) (GLhandleARB shaderObj); +typedef GLhandleARB (APIENTRYP PFNGLCREATEPROGRAMOBJECTARBPROC) (void); +typedef void (APIENTRYP PFNGLATTACHOBJECTARBPROC) (GLhandleARB containerObj, GLhandleARB obj); +typedef void (APIENTRYP PFNGLLINKPROGRAMARBPROC) (GLhandleARB programObj); +typedef void (APIENTRYP PFNGLUSEPROGRAMOBJECTARBPROC) (GLhandleARB programObj); +typedef void (APIENTRYP PFNGLVALIDATEPROGRAMARBPROC) (GLhandleARB programObj); +typedef void (APIENTRYP PFNGLUNIFORM1FARBPROC) (GLint location, GLfloat v0); +typedef void (APIENTRYP PFNGLUNIFORM2FARBPROC) (GLint location, GLfloat v0, GLfloat v1); +typedef void (APIENTRYP PFNGLUNIFORM3FARBPROC) (GLint location, GLfloat v0, GLfloat v1, GLfloat v2); +typedef void (APIENTRYP PFNGLUNIFORM4FARBPROC) (GLint location, GLfloat v0, GLfloat v1, GLfloat v2, GLfloat v3); +typedef void (APIENTRYP PFNGLUNIFORM1IARBPROC) (GLint location, GLint v0); +typedef void (APIENTRYP PFNGLUNIFORM2IARBPROC) (GLint location, GLint v0, GLint v1); +typedef void (APIENTRYP PFNGLUNIFORM3IARBPROC) (GLint location, GLint v0, GLint v1, GLint v2); +typedef void (APIENTRYP PFNGLUNIFORM4IARBPROC) (GLint location, GLint v0, GLint v1, GLint v2, GLint v3); +typedef void (APIENTRYP PFNGLUNIFORM1FVARBPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM2FVARBPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM3FVARBPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM4FVARBPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM1IVARBPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORM2IVARBPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORM3IVARBPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORM4IVARBPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX2FVARBPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX3FVARBPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX4FVARBPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLGETOBJECTPARAMETERFVARBPROC) (GLhandleARB obj, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETOBJECTPARAMETERIVARBPROC) (GLhandleARB obj, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETINFOLOGARBPROC) (GLhandleARB obj, GLsizei maxLength, GLsizei *length, GLcharARB *infoLog); +typedef void (APIENTRYP PFNGLGETATTACHEDOBJECTSARBPROC) (GLhandleARB containerObj, GLsizei maxCount, GLsizei *count, GLhandleARB *obj); +typedef GLint (APIENTRYP PFNGLGETUNIFORMLOCATIONARBPROC) (GLhandleARB programObj, const GLcharARB *name); +typedef void (APIENTRYP PFNGLGETACTIVEUNIFORMARBPROC) (GLhandleARB programObj, GLuint index, GLsizei maxLength, GLsizei *length, GLint *size, GLenum *type, GLcharARB *name); +typedef void (APIENTRYP PFNGLGETUNIFORMFVARBPROC) (GLhandleARB programObj, GLint location, GLfloat *params); +typedef void (APIENTRYP PFNGLGETUNIFORMIVARBPROC) (GLhandleARB programObj, GLint location, GLint *params); +typedef void (APIENTRYP PFNGLGETSHADERSOURCEARBPROC) (GLhandleARB obj, GLsizei maxLength, GLsizei *length, GLcharARB *source); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDeleteObjectARB (GLhandleARB obj); +GLAPI GLhandleARB APIENTRY glGetHandleARB (GLenum pname); +GLAPI void APIENTRY glDetachObjectARB (GLhandleARB containerObj, GLhandleARB attachedObj); +GLAPI GLhandleARB APIENTRY glCreateShaderObjectARB (GLenum shaderType); +GLAPI void APIENTRY glShaderSourceARB (GLhandleARB shaderObj, GLsizei count, const GLcharARB **string, const GLint *length); +GLAPI void APIENTRY glCompileShaderARB (GLhandleARB shaderObj); +GLAPI GLhandleARB APIENTRY glCreateProgramObjectARB (void); +GLAPI void APIENTRY glAttachObjectARB (GLhandleARB containerObj, GLhandleARB obj); +GLAPI void APIENTRY glLinkProgramARB (GLhandleARB programObj); +GLAPI void APIENTRY glUseProgramObjectARB (GLhandleARB programObj); +GLAPI void APIENTRY glValidateProgramARB (GLhandleARB programObj); +GLAPI void APIENTRY glUniform1fARB (GLint location, GLfloat v0); +GLAPI void APIENTRY glUniform2fARB (GLint location, GLfloat v0, GLfloat v1); +GLAPI void APIENTRY glUniform3fARB (GLint location, GLfloat v0, GLfloat v1, GLfloat v2); +GLAPI void APIENTRY glUniform4fARB (GLint location, GLfloat v0, GLfloat v1, GLfloat v2, GLfloat v3); +GLAPI void APIENTRY glUniform1iARB (GLint location, GLint v0); +GLAPI void APIENTRY glUniform2iARB (GLint location, GLint v0, GLint v1); +GLAPI void APIENTRY glUniform3iARB (GLint location, GLint v0, GLint v1, GLint v2); +GLAPI void APIENTRY glUniform4iARB (GLint location, GLint v0, GLint v1, GLint v2, GLint v3); +GLAPI void APIENTRY glUniform1fvARB (GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glUniform2fvARB (GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glUniform3fvARB (GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glUniform4fvARB (GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glUniform1ivARB (GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glUniform2ivARB (GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glUniform3ivARB (GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glUniform4ivARB (GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glUniformMatrix2fvARB (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glUniformMatrix3fvARB (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glUniformMatrix4fvARB (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glGetObjectParameterfvARB (GLhandleARB obj, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetObjectParameterivARB (GLhandleARB obj, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetInfoLogARB (GLhandleARB obj, GLsizei maxLength, GLsizei *length, GLcharARB *infoLog); +GLAPI void APIENTRY glGetAttachedObjectsARB (GLhandleARB containerObj, GLsizei maxCount, GLsizei *count, GLhandleARB *obj); +GLAPI GLint APIENTRY glGetUniformLocationARB (GLhandleARB programObj, const GLcharARB *name); +GLAPI void APIENTRY glGetActiveUniformARB (GLhandleARB programObj, GLuint index, GLsizei maxLength, GLsizei *length, GLint *size, GLenum *type, GLcharARB *name); +GLAPI void APIENTRY glGetUniformfvARB (GLhandleARB programObj, GLint location, GLfloat *params); +GLAPI void APIENTRY glGetUniformivARB (GLhandleARB programObj, GLint location, GLint *params); +GLAPI void APIENTRY glGetShaderSourceARB (GLhandleARB obj, GLsizei maxLength, GLsizei *length, GLcharARB *source); +#endif +#endif /* GL_ARB_shader_objects */ + +#ifndef GL_ARB_shader_precision +#define GL_ARB_shader_precision 1 +#endif /* GL_ARB_shader_precision */ + +#ifndef GL_ARB_shader_stencil_export +#define GL_ARB_shader_stencil_export 1 +#endif /* GL_ARB_shader_stencil_export */ + +#ifndef GL_ARB_shader_storage_buffer_object +#define GL_ARB_shader_storage_buffer_object 1 +#endif /* GL_ARB_shader_storage_buffer_object */ + +#ifndef GL_ARB_shader_subroutine +#define GL_ARB_shader_subroutine 1 +#endif /* GL_ARB_shader_subroutine */ + +#ifndef GL_ARB_shader_texture_image_samples +#define GL_ARB_shader_texture_image_samples 1 +#endif /* GL_ARB_shader_texture_image_samples */ + +#ifndef GL_ARB_shader_texture_lod +#define GL_ARB_shader_texture_lod 1 +#endif /* GL_ARB_shader_texture_lod */ + +#ifndef GL_ARB_shader_viewport_layer_array +#define GL_ARB_shader_viewport_layer_array 1 +#endif /* GL_ARB_shader_viewport_layer_array */ + +#ifndef GL_ARB_shading_language_100 +#define GL_ARB_shading_language_100 1 +#define GL_SHADING_LANGUAGE_VERSION_ARB 0x8B8C +#endif /* GL_ARB_shading_language_100 */ + +#ifndef GL_ARB_shading_language_420pack +#define GL_ARB_shading_language_420pack 1 +#endif /* GL_ARB_shading_language_420pack */ + +#ifndef GL_ARB_shading_language_include +#define GL_ARB_shading_language_include 1 +#define GL_SHADER_INCLUDE_ARB 0x8DAE +#define GL_NAMED_STRING_LENGTH_ARB 0x8DE9 +#define GL_NAMED_STRING_TYPE_ARB 0x8DEA +typedef void (APIENTRYP PFNGLNAMEDSTRINGARBPROC) (GLenum type, GLint namelen, const GLchar *name, GLint stringlen, const GLchar *string); +typedef void (APIENTRYP PFNGLDELETENAMEDSTRINGARBPROC) (GLint namelen, const GLchar *name); +typedef void (APIENTRYP PFNGLCOMPILESHADERINCLUDEARBPROC) (GLuint shader, GLsizei count, const GLchar *const*path, const GLint *length); +typedef GLboolean (APIENTRYP PFNGLISNAMEDSTRINGARBPROC) (GLint namelen, const GLchar *name); +typedef void (APIENTRYP PFNGLGETNAMEDSTRINGARBPROC) (GLint namelen, const GLchar *name, GLsizei bufSize, GLint *stringlen, GLchar *string); +typedef void (APIENTRYP PFNGLGETNAMEDSTRINGIVARBPROC) (GLint namelen, const GLchar *name, GLenum pname, GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glNamedStringARB (GLenum type, GLint namelen, const GLchar *name, GLint stringlen, const GLchar *string); +GLAPI void APIENTRY glDeleteNamedStringARB (GLint namelen, const GLchar *name); +GLAPI void APIENTRY glCompileShaderIncludeARB (GLuint shader, GLsizei count, const GLchar *const*path, const GLint *length); +GLAPI GLboolean APIENTRY glIsNamedStringARB (GLint namelen, const GLchar *name); +GLAPI void APIENTRY glGetNamedStringARB (GLint namelen, const GLchar *name, GLsizei bufSize, GLint *stringlen, GLchar *string); +GLAPI void APIENTRY glGetNamedStringivARB (GLint namelen, const GLchar *name, GLenum pname, GLint *params); +#endif +#endif /* GL_ARB_shading_language_include */ + +#ifndef GL_ARB_shading_language_packing +#define GL_ARB_shading_language_packing 1 +#endif /* GL_ARB_shading_language_packing */ + +#ifndef GL_ARB_shadow +#define GL_ARB_shadow 1 +#define GL_TEXTURE_COMPARE_MODE_ARB 0x884C +#define GL_TEXTURE_COMPARE_FUNC_ARB 0x884D +#define GL_COMPARE_R_TO_TEXTURE_ARB 0x884E +#endif /* GL_ARB_shadow */ + +#ifndef GL_ARB_shadow_ambient +#define GL_ARB_shadow_ambient 1 +#define GL_TEXTURE_COMPARE_FAIL_VALUE_ARB 0x80BF +#endif /* GL_ARB_shadow_ambient */ + +#ifndef GL_ARB_sparse_buffer +#define GL_ARB_sparse_buffer 1 +#define GL_SPARSE_STORAGE_BIT_ARB 0x0400 +#define GL_SPARSE_BUFFER_PAGE_SIZE_ARB 0x82F8 +typedef void (APIENTRYP PFNGLBUFFERPAGECOMMITMENTARBPROC) (GLenum target, GLintptr offset, GLsizeiptr size, GLboolean commit); +typedef void (APIENTRYP PFNGLNAMEDBUFFERPAGECOMMITMENTEXTPROC) (GLuint buffer, GLintptr offset, GLsizeiptr size, GLboolean commit); +typedef void (APIENTRYP PFNGLNAMEDBUFFERPAGECOMMITMENTARBPROC) (GLuint buffer, GLintptr offset, GLsizeiptr size, GLboolean commit); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBufferPageCommitmentARB (GLenum target, GLintptr offset, GLsizeiptr size, GLboolean commit); +GLAPI void APIENTRY glNamedBufferPageCommitmentEXT (GLuint buffer, GLintptr offset, GLsizeiptr size, GLboolean commit); +GLAPI void APIENTRY glNamedBufferPageCommitmentARB (GLuint buffer, GLintptr offset, GLsizeiptr size, GLboolean commit); +#endif +#endif /* GL_ARB_sparse_buffer */ + +#ifndef GL_ARB_sparse_texture +#define GL_ARB_sparse_texture 1 +#define GL_TEXTURE_SPARSE_ARB 0x91A6 +#define GL_VIRTUAL_PAGE_SIZE_INDEX_ARB 0x91A7 +#define GL_NUM_SPARSE_LEVELS_ARB 0x91AA +#define GL_NUM_VIRTUAL_PAGE_SIZES_ARB 0x91A8 +#define GL_VIRTUAL_PAGE_SIZE_X_ARB 0x9195 +#define GL_VIRTUAL_PAGE_SIZE_Y_ARB 0x9196 +#define GL_VIRTUAL_PAGE_SIZE_Z_ARB 0x9197 +#define GL_MAX_SPARSE_TEXTURE_SIZE_ARB 0x9198 +#define GL_MAX_SPARSE_3D_TEXTURE_SIZE_ARB 0x9199 +#define GL_MAX_SPARSE_ARRAY_TEXTURE_LAYERS_ARB 0x919A +#define GL_SPARSE_TEXTURE_FULL_ARRAY_CUBE_MIPMAPS_ARB 0x91A9 +typedef void (APIENTRYP PFNGLTEXPAGECOMMITMENTARBPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLboolean commit); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTexPageCommitmentARB (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLboolean commit); +#endif +#endif /* GL_ARB_sparse_texture */ + +#ifndef GL_ARB_sparse_texture2 +#define GL_ARB_sparse_texture2 1 +#endif /* GL_ARB_sparse_texture2 */ + +#ifndef GL_ARB_sparse_texture_clamp +#define GL_ARB_sparse_texture_clamp 1 +#endif /* GL_ARB_sparse_texture_clamp */ + +#ifndef GL_ARB_spirv_extensions +#define GL_ARB_spirv_extensions 1 +#endif /* GL_ARB_spirv_extensions */ + +#ifndef GL_ARB_stencil_texturing +#define GL_ARB_stencil_texturing 1 +#endif /* GL_ARB_stencil_texturing */ + +#ifndef GL_ARB_sync +#define GL_ARB_sync 1 +#endif /* GL_ARB_sync */ + +#ifndef GL_ARB_tessellation_shader +#define GL_ARB_tessellation_shader 1 +#endif /* GL_ARB_tessellation_shader */ + +#ifndef GL_ARB_texture_barrier +#define GL_ARB_texture_barrier 1 +#endif /* GL_ARB_texture_barrier */ + +#ifndef GL_ARB_texture_border_clamp +#define GL_ARB_texture_border_clamp 1 +#define GL_CLAMP_TO_BORDER_ARB 0x812D +#endif /* GL_ARB_texture_border_clamp */ + +#ifndef GL_ARB_texture_buffer_object +#define GL_ARB_texture_buffer_object 1 +#define GL_TEXTURE_BUFFER_ARB 0x8C2A +#define GL_MAX_TEXTURE_BUFFER_SIZE_ARB 0x8C2B +#define GL_TEXTURE_BINDING_BUFFER_ARB 0x8C2C +#define GL_TEXTURE_BUFFER_DATA_STORE_BINDING_ARB 0x8C2D +#define GL_TEXTURE_BUFFER_FORMAT_ARB 0x8C2E +typedef void (APIENTRYP PFNGLTEXBUFFERARBPROC) (GLenum target, GLenum internalformat, GLuint buffer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTexBufferARB (GLenum target, GLenum internalformat, GLuint buffer); +#endif +#endif /* GL_ARB_texture_buffer_object */ + +#ifndef GL_ARB_texture_buffer_object_rgb32 +#define GL_ARB_texture_buffer_object_rgb32 1 +#endif /* GL_ARB_texture_buffer_object_rgb32 */ + +#ifndef GL_ARB_texture_buffer_range +#define GL_ARB_texture_buffer_range 1 +#endif /* GL_ARB_texture_buffer_range */ + +#ifndef GL_ARB_texture_compression +#define GL_ARB_texture_compression 1 +#define GL_COMPRESSED_ALPHA_ARB 0x84E9 +#define GL_COMPRESSED_LUMINANCE_ARB 0x84EA +#define GL_COMPRESSED_LUMINANCE_ALPHA_ARB 0x84EB +#define GL_COMPRESSED_INTENSITY_ARB 0x84EC +#define GL_COMPRESSED_RGB_ARB 0x84ED +#define GL_COMPRESSED_RGBA_ARB 0x84EE +#define GL_TEXTURE_COMPRESSION_HINT_ARB 0x84EF +#define GL_TEXTURE_COMPRESSED_IMAGE_SIZE_ARB 0x86A0 +#define GL_TEXTURE_COMPRESSED_ARB 0x86A1 +#define GL_NUM_COMPRESSED_TEXTURE_FORMATS_ARB 0x86A2 +#define GL_COMPRESSED_TEXTURE_FORMATS_ARB 0x86A3 +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXIMAGE3DARBPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLsizei imageSize, const void *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXIMAGE2DARBPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLsizei imageSize, const void *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXIMAGE1DARBPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLint border, GLsizei imageSize, const void *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE3DARBPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const void *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE2DARBPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const void *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE1DARBPROC) (GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLsizei imageSize, const void *data); +typedef void (APIENTRYP PFNGLGETCOMPRESSEDTEXIMAGEARBPROC) (GLenum target, GLint level, void *img); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCompressedTexImage3DARB (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLsizei imageSize, const void *data); +GLAPI void APIENTRY glCompressedTexImage2DARB (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLsizei imageSize, const void *data); +GLAPI void APIENTRY glCompressedTexImage1DARB (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLint border, GLsizei imageSize, const void *data); +GLAPI void APIENTRY glCompressedTexSubImage3DARB (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const void *data); +GLAPI void APIENTRY glCompressedTexSubImage2DARB (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const void *data); +GLAPI void APIENTRY glCompressedTexSubImage1DARB (GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLsizei imageSize, const void *data); +GLAPI void APIENTRY glGetCompressedTexImageARB (GLenum target, GLint level, void *img); +#endif +#endif /* GL_ARB_texture_compression */ + +#ifndef GL_ARB_texture_compression_bptc +#define GL_ARB_texture_compression_bptc 1 +#define GL_COMPRESSED_RGBA_BPTC_UNORM_ARB 0x8E8C +#define GL_COMPRESSED_SRGB_ALPHA_BPTC_UNORM_ARB 0x8E8D +#define GL_COMPRESSED_RGB_BPTC_SIGNED_FLOAT_ARB 0x8E8E +#define GL_COMPRESSED_RGB_BPTC_UNSIGNED_FLOAT_ARB 0x8E8F +#endif /* GL_ARB_texture_compression_bptc */ + +#ifndef GL_ARB_texture_compression_rgtc +#define GL_ARB_texture_compression_rgtc 1 +#endif /* GL_ARB_texture_compression_rgtc */ + +#ifndef GL_ARB_texture_cube_map +#define GL_ARB_texture_cube_map 1 +#define GL_NORMAL_MAP_ARB 0x8511 +#define GL_REFLECTION_MAP_ARB 0x8512 +#define GL_TEXTURE_CUBE_MAP_ARB 0x8513 +#define GL_TEXTURE_BINDING_CUBE_MAP_ARB 0x8514 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_X_ARB 0x8515 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_X_ARB 0x8516 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Y_ARB 0x8517 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Y_ARB 0x8518 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Z_ARB 0x8519 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Z_ARB 0x851A +#define GL_PROXY_TEXTURE_CUBE_MAP_ARB 0x851B +#define GL_MAX_CUBE_MAP_TEXTURE_SIZE_ARB 0x851C +#endif /* GL_ARB_texture_cube_map */ + +#ifndef GL_ARB_texture_cube_map_array +#define GL_ARB_texture_cube_map_array 1 +#define GL_TEXTURE_CUBE_MAP_ARRAY_ARB 0x9009 +#define GL_TEXTURE_BINDING_CUBE_MAP_ARRAY_ARB 0x900A +#define GL_PROXY_TEXTURE_CUBE_MAP_ARRAY_ARB 0x900B +#define GL_SAMPLER_CUBE_MAP_ARRAY_ARB 0x900C +#define GL_SAMPLER_CUBE_MAP_ARRAY_SHADOW_ARB 0x900D +#define GL_INT_SAMPLER_CUBE_MAP_ARRAY_ARB 0x900E +#define GL_UNSIGNED_INT_SAMPLER_CUBE_MAP_ARRAY_ARB 0x900F +#endif /* GL_ARB_texture_cube_map_array */ + +#ifndef GL_ARB_texture_env_add +#define GL_ARB_texture_env_add 1 +#endif /* GL_ARB_texture_env_add */ + +#ifndef GL_ARB_texture_env_combine +#define GL_ARB_texture_env_combine 1 +#define GL_COMBINE_ARB 0x8570 +#define GL_COMBINE_RGB_ARB 0x8571 +#define GL_COMBINE_ALPHA_ARB 0x8572 +#define GL_SOURCE0_RGB_ARB 0x8580 +#define GL_SOURCE1_RGB_ARB 0x8581 +#define GL_SOURCE2_RGB_ARB 0x8582 +#define GL_SOURCE0_ALPHA_ARB 0x8588 +#define GL_SOURCE1_ALPHA_ARB 0x8589 +#define GL_SOURCE2_ALPHA_ARB 0x858A +#define GL_OPERAND0_RGB_ARB 0x8590 +#define GL_OPERAND1_RGB_ARB 0x8591 +#define GL_OPERAND2_RGB_ARB 0x8592 +#define GL_OPERAND0_ALPHA_ARB 0x8598 +#define GL_OPERAND1_ALPHA_ARB 0x8599 +#define GL_OPERAND2_ALPHA_ARB 0x859A +#define GL_RGB_SCALE_ARB 0x8573 +#define GL_ADD_SIGNED_ARB 0x8574 +#define GL_INTERPOLATE_ARB 0x8575 +#define GL_SUBTRACT_ARB 0x84E7 +#define GL_CONSTANT_ARB 0x8576 +#define GL_PRIMARY_COLOR_ARB 0x8577 +#define GL_PREVIOUS_ARB 0x8578 +#endif /* GL_ARB_texture_env_combine */ + +#ifndef GL_ARB_texture_env_crossbar +#define GL_ARB_texture_env_crossbar 1 +#endif /* GL_ARB_texture_env_crossbar */ + +#ifndef GL_ARB_texture_env_dot3 +#define GL_ARB_texture_env_dot3 1 +#define GL_DOT3_RGB_ARB 0x86AE +#define GL_DOT3_RGBA_ARB 0x86AF +#endif /* GL_ARB_texture_env_dot3 */ + +#ifndef GL_ARB_texture_filter_anisotropic +#define GL_ARB_texture_filter_anisotropic 1 +#endif /* GL_ARB_texture_filter_anisotropic */ + +#ifndef GL_ARB_texture_filter_minmax +#define GL_ARB_texture_filter_minmax 1 +#define GL_TEXTURE_REDUCTION_MODE_ARB 0x9366 +#define GL_WEIGHTED_AVERAGE_ARB 0x9367 +#endif /* GL_ARB_texture_filter_minmax */ + +#ifndef GL_ARB_texture_float +#define GL_ARB_texture_float 1 +#define GL_TEXTURE_RED_TYPE_ARB 0x8C10 +#define GL_TEXTURE_GREEN_TYPE_ARB 0x8C11 +#define GL_TEXTURE_BLUE_TYPE_ARB 0x8C12 +#define GL_TEXTURE_ALPHA_TYPE_ARB 0x8C13 +#define GL_TEXTURE_LUMINANCE_TYPE_ARB 0x8C14 +#define GL_TEXTURE_INTENSITY_TYPE_ARB 0x8C15 +#define GL_TEXTURE_DEPTH_TYPE_ARB 0x8C16 +#define GL_UNSIGNED_NORMALIZED_ARB 0x8C17 +#define GL_RGBA32F_ARB 0x8814 +#define GL_RGB32F_ARB 0x8815 +#define GL_ALPHA32F_ARB 0x8816 +#define GL_INTENSITY32F_ARB 0x8817 +#define GL_LUMINANCE32F_ARB 0x8818 +#define GL_LUMINANCE_ALPHA32F_ARB 0x8819 +#define GL_RGBA16F_ARB 0x881A +#define GL_RGB16F_ARB 0x881B +#define GL_ALPHA16F_ARB 0x881C +#define GL_INTENSITY16F_ARB 0x881D +#define GL_LUMINANCE16F_ARB 0x881E +#define GL_LUMINANCE_ALPHA16F_ARB 0x881F +#endif /* GL_ARB_texture_float */ + +#ifndef GL_ARB_texture_gather +#define GL_ARB_texture_gather 1 +#define GL_MIN_PROGRAM_TEXTURE_GATHER_OFFSET_ARB 0x8E5E +#define GL_MAX_PROGRAM_TEXTURE_GATHER_OFFSET_ARB 0x8E5F +#define GL_MAX_PROGRAM_TEXTURE_GATHER_COMPONENTS_ARB 0x8F9F +#endif /* GL_ARB_texture_gather */ + +#ifndef GL_ARB_texture_mirror_clamp_to_edge +#define GL_ARB_texture_mirror_clamp_to_edge 1 +#endif /* GL_ARB_texture_mirror_clamp_to_edge */ + +#ifndef GL_ARB_texture_mirrored_repeat +#define GL_ARB_texture_mirrored_repeat 1 +#define GL_MIRRORED_REPEAT_ARB 0x8370 +#endif /* GL_ARB_texture_mirrored_repeat */ + +#ifndef GL_ARB_texture_multisample +#define GL_ARB_texture_multisample 1 +#endif /* GL_ARB_texture_multisample */ + +#ifndef GL_ARB_texture_non_power_of_two +#define GL_ARB_texture_non_power_of_two 1 +#endif /* GL_ARB_texture_non_power_of_two */ + +#ifndef GL_ARB_texture_query_levels +#define GL_ARB_texture_query_levels 1 +#endif /* GL_ARB_texture_query_levels */ + +#ifndef GL_ARB_texture_query_lod +#define GL_ARB_texture_query_lod 1 +#endif /* GL_ARB_texture_query_lod */ + +#ifndef GL_ARB_texture_rectangle +#define GL_ARB_texture_rectangle 1 +#define GL_TEXTURE_RECTANGLE_ARB 0x84F5 +#define GL_TEXTURE_BINDING_RECTANGLE_ARB 0x84F6 +#define GL_PROXY_TEXTURE_RECTANGLE_ARB 0x84F7 +#define GL_MAX_RECTANGLE_TEXTURE_SIZE_ARB 0x84F8 +#endif /* GL_ARB_texture_rectangle */ + +#ifndef GL_ARB_texture_rg +#define GL_ARB_texture_rg 1 +#endif /* GL_ARB_texture_rg */ + +#ifndef GL_ARB_texture_rgb10_a2ui +#define GL_ARB_texture_rgb10_a2ui 1 +#endif /* GL_ARB_texture_rgb10_a2ui */ + +#ifndef GL_ARB_texture_stencil8 +#define GL_ARB_texture_stencil8 1 +#endif /* GL_ARB_texture_stencil8 */ + +#ifndef GL_ARB_texture_storage +#define GL_ARB_texture_storage 1 +#endif /* GL_ARB_texture_storage */ + +#ifndef GL_ARB_texture_storage_multisample +#define GL_ARB_texture_storage_multisample 1 +#endif /* GL_ARB_texture_storage_multisample */ + +#ifndef GL_ARB_texture_swizzle +#define GL_ARB_texture_swizzle 1 +#endif /* GL_ARB_texture_swizzle */ + +#ifndef GL_ARB_texture_view +#define GL_ARB_texture_view 1 +#endif /* GL_ARB_texture_view */ + +#ifndef GL_ARB_timer_query +#define GL_ARB_timer_query 1 +#endif /* GL_ARB_timer_query */ + +#ifndef GL_ARB_transform_feedback2 +#define GL_ARB_transform_feedback2 1 +#endif /* GL_ARB_transform_feedback2 */ + +#ifndef GL_ARB_transform_feedback3 +#define GL_ARB_transform_feedback3 1 +#endif /* GL_ARB_transform_feedback3 */ + +#ifndef GL_ARB_transform_feedback_instanced +#define GL_ARB_transform_feedback_instanced 1 +#endif /* GL_ARB_transform_feedback_instanced */ + +#ifndef GL_ARB_transform_feedback_overflow_query +#define GL_ARB_transform_feedback_overflow_query 1 +#define GL_TRANSFORM_FEEDBACK_OVERFLOW_ARB 0x82EC +#define GL_TRANSFORM_FEEDBACK_STREAM_OVERFLOW_ARB 0x82ED +#endif /* GL_ARB_transform_feedback_overflow_query */ + +#ifndef GL_ARB_transpose_matrix +#define GL_ARB_transpose_matrix 1 +#define GL_TRANSPOSE_MODELVIEW_MATRIX_ARB 0x84E3 +#define GL_TRANSPOSE_PROJECTION_MATRIX_ARB 0x84E4 +#define GL_TRANSPOSE_TEXTURE_MATRIX_ARB 0x84E5 +#define GL_TRANSPOSE_COLOR_MATRIX_ARB 0x84E6 +typedef void (APIENTRYP PFNGLLOADTRANSPOSEMATRIXFARBPROC) (const GLfloat *m); +typedef void (APIENTRYP PFNGLLOADTRANSPOSEMATRIXDARBPROC) (const GLdouble *m); +typedef void (APIENTRYP PFNGLMULTTRANSPOSEMATRIXFARBPROC) (const GLfloat *m); +typedef void (APIENTRYP PFNGLMULTTRANSPOSEMATRIXDARBPROC) (const GLdouble *m); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glLoadTransposeMatrixfARB (const GLfloat *m); +GLAPI void APIENTRY glLoadTransposeMatrixdARB (const GLdouble *m); +GLAPI void APIENTRY glMultTransposeMatrixfARB (const GLfloat *m); +GLAPI void APIENTRY glMultTransposeMatrixdARB (const GLdouble *m); +#endif +#endif /* GL_ARB_transpose_matrix */ + +#ifndef GL_ARB_uniform_buffer_object +#define GL_ARB_uniform_buffer_object 1 +#endif /* GL_ARB_uniform_buffer_object */ + +#ifndef GL_ARB_vertex_array_bgra +#define GL_ARB_vertex_array_bgra 1 +#endif /* GL_ARB_vertex_array_bgra */ + +#ifndef GL_ARB_vertex_array_object +#define GL_ARB_vertex_array_object 1 +#endif /* GL_ARB_vertex_array_object */ + +#ifndef GL_ARB_vertex_attrib_64bit +#define GL_ARB_vertex_attrib_64bit 1 +#endif /* GL_ARB_vertex_attrib_64bit */ + +#ifndef GL_ARB_vertex_attrib_binding +#define GL_ARB_vertex_attrib_binding 1 +#endif /* GL_ARB_vertex_attrib_binding */ + +#ifndef GL_ARB_vertex_blend +#define GL_ARB_vertex_blend 1 +#define GL_MAX_VERTEX_UNITS_ARB 0x86A4 +#define GL_ACTIVE_VERTEX_UNITS_ARB 0x86A5 +#define GL_WEIGHT_SUM_UNITY_ARB 0x86A6 +#define GL_VERTEX_BLEND_ARB 0x86A7 +#define GL_CURRENT_WEIGHT_ARB 0x86A8 +#define GL_WEIGHT_ARRAY_TYPE_ARB 0x86A9 +#define GL_WEIGHT_ARRAY_STRIDE_ARB 0x86AA +#define GL_WEIGHT_ARRAY_SIZE_ARB 0x86AB +#define GL_WEIGHT_ARRAY_POINTER_ARB 0x86AC +#define GL_WEIGHT_ARRAY_ARB 0x86AD +#define GL_MODELVIEW0_ARB 0x1700 +#define GL_MODELVIEW1_ARB 0x850A +#define GL_MODELVIEW2_ARB 0x8722 +#define GL_MODELVIEW3_ARB 0x8723 +#define GL_MODELVIEW4_ARB 0x8724 +#define GL_MODELVIEW5_ARB 0x8725 +#define GL_MODELVIEW6_ARB 0x8726 +#define GL_MODELVIEW7_ARB 0x8727 +#define GL_MODELVIEW8_ARB 0x8728 +#define GL_MODELVIEW9_ARB 0x8729 +#define GL_MODELVIEW10_ARB 0x872A +#define GL_MODELVIEW11_ARB 0x872B +#define GL_MODELVIEW12_ARB 0x872C +#define GL_MODELVIEW13_ARB 0x872D +#define GL_MODELVIEW14_ARB 0x872E +#define GL_MODELVIEW15_ARB 0x872F +#define GL_MODELVIEW16_ARB 0x8730 +#define GL_MODELVIEW17_ARB 0x8731 +#define GL_MODELVIEW18_ARB 0x8732 +#define GL_MODELVIEW19_ARB 0x8733 +#define GL_MODELVIEW20_ARB 0x8734 +#define GL_MODELVIEW21_ARB 0x8735 +#define GL_MODELVIEW22_ARB 0x8736 +#define GL_MODELVIEW23_ARB 0x8737 +#define GL_MODELVIEW24_ARB 0x8738 +#define GL_MODELVIEW25_ARB 0x8739 +#define GL_MODELVIEW26_ARB 0x873A +#define GL_MODELVIEW27_ARB 0x873B +#define GL_MODELVIEW28_ARB 0x873C +#define GL_MODELVIEW29_ARB 0x873D +#define GL_MODELVIEW30_ARB 0x873E +#define GL_MODELVIEW31_ARB 0x873F +typedef void (APIENTRYP PFNGLWEIGHTBVARBPROC) (GLint size, const GLbyte *weights); +typedef void (APIENTRYP PFNGLWEIGHTSVARBPROC) (GLint size, const GLshort *weights); +typedef void (APIENTRYP PFNGLWEIGHTIVARBPROC) (GLint size, const GLint *weights); +typedef void (APIENTRYP PFNGLWEIGHTFVARBPROC) (GLint size, const GLfloat *weights); +typedef void (APIENTRYP PFNGLWEIGHTDVARBPROC) (GLint size, const GLdouble *weights); +typedef void (APIENTRYP PFNGLWEIGHTUBVARBPROC) (GLint size, const GLubyte *weights); +typedef void (APIENTRYP PFNGLWEIGHTUSVARBPROC) (GLint size, const GLushort *weights); +typedef void (APIENTRYP PFNGLWEIGHTUIVARBPROC) (GLint size, const GLuint *weights); +typedef void (APIENTRYP PFNGLWEIGHTPOINTERARBPROC) (GLint size, GLenum type, GLsizei stride, const void *pointer); +typedef void (APIENTRYP PFNGLVERTEXBLENDARBPROC) (GLint count); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glWeightbvARB (GLint size, const GLbyte *weights); +GLAPI void APIENTRY glWeightsvARB (GLint size, const GLshort *weights); +GLAPI void APIENTRY glWeightivARB (GLint size, const GLint *weights); +GLAPI void APIENTRY glWeightfvARB (GLint size, const GLfloat *weights); +GLAPI void APIENTRY glWeightdvARB (GLint size, const GLdouble *weights); +GLAPI void APIENTRY glWeightubvARB (GLint size, const GLubyte *weights); +GLAPI void APIENTRY glWeightusvARB (GLint size, const GLushort *weights); +GLAPI void APIENTRY glWeightuivARB (GLint size, const GLuint *weights); +GLAPI void APIENTRY glWeightPointerARB (GLint size, GLenum type, GLsizei stride, const void *pointer); +GLAPI void APIENTRY glVertexBlendARB (GLint count); +#endif +#endif /* GL_ARB_vertex_blend */ + +#ifndef GL_ARB_vertex_buffer_object +#define GL_ARB_vertex_buffer_object 1 +typedef khronos_ssize_t GLsizeiptrARB; +typedef khronos_intptr_t GLintptrARB; +#define GL_BUFFER_SIZE_ARB 0x8764 +#define GL_BUFFER_USAGE_ARB 0x8765 +#define GL_ARRAY_BUFFER_ARB 0x8892 +#define GL_ELEMENT_ARRAY_BUFFER_ARB 0x8893 +#define GL_ARRAY_BUFFER_BINDING_ARB 0x8894 +#define GL_ELEMENT_ARRAY_BUFFER_BINDING_ARB 0x8895 +#define GL_VERTEX_ARRAY_BUFFER_BINDING_ARB 0x8896 +#define GL_NORMAL_ARRAY_BUFFER_BINDING_ARB 0x8897 +#define GL_COLOR_ARRAY_BUFFER_BINDING_ARB 0x8898 +#define GL_INDEX_ARRAY_BUFFER_BINDING_ARB 0x8899 +#define GL_TEXTURE_COORD_ARRAY_BUFFER_BINDING_ARB 0x889A +#define GL_EDGE_FLAG_ARRAY_BUFFER_BINDING_ARB 0x889B +#define GL_SECONDARY_COLOR_ARRAY_BUFFER_BINDING_ARB 0x889C +#define GL_FOG_COORDINATE_ARRAY_BUFFER_BINDING_ARB 0x889D +#define GL_WEIGHT_ARRAY_BUFFER_BINDING_ARB 0x889E +#define GL_VERTEX_ATTRIB_ARRAY_BUFFER_BINDING_ARB 0x889F +#define GL_READ_ONLY_ARB 0x88B8 +#define GL_WRITE_ONLY_ARB 0x88B9 +#define GL_READ_WRITE_ARB 0x88BA +#define GL_BUFFER_ACCESS_ARB 0x88BB +#define GL_BUFFER_MAPPED_ARB 0x88BC +#define GL_BUFFER_MAP_POINTER_ARB 0x88BD +#define GL_STREAM_DRAW_ARB 0x88E0 +#define GL_STREAM_READ_ARB 0x88E1 +#define GL_STREAM_COPY_ARB 0x88E2 +#define GL_STATIC_DRAW_ARB 0x88E4 +#define GL_STATIC_READ_ARB 0x88E5 +#define GL_STATIC_COPY_ARB 0x88E6 +#define GL_DYNAMIC_DRAW_ARB 0x88E8 +#define GL_DYNAMIC_READ_ARB 0x88E9 +#define GL_DYNAMIC_COPY_ARB 0x88EA +typedef void (APIENTRYP PFNGLBINDBUFFERARBPROC) (GLenum target, GLuint buffer); +typedef void (APIENTRYP PFNGLDELETEBUFFERSARBPROC) (GLsizei n, const GLuint *buffers); +typedef void (APIENTRYP PFNGLGENBUFFERSARBPROC) (GLsizei n, GLuint *buffers); +typedef GLboolean (APIENTRYP PFNGLISBUFFERARBPROC) (GLuint buffer); +typedef void (APIENTRYP PFNGLBUFFERDATAARBPROC) (GLenum target, GLsizeiptrARB size, const void *data, GLenum usage); +typedef void (APIENTRYP PFNGLBUFFERSUBDATAARBPROC) (GLenum target, GLintptrARB offset, GLsizeiptrARB size, const void *data); +typedef void (APIENTRYP PFNGLGETBUFFERSUBDATAARBPROC) (GLenum target, GLintptrARB offset, GLsizeiptrARB size, void *data); +typedef void *(APIENTRYP PFNGLMAPBUFFERARBPROC) (GLenum target, GLenum access); +typedef GLboolean (APIENTRYP PFNGLUNMAPBUFFERARBPROC) (GLenum target); +typedef void (APIENTRYP PFNGLGETBUFFERPARAMETERIVARBPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETBUFFERPOINTERVARBPROC) (GLenum target, GLenum pname, void **params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBindBufferARB (GLenum target, GLuint buffer); +GLAPI void APIENTRY glDeleteBuffersARB (GLsizei n, const GLuint *buffers); +GLAPI void APIENTRY glGenBuffersARB (GLsizei n, GLuint *buffers); +GLAPI GLboolean APIENTRY glIsBufferARB (GLuint buffer); +GLAPI void APIENTRY glBufferDataARB (GLenum target, GLsizeiptrARB size, const void *data, GLenum usage); +GLAPI void APIENTRY glBufferSubDataARB (GLenum target, GLintptrARB offset, GLsizeiptrARB size, const void *data); +GLAPI void APIENTRY glGetBufferSubDataARB (GLenum target, GLintptrARB offset, GLsizeiptrARB size, void *data); +GLAPI void *APIENTRY glMapBufferARB (GLenum target, GLenum access); +GLAPI GLboolean APIENTRY glUnmapBufferARB (GLenum target); +GLAPI void APIENTRY glGetBufferParameterivARB (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetBufferPointervARB (GLenum target, GLenum pname, void **params); +#endif +#endif /* GL_ARB_vertex_buffer_object */ + +#ifndef GL_ARB_vertex_program +#define GL_ARB_vertex_program 1 +#define GL_COLOR_SUM_ARB 0x8458 +#define GL_VERTEX_PROGRAM_ARB 0x8620 +#define GL_VERTEX_ATTRIB_ARRAY_ENABLED_ARB 0x8622 +#define GL_VERTEX_ATTRIB_ARRAY_SIZE_ARB 0x8623 +#define GL_VERTEX_ATTRIB_ARRAY_STRIDE_ARB 0x8624 +#define GL_VERTEX_ATTRIB_ARRAY_TYPE_ARB 0x8625 +#define GL_CURRENT_VERTEX_ATTRIB_ARB 0x8626 +#define GL_VERTEX_PROGRAM_POINT_SIZE_ARB 0x8642 +#define GL_VERTEX_PROGRAM_TWO_SIDE_ARB 0x8643 +#define GL_VERTEX_ATTRIB_ARRAY_POINTER_ARB 0x8645 +#define GL_MAX_VERTEX_ATTRIBS_ARB 0x8869 +#define GL_VERTEX_ATTRIB_ARRAY_NORMALIZED_ARB 0x886A +#define GL_PROGRAM_ADDRESS_REGISTERS_ARB 0x88B0 +#define GL_MAX_PROGRAM_ADDRESS_REGISTERS_ARB 0x88B1 +#define GL_PROGRAM_NATIVE_ADDRESS_REGISTERS_ARB 0x88B2 +#define GL_MAX_PROGRAM_NATIVE_ADDRESS_REGISTERS_ARB 0x88B3 +typedef void (APIENTRYP PFNGLVERTEXATTRIB1DARBPROC) (GLuint index, GLdouble x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1DVARBPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1FARBPROC) (GLuint index, GLfloat x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1FVARBPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1SARBPROC) (GLuint index, GLshort x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1SVARBPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2DARBPROC) (GLuint index, GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2DVARBPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2FARBPROC) (GLuint index, GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2FVARBPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2SARBPROC) (GLuint index, GLshort x, GLshort y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2SVARBPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3DARBPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3DVARBPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3FARBPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3FVARBPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3SARBPROC) (GLuint index, GLshort x, GLshort y, GLshort z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3SVARBPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NBVARBPROC) (GLuint index, const GLbyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NIVARBPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NSVARBPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUBARBPROC) (GLuint index, GLubyte x, GLubyte y, GLubyte z, GLubyte w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUBVARBPROC) (GLuint index, const GLubyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUIVARBPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUSVARBPROC) (GLuint index, const GLushort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4BVARBPROC) (GLuint index, const GLbyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4DARBPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4DVARBPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4FARBPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4FVARBPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4IVARBPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4SARBPROC) (GLuint index, GLshort x, GLshort y, GLshort z, GLshort w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4SVARBPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4UBVARBPROC) (GLuint index, const GLubyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4UIVARBPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4USVARBPROC) (GLuint index, const GLushort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBPOINTERARBPROC) (GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride, const void *pointer); +typedef void (APIENTRYP PFNGLENABLEVERTEXATTRIBARRAYARBPROC) (GLuint index); +typedef void (APIENTRYP PFNGLDISABLEVERTEXATTRIBARRAYARBPROC) (GLuint index); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBDVARBPROC) (GLuint index, GLenum pname, GLdouble *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBFVARBPROC) (GLuint index, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBIVARBPROC) (GLuint index, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBPOINTERVARBPROC) (GLuint index, GLenum pname, void **pointer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexAttrib1dARB (GLuint index, GLdouble x); +GLAPI void APIENTRY glVertexAttrib1dvARB (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttrib1fARB (GLuint index, GLfloat x); +GLAPI void APIENTRY glVertexAttrib1fvARB (GLuint index, const GLfloat *v); +GLAPI void APIENTRY glVertexAttrib1sARB (GLuint index, GLshort x); +GLAPI void APIENTRY glVertexAttrib1svARB (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttrib2dARB (GLuint index, GLdouble x, GLdouble y); +GLAPI void APIENTRY glVertexAttrib2dvARB (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttrib2fARB (GLuint index, GLfloat x, GLfloat y); +GLAPI void APIENTRY glVertexAttrib2fvARB (GLuint index, const GLfloat *v); +GLAPI void APIENTRY glVertexAttrib2sARB (GLuint index, GLshort x, GLshort y); +GLAPI void APIENTRY glVertexAttrib2svARB (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttrib3dARB (GLuint index, GLdouble x, GLdouble y, GLdouble z); +GLAPI void APIENTRY glVertexAttrib3dvARB (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttrib3fARB (GLuint index, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glVertexAttrib3fvARB (GLuint index, const GLfloat *v); +GLAPI void APIENTRY glVertexAttrib3sARB (GLuint index, GLshort x, GLshort y, GLshort z); +GLAPI void APIENTRY glVertexAttrib3svARB (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttrib4NbvARB (GLuint index, const GLbyte *v); +GLAPI void APIENTRY glVertexAttrib4NivARB (GLuint index, const GLint *v); +GLAPI void APIENTRY glVertexAttrib4NsvARB (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttrib4NubARB (GLuint index, GLubyte x, GLubyte y, GLubyte z, GLubyte w); +GLAPI void APIENTRY glVertexAttrib4NubvARB (GLuint index, const GLubyte *v); +GLAPI void APIENTRY glVertexAttrib4NuivARB (GLuint index, const GLuint *v); +GLAPI void APIENTRY glVertexAttrib4NusvARB (GLuint index, const GLushort *v); +GLAPI void APIENTRY glVertexAttrib4bvARB (GLuint index, const GLbyte *v); +GLAPI void APIENTRY glVertexAttrib4dARB (GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +GLAPI void APIENTRY glVertexAttrib4dvARB (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttrib4fARB (GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +GLAPI void APIENTRY glVertexAttrib4fvARB (GLuint index, const GLfloat *v); +GLAPI void APIENTRY glVertexAttrib4ivARB (GLuint index, const GLint *v); +GLAPI void APIENTRY glVertexAttrib4sARB (GLuint index, GLshort x, GLshort y, GLshort z, GLshort w); +GLAPI void APIENTRY glVertexAttrib4svARB (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttrib4ubvARB (GLuint index, const GLubyte *v); +GLAPI void APIENTRY glVertexAttrib4uivARB (GLuint index, const GLuint *v); +GLAPI void APIENTRY glVertexAttrib4usvARB (GLuint index, const GLushort *v); +GLAPI void APIENTRY glVertexAttribPointerARB (GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride, const void *pointer); +GLAPI void APIENTRY glEnableVertexAttribArrayARB (GLuint index); +GLAPI void APIENTRY glDisableVertexAttribArrayARB (GLuint index); +GLAPI void APIENTRY glGetVertexAttribdvARB (GLuint index, GLenum pname, GLdouble *params); +GLAPI void APIENTRY glGetVertexAttribfvARB (GLuint index, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetVertexAttribivARB (GLuint index, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetVertexAttribPointervARB (GLuint index, GLenum pname, void **pointer); +#endif +#endif /* GL_ARB_vertex_program */ + +#ifndef GL_ARB_vertex_shader +#define GL_ARB_vertex_shader 1 +#define GL_VERTEX_SHADER_ARB 0x8B31 +#define GL_MAX_VERTEX_UNIFORM_COMPONENTS_ARB 0x8B4A +#define GL_MAX_VARYING_FLOATS_ARB 0x8B4B +#define GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS_ARB 0x8B4C +#define GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS_ARB 0x8B4D +#define GL_OBJECT_ACTIVE_ATTRIBUTES_ARB 0x8B89 +#define GL_OBJECT_ACTIVE_ATTRIBUTE_MAX_LENGTH_ARB 0x8B8A +typedef void (APIENTRYP PFNGLBINDATTRIBLOCATIONARBPROC) (GLhandleARB programObj, GLuint index, const GLcharARB *name); +typedef void (APIENTRYP PFNGLGETACTIVEATTRIBARBPROC) (GLhandleARB programObj, GLuint index, GLsizei maxLength, GLsizei *length, GLint *size, GLenum *type, GLcharARB *name); +typedef GLint (APIENTRYP PFNGLGETATTRIBLOCATIONARBPROC) (GLhandleARB programObj, const GLcharARB *name); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBindAttribLocationARB (GLhandleARB programObj, GLuint index, const GLcharARB *name); +GLAPI void APIENTRY glGetActiveAttribARB (GLhandleARB programObj, GLuint index, GLsizei maxLength, GLsizei *length, GLint *size, GLenum *type, GLcharARB *name); +GLAPI GLint APIENTRY glGetAttribLocationARB (GLhandleARB programObj, const GLcharARB *name); +#endif +#endif /* GL_ARB_vertex_shader */ + +#ifndef GL_ARB_vertex_type_10f_11f_11f_rev +#define GL_ARB_vertex_type_10f_11f_11f_rev 1 +#endif /* GL_ARB_vertex_type_10f_11f_11f_rev */ + +#ifndef GL_ARB_vertex_type_2_10_10_10_rev +#define GL_ARB_vertex_type_2_10_10_10_rev 1 +#endif /* GL_ARB_vertex_type_2_10_10_10_rev */ + +#ifndef GL_ARB_viewport_array +#define GL_ARB_viewport_array 1 +typedef void (APIENTRYP PFNGLDEPTHRANGEARRAYDVNVPROC) (GLuint first, GLsizei count, const GLdouble *v); +typedef void (APIENTRYP PFNGLDEPTHRANGEINDEXEDDNVPROC) (GLuint index, GLdouble n, GLdouble f); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDepthRangeArraydvNV (GLuint first, GLsizei count, const GLdouble *v); +GLAPI void APIENTRY glDepthRangeIndexeddNV (GLuint index, GLdouble n, GLdouble f); +#endif +#endif /* GL_ARB_viewport_array */ + +#ifndef GL_ARB_window_pos +#define GL_ARB_window_pos 1 +typedef void (APIENTRYP PFNGLWINDOWPOS2DARBPROC) (GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLWINDOWPOS2DVARBPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2FARBPROC) (GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLWINDOWPOS2FVARBPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2IARBPROC) (GLint x, GLint y); +typedef void (APIENTRYP PFNGLWINDOWPOS2IVARBPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2SARBPROC) (GLshort x, GLshort y); +typedef void (APIENTRYP PFNGLWINDOWPOS2SVARBPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3DARBPROC) (GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLWINDOWPOS3DVARBPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3FARBPROC) (GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLWINDOWPOS3FVARBPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3IARBPROC) (GLint x, GLint y, GLint z); +typedef void (APIENTRYP PFNGLWINDOWPOS3IVARBPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3SARBPROC) (GLshort x, GLshort y, GLshort z); +typedef void (APIENTRYP PFNGLWINDOWPOS3SVARBPROC) (const GLshort *v); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glWindowPos2dARB (GLdouble x, GLdouble y); +GLAPI void APIENTRY glWindowPos2dvARB (const GLdouble *v); +GLAPI void APIENTRY glWindowPos2fARB (GLfloat x, GLfloat y); +GLAPI void APIENTRY glWindowPos2fvARB (const GLfloat *v); +GLAPI void APIENTRY glWindowPos2iARB (GLint x, GLint y); +GLAPI void APIENTRY glWindowPos2ivARB (const GLint *v); +GLAPI void APIENTRY glWindowPos2sARB (GLshort x, GLshort y); +GLAPI void APIENTRY glWindowPos2svARB (const GLshort *v); +GLAPI void APIENTRY glWindowPos3dARB (GLdouble x, GLdouble y, GLdouble z); +GLAPI void APIENTRY glWindowPos3dvARB (const GLdouble *v); +GLAPI void APIENTRY glWindowPos3fARB (GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glWindowPos3fvARB (const GLfloat *v); +GLAPI void APIENTRY glWindowPos3iARB (GLint x, GLint y, GLint z); +GLAPI void APIENTRY glWindowPos3ivARB (const GLint *v); +GLAPI void APIENTRY glWindowPos3sARB (GLshort x, GLshort y, GLshort z); +GLAPI void APIENTRY glWindowPos3svARB (const GLshort *v); +#endif +#endif /* GL_ARB_window_pos */ + +#ifndef GL_KHR_blend_equation_advanced +#define GL_KHR_blend_equation_advanced 1 +#define GL_MULTIPLY_KHR 0x9294 +#define GL_SCREEN_KHR 0x9295 +#define GL_OVERLAY_KHR 0x9296 +#define GL_DARKEN_KHR 0x9297 +#define GL_LIGHTEN_KHR 0x9298 +#define GL_COLORDODGE_KHR 0x9299 +#define GL_COLORBURN_KHR 0x929A +#define GL_HARDLIGHT_KHR 0x929B +#define GL_SOFTLIGHT_KHR 0x929C +#define GL_DIFFERENCE_KHR 0x929E +#define GL_EXCLUSION_KHR 0x92A0 +#define GL_HSL_HUE_KHR 0x92AD +#define GL_HSL_SATURATION_KHR 0x92AE +#define GL_HSL_COLOR_KHR 0x92AF +#define GL_HSL_LUMINOSITY_KHR 0x92B0 +typedef void (APIENTRYP PFNGLBLENDBARRIERKHRPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendBarrierKHR (void); +#endif +#endif /* GL_KHR_blend_equation_advanced */ + +#ifndef GL_KHR_blend_equation_advanced_coherent +#define GL_KHR_blend_equation_advanced_coherent 1 +#define GL_BLEND_ADVANCED_COHERENT_KHR 0x9285 +#endif /* GL_KHR_blend_equation_advanced_coherent */ + +#ifndef GL_KHR_context_flush_control +#define GL_KHR_context_flush_control 1 +#endif /* GL_KHR_context_flush_control */ + +#ifndef GL_KHR_debug +#define GL_KHR_debug 1 +#endif /* GL_KHR_debug */ + +#ifndef GL_KHR_no_error +#define GL_KHR_no_error 1 +#define GL_CONTEXT_FLAG_NO_ERROR_BIT_KHR 0x00000008 +#endif /* GL_KHR_no_error */ + +#ifndef GL_KHR_parallel_shader_compile +#define GL_KHR_parallel_shader_compile 1 +#define GL_MAX_SHADER_COMPILER_THREADS_KHR 0x91B0 +#define GL_COMPLETION_STATUS_KHR 0x91B1 +typedef void (APIENTRYP PFNGLMAXSHADERCOMPILERTHREADSKHRPROC) (GLuint count); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMaxShaderCompilerThreadsKHR (GLuint count); +#endif +#endif /* GL_KHR_parallel_shader_compile */ + +#ifndef GL_KHR_robust_buffer_access_behavior +#define GL_KHR_robust_buffer_access_behavior 1 +#endif /* GL_KHR_robust_buffer_access_behavior */ + +#ifndef GL_KHR_robustness +#define GL_KHR_robustness 1 +#define GL_CONTEXT_ROBUST_ACCESS 0x90F3 +#endif /* GL_KHR_robustness */ + +#ifndef GL_KHR_shader_subgroup +#define GL_KHR_shader_subgroup 1 +#define GL_SUBGROUP_SIZE_KHR 0x9532 +#define GL_SUBGROUP_SUPPORTED_STAGES_KHR 0x9533 +#define GL_SUBGROUP_SUPPORTED_FEATURES_KHR 0x9534 +#define GL_SUBGROUP_QUAD_ALL_STAGES_KHR 0x9535 +#define GL_SUBGROUP_FEATURE_BASIC_BIT_KHR 0x00000001 +#define GL_SUBGROUP_FEATURE_VOTE_BIT_KHR 0x00000002 +#define GL_SUBGROUP_FEATURE_ARITHMETIC_BIT_KHR 0x00000004 +#define GL_SUBGROUP_FEATURE_BALLOT_BIT_KHR 0x00000008 +#define GL_SUBGROUP_FEATURE_SHUFFLE_BIT_KHR 0x00000010 +#define GL_SUBGROUP_FEATURE_SHUFFLE_RELATIVE_BIT_KHR 0x00000020 +#define GL_SUBGROUP_FEATURE_CLUSTERED_BIT_KHR 0x00000040 +#define GL_SUBGROUP_FEATURE_QUAD_BIT_KHR 0x00000080 +#endif /* GL_KHR_shader_subgroup */ + +#ifndef GL_KHR_texture_compression_astc_hdr +#define GL_KHR_texture_compression_astc_hdr 1 +#define GL_COMPRESSED_RGBA_ASTC_4x4_KHR 0x93B0 +#define GL_COMPRESSED_RGBA_ASTC_5x4_KHR 0x93B1 +#define GL_COMPRESSED_RGBA_ASTC_5x5_KHR 0x93B2 +#define GL_COMPRESSED_RGBA_ASTC_6x5_KHR 0x93B3 +#define GL_COMPRESSED_RGBA_ASTC_6x6_KHR 0x93B4 +#define GL_COMPRESSED_RGBA_ASTC_8x5_KHR 0x93B5 +#define GL_COMPRESSED_RGBA_ASTC_8x6_KHR 0x93B6 +#define GL_COMPRESSED_RGBA_ASTC_8x8_KHR 0x93B7 +#define GL_COMPRESSED_RGBA_ASTC_10x5_KHR 0x93B8 +#define GL_COMPRESSED_RGBA_ASTC_10x6_KHR 0x93B9 +#define GL_COMPRESSED_RGBA_ASTC_10x8_KHR 0x93BA +#define GL_COMPRESSED_RGBA_ASTC_10x10_KHR 0x93BB +#define GL_COMPRESSED_RGBA_ASTC_12x10_KHR 0x93BC +#define GL_COMPRESSED_RGBA_ASTC_12x12_KHR 0x93BD +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR 0x93D0 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_5x4_KHR 0x93D1 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_5x5_KHR 0x93D2 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_6x5_KHR 0x93D3 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_6x6_KHR 0x93D4 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_8x5_KHR 0x93D5 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_8x6_KHR 0x93D6 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_8x8_KHR 0x93D7 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x5_KHR 0x93D8 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x6_KHR 0x93D9 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x8_KHR 0x93DA +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x10_KHR 0x93DB +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_12x10_KHR 0x93DC +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_12x12_KHR 0x93DD +#endif /* GL_KHR_texture_compression_astc_hdr */ + +#ifndef GL_KHR_texture_compression_astc_ldr +#define GL_KHR_texture_compression_astc_ldr 1 +#endif /* GL_KHR_texture_compression_astc_ldr */ + +#ifndef GL_KHR_texture_compression_astc_sliced_3d +#define GL_KHR_texture_compression_astc_sliced_3d 1 +#endif /* GL_KHR_texture_compression_astc_sliced_3d */ + +#ifndef GL_OES_byte_coordinates +#define GL_OES_byte_coordinates 1 +typedef void (APIENTRYP PFNGLMULTITEXCOORD1BOESPROC) (GLenum texture, GLbyte s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1BVOESPROC) (GLenum texture, const GLbyte *coords); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2BOESPROC) (GLenum texture, GLbyte s, GLbyte t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2BVOESPROC) (GLenum texture, const GLbyte *coords); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3BOESPROC) (GLenum texture, GLbyte s, GLbyte t, GLbyte r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3BVOESPROC) (GLenum texture, const GLbyte *coords); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4BOESPROC) (GLenum texture, GLbyte s, GLbyte t, GLbyte r, GLbyte q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4BVOESPROC) (GLenum texture, const GLbyte *coords); +typedef void (APIENTRYP PFNGLTEXCOORD1BOESPROC) (GLbyte s); +typedef void (APIENTRYP PFNGLTEXCOORD1BVOESPROC) (const GLbyte *coords); +typedef void (APIENTRYP PFNGLTEXCOORD2BOESPROC) (GLbyte s, GLbyte t); +typedef void (APIENTRYP PFNGLTEXCOORD2BVOESPROC) (const GLbyte *coords); +typedef void (APIENTRYP PFNGLTEXCOORD3BOESPROC) (GLbyte s, GLbyte t, GLbyte r); +typedef void (APIENTRYP PFNGLTEXCOORD3BVOESPROC) (const GLbyte *coords); +typedef void (APIENTRYP PFNGLTEXCOORD4BOESPROC) (GLbyte s, GLbyte t, GLbyte r, GLbyte q); +typedef void (APIENTRYP PFNGLTEXCOORD4BVOESPROC) (const GLbyte *coords); +typedef void (APIENTRYP PFNGLVERTEX2BOESPROC) (GLbyte x, GLbyte y); +typedef void (APIENTRYP PFNGLVERTEX2BVOESPROC) (const GLbyte *coords); +typedef void (APIENTRYP PFNGLVERTEX3BOESPROC) (GLbyte x, GLbyte y, GLbyte z); +typedef void (APIENTRYP PFNGLVERTEX3BVOESPROC) (const GLbyte *coords); +typedef void (APIENTRYP PFNGLVERTEX4BOESPROC) (GLbyte x, GLbyte y, GLbyte z, GLbyte w); +typedef void (APIENTRYP PFNGLVERTEX4BVOESPROC) (const GLbyte *coords); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMultiTexCoord1bOES (GLenum texture, GLbyte s); +GLAPI void APIENTRY glMultiTexCoord1bvOES (GLenum texture, const GLbyte *coords); +GLAPI void APIENTRY glMultiTexCoord2bOES (GLenum texture, GLbyte s, GLbyte t); +GLAPI void APIENTRY glMultiTexCoord2bvOES (GLenum texture, const GLbyte *coords); +GLAPI void APIENTRY glMultiTexCoord3bOES (GLenum texture, GLbyte s, GLbyte t, GLbyte r); +GLAPI void APIENTRY glMultiTexCoord3bvOES (GLenum texture, const GLbyte *coords); +GLAPI void APIENTRY glMultiTexCoord4bOES (GLenum texture, GLbyte s, GLbyte t, GLbyte r, GLbyte q); +GLAPI void APIENTRY glMultiTexCoord4bvOES (GLenum texture, const GLbyte *coords); +GLAPI void APIENTRY glTexCoord1bOES (GLbyte s); +GLAPI void APIENTRY glTexCoord1bvOES (const GLbyte *coords); +GLAPI void APIENTRY glTexCoord2bOES (GLbyte s, GLbyte t); +GLAPI void APIENTRY glTexCoord2bvOES (const GLbyte *coords); +GLAPI void APIENTRY glTexCoord3bOES (GLbyte s, GLbyte t, GLbyte r); +GLAPI void APIENTRY glTexCoord3bvOES (const GLbyte *coords); +GLAPI void APIENTRY glTexCoord4bOES (GLbyte s, GLbyte t, GLbyte r, GLbyte q); +GLAPI void APIENTRY glTexCoord4bvOES (const GLbyte *coords); +GLAPI void APIENTRY glVertex2bOES (GLbyte x, GLbyte y); +GLAPI void APIENTRY glVertex2bvOES (const GLbyte *coords); +GLAPI void APIENTRY glVertex3bOES (GLbyte x, GLbyte y, GLbyte z); +GLAPI void APIENTRY glVertex3bvOES (const GLbyte *coords); +GLAPI void APIENTRY glVertex4bOES (GLbyte x, GLbyte y, GLbyte z, GLbyte w); +GLAPI void APIENTRY glVertex4bvOES (const GLbyte *coords); +#endif +#endif /* GL_OES_byte_coordinates */ + +#ifndef GL_OES_compressed_paletted_texture +#define GL_OES_compressed_paletted_texture 1 +#define GL_PALETTE4_RGB8_OES 0x8B90 +#define GL_PALETTE4_RGBA8_OES 0x8B91 +#define GL_PALETTE4_R5_G6_B5_OES 0x8B92 +#define GL_PALETTE4_RGBA4_OES 0x8B93 +#define GL_PALETTE4_RGB5_A1_OES 0x8B94 +#define GL_PALETTE8_RGB8_OES 0x8B95 +#define GL_PALETTE8_RGBA8_OES 0x8B96 +#define GL_PALETTE8_R5_G6_B5_OES 0x8B97 +#define GL_PALETTE8_RGBA4_OES 0x8B98 +#define GL_PALETTE8_RGB5_A1_OES 0x8B99 +#endif /* GL_OES_compressed_paletted_texture */ + +#ifndef GL_OES_fixed_point +#define GL_OES_fixed_point 1 +typedef khronos_int32_t GLfixed; +#define GL_FIXED_OES 0x140C +typedef void (APIENTRYP PFNGLALPHAFUNCXOESPROC) (GLenum func, GLfixed ref); +typedef void (APIENTRYP PFNGLCLEARCOLORXOESPROC) (GLfixed red, GLfixed green, GLfixed blue, GLfixed alpha); +typedef void (APIENTRYP PFNGLCLEARDEPTHXOESPROC) (GLfixed depth); +typedef void (APIENTRYP PFNGLCLIPPLANEXOESPROC) (GLenum plane, const GLfixed *equation); +typedef void (APIENTRYP PFNGLCOLOR4XOESPROC) (GLfixed red, GLfixed green, GLfixed blue, GLfixed alpha); +typedef void (APIENTRYP PFNGLDEPTHRANGEXOESPROC) (GLfixed n, GLfixed f); +typedef void (APIENTRYP PFNGLFOGXOESPROC) (GLenum pname, GLfixed param); +typedef void (APIENTRYP PFNGLFOGXVOESPROC) (GLenum pname, const GLfixed *param); +typedef void (APIENTRYP PFNGLFRUSTUMXOESPROC) (GLfixed l, GLfixed r, GLfixed b, GLfixed t, GLfixed n, GLfixed f); +typedef void (APIENTRYP PFNGLGETCLIPPLANEXOESPROC) (GLenum plane, GLfixed *equation); +typedef void (APIENTRYP PFNGLGETFIXEDVOESPROC) (GLenum pname, GLfixed *params); +typedef void (APIENTRYP PFNGLGETTEXENVXVOESPROC) (GLenum target, GLenum pname, GLfixed *params); +typedef void (APIENTRYP PFNGLGETTEXPARAMETERXVOESPROC) (GLenum target, GLenum pname, GLfixed *params); +typedef void (APIENTRYP PFNGLLIGHTMODELXOESPROC) (GLenum pname, GLfixed param); +typedef void (APIENTRYP PFNGLLIGHTMODELXVOESPROC) (GLenum pname, const GLfixed *param); +typedef void (APIENTRYP PFNGLLIGHTXOESPROC) (GLenum light, GLenum pname, GLfixed param); +typedef void (APIENTRYP PFNGLLIGHTXVOESPROC) (GLenum light, GLenum pname, const GLfixed *params); +typedef void (APIENTRYP PFNGLLINEWIDTHXOESPROC) (GLfixed width); +typedef void (APIENTRYP PFNGLLOADMATRIXXOESPROC) (const GLfixed *m); +typedef void (APIENTRYP PFNGLMATERIALXOESPROC) (GLenum face, GLenum pname, GLfixed param); +typedef void (APIENTRYP PFNGLMATERIALXVOESPROC) (GLenum face, GLenum pname, const GLfixed *param); +typedef void (APIENTRYP PFNGLMULTMATRIXXOESPROC) (const GLfixed *m); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4XOESPROC) (GLenum texture, GLfixed s, GLfixed t, GLfixed r, GLfixed q); +typedef void (APIENTRYP PFNGLNORMAL3XOESPROC) (GLfixed nx, GLfixed ny, GLfixed nz); +typedef void (APIENTRYP PFNGLORTHOXOESPROC) (GLfixed l, GLfixed r, GLfixed b, GLfixed t, GLfixed n, GLfixed f); +typedef void (APIENTRYP PFNGLPOINTPARAMETERXVOESPROC) (GLenum pname, const GLfixed *params); +typedef void (APIENTRYP PFNGLPOINTSIZEXOESPROC) (GLfixed size); +typedef void (APIENTRYP PFNGLPOLYGONOFFSETXOESPROC) (GLfixed factor, GLfixed units); +typedef void (APIENTRYP PFNGLROTATEXOESPROC) (GLfixed angle, GLfixed x, GLfixed y, GLfixed z); +typedef void (APIENTRYP PFNGLSCALEXOESPROC) (GLfixed x, GLfixed y, GLfixed z); +typedef void (APIENTRYP PFNGLTEXENVXOESPROC) (GLenum target, GLenum pname, GLfixed param); +typedef void (APIENTRYP PFNGLTEXENVXVOESPROC) (GLenum target, GLenum pname, const GLfixed *params); +typedef void (APIENTRYP PFNGLTEXPARAMETERXOESPROC) (GLenum target, GLenum pname, GLfixed param); +typedef void (APIENTRYP PFNGLTEXPARAMETERXVOESPROC) (GLenum target, GLenum pname, const GLfixed *params); +typedef void (APIENTRYP PFNGLTRANSLATEXOESPROC) (GLfixed x, GLfixed y, GLfixed z); +typedef void (APIENTRYP PFNGLACCUMXOESPROC) (GLenum op, GLfixed value); +typedef void (APIENTRYP PFNGLBITMAPXOESPROC) (GLsizei width, GLsizei height, GLfixed xorig, GLfixed yorig, GLfixed xmove, GLfixed ymove, const GLubyte *bitmap); +typedef void (APIENTRYP PFNGLBLENDCOLORXOESPROC) (GLfixed red, GLfixed green, GLfixed blue, GLfixed alpha); +typedef void (APIENTRYP PFNGLCLEARACCUMXOESPROC) (GLfixed red, GLfixed green, GLfixed blue, GLfixed alpha); +typedef void (APIENTRYP PFNGLCOLOR3XOESPROC) (GLfixed red, GLfixed green, GLfixed blue); +typedef void (APIENTRYP PFNGLCOLOR3XVOESPROC) (const GLfixed *components); +typedef void (APIENTRYP PFNGLCOLOR4XVOESPROC) (const GLfixed *components); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERXOESPROC) (GLenum target, GLenum pname, GLfixed param); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERXVOESPROC) (GLenum target, GLenum pname, const GLfixed *params); +typedef void (APIENTRYP PFNGLEVALCOORD1XOESPROC) (GLfixed u); +typedef void (APIENTRYP PFNGLEVALCOORD1XVOESPROC) (const GLfixed *coords); +typedef void (APIENTRYP PFNGLEVALCOORD2XOESPROC) (GLfixed u, GLfixed v); +typedef void (APIENTRYP PFNGLEVALCOORD2XVOESPROC) (const GLfixed *coords); +typedef void (APIENTRYP PFNGLFEEDBACKBUFFERXOESPROC) (GLsizei n, GLenum type, const GLfixed *buffer); +typedef void (APIENTRYP PFNGLGETCONVOLUTIONPARAMETERXVOESPROC) (GLenum target, GLenum pname, GLfixed *params); +typedef void (APIENTRYP PFNGLGETHISTOGRAMPARAMETERXVOESPROC) (GLenum target, GLenum pname, GLfixed *params); +typedef void (APIENTRYP PFNGLGETLIGHTXOESPROC) (GLenum light, GLenum pname, GLfixed *params); +typedef void (APIENTRYP PFNGLGETMAPXVOESPROC) (GLenum target, GLenum query, GLfixed *v); +typedef void (APIENTRYP PFNGLGETMATERIALXOESPROC) (GLenum face, GLenum pname, GLfixed param); +typedef void (APIENTRYP PFNGLGETPIXELMAPXVPROC) (GLenum map, GLint size, GLfixed *values); +typedef void (APIENTRYP PFNGLGETTEXGENXVOESPROC) (GLenum coord, GLenum pname, GLfixed *params); +typedef void (APIENTRYP PFNGLGETTEXLEVELPARAMETERXVOESPROC) (GLenum target, GLint level, GLenum pname, GLfixed *params); +typedef void (APIENTRYP PFNGLINDEXXOESPROC) (GLfixed component); +typedef void (APIENTRYP PFNGLINDEXXVOESPROC) (const GLfixed *component); +typedef void (APIENTRYP PFNGLLOADTRANSPOSEMATRIXXOESPROC) (const GLfixed *m); +typedef void (APIENTRYP PFNGLMAP1XOESPROC) (GLenum target, GLfixed u1, GLfixed u2, GLint stride, GLint order, GLfixed points); +typedef void (APIENTRYP PFNGLMAP2XOESPROC) (GLenum target, GLfixed u1, GLfixed u2, GLint ustride, GLint uorder, GLfixed v1, GLfixed v2, GLint vstride, GLint vorder, GLfixed points); +typedef void (APIENTRYP PFNGLMAPGRID1XOESPROC) (GLint n, GLfixed u1, GLfixed u2); +typedef void (APIENTRYP PFNGLMAPGRID2XOESPROC) (GLint n, GLfixed u1, GLfixed u2, GLfixed v1, GLfixed v2); +typedef void (APIENTRYP PFNGLMULTTRANSPOSEMATRIXXOESPROC) (const GLfixed *m); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1XOESPROC) (GLenum texture, GLfixed s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1XVOESPROC) (GLenum texture, const GLfixed *coords); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2XOESPROC) (GLenum texture, GLfixed s, GLfixed t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2XVOESPROC) (GLenum texture, const GLfixed *coords); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3XOESPROC) (GLenum texture, GLfixed s, GLfixed t, GLfixed r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3XVOESPROC) (GLenum texture, const GLfixed *coords); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4XVOESPROC) (GLenum texture, const GLfixed *coords); +typedef void (APIENTRYP PFNGLNORMAL3XVOESPROC) (const GLfixed *coords); +typedef void (APIENTRYP PFNGLPASSTHROUGHXOESPROC) (GLfixed token); +typedef void (APIENTRYP PFNGLPIXELMAPXPROC) (GLenum map, GLint size, const GLfixed *values); +typedef void (APIENTRYP PFNGLPIXELSTOREXPROC) (GLenum pname, GLfixed param); +typedef void (APIENTRYP PFNGLPIXELTRANSFERXOESPROC) (GLenum pname, GLfixed param); +typedef void (APIENTRYP PFNGLPIXELZOOMXOESPROC) (GLfixed xfactor, GLfixed yfactor); +typedef void (APIENTRYP PFNGLPRIORITIZETEXTURESXOESPROC) (GLsizei n, const GLuint *textures, const GLfixed *priorities); +typedef void (APIENTRYP PFNGLRASTERPOS2XOESPROC) (GLfixed x, GLfixed y); +typedef void (APIENTRYP PFNGLRASTERPOS2XVOESPROC) (const GLfixed *coords); +typedef void (APIENTRYP PFNGLRASTERPOS3XOESPROC) (GLfixed x, GLfixed y, GLfixed z); +typedef void (APIENTRYP PFNGLRASTERPOS3XVOESPROC) (const GLfixed *coords); +typedef void (APIENTRYP PFNGLRASTERPOS4XOESPROC) (GLfixed x, GLfixed y, GLfixed z, GLfixed w); +typedef void (APIENTRYP PFNGLRASTERPOS4XVOESPROC) (const GLfixed *coords); +typedef void (APIENTRYP PFNGLRECTXOESPROC) (GLfixed x1, GLfixed y1, GLfixed x2, GLfixed y2); +typedef void (APIENTRYP PFNGLRECTXVOESPROC) (const GLfixed *v1, const GLfixed *v2); +typedef void (APIENTRYP PFNGLTEXCOORD1XOESPROC) (GLfixed s); +typedef void (APIENTRYP PFNGLTEXCOORD1XVOESPROC) (const GLfixed *coords); +typedef void (APIENTRYP PFNGLTEXCOORD2XOESPROC) (GLfixed s, GLfixed t); +typedef void (APIENTRYP PFNGLTEXCOORD2XVOESPROC) (const GLfixed *coords); +typedef void (APIENTRYP PFNGLTEXCOORD3XOESPROC) (GLfixed s, GLfixed t, GLfixed r); +typedef void (APIENTRYP PFNGLTEXCOORD3XVOESPROC) (const GLfixed *coords); +typedef void (APIENTRYP PFNGLTEXCOORD4XOESPROC) (GLfixed s, GLfixed t, GLfixed r, GLfixed q); +typedef void (APIENTRYP PFNGLTEXCOORD4XVOESPROC) (const GLfixed *coords); +typedef void (APIENTRYP PFNGLTEXGENXOESPROC) (GLenum coord, GLenum pname, GLfixed param); +typedef void (APIENTRYP PFNGLTEXGENXVOESPROC) (GLenum coord, GLenum pname, const GLfixed *params); +typedef void (APIENTRYP PFNGLVERTEX2XOESPROC) (GLfixed x); +typedef void (APIENTRYP PFNGLVERTEX2XVOESPROC) (const GLfixed *coords); +typedef void (APIENTRYP PFNGLVERTEX3XOESPROC) (GLfixed x, GLfixed y); +typedef void (APIENTRYP PFNGLVERTEX3XVOESPROC) (const GLfixed *coords); +typedef void (APIENTRYP PFNGLVERTEX4XOESPROC) (GLfixed x, GLfixed y, GLfixed z); +typedef void (APIENTRYP PFNGLVERTEX4XVOESPROC) (const GLfixed *coords); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glAlphaFuncxOES (GLenum func, GLfixed ref); +GLAPI void APIENTRY glClearColorxOES (GLfixed red, GLfixed green, GLfixed blue, GLfixed alpha); +GLAPI void APIENTRY glClearDepthxOES (GLfixed depth); +GLAPI void APIENTRY glClipPlanexOES (GLenum plane, const GLfixed *equation); +GLAPI void APIENTRY glColor4xOES (GLfixed red, GLfixed green, GLfixed blue, GLfixed alpha); +GLAPI void APIENTRY glDepthRangexOES (GLfixed n, GLfixed f); +GLAPI void APIENTRY glFogxOES (GLenum pname, GLfixed param); +GLAPI void APIENTRY glFogxvOES (GLenum pname, const GLfixed *param); +GLAPI void APIENTRY glFrustumxOES (GLfixed l, GLfixed r, GLfixed b, GLfixed t, GLfixed n, GLfixed f); +GLAPI void APIENTRY glGetClipPlanexOES (GLenum plane, GLfixed *equation); +GLAPI void APIENTRY glGetFixedvOES (GLenum pname, GLfixed *params); +GLAPI void APIENTRY glGetTexEnvxvOES (GLenum target, GLenum pname, GLfixed *params); +GLAPI void APIENTRY glGetTexParameterxvOES (GLenum target, GLenum pname, GLfixed *params); +GLAPI void APIENTRY glLightModelxOES (GLenum pname, GLfixed param); +GLAPI void APIENTRY glLightModelxvOES (GLenum pname, const GLfixed *param); +GLAPI void APIENTRY glLightxOES (GLenum light, GLenum pname, GLfixed param); +GLAPI void APIENTRY glLightxvOES (GLenum light, GLenum pname, const GLfixed *params); +GLAPI void APIENTRY glLineWidthxOES (GLfixed width); +GLAPI void APIENTRY glLoadMatrixxOES (const GLfixed *m); +GLAPI void APIENTRY glMaterialxOES (GLenum face, GLenum pname, GLfixed param); +GLAPI void APIENTRY glMaterialxvOES (GLenum face, GLenum pname, const GLfixed *param); +GLAPI void APIENTRY glMultMatrixxOES (const GLfixed *m); +GLAPI void APIENTRY glMultiTexCoord4xOES (GLenum texture, GLfixed s, GLfixed t, GLfixed r, GLfixed q); +GLAPI void APIENTRY glNormal3xOES (GLfixed nx, GLfixed ny, GLfixed nz); +GLAPI void APIENTRY glOrthoxOES (GLfixed l, GLfixed r, GLfixed b, GLfixed t, GLfixed n, GLfixed f); +GLAPI void APIENTRY glPointParameterxvOES (GLenum pname, const GLfixed *params); +GLAPI void APIENTRY glPointSizexOES (GLfixed size); +GLAPI void APIENTRY glPolygonOffsetxOES (GLfixed factor, GLfixed units); +GLAPI void APIENTRY glRotatexOES (GLfixed angle, GLfixed x, GLfixed y, GLfixed z); +GLAPI void APIENTRY glScalexOES (GLfixed x, GLfixed y, GLfixed z); +GLAPI void APIENTRY glTexEnvxOES (GLenum target, GLenum pname, GLfixed param); +GLAPI void APIENTRY glTexEnvxvOES (GLenum target, GLenum pname, const GLfixed *params); +GLAPI void APIENTRY glTexParameterxOES (GLenum target, GLenum pname, GLfixed param); +GLAPI void APIENTRY glTexParameterxvOES (GLenum target, GLenum pname, const GLfixed *params); +GLAPI void APIENTRY glTranslatexOES (GLfixed x, GLfixed y, GLfixed z); +GLAPI void APIENTRY glAccumxOES (GLenum op, GLfixed value); +GLAPI void APIENTRY glBitmapxOES (GLsizei width, GLsizei height, GLfixed xorig, GLfixed yorig, GLfixed xmove, GLfixed ymove, const GLubyte *bitmap); +GLAPI void APIENTRY glBlendColorxOES (GLfixed red, GLfixed green, GLfixed blue, GLfixed alpha); +GLAPI void APIENTRY glClearAccumxOES (GLfixed red, GLfixed green, GLfixed blue, GLfixed alpha); +GLAPI void APIENTRY glColor3xOES (GLfixed red, GLfixed green, GLfixed blue); +GLAPI void APIENTRY glColor3xvOES (const GLfixed *components); +GLAPI void APIENTRY glColor4xvOES (const GLfixed *components); +GLAPI void APIENTRY glConvolutionParameterxOES (GLenum target, GLenum pname, GLfixed param); +GLAPI void APIENTRY glConvolutionParameterxvOES (GLenum target, GLenum pname, const GLfixed *params); +GLAPI void APIENTRY glEvalCoord1xOES (GLfixed u); +GLAPI void APIENTRY glEvalCoord1xvOES (const GLfixed *coords); +GLAPI void APIENTRY glEvalCoord2xOES (GLfixed u, GLfixed v); +GLAPI void APIENTRY glEvalCoord2xvOES (const GLfixed *coords); +GLAPI void APIENTRY glFeedbackBufferxOES (GLsizei n, GLenum type, const GLfixed *buffer); +GLAPI void APIENTRY glGetConvolutionParameterxvOES (GLenum target, GLenum pname, GLfixed *params); +GLAPI void APIENTRY glGetHistogramParameterxvOES (GLenum target, GLenum pname, GLfixed *params); +GLAPI void APIENTRY glGetLightxOES (GLenum light, GLenum pname, GLfixed *params); +GLAPI void APIENTRY glGetMapxvOES (GLenum target, GLenum query, GLfixed *v); +GLAPI void APIENTRY glGetMaterialxOES (GLenum face, GLenum pname, GLfixed param); +GLAPI void APIENTRY glGetPixelMapxv (GLenum map, GLint size, GLfixed *values); +GLAPI void APIENTRY glGetTexGenxvOES (GLenum coord, GLenum pname, GLfixed *params); +GLAPI void APIENTRY glGetTexLevelParameterxvOES (GLenum target, GLint level, GLenum pname, GLfixed *params); +GLAPI void APIENTRY glIndexxOES (GLfixed component); +GLAPI void APIENTRY glIndexxvOES (const GLfixed *component); +GLAPI void APIENTRY glLoadTransposeMatrixxOES (const GLfixed *m); +GLAPI void APIENTRY glMap1xOES (GLenum target, GLfixed u1, GLfixed u2, GLint stride, GLint order, GLfixed points); +GLAPI void APIENTRY glMap2xOES (GLenum target, GLfixed u1, GLfixed u2, GLint ustride, GLint uorder, GLfixed v1, GLfixed v2, GLint vstride, GLint vorder, GLfixed points); +GLAPI void APIENTRY glMapGrid1xOES (GLint n, GLfixed u1, GLfixed u2); +GLAPI void APIENTRY glMapGrid2xOES (GLint n, GLfixed u1, GLfixed u2, GLfixed v1, GLfixed v2); +GLAPI void APIENTRY glMultTransposeMatrixxOES (const GLfixed *m); +GLAPI void APIENTRY glMultiTexCoord1xOES (GLenum texture, GLfixed s); +GLAPI void APIENTRY glMultiTexCoord1xvOES (GLenum texture, const GLfixed *coords); +GLAPI void APIENTRY glMultiTexCoord2xOES (GLenum texture, GLfixed s, GLfixed t); +GLAPI void APIENTRY glMultiTexCoord2xvOES (GLenum texture, const GLfixed *coords); +GLAPI void APIENTRY glMultiTexCoord3xOES (GLenum texture, GLfixed s, GLfixed t, GLfixed r); +GLAPI void APIENTRY glMultiTexCoord3xvOES (GLenum texture, const GLfixed *coords); +GLAPI void APIENTRY glMultiTexCoord4xvOES (GLenum texture, const GLfixed *coords); +GLAPI void APIENTRY glNormal3xvOES (const GLfixed *coords); +GLAPI void APIENTRY glPassThroughxOES (GLfixed token); +GLAPI void APIENTRY glPixelMapx (GLenum map, GLint size, const GLfixed *values); +GLAPI void APIENTRY glPixelStorex (GLenum pname, GLfixed param); +GLAPI void APIENTRY glPixelTransferxOES (GLenum pname, GLfixed param); +GLAPI void APIENTRY glPixelZoomxOES (GLfixed xfactor, GLfixed yfactor); +GLAPI void APIENTRY glPrioritizeTexturesxOES (GLsizei n, const GLuint *textures, const GLfixed *priorities); +GLAPI void APIENTRY glRasterPos2xOES (GLfixed x, GLfixed y); +GLAPI void APIENTRY glRasterPos2xvOES (const GLfixed *coords); +GLAPI void APIENTRY glRasterPos3xOES (GLfixed x, GLfixed y, GLfixed z); +GLAPI void APIENTRY glRasterPos3xvOES (const GLfixed *coords); +GLAPI void APIENTRY glRasterPos4xOES (GLfixed x, GLfixed y, GLfixed z, GLfixed w); +GLAPI void APIENTRY glRasterPos4xvOES (const GLfixed *coords); +GLAPI void APIENTRY glRectxOES (GLfixed x1, GLfixed y1, GLfixed x2, GLfixed y2); +GLAPI void APIENTRY glRectxvOES (const GLfixed *v1, const GLfixed *v2); +GLAPI void APIENTRY glTexCoord1xOES (GLfixed s); +GLAPI void APIENTRY glTexCoord1xvOES (const GLfixed *coords); +GLAPI void APIENTRY glTexCoord2xOES (GLfixed s, GLfixed t); +GLAPI void APIENTRY glTexCoord2xvOES (const GLfixed *coords); +GLAPI void APIENTRY glTexCoord3xOES (GLfixed s, GLfixed t, GLfixed r); +GLAPI void APIENTRY glTexCoord3xvOES (const GLfixed *coords); +GLAPI void APIENTRY glTexCoord4xOES (GLfixed s, GLfixed t, GLfixed r, GLfixed q); +GLAPI void APIENTRY glTexCoord4xvOES (const GLfixed *coords); +GLAPI void APIENTRY glTexGenxOES (GLenum coord, GLenum pname, GLfixed param); +GLAPI void APIENTRY glTexGenxvOES (GLenum coord, GLenum pname, const GLfixed *params); +GLAPI void APIENTRY glVertex2xOES (GLfixed x); +GLAPI void APIENTRY glVertex2xvOES (const GLfixed *coords); +GLAPI void APIENTRY glVertex3xOES (GLfixed x, GLfixed y); +GLAPI void APIENTRY glVertex3xvOES (const GLfixed *coords); +GLAPI void APIENTRY glVertex4xOES (GLfixed x, GLfixed y, GLfixed z); +GLAPI void APIENTRY glVertex4xvOES (const GLfixed *coords); +#endif +#endif /* GL_OES_fixed_point */ + +#ifndef GL_OES_query_matrix +#define GL_OES_query_matrix 1 +typedef GLbitfield (APIENTRYP PFNGLQUERYMATRIXXOESPROC) (GLfixed *mantissa, GLint *exponent); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLbitfield APIENTRY glQueryMatrixxOES (GLfixed *mantissa, GLint *exponent); +#endif +#endif /* GL_OES_query_matrix */ + +#ifndef GL_OES_read_format +#define GL_OES_read_format 1 +#define GL_IMPLEMENTATION_COLOR_READ_TYPE_OES 0x8B9A +#define GL_IMPLEMENTATION_COLOR_READ_FORMAT_OES 0x8B9B +#endif /* GL_OES_read_format */ + +#ifndef GL_OES_single_precision +#define GL_OES_single_precision 1 +typedef void (APIENTRYP PFNGLCLEARDEPTHFOESPROC) (GLclampf depth); +typedef void (APIENTRYP PFNGLCLIPPLANEFOESPROC) (GLenum plane, const GLfloat *equation); +typedef void (APIENTRYP PFNGLDEPTHRANGEFOESPROC) (GLclampf n, GLclampf f); +typedef void (APIENTRYP PFNGLFRUSTUMFOESPROC) (GLfloat l, GLfloat r, GLfloat b, GLfloat t, GLfloat n, GLfloat f); +typedef void (APIENTRYP PFNGLGETCLIPPLANEFOESPROC) (GLenum plane, GLfloat *equation); +typedef void (APIENTRYP PFNGLORTHOFOESPROC) (GLfloat l, GLfloat r, GLfloat b, GLfloat t, GLfloat n, GLfloat f); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glClearDepthfOES (GLclampf depth); +GLAPI void APIENTRY glClipPlanefOES (GLenum plane, const GLfloat *equation); +GLAPI void APIENTRY glDepthRangefOES (GLclampf n, GLclampf f); +GLAPI void APIENTRY glFrustumfOES (GLfloat l, GLfloat r, GLfloat b, GLfloat t, GLfloat n, GLfloat f); +GLAPI void APIENTRY glGetClipPlanefOES (GLenum plane, GLfloat *equation); +GLAPI void APIENTRY glOrthofOES (GLfloat l, GLfloat r, GLfloat b, GLfloat t, GLfloat n, GLfloat f); +#endif +#endif /* GL_OES_single_precision */ + +#ifndef GL_3DFX_multisample +#define GL_3DFX_multisample 1 +#define GL_MULTISAMPLE_3DFX 0x86B2 +#define GL_SAMPLE_BUFFERS_3DFX 0x86B3 +#define GL_SAMPLES_3DFX 0x86B4 +#define GL_MULTISAMPLE_BIT_3DFX 0x20000000 +#endif /* GL_3DFX_multisample */ + +#ifndef GL_3DFX_tbuffer +#define GL_3DFX_tbuffer 1 +typedef void (APIENTRYP PFNGLTBUFFERMASK3DFXPROC) (GLuint mask); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTbufferMask3DFX (GLuint mask); +#endif +#endif /* GL_3DFX_tbuffer */ + +#ifndef GL_3DFX_texture_compression_FXT1 +#define GL_3DFX_texture_compression_FXT1 1 +#define GL_COMPRESSED_RGB_FXT1_3DFX 0x86B0 +#define GL_COMPRESSED_RGBA_FXT1_3DFX 0x86B1 +#endif /* GL_3DFX_texture_compression_FXT1 */ + +#ifndef GL_AMD_blend_minmax_factor +#define GL_AMD_blend_minmax_factor 1 +#define GL_FACTOR_MIN_AMD 0x901C +#define GL_FACTOR_MAX_AMD 0x901D +#endif /* GL_AMD_blend_minmax_factor */ + +#ifndef GL_AMD_conservative_depth +#define GL_AMD_conservative_depth 1 +#endif /* GL_AMD_conservative_depth */ + +#ifndef GL_AMD_debug_output +#define GL_AMD_debug_output 1 +typedef void (APIENTRY *GLDEBUGPROCAMD)(GLuint id,GLenum category,GLenum severity,GLsizei length,const GLchar *message,void *userParam); +#define GL_MAX_DEBUG_MESSAGE_LENGTH_AMD 0x9143 +#define GL_MAX_DEBUG_LOGGED_MESSAGES_AMD 0x9144 +#define GL_DEBUG_LOGGED_MESSAGES_AMD 0x9145 +#define GL_DEBUG_SEVERITY_HIGH_AMD 0x9146 +#define GL_DEBUG_SEVERITY_MEDIUM_AMD 0x9147 +#define GL_DEBUG_SEVERITY_LOW_AMD 0x9148 +#define GL_DEBUG_CATEGORY_API_ERROR_AMD 0x9149 +#define GL_DEBUG_CATEGORY_WINDOW_SYSTEM_AMD 0x914A +#define GL_DEBUG_CATEGORY_DEPRECATION_AMD 0x914B +#define GL_DEBUG_CATEGORY_UNDEFINED_BEHAVIOR_AMD 0x914C +#define GL_DEBUG_CATEGORY_PERFORMANCE_AMD 0x914D +#define GL_DEBUG_CATEGORY_SHADER_COMPILER_AMD 0x914E +#define GL_DEBUG_CATEGORY_APPLICATION_AMD 0x914F +#define GL_DEBUG_CATEGORY_OTHER_AMD 0x9150 +typedef void (APIENTRYP PFNGLDEBUGMESSAGEENABLEAMDPROC) (GLenum category, GLenum severity, GLsizei count, const GLuint *ids, GLboolean enabled); +typedef void (APIENTRYP PFNGLDEBUGMESSAGEINSERTAMDPROC) (GLenum category, GLenum severity, GLuint id, GLsizei length, const GLchar *buf); +typedef void (APIENTRYP PFNGLDEBUGMESSAGECALLBACKAMDPROC) (GLDEBUGPROCAMD callback, void *userParam); +typedef GLuint (APIENTRYP PFNGLGETDEBUGMESSAGELOGAMDPROC) (GLuint count, GLsizei bufSize, GLenum *categories, GLuint *severities, GLuint *ids, GLsizei *lengths, GLchar *message); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDebugMessageEnableAMD (GLenum category, GLenum severity, GLsizei count, const GLuint *ids, GLboolean enabled); +GLAPI void APIENTRY glDebugMessageInsertAMD (GLenum category, GLenum severity, GLuint id, GLsizei length, const GLchar *buf); +GLAPI void APIENTRY glDebugMessageCallbackAMD (GLDEBUGPROCAMD callback, void *userParam); +GLAPI GLuint APIENTRY glGetDebugMessageLogAMD (GLuint count, GLsizei bufSize, GLenum *categories, GLuint *severities, GLuint *ids, GLsizei *lengths, GLchar *message); +#endif +#endif /* GL_AMD_debug_output */ + +#ifndef GL_AMD_depth_clamp_separate +#define GL_AMD_depth_clamp_separate 1 +#define GL_DEPTH_CLAMP_NEAR_AMD 0x901E +#define GL_DEPTH_CLAMP_FAR_AMD 0x901F +#endif /* GL_AMD_depth_clamp_separate */ + +#ifndef GL_AMD_draw_buffers_blend +#define GL_AMD_draw_buffers_blend 1 +typedef void (APIENTRYP PFNGLBLENDFUNCINDEXEDAMDPROC) (GLuint buf, GLenum src, GLenum dst); +typedef void (APIENTRYP PFNGLBLENDFUNCSEPARATEINDEXEDAMDPROC) (GLuint buf, GLenum srcRGB, GLenum dstRGB, GLenum srcAlpha, GLenum dstAlpha); +typedef void (APIENTRYP PFNGLBLENDEQUATIONINDEXEDAMDPROC) (GLuint buf, GLenum mode); +typedef void (APIENTRYP PFNGLBLENDEQUATIONSEPARATEINDEXEDAMDPROC) (GLuint buf, GLenum modeRGB, GLenum modeAlpha); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendFuncIndexedAMD (GLuint buf, GLenum src, GLenum dst); +GLAPI void APIENTRY glBlendFuncSeparateIndexedAMD (GLuint buf, GLenum srcRGB, GLenum dstRGB, GLenum srcAlpha, GLenum dstAlpha); +GLAPI void APIENTRY glBlendEquationIndexedAMD (GLuint buf, GLenum mode); +GLAPI void APIENTRY glBlendEquationSeparateIndexedAMD (GLuint buf, GLenum modeRGB, GLenum modeAlpha); +#endif +#endif /* GL_AMD_draw_buffers_blend */ + +#ifndef GL_AMD_framebuffer_multisample_advanced +#define GL_AMD_framebuffer_multisample_advanced 1 +#define GL_RENDERBUFFER_STORAGE_SAMPLES_AMD 0x91B2 +#define GL_MAX_COLOR_FRAMEBUFFER_SAMPLES_AMD 0x91B3 +#define GL_MAX_COLOR_FRAMEBUFFER_STORAGE_SAMPLES_AMD 0x91B4 +#define GL_MAX_DEPTH_STENCIL_FRAMEBUFFER_SAMPLES_AMD 0x91B5 +#define GL_NUM_SUPPORTED_MULTISAMPLE_MODES_AMD 0x91B6 +#define GL_SUPPORTED_MULTISAMPLE_MODES_AMD 0x91B7 +typedef void (APIENTRYP PFNGLRENDERBUFFERSTORAGEMULTISAMPLEADVANCEDAMDPROC) (GLenum target, GLsizei samples, GLsizei storageSamples, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLNAMEDRENDERBUFFERSTORAGEMULTISAMPLEADVANCEDAMDPROC) (GLuint renderbuffer, GLsizei samples, GLsizei storageSamples, GLenum internalformat, GLsizei width, GLsizei height); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glRenderbufferStorageMultisampleAdvancedAMD (GLenum target, GLsizei samples, GLsizei storageSamples, GLenum internalformat, GLsizei width, GLsizei height); +GLAPI void APIENTRY glNamedRenderbufferStorageMultisampleAdvancedAMD (GLuint renderbuffer, GLsizei samples, GLsizei storageSamples, GLenum internalformat, GLsizei width, GLsizei height); +#endif +#endif /* GL_AMD_framebuffer_multisample_advanced */ + +#ifndef GL_AMD_framebuffer_sample_positions +#define GL_AMD_framebuffer_sample_positions 1 +#define GL_SUBSAMPLE_DISTANCE_AMD 0x883F +#define GL_PIXELS_PER_SAMPLE_PATTERN_X_AMD 0x91AE +#define GL_PIXELS_PER_SAMPLE_PATTERN_Y_AMD 0x91AF +#define GL_ALL_PIXELS_AMD 0xFFFFFFFF +typedef void (APIENTRYP PFNGLFRAMEBUFFERSAMPLEPOSITIONSFVAMDPROC) (GLenum target, GLuint numsamples, GLuint pixelindex, const GLfloat *values); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERSAMPLEPOSITIONSFVAMDPROC) (GLuint framebuffer, GLuint numsamples, GLuint pixelindex, const GLfloat *values); +typedef void (APIENTRYP PFNGLGETFRAMEBUFFERPARAMETERFVAMDPROC) (GLenum target, GLenum pname, GLuint numsamples, GLuint pixelindex, GLsizei size, GLfloat *values); +typedef void (APIENTRYP PFNGLGETNAMEDFRAMEBUFFERPARAMETERFVAMDPROC) (GLuint framebuffer, GLenum pname, GLuint numsamples, GLuint pixelindex, GLsizei size, GLfloat *values); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFramebufferSamplePositionsfvAMD (GLenum target, GLuint numsamples, GLuint pixelindex, const GLfloat *values); +GLAPI void APIENTRY glNamedFramebufferSamplePositionsfvAMD (GLuint framebuffer, GLuint numsamples, GLuint pixelindex, const GLfloat *values); +GLAPI void APIENTRY glGetFramebufferParameterfvAMD (GLenum target, GLenum pname, GLuint numsamples, GLuint pixelindex, GLsizei size, GLfloat *values); +GLAPI void APIENTRY glGetNamedFramebufferParameterfvAMD (GLuint framebuffer, GLenum pname, GLuint numsamples, GLuint pixelindex, GLsizei size, GLfloat *values); +#endif +#endif /* GL_AMD_framebuffer_sample_positions */ + +#ifndef GL_AMD_gcn_shader +#define GL_AMD_gcn_shader 1 +#endif /* GL_AMD_gcn_shader */ + +#ifndef GL_AMD_gpu_shader_half_float +#define GL_AMD_gpu_shader_half_float 1 +#define GL_FLOAT16_NV 0x8FF8 +#define GL_FLOAT16_VEC2_NV 0x8FF9 +#define GL_FLOAT16_VEC3_NV 0x8FFA +#define GL_FLOAT16_VEC4_NV 0x8FFB +#define GL_FLOAT16_MAT2_AMD 0x91C5 +#define GL_FLOAT16_MAT3_AMD 0x91C6 +#define GL_FLOAT16_MAT4_AMD 0x91C7 +#define GL_FLOAT16_MAT2x3_AMD 0x91C8 +#define GL_FLOAT16_MAT2x4_AMD 0x91C9 +#define GL_FLOAT16_MAT3x2_AMD 0x91CA +#define GL_FLOAT16_MAT3x4_AMD 0x91CB +#define GL_FLOAT16_MAT4x2_AMD 0x91CC +#define GL_FLOAT16_MAT4x3_AMD 0x91CD +#endif /* GL_AMD_gpu_shader_half_float */ + +#ifndef GL_AMD_gpu_shader_int16 +#define GL_AMD_gpu_shader_int16 1 +#endif /* GL_AMD_gpu_shader_int16 */ + +#ifndef GL_AMD_gpu_shader_int64 +#define GL_AMD_gpu_shader_int64 1 +typedef khronos_int64_t GLint64EXT; +#define GL_INT64_NV 0x140E +#define GL_UNSIGNED_INT64_NV 0x140F +#define GL_INT8_NV 0x8FE0 +#define GL_INT8_VEC2_NV 0x8FE1 +#define GL_INT8_VEC3_NV 0x8FE2 +#define GL_INT8_VEC4_NV 0x8FE3 +#define GL_INT16_NV 0x8FE4 +#define GL_INT16_VEC2_NV 0x8FE5 +#define GL_INT16_VEC3_NV 0x8FE6 +#define GL_INT16_VEC4_NV 0x8FE7 +#define GL_INT64_VEC2_NV 0x8FE9 +#define GL_INT64_VEC3_NV 0x8FEA +#define GL_INT64_VEC4_NV 0x8FEB +#define GL_UNSIGNED_INT8_NV 0x8FEC +#define GL_UNSIGNED_INT8_VEC2_NV 0x8FED +#define GL_UNSIGNED_INT8_VEC3_NV 0x8FEE +#define GL_UNSIGNED_INT8_VEC4_NV 0x8FEF +#define GL_UNSIGNED_INT16_NV 0x8FF0 +#define GL_UNSIGNED_INT16_VEC2_NV 0x8FF1 +#define GL_UNSIGNED_INT16_VEC3_NV 0x8FF2 +#define GL_UNSIGNED_INT16_VEC4_NV 0x8FF3 +#define GL_UNSIGNED_INT64_VEC2_NV 0x8FF5 +#define GL_UNSIGNED_INT64_VEC3_NV 0x8FF6 +#define GL_UNSIGNED_INT64_VEC4_NV 0x8FF7 +typedef void (APIENTRYP PFNGLUNIFORM1I64NVPROC) (GLint location, GLint64EXT x); +typedef void (APIENTRYP PFNGLUNIFORM2I64NVPROC) (GLint location, GLint64EXT x, GLint64EXT y); +typedef void (APIENTRYP PFNGLUNIFORM3I64NVPROC) (GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z); +typedef void (APIENTRYP PFNGLUNIFORM4I64NVPROC) (GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z, GLint64EXT w); +typedef void (APIENTRYP PFNGLUNIFORM1I64VNVPROC) (GLint location, GLsizei count, const GLint64EXT *value); +typedef void (APIENTRYP PFNGLUNIFORM2I64VNVPROC) (GLint location, GLsizei count, const GLint64EXT *value); +typedef void (APIENTRYP PFNGLUNIFORM3I64VNVPROC) (GLint location, GLsizei count, const GLint64EXT *value); +typedef void (APIENTRYP PFNGLUNIFORM4I64VNVPROC) (GLint location, GLsizei count, const GLint64EXT *value); +typedef void (APIENTRYP PFNGLUNIFORM1UI64NVPROC) (GLint location, GLuint64EXT x); +typedef void (APIENTRYP PFNGLUNIFORM2UI64NVPROC) (GLint location, GLuint64EXT x, GLuint64EXT y); +typedef void (APIENTRYP PFNGLUNIFORM3UI64NVPROC) (GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z); +typedef void (APIENTRYP PFNGLUNIFORM4UI64NVPROC) (GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z, GLuint64EXT w); +typedef void (APIENTRYP PFNGLUNIFORM1UI64VNVPROC) (GLint location, GLsizei count, const GLuint64EXT *value); +typedef void (APIENTRYP PFNGLUNIFORM2UI64VNVPROC) (GLint location, GLsizei count, const GLuint64EXT *value); +typedef void (APIENTRYP PFNGLUNIFORM3UI64VNVPROC) (GLint location, GLsizei count, const GLuint64EXT *value); +typedef void (APIENTRYP PFNGLUNIFORM4UI64VNVPROC) (GLint location, GLsizei count, const GLuint64EXT *value); +typedef void (APIENTRYP PFNGLGETUNIFORMI64VNVPROC) (GLuint program, GLint location, GLint64EXT *params); +typedef void (APIENTRYP PFNGLGETUNIFORMUI64VNVPROC) (GLuint program, GLint location, GLuint64EXT *params); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1I64NVPROC) (GLuint program, GLint location, GLint64EXT x); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2I64NVPROC) (GLuint program, GLint location, GLint64EXT x, GLint64EXT y); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3I64NVPROC) (GLuint program, GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4I64NVPROC) (GLuint program, GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z, GLint64EXT w); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1I64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2I64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3I64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4I64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1UI64NVPROC) (GLuint program, GLint location, GLuint64EXT x); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2UI64NVPROC) (GLuint program, GLint location, GLuint64EXT x, GLuint64EXT y); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3UI64NVPROC) (GLuint program, GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4UI64NVPROC) (GLuint program, GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z, GLuint64EXT w); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1UI64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2UI64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3UI64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4UI64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glUniform1i64NV (GLint location, GLint64EXT x); +GLAPI void APIENTRY glUniform2i64NV (GLint location, GLint64EXT x, GLint64EXT y); +GLAPI void APIENTRY glUniform3i64NV (GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z); +GLAPI void APIENTRY glUniform4i64NV (GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z, GLint64EXT w); +GLAPI void APIENTRY glUniform1i64vNV (GLint location, GLsizei count, const GLint64EXT *value); +GLAPI void APIENTRY glUniform2i64vNV (GLint location, GLsizei count, const GLint64EXT *value); +GLAPI void APIENTRY glUniform3i64vNV (GLint location, GLsizei count, const GLint64EXT *value); +GLAPI void APIENTRY glUniform4i64vNV (GLint location, GLsizei count, const GLint64EXT *value); +GLAPI void APIENTRY glUniform1ui64NV (GLint location, GLuint64EXT x); +GLAPI void APIENTRY glUniform2ui64NV (GLint location, GLuint64EXT x, GLuint64EXT y); +GLAPI void APIENTRY glUniform3ui64NV (GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z); +GLAPI void APIENTRY glUniform4ui64NV (GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z, GLuint64EXT w); +GLAPI void APIENTRY glUniform1ui64vNV (GLint location, GLsizei count, const GLuint64EXT *value); +GLAPI void APIENTRY glUniform2ui64vNV (GLint location, GLsizei count, const GLuint64EXT *value); +GLAPI void APIENTRY glUniform3ui64vNV (GLint location, GLsizei count, const GLuint64EXT *value); +GLAPI void APIENTRY glUniform4ui64vNV (GLint location, GLsizei count, const GLuint64EXT *value); +GLAPI void APIENTRY glGetUniformi64vNV (GLuint program, GLint location, GLint64EXT *params); +GLAPI void APIENTRY glGetUniformui64vNV (GLuint program, GLint location, GLuint64EXT *params); +GLAPI void APIENTRY glProgramUniform1i64NV (GLuint program, GLint location, GLint64EXT x); +GLAPI void APIENTRY glProgramUniform2i64NV (GLuint program, GLint location, GLint64EXT x, GLint64EXT y); +GLAPI void APIENTRY glProgramUniform3i64NV (GLuint program, GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z); +GLAPI void APIENTRY glProgramUniform4i64NV (GLuint program, GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z, GLint64EXT w); +GLAPI void APIENTRY glProgramUniform1i64vNV (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +GLAPI void APIENTRY glProgramUniform2i64vNV (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +GLAPI void APIENTRY glProgramUniform3i64vNV (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +GLAPI void APIENTRY glProgramUniform4i64vNV (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +GLAPI void APIENTRY glProgramUniform1ui64NV (GLuint program, GLint location, GLuint64EXT x); +GLAPI void APIENTRY glProgramUniform2ui64NV (GLuint program, GLint location, GLuint64EXT x, GLuint64EXT y); +GLAPI void APIENTRY glProgramUniform3ui64NV (GLuint program, GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z); +GLAPI void APIENTRY glProgramUniform4ui64NV (GLuint program, GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z, GLuint64EXT w); +GLAPI void APIENTRY glProgramUniform1ui64vNV (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +GLAPI void APIENTRY glProgramUniform2ui64vNV (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +GLAPI void APIENTRY glProgramUniform3ui64vNV (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +GLAPI void APIENTRY glProgramUniform4ui64vNV (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +#endif +#endif /* GL_AMD_gpu_shader_int64 */ + +#ifndef GL_AMD_interleaved_elements +#define GL_AMD_interleaved_elements 1 +#define GL_VERTEX_ELEMENT_SWIZZLE_AMD 0x91A4 +#define GL_VERTEX_ID_SWIZZLE_AMD 0x91A5 +typedef void (APIENTRYP PFNGLVERTEXATTRIBPARAMETERIAMDPROC) (GLuint index, GLenum pname, GLint param); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexAttribParameteriAMD (GLuint index, GLenum pname, GLint param); +#endif +#endif /* GL_AMD_interleaved_elements */ + +#ifndef GL_AMD_multi_draw_indirect +#define GL_AMD_multi_draw_indirect 1 +typedef void (APIENTRYP PFNGLMULTIDRAWARRAYSINDIRECTAMDPROC) (GLenum mode, const void *indirect, GLsizei primcount, GLsizei stride); +typedef void (APIENTRYP PFNGLMULTIDRAWELEMENTSINDIRECTAMDPROC) (GLenum mode, GLenum type, const void *indirect, GLsizei primcount, GLsizei stride); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMultiDrawArraysIndirectAMD (GLenum mode, const void *indirect, GLsizei primcount, GLsizei stride); +GLAPI void APIENTRY glMultiDrawElementsIndirectAMD (GLenum mode, GLenum type, const void *indirect, GLsizei primcount, GLsizei stride); +#endif +#endif /* GL_AMD_multi_draw_indirect */ + +#ifndef GL_AMD_name_gen_delete +#define GL_AMD_name_gen_delete 1 +#define GL_DATA_BUFFER_AMD 0x9151 +#define GL_PERFORMANCE_MONITOR_AMD 0x9152 +#define GL_QUERY_OBJECT_AMD 0x9153 +#define GL_VERTEX_ARRAY_OBJECT_AMD 0x9154 +#define GL_SAMPLER_OBJECT_AMD 0x9155 +typedef void (APIENTRYP PFNGLGENNAMESAMDPROC) (GLenum identifier, GLuint num, GLuint *names); +typedef void (APIENTRYP PFNGLDELETENAMESAMDPROC) (GLenum identifier, GLuint num, const GLuint *names); +typedef GLboolean (APIENTRYP PFNGLISNAMEAMDPROC) (GLenum identifier, GLuint name); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGenNamesAMD (GLenum identifier, GLuint num, GLuint *names); +GLAPI void APIENTRY glDeleteNamesAMD (GLenum identifier, GLuint num, const GLuint *names); +GLAPI GLboolean APIENTRY glIsNameAMD (GLenum identifier, GLuint name); +#endif +#endif /* GL_AMD_name_gen_delete */ + +#ifndef GL_AMD_occlusion_query_event +#define GL_AMD_occlusion_query_event 1 +#define GL_OCCLUSION_QUERY_EVENT_MASK_AMD 0x874F +#define GL_QUERY_DEPTH_PASS_EVENT_BIT_AMD 0x00000001 +#define GL_QUERY_DEPTH_FAIL_EVENT_BIT_AMD 0x00000002 +#define GL_QUERY_STENCIL_FAIL_EVENT_BIT_AMD 0x00000004 +#define GL_QUERY_DEPTH_BOUNDS_FAIL_EVENT_BIT_AMD 0x00000008 +#define GL_QUERY_ALL_EVENT_BITS_AMD 0xFFFFFFFF +typedef void (APIENTRYP PFNGLQUERYOBJECTPARAMETERUIAMDPROC) (GLenum target, GLuint id, GLenum pname, GLuint param); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glQueryObjectParameteruiAMD (GLenum target, GLuint id, GLenum pname, GLuint param); +#endif +#endif /* GL_AMD_occlusion_query_event */ + +#ifndef GL_AMD_performance_monitor +#define GL_AMD_performance_monitor 1 +#define GL_COUNTER_TYPE_AMD 0x8BC0 +#define GL_COUNTER_RANGE_AMD 0x8BC1 +#define GL_UNSIGNED_INT64_AMD 0x8BC2 +#define GL_PERCENTAGE_AMD 0x8BC3 +#define GL_PERFMON_RESULT_AVAILABLE_AMD 0x8BC4 +#define GL_PERFMON_RESULT_SIZE_AMD 0x8BC5 +#define GL_PERFMON_RESULT_AMD 0x8BC6 +typedef void (APIENTRYP PFNGLGETPERFMONITORGROUPSAMDPROC) (GLint *numGroups, GLsizei groupsSize, GLuint *groups); +typedef void (APIENTRYP PFNGLGETPERFMONITORCOUNTERSAMDPROC) (GLuint group, GLint *numCounters, GLint *maxActiveCounters, GLsizei counterSize, GLuint *counters); +typedef void (APIENTRYP PFNGLGETPERFMONITORGROUPSTRINGAMDPROC) (GLuint group, GLsizei bufSize, GLsizei *length, GLchar *groupString); +typedef void (APIENTRYP PFNGLGETPERFMONITORCOUNTERSTRINGAMDPROC) (GLuint group, GLuint counter, GLsizei bufSize, GLsizei *length, GLchar *counterString); +typedef void (APIENTRYP PFNGLGETPERFMONITORCOUNTERINFOAMDPROC) (GLuint group, GLuint counter, GLenum pname, void *data); +typedef void (APIENTRYP PFNGLGENPERFMONITORSAMDPROC) (GLsizei n, GLuint *monitors); +typedef void (APIENTRYP PFNGLDELETEPERFMONITORSAMDPROC) (GLsizei n, GLuint *monitors); +typedef void (APIENTRYP PFNGLSELECTPERFMONITORCOUNTERSAMDPROC) (GLuint monitor, GLboolean enable, GLuint group, GLint numCounters, GLuint *counterList); +typedef void (APIENTRYP PFNGLBEGINPERFMONITORAMDPROC) (GLuint monitor); +typedef void (APIENTRYP PFNGLENDPERFMONITORAMDPROC) (GLuint monitor); +typedef void (APIENTRYP PFNGLGETPERFMONITORCOUNTERDATAAMDPROC) (GLuint monitor, GLenum pname, GLsizei dataSize, GLuint *data, GLint *bytesWritten); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGetPerfMonitorGroupsAMD (GLint *numGroups, GLsizei groupsSize, GLuint *groups); +GLAPI void APIENTRY glGetPerfMonitorCountersAMD (GLuint group, GLint *numCounters, GLint *maxActiveCounters, GLsizei counterSize, GLuint *counters); +GLAPI void APIENTRY glGetPerfMonitorGroupStringAMD (GLuint group, GLsizei bufSize, GLsizei *length, GLchar *groupString); +GLAPI void APIENTRY glGetPerfMonitorCounterStringAMD (GLuint group, GLuint counter, GLsizei bufSize, GLsizei *length, GLchar *counterString); +GLAPI void APIENTRY glGetPerfMonitorCounterInfoAMD (GLuint group, GLuint counter, GLenum pname, void *data); +GLAPI void APIENTRY glGenPerfMonitorsAMD (GLsizei n, GLuint *monitors); +GLAPI void APIENTRY glDeletePerfMonitorsAMD (GLsizei n, GLuint *monitors); +GLAPI void APIENTRY glSelectPerfMonitorCountersAMD (GLuint monitor, GLboolean enable, GLuint group, GLint numCounters, GLuint *counterList); +GLAPI void APIENTRY glBeginPerfMonitorAMD (GLuint monitor); +GLAPI void APIENTRY glEndPerfMonitorAMD (GLuint monitor); +GLAPI void APIENTRY glGetPerfMonitorCounterDataAMD (GLuint monitor, GLenum pname, GLsizei dataSize, GLuint *data, GLint *bytesWritten); +#endif +#endif /* GL_AMD_performance_monitor */ + +#ifndef GL_AMD_pinned_memory +#define GL_AMD_pinned_memory 1 +#define GL_EXTERNAL_VIRTUAL_MEMORY_BUFFER_AMD 0x9160 +#endif /* GL_AMD_pinned_memory */ + +#ifndef GL_AMD_query_buffer_object +#define GL_AMD_query_buffer_object 1 +#define GL_QUERY_BUFFER_AMD 0x9192 +#define GL_QUERY_BUFFER_BINDING_AMD 0x9193 +#define GL_QUERY_RESULT_NO_WAIT_AMD 0x9194 +#endif /* GL_AMD_query_buffer_object */ + +#ifndef GL_AMD_sample_positions +#define GL_AMD_sample_positions 1 +typedef void (APIENTRYP PFNGLSETMULTISAMPLEFVAMDPROC) (GLenum pname, GLuint index, const GLfloat *val); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSetMultisamplefvAMD (GLenum pname, GLuint index, const GLfloat *val); +#endif +#endif /* GL_AMD_sample_positions */ + +#ifndef GL_AMD_seamless_cubemap_per_texture +#define GL_AMD_seamless_cubemap_per_texture 1 +#endif /* GL_AMD_seamless_cubemap_per_texture */ + +#ifndef GL_AMD_shader_atomic_counter_ops +#define GL_AMD_shader_atomic_counter_ops 1 +#endif /* GL_AMD_shader_atomic_counter_ops */ + +#ifndef GL_AMD_shader_ballot +#define GL_AMD_shader_ballot 1 +#endif /* GL_AMD_shader_ballot */ + +#ifndef GL_AMD_shader_explicit_vertex_parameter +#define GL_AMD_shader_explicit_vertex_parameter 1 +#endif /* GL_AMD_shader_explicit_vertex_parameter */ + +#ifndef GL_AMD_shader_gpu_shader_half_float_fetch +#define GL_AMD_shader_gpu_shader_half_float_fetch 1 +#endif /* GL_AMD_shader_gpu_shader_half_float_fetch */ + +#ifndef GL_AMD_shader_image_load_store_lod +#define GL_AMD_shader_image_load_store_lod 1 +#endif /* GL_AMD_shader_image_load_store_lod */ + +#ifndef GL_AMD_shader_stencil_export +#define GL_AMD_shader_stencil_export 1 +#endif /* GL_AMD_shader_stencil_export */ + +#ifndef GL_AMD_shader_trinary_minmax +#define GL_AMD_shader_trinary_minmax 1 +#endif /* GL_AMD_shader_trinary_minmax */ + +#ifndef GL_AMD_sparse_texture +#define GL_AMD_sparse_texture 1 +#define GL_VIRTUAL_PAGE_SIZE_X_AMD 0x9195 +#define GL_VIRTUAL_PAGE_SIZE_Y_AMD 0x9196 +#define GL_VIRTUAL_PAGE_SIZE_Z_AMD 0x9197 +#define GL_MAX_SPARSE_TEXTURE_SIZE_AMD 0x9198 +#define GL_MAX_SPARSE_3D_TEXTURE_SIZE_AMD 0x9199 +#define GL_MAX_SPARSE_ARRAY_TEXTURE_LAYERS 0x919A +#define GL_MIN_SPARSE_LEVEL_AMD 0x919B +#define GL_MIN_LOD_WARNING_AMD 0x919C +#define GL_TEXTURE_STORAGE_SPARSE_BIT_AMD 0x00000001 +typedef void (APIENTRYP PFNGLTEXSTORAGESPARSEAMDPROC) (GLenum target, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLsizei layers, GLbitfield flags); +typedef void (APIENTRYP PFNGLTEXTURESTORAGESPARSEAMDPROC) (GLuint texture, GLenum target, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLsizei layers, GLbitfield flags); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTexStorageSparseAMD (GLenum target, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLsizei layers, GLbitfield flags); +GLAPI void APIENTRY glTextureStorageSparseAMD (GLuint texture, GLenum target, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLsizei layers, GLbitfield flags); +#endif +#endif /* GL_AMD_sparse_texture */ + +#ifndef GL_AMD_stencil_operation_extended +#define GL_AMD_stencil_operation_extended 1 +#define GL_SET_AMD 0x874A +#define GL_REPLACE_VALUE_AMD 0x874B +#define GL_STENCIL_OP_VALUE_AMD 0x874C +#define GL_STENCIL_BACK_OP_VALUE_AMD 0x874D +typedef void (APIENTRYP PFNGLSTENCILOPVALUEAMDPROC) (GLenum face, GLuint value); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glStencilOpValueAMD (GLenum face, GLuint value); +#endif +#endif /* GL_AMD_stencil_operation_extended */ + +#ifndef GL_AMD_texture_gather_bias_lod +#define GL_AMD_texture_gather_bias_lod 1 +#endif /* GL_AMD_texture_gather_bias_lod */ + +#ifndef GL_AMD_texture_texture4 +#define GL_AMD_texture_texture4 1 +#endif /* GL_AMD_texture_texture4 */ + +#ifndef GL_AMD_transform_feedback3_lines_triangles +#define GL_AMD_transform_feedback3_lines_triangles 1 +#endif /* GL_AMD_transform_feedback3_lines_triangles */ + +#ifndef GL_AMD_transform_feedback4 +#define GL_AMD_transform_feedback4 1 +#define GL_STREAM_RASTERIZATION_AMD 0x91A0 +#endif /* GL_AMD_transform_feedback4 */ + +#ifndef GL_AMD_vertex_shader_layer +#define GL_AMD_vertex_shader_layer 1 +#endif /* GL_AMD_vertex_shader_layer */ + +#ifndef GL_AMD_vertex_shader_tessellator +#define GL_AMD_vertex_shader_tessellator 1 +#define GL_SAMPLER_BUFFER_AMD 0x9001 +#define GL_INT_SAMPLER_BUFFER_AMD 0x9002 +#define GL_UNSIGNED_INT_SAMPLER_BUFFER_AMD 0x9003 +#define GL_TESSELLATION_MODE_AMD 0x9004 +#define GL_TESSELLATION_FACTOR_AMD 0x9005 +#define GL_DISCRETE_AMD 0x9006 +#define GL_CONTINUOUS_AMD 0x9007 +typedef void (APIENTRYP PFNGLTESSELLATIONFACTORAMDPROC) (GLfloat factor); +typedef void (APIENTRYP PFNGLTESSELLATIONMODEAMDPROC) (GLenum mode); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTessellationFactorAMD (GLfloat factor); +GLAPI void APIENTRY glTessellationModeAMD (GLenum mode); +#endif +#endif /* GL_AMD_vertex_shader_tessellator */ + +#ifndef GL_AMD_vertex_shader_viewport_index +#define GL_AMD_vertex_shader_viewport_index 1 +#endif /* GL_AMD_vertex_shader_viewport_index */ + +#ifndef GL_APPLE_aux_depth_stencil +#define GL_APPLE_aux_depth_stencil 1 +#define GL_AUX_DEPTH_STENCIL_APPLE 0x8A14 +#endif /* GL_APPLE_aux_depth_stencil */ + +#ifndef GL_APPLE_client_storage +#define GL_APPLE_client_storage 1 +#define GL_UNPACK_CLIENT_STORAGE_APPLE 0x85B2 +#endif /* GL_APPLE_client_storage */ + +#ifndef GL_APPLE_element_array +#define GL_APPLE_element_array 1 +#define GL_ELEMENT_ARRAY_APPLE 0x8A0C +#define GL_ELEMENT_ARRAY_TYPE_APPLE 0x8A0D +#define GL_ELEMENT_ARRAY_POINTER_APPLE 0x8A0E +typedef void (APIENTRYP PFNGLELEMENTPOINTERAPPLEPROC) (GLenum type, const void *pointer); +typedef void (APIENTRYP PFNGLDRAWELEMENTARRAYAPPLEPROC) (GLenum mode, GLint first, GLsizei count); +typedef void (APIENTRYP PFNGLDRAWRANGEELEMENTARRAYAPPLEPROC) (GLenum mode, GLuint start, GLuint end, GLint first, GLsizei count); +typedef void (APIENTRYP PFNGLMULTIDRAWELEMENTARRAYAPPLEPROC) (GLenum mode, const GLint *first, const GLsizei *count, GLsizei primcount); +typedef void (APIENTRYP PFNGLMULTIDRAWRANGEELEMENTARRAYAPPLEPROC) (GLenum mode, GLuint start, GLuint end, const GLint *first, const GLsizei *count, GLsizei primcount); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glElementPointerAPPLE (GLenum type, const void *pointer); +GLAPI void APIENTRY glDrawElementArrayAPPLE (GLenum mode, GLint first, GLsizei count); +GLAPI void APIENTRY glDrawRangeElementArrayAPPLE (GLenum mode, GLuint start, GLuint end, GLint first, GLsizei count); +GLAPI void APIENTRY glMultiDrawElementArrayAPPLE (GLenum mode, const GLint *first, const GLsizei *count, GLsizei primcount); +GLAPI void APIENTRY glMultiDrawRangeElementArrayAPPLE (GLenum mode, GLuint start, GLuint end, const GLint *first, const GLsizei *count, GLsizei primcount); +#endif +#endif /* GL_APPLE_element_array */ + +#ifndef GL_APPLE_fence +#define GL_APPLE_fence 1 +#define GL_DRAW_PIXELS_APPLE 0x8A0A +#define GL_FENCE_APPLE 0x8A0B +typedef void (APIENTRYP PFNGLGENFENCESAPPLEPROC) (GLsizei n, GLuint *fences); +typedef void (APIENTRYP PFNGLDELETEFENCESAPPLEPROC) (GLsizei n, const GLuint *fences); +typedef void (APIENTRYP PFNGLSETFENCEAPPLEPROC) (GLuint fence); +typedef GLboolean (APIENTRYP PFNGLISFENCEAPPLEPROC) (GLuint fence); +typedef GLboolean (APIENTRYP PFNGLTESTFENCEAPPLEPROC) (GLuint fence); +typedef void (APIENTRYP PFNGLFINISHFENCEAPPLEPROC) (GLuint fence); +typedef GLboolean (APIENTRYP PFNGLTESTOBJECTAPPLEPROC) (GLenum object, GLuint name); +typedef void (APIENTRYP PFNGLFINISHOBJECTAPPLEPROC) (GLenum object, GLint name); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGenFencesAPPLE (GLsizei n, GLuint *fences); +GLAPI void APIENTRY glDeleteFencesAPPLE (GLsizei n, const GLuint *fences); +GLAPI void APIENTRY glSetFenceAPPLE (GLuint fence); +GLAPI GLboolean APIENTRY glIsFenceAPPLE (GLuint fence); +GLAPI GLboolean APIENTRY glTestFenceAPPLE (GLuint fence); +GLAPI void APIENTRY glFinishFenceAPPLE (GLuint fence); +GLAPI GLboolean APIENTRY glTestObjectAPPLE (GLenum object, GLuint name); +GLAPI void APIENTRY glFinishObjectAPPLE (GLenum object, GLint name); +#endif +#endif /* GL_APPLE_fence */ + +#ifndef GL_APPLE_float_pixels +#define GL_APPLE_float_pixels 1 +#define GL_HALF_APPLE 0x140B +#define GL_RGBA_FLOAT32_APPLE 0x8814 +#define GL_RGB_FLOAT32_APPLE 0x8815 +#define GL_ALPHA_FLOAT32_APPLE 0x8816 +#define GL_INTENSITY_FLOAT32_APPLE 0x8817 +#define GL_LUMINANCE_FLOAT32_APPLE 0x8818 +#define GL_LUMINANCE_ALPHA_FLOAT32_APPLE 0x8819 +#define GL_RGBA_FLOAT16_APPLE 0x881A +#define GL_RGB_FLOAT16_APPLE 0x881B +#define GL_ALPHA_FLOAT16_APPLE 0x881C +#define GL_INTENSITY_FLOAT16_APPLE 0x881D +#define GL_LUMINANCE_FLOAT16_APPLE 0x881E +#define GL_LUMINANCE_ALPHA_FLOAT16_APPLE 0x881F +#define GL_COLOR_FLOAT_APPLE 0x8A0F +#endif /* GL_APPLE_float_pixels */ + +#ifndef GL_APPLE_flush_buffer_range +#define GL_APPLE_flush_buffer_range 1 +#define GL_BUFFER_SERIALIZED_MODIFY_APPLE 0x8A12 +#define GL_BUFFER_FLUSHING_UNMAP_APPLE 0x8A13 +typedef void (APIENTRYP PFNGLBUFFERPARAMETERIAPPLEPROC) (GLenum target, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLFLUSHMAPPEDBUFFERRANGEAPPLEPROC) (GLenum target, GLintptr offset, GLsizeiptr size); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBufferParameteriAPPLE (GLenum target, GLenum pname, GLint param); +GLAPI void APIENTRY glFlushMappedBufferRangeAPPLE (GLenum target, GLintptr offset, GLsizeiptr size); +#endif +#endif /* GL_APPLE_flush_buffer_range */ + +#ifndef GL_APPLE_object_purgeable +#define GL_APPLE_object_purgeable 1 +#define GL_BUFFER_OBJECT_APPLE 0x85B3 +#define GL_RELEASED_APPLE 0x8A19 +#define GL_VOLATILE_APPLE 0x8A1A +#define GL_RETAINED_APPLE 0x8A1B +#define GL_UNDEFINED_APPLE 0x8A1C +#define GL_PURGEABLE_APPLE 0x8A1D +typedef GLenum (APIENTRYP PFNGLOBJECTPURGEABLEAPPLEPROC) (GLenum objectType, GLuint name, GLenum option); +typedef GLenum (APIENTRYP PFNGLOBJECTUNPURGEABLEAPPLEPROC) (GLenum objectType, GLuint name, GLenum option); +typedef void (APIENTRYP PFNGLGETOBJECTPARAMETERIVAPPLEPROC) (GLenum objectType, GLuint name, GLenum pname, GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLenum APIENTRY glObjectPurgeableAPPLE (GLenum objectType, GLuint name, GLenum option); +GLAPI GLenum APIENTRY glObjectUnpurgeableAPPLE (GLenum objectType, GLuint name, GLenum option); +GLAPI void APIENTRY glGetObjectParameterivAPPLE (GLenum objectType, GLuint name, GLenum pname, GLint *params); +#endif +#endif /* GL_APPLE_object_purgeable */ + +#ifndef GL_APPLE_rgb_422 +#define GL_APPLE_rgb_422 1 +#define GL_RGB_422_APPLE 0x8A1F +#define GL_UNSIGNED_SHORT_8_8_APPLE 0x85BA +#define GL_UNSIGNED_SHORT_8_8_REV_APPLE 0x85BB +#define GL_RGB_RAW_422_APPLE 0x8A51 +#endif /* GL_APPLE_rgb_422 */ + +#ifndef GL_APPLE_row_bytes +#define GL_APPLE_row_bytes 1 +#define GL_PACK_ROW_BYTES_APPLE 0x8A15 +#define GL_UNPACK_ROW_BYTES_APPLE 0x8A16 +#endif /* GL_APPLE_row_bytes */ + +#ifndef GL_APPLE_specular_vector +#define GL_APPLE_specular_vector 1 +#define GL_LIGHT_MODEL_SPECULAR_VECTOR_APPLE 0x85B0 +#endif /* GL_APPLE_specular_vector */ + +#ifndef GL_APPLE_texture_range +#define GL_APPLE_texture_range 1 +#define GL_TEXTURE_RANGE_LENGTH_APPLE 0x85B7 +#define GL_TEXTURE_RANGE_POINTER_APPLE 0x85B8 +#define GL_TEXTURE_STORAGE_HINT_APPLE 0x85BC +#define GL_STORAGE_PRIVATE_APPLE 0x85BD +#define GL_STORAGE_CACHED_APPLE 0x85BE +#define GL_STORAGE_SHARED_APPLE 0x85BF +typedef void (APIENTRYP PFNGLTEXTURERANGEAPPLEPROC) (GLenum target, GLsizei length, const void *pointer); +typedef void (APIENTRYP PFNGLGETTEXPARAMETERPOINTERVAPPLEPROC) (GLenum target, GLenum pname, void **params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTextureRangeAPPLE (GLenum target, GLsizei length, const void *pointer); +GLAPI void APIENTRY glGetTexParameterPointervAPPLE (GLenum target, GLenum pname, void **params); +#endif +#endif /* GL_APPLE_texture_range */ + +#ifndef GL_APPLE_transform_hint +#define GL_APPLE_transform_hint 1 +#define GL_TRANSFORM_HINT_APPLE 0x85B1 +#endif /* GL_APPLE_transform_hint */ + +#ifndef GL_APPLE_vertex_array_object +#define GL_APPLE_vertex_array_object 1 +#define GL_VERTEX_ARRAY_BINDING_APPLE 0x85B5 +typedef void (APIENTRYP PFNGLBINDVERTEXARRAYAPPLEPROC) (GLuint array); +typedef void (APIENTRYP PFNGLDELETEVERTEXARRAYSAPPLEPROC) (GLsizei n, const GLuint *arrays); +typedef void (APIENTRYP PFNGLGENVERTEXARRAYSAPPLEPROC) (GLsizei n, GLuint *arrays); +typedef GLboolean (APIENTRYP PFNGLISVERTEXARRAYAPPLEPROC) (GLuint array); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBindVertexArrayAPPLE (GLuint array); +GLAPI void APIENTRY glDeleteVertexArraysAPPLE (GLsizei n, const GLuint *arrays); +GLAPI void APIENTRY glGenVertexArraysAPPLE (GLsizei n, GLuint *arrays); +GLAPI GLboolean APIENTRY glIsVertexArrayAPPLE (GLuint array); +#endif +#endif /* GL_APPLE_vertex_array_object */ + +#ifndef GL_APPLE_vertex_array_range +#define GL_APPLE_vertex_array_range 1 +#define GL_VERTEX_ARRAY_RANGE_APPLE 0x851D +#define GL_VERTEX_ARRAY_RANGE_LENGTH_APPLE 0x851E +#define GL_VERTEX_ARRAY_STORAGE_HINT_APPLE 0x851F +#define GL_VERTEX_ARRAY_RANGE_POINTER_APPLE 0x8521 +#define GL_STORAGE_CLIENT_APPLE 0x85B4 +typedef void (APIENTRYP PFNGLVERTEXARRAYRANGEAPPLEPROC) (GLsizei length, void *pointer); +typedef void (APIENTRYP PFNGLFLUSHVERTEXARRAYRANGEAPPLEPROC) (GLsizei length, void *pointer); +typedef void (APIENTRYP PFNGLVERTEXARRAYPARAMETERIAPPLEPROC) (GLenum pname, GLint param); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexArrayRangeAPPLE (GLsizei length, void *pointer); +GLAPI void APIENTRY glFlushVertexArrayRangeAPPLE (GLsizei length, void *pointer); +GLAPI void APIENTRY glVertexArrayParameteriAPPLE (GLenum pname, GLint param); +#endif +#endif /* GL_APPLE_vertex_array_range */ + +#ifndef GL_APPLE_vertex_program_evaluators +#define GL_APPLE_vertex_program_evaluators 1 +#define GL_VERTEX_ATTRIB_MAP1_APPLE 0x8A00 +#define GL_VERTEX_ATTRIB_MAP2_APPLE 0x8A01 +#define GL_VERTEX_ATTRIB_MAP1_SIZE_APPLE 0x8A02 +#define GL_VERTEX_ATTRIB_MAP1_COEFF_APPLE 0x8A03 +#define GL_VERTEX_ATTRIB_MAP1_ORDER_APPLE 0x8A04 +#define GL_VERTEX_ATTRIB_MAP1_DOMAIN_APPLE 0x8A05 +#define GL_VERTEX_ATTRIB_MAP2_SIZE_APPLE 0x8A06 +#define GL_VERTEX_ATTRIB_MAP2_COEFF_APPLE 0x8A07 +#define GL_VERTEX_ATTRIB_MAP2_ORDER_APPLE 0x8A08 +#define GL_VERTEX_ATTRIB_MAP2_DOMAIN_APPLE 0x8A09 +typedef void (APIENTRYP PFNGLENABLEVERTEXATTRIBAPPLEPROC) (GLuint index, GLenum pname); +typedef void (APIENTRYP PFNGLDISABLEVERTEXATTRIBAPPLEPROC) (GLuint index, GLenum pname); +typedef GLboolean (APIENTRYP PFNGLISVERTEXATTRIBENABLEDAPPLEPROC) (GLuint index, GLenum pname); +typedef void (APIENTRYP PFNGLMAPVERTEXATTRIB1DAPPLEPROC) (GLuint index, GLuint size, GLdouble u1, GLdouble u2, GLint stride, GLint order, const GLdouble *points); +typedef void (APIENTRYP PFNGLMAPVERTEXATTRIB1FAPPLEPROC) (GLuint index, GLuint size, GLfloat u1, GLfloat u2, GLint stride, GLint order, const GLfloat *points); +typedef void (APIENTRYP PFNGLMAPVERTEXATTRIB2DAPPLEPROC) (GLuint index, GLuint size, GLdouble u1, GLdouble u2, GLint ustride, GLint uorder, GLdouble v1, GLdouble v2, GLint vstride, GLint vorder, const GLdouble *points); +typedef void (APIENTRYP PFNGLMAPVERTEXATTRIB2FAPPLEPROC) (GLuint index, GLuint size, GLfloat u1, GLfloat u2, GLint ustride, GLint uorder, GLfloat v1, GLfloat v2, GLint vstride, GLint vorder, const GLfloat *points); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glEnableVertexAttribAPPLE (GLuint index, GLenum pname); +GLAPI void APIENTRY glDisableVertexAttribAPPLE (GLuint index, GLenum pname); +GLAPI GLboolean APIENTRY glIsVertexAttribEnabledAPPLE (GLuint index, GLenum pname); +GLAPI void APIENTRY glMapVertexAttrib1dAPPLE (GLuint index, GLuint size, GLdouble u1, GLdouble u2, GLint stride, GLint order, const GLdouble *points); +GLAPI void APIENTRY glMapVertexAttrib1fAPPLE (GLuint index, GLuint size, GLfloat u1, GLfloat u2, GLint stride, GLint order, const GLfloat *points); +GLAPI void APIENTRY glMapVertexAttrib2dAPPLE (GLuint index, GLuint size, GLdouble u1, GLdouble u2, GLint ustride, GLint uorder, GLdouble v1, GLdouble v2, GLint vstride, GLint vorder, const GLdouble *points); +GLAPI void APIENTRY glMapVertexAttrib2fAPPLE (GLuint index, GLuint size, GLfloat u1, GLfloat u2, GLint ustride, GLint uorder, GLfloat v1, GLfloat v2, GLint vstride, GLint vorder, const GLfloat *points); +#endif +#endif /* GL_APPLE_vertex_program_evaluators */ + +#ifndef GL_APPLE_ycbcr_422 +#define GL_APPLE_ycbcr_422 1 +#define GL_YCBCR_422_APPLE 0x85B9 +#endif /* GL_APPLE_ycbcr_422 */ + +#ifndef GL_ATI_draw_buffers +#define GL_ATI_draw_buffers 1 +#define GL_MAX_DRAW_BUFFERS_ATI 0x8824 +#define GL_DRAW_BUFFER0_ATI 0x8825 +#define GL_DRAW_BUFFER1_ATI 0x8826 +#define GL_DRAW_BUFFER2_ATI 0x8827 +#define GL_DRAW_BUFFER3_ATI 0x8828 +#define GL_DRAW_BUFFER4_ATI 0x8829 +#define GL_DRAW_BUFFER5_ATI 0x882A +#define GL_DRAW_BUFFER6_ATI 0x882B +#define GL_DRAW_BUFFER7_ATI 0x882C +#define GL_DRAW_BUFFER8_ATI 0x882D +#define GL_DRAW_BUFFER9_ATI 0x882E +#define GL_DRAW_BUFFER10_ATI 0x882F +#define GL_DRAW_BUFFER11_ATI 0x8830 +#define GL_DRAW_BUFFER12_ATI 0x8831 +#define GL_DRAW_BUFFER13_ATI 0x8832 +#define GL_DRAW_BUFFER14_ATI 0x8833 +#define GL_DRAW_BUFFER15_ATI 0x8834 +typedef void (APIENTRYP PFNGLDRAWBUFFERSATIPROC) (GLsizei n, const GLenum *bufs); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawBuffersATI (GLsizei n, const GLenum *bufs); +#endif +#endif /* GL_ATI_draw_buffers */ + +#ifndef GL_ATI_element_array +#define GL_ATI_element_array 1 +#define GL_ELEMENT_ARRAY_ATI 0x8768 +#define GL_ELEMENT_ARRAY_TYPE_ATI 0x8769 +#define GL_ELEMENT_ARRAY_POINTER_ATI 0x876A +typedef void (APIENTRYP PFNGLELEMENTPOINTERATIPROC) (GLenum type, const void *pointer); +typedef void (APIENTRYP PFNGLDRAWELEMENTARRAYATIPROC) (GLenum mode, GLsizei count); +typedef void (APIENTRYP PFNGLDRAWRANGEELEMENTARRAYATIPROC) (GLenum mode, GLuint start, GLuint end, GLsizei count); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glElementPointerATI (GLenum type, const void *pointer); +GLAPI void APIENTRY glDrawElementArrayATI (GLenum mode, GLsizei count); +GLAPI void APIENTRY glDrawRangeElementArrayATI (GLenum mode, GLuint start, GLuint end, GLsizei count); +#endif +#endif /* GL_ATI_element_array */ + +#ifndef GL_ATI_envmap_bumpmap +#define GL_ATI_envmap_bumpmap 1 +#define GL_BUMP_ROT_MATRIX_ATI 0x8775 +#define GL_BUMP_ROT_MATRIX_SIZE_ATI 0x8776 +#define GL_BUMP_NUM_TEX_UNITS_ATI 0x8777 +#define GL_BUMP_TEX_UNITS_ATI 0x8778 +#define GL_DUDV_ATI 0x8779 +#define GL_DU8DV8_ATI 0x877A +#define GL_BUMP_ENVMAP_ATI 0x877B +#define GL_BUMP_TARGET_ATI 0x877C +typedef void (APIENTRYP PFNGLTEXBUMPPARAMETERIVATIPROC) (GLenum pname, const GLint *param); +typedef void (APIENTRYP PFNGLTEXBUMPPARAMETERFVATIPROC) (GLenum pname, const GLfloat *param); +typedef void (APIENTRYP PFNGLGETTEXBUMPPARAMETERIVATIPROC) (GLenum pname, GLint *param); +typedef void (APIENTRYP PFNGLGETTEXBUMPPARAMETERFVATIPROC) (GLenum pname, GLfloat *param); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTexBumpParameterivATI (GLenum pname, const GLint *param); +GLAPI void APIENTRY glTexBumpParameterfvATI (GLenum pname, const GLfloat *param); +GLAPI void APIENTRY glGetTexBumpParameterivATI (GLenum pname, GLint *param); +GLAPI void APIENTRY glGetTexBumpParameterfvATI (GLenum pname, GLfloat *param); +#endif +#endif /* GL_ATI_envmap_bumpmap */ + +#ifndef GL_ATI_fragment_shader +#define GL_ATI_fragment_shader 1 +#define GL_FRAGMENT_SHADER_ATI 0x8920 +#define GL_REG_0_ATI 0x8921 +#define GL_REG_1_ATI 0x8922 +#define GL_REG_2_ATI 0x8923 +#define GL_REG_3_ATI 0x8924 +#define GL_REG_4_ATI 0x8925 +#define GL_REG_5_ATI 0x8926 +#define GL_REG_6_ATI 0x8927 +#define GL_REG_7_ATI 0x8928 +#define GL_REG_8_ATI 0x8929 +#define GL_REG_9_ATI 0x892A +#define GL_REG_10_ATI 0x892B +#define GL_REG_11_ATI 0x892C +#define GL_REG_12_ATI 0x892D +#define GL_REG_13_ATI 0x892E +#define GL_REG_14_ATI 0x892F +#define GL_REG_15_ATI 0x8930 +#define GL_REG_16_ATI 0x8931 +#define GL_REG_17_ATI 0x8932 +#define GL_REG_18_ATI 0x8933 +#define GL_REG_19_ATI 0x8934 +#define GL_REG_20_ATI 0x8935 +#define GL_REG_21_ATI 0x8936 +#define GL_REG_22_ATI 0x8937 +#define GL_REG_23_ATI 0x8938 +#define GL_REG_24_ATI 0x8939 +#define GL_REG_25_ATI 0x893A +#define GL_REG_26_ATI 0x893B +#define GL_REG_27_ATI 0x893C +#define GL_REG_28_ATI 0x893D +#define GL_REG_29_ATI 0x893E +#define GL_REG_30_ATI 0x893F +#define GL_REG_31_ATI 0x8940 +#define GL_CON_0_ATI 0x8941 +#define GL_CON_1_ATI 0x8942 +#define GL_CON_2_ATI 0x8943 +#define GL_CON_3_ATI 0x8944 +#define GL_CON_4_ATI 0x8945 +#define GL_CON_5_ATI 0x8946 +#define GL_CON_6_ATI 0x8947 +#define GL_CON_7_ATI 0x8948 +#define GL_CON_8_ATI 0x8949 +#define GL_CON_9_ATI 0x894A +#define GL_CON_10_ATI 0x894B +#define GL_CON_11_ATI 0x894C +#define GL_CON_12_ATI 0x894D +#define GL_CON_13_ATI 0x894E +#define GL_CON_14_ATI 0x894F +#define GL_CON_15_ATI 0x8950 +#define GL_CON_16_ATI 0x8951 +#define GL_CON_17_ATI 0x8952 +#define GL_CON_18_ATI 0x8953 +#define GL_CON_19_ATI 0x8954 +#define GL_CON_20_ATI 0x8955 +#define GL_CON_21_ATI 0x8956 +#define GL_CON_22_ATI 0x8957 +#define GL_CON_23_ATI 0x8958 +#define GL_CON_24_ATI 0x8959 +#define GL_CON_25_ATI 0x895A +#define GL_CON_26_ATI 0x895B +#define GL_CON_27_ATI 0x895C +#define GL_CON_28_ATI 0x895D +#define GL_CON_29_ATI 0x895E +#define GL_CON_30_ATI 0x895F +#define GL_CON_31_ATI 0x8960 +#define GL_MOV_ATI 0x8961 +#define GL_ADD_ATI 0x8963 +#define GL_MUL_ATI 0x8964 +#define GL_SUB_ATI 0x8965 +#define GL_DOT3_ATI 0x8966 +#define GL_DOT4_ATI 0x8967 +#define GL_MAD_ATI 0x8968 +#define GL_LERP_ATI 0x8969 +#define GL_CND_ATI 0x896A +#define GL_CND0_ATI 0x896B +#define GL_DOT2_ADD_ATI 0x896C +#define GL_SECONDARY_INTERPOLATOR_ATI 0x896D +#define GL_NUM_FRAGMENT_REGISTERS_ATI 0x896E +#define GL_NUM_FRAGMENT_CONSTANTS_ATI 0x896F +#define GL_NUM_PASSES_ATI 0x8970 +#define GL_NUM_INSTRUCTIONS_PER_PASS_ATI 0x8971 +#define GL_NUM_INSTRUCTIONS_TOTAL_ATI 0x8972 +#define GL_NUM_INPUT_INTERPOLATOR_COMPONENTS_ATI 0x8973 +#define GL_NUM_LOOPBACK_COMPONENTS_ATI 0x8974 +#define GL_COLOR_ALPHA_PAIRING_ATI 0x8975 +#define GL_SWIZZLE_STR_ATI 0x8976 +#define GL_SWIZZLE_STQ_ATI 0x8977 +#define GL_SWIZZLE_STR_DR_ATI 0x8978 +#define GL_SWIZZLE_STQ_DQ_ATI 0x8979 +#define GL_SWIZZLE_STRQ_ATI 0x897A +#define GL_SWIZZLE_STRQ_DQ_ATI 0x897B +#define GL_RED_BIT_ATI 0x00000001 +#define GL_GREEN_BIT_ATI 0x00000002 +#define GL_BLUE_BIT_ATI 0x00000004 +#define GL_2X_BIT_ATI 0x00000001 +#define GL_4X_BIT_ATI 0x00000002 +#define GL_8X_BIT_ATI 0x00000004 +#define GL_HALF_BIT_ATI 0x00000008 +#define GL_QUARTER_BIT_ATI 0x00000010 +#define GL_EIGHTH_BIT_ATI 0x00000020 +#define GL_SATURATE_BIT_ATI 0x00000040 +#define GL_COMP_BIT_ATI 0x00000002 +#define GL_NEGATE_BIT_ATI 0x00000004 +#define GL_BIAS_BIT_ATI 0x00000008 +typedef GLuint (APIENTRYP PFNGLGENFRAGMENTSHADERSATIPROC) (GLuint range); +typedef void (APIENTRYP PFNGLBINDFRAGMENTSHADERATIPROC) (GLuint id); +typedef void (APIENTRYP PFNGLDELETEFRAGMENTSHADERATIPROC) (GLuint id); +typedef void (APIENTRYP PFNGLBEGINFRAGMENTSHADERATIPROC) (void); +typedef void (APIENTRYP PFNGLENDFRAGMENTSHADERATIPROC) (void); +typedef void (APIENTRYP PFNGLPASSTEXCOORDATIPROC) (GLuint dst, GLuint coord, GLenum swizzle); +typedef void (APIENTRYP PFNGLSAMPLEMAPATIPROC) (GLuint dst, GLuint interp, GLenum swizzle); +typedef void (APIENTRYP PFNGLCOLORFRAGMENTOP1ATIPROC) (GLenum op, GLuint dst, GLuint dstMask, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod); +typedef void (APIENTRYP PFNGLCOLORFRAGMENTOP2ATIPROC) (GLenum op, GLuint dst, GLuint dstMask, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod, GLuint arg2, GLuint arg2Rep, GLuint arg2Mod); +typedef void (APIENTRYP PFNGLCOLORFRAGMENTOP3ATIPROC) (GLenum op, GLuint dst, GLuint dstMask, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod, GLuint arg2, GLuint arg2Rep, GLuint arg2Mod, GLuint arg3, GLuint arg3Rep, GLuint arg3Mod); +typedef void (APIENTRYP PFNGLALPHAFRAGMENTOP1ATIPROC) (GLenum op, GLuint dst, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod); +typedef void (APIENTRYP PFNGLALPHAFRAGMENTOP2ATIPROC) (GLenum op, GLuint dst, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod, GLuint arg2, GLuint arg2Rep, GLuint arg2Mod); +typedef void (APIENTRYP PFNGLALPHAFRAGMENTOP3ATIPROC) (GLenum op, GLuint dst, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod, GLuint arg2, GLuint arg2Rep, GLuint arg2Mod, GLuint arg3, GLuint arg3Rep, GLuint arg3Mod); +typedef void (APIENTRYP PFNGLSETFRAGMENTSHADERCONSTANTATIPROC) (GLuint dst, const GLfloat *value); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLuint APIENTRY glGenFragmentShadersATI (GLuint range); +GLAPI void APIENTRY glBindFragmentShaderATI (GLuint id); +GLAPI void APIENTRY glDeleteFragmentShaderATI (GLuint id); +GLAPI void APIENTRY glBeginFragmentShaderATI (void); +GLAPI void APIENTRY glEndFragmentShaderATI (void); +GLAPI void APIENTRY glPassTexCoordATI (GLuint dst, GLuint coord, GLenum swizzle); +GLAPI void APIENTRY glSampleMapATI (GLuint dst, GLuint interp, GLenum swizzle); +GLAPI void APIENTRY glColorFragmentOp1ATI (GLenum op, GLuint dst, GLuint dstMask, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod); +GLAPI void APIENTRY glColorFragmentOp2ATI (GLenum op, GLuint dst, GLuint dstMask, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod, GLuint arg2, GLuint arg2Rep, GLuint arg2Mod); +GLAPI void APIENTRY glColorFragmentOp3ATI (GLenum op, GLuint dst, GLuint dstMask, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod, GLuint arg2, GLuint arg2Rep, GLuint arg2Mod, GLuint arg3, GLuint arg3Rep, GLuint arg3Mod); +GLAPI void APIENTRY glAlphaFragmentOp1ATI (GLenum op, GLuint dst, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod); +GLAPI void APIENTRY glAlphaFragmentOp2ATI (GLenum op, GLuint dst, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod, GLuint arg2, GLuint arg2Rep, GLuint arg2Mod); +GLAPI void APIENTRY glAlphaFragmentOp3ATI (GLenum op, GLuint dst, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod, GLuint arg2, GLuint arg2Rep, GLuint arg2Mod, GLuint arg3, GLuint arg3Rep, GLuint arg3Mod); +GLAPI void APIENTRY glSetFragmentShaderConstantATI (GLuint dst, const GLfloat *value); +#endif +#endif /* GL_ATI_fragment_shader */ + +#ifndef GL_ATI_map_object_buffer +#define GL_ATI_map_object_buffer 1 +typedef void *(APIENTRYP PFNGLMAPOBJECTBUFFERATIPROC) (GLuint buffer); +typedef void (APIENTRYP PFNGLUNMAPOBJECTBUFFERATIPROC) (GLuint buffer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void *APIENTRY glMapObjectBufferATI (GLuint buffer); +GLAPI void APIENTRY glUnmapObjectBufferATI (GLuint buffer); +#endif +#endif /* GL_ATI_map_object_buffer */ + +#ifndef GL_ATI_meminfo +#define GL_ATI_meminfo 1 +#define GL_VBO_FREE_MEMORY_ATI 0x87FB +#define GL_TEXTURE_FREE_MEMORY_ATI 0x87FC +#define GL_RENDERBUFFER_FREE_MEMORY_ATI 0x87FD +#endif /* GL_ATI_meminfo */ + +#ifndef GL_ATI_pixel_format_float +#define GL_ATI_pixel_format_float 1 +#define GL_RGBA_FLOAT_MODE_ATI 0x8820 +#define GL_COLOR_CLEAR_UNCLAMPED_VALUE_ATI 0x8835 +#endif /* GL_ATI_pixel_format_float */ + +#ifndef GL_ATI_pn_triangles +#define GL_ATI_pn_triangles 1 +#define GL_PN_TRIANGLES_ATI 0x87F0 +#define GL_MAX_PN_TRIANGLES_TESSELATION_LEVEL_ATI 0x87F1 +#define GL_PN_TRIANGLES_POINT_MODE_ATI 0x87F2 +#define GL_PN_TRIANGLES_NORMAL_MODE_ATI 0x87F3 +#define GL_PN_TRIANGLES_TESSELATION_LEVEL_ATI 0x87F4 +#define GL_PN_TRIANGLES_POINT_MODE_LINEAR_ATI 0x87F5 +#define GL_PN_TRIANGLES_POINT_MODE_CUBIC_ATI 0x87F6 +#define GL_PN_TRIANGLES_NORMAL_MODE_LINEAR_ATI 0x87F7 +#define GL_PN_TRIANGLES_NORMAL_MODE_QUADRATIC_ATI 0x87F8 +typedef void (APIENTRYP PFNGLPNTRIANGLESIATIPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLPNTRIANGLESFATIPROC) (GLenum pname, GLfloat param); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPNTrianglesiATI (GLenum pname, GLint param); +GLAPI void APIENTRY glPNTrianglesfATI (GLenum pname, GLfloat param); +#endif +#endif /* GL_ATI_pn_triangles */ + +#ifndef GL_ATI_separate_stencil +#define GL_ATI_separate_stencil 1 +#define GL_STENCIL_BACK_FUNC_ATI 0x8800 +#define GL_STENCIL_BACK_FAIL_ATI 0x8801 +#define GL_STENCIL_BACK_PASS_DEPTH_FAIL_ATI 0x8802 +#define GL_STENCIL_BACK_PASS_DEPTH_PASS_ATI 0x8803 +typedef void (APIENTRYP PFNGLSTENCILOPSEPARATEATIPROC) (GLenum face, GLenum sfail, GLenum dpfail, GLenum dppass); +typedef void (APIENTRYP PFNGLSTENCILFUNCSEPARATEATIPROC) (GLenum frontfunc, GLenum backfunc, GLint ref, GLuint mask); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glStencilOpSeparateATI (GLenum face, GLenum sfail, GLenum dpfail, GLenum dppass); +GLAPI void APIENTRY glStencilFuncSeparateATI (GLenum frontfunc, GLenum backfunc, GLint ref, GLuint mask); +#endif +#endif /* GL_ATI_separate_stencil */ + +#ifndef GL_ATI_text_fragment_shader +#define GL_ATI_text_fragment_shader 1 +#define GL_TEXT_FRAGMENT_SHADER_ATI 0x8200 +#endif /* GL_ATI_text_fragment_shader */ + +#ifndef GL_ATI_texture_env_combine3 +#define GL_ATI_texture_env_combine3 1 +#define GL_MODULATE_ADD_ATI 0x8744 +#define GL_MODULATE_SIGNED_ADD_ATI 0x8745 +#define GL_MODULATE_SUBTRACT_ATI 0x8746 +#endif /* GL_ATI_texture_env_combine3 */ + +#ifndef GL_ATI_texture_float +#define GL_ATI_texture_float 1 +#define GL_RGBA_FLOAT32_ATI 0x8814 +#define GL_RGB_FLOAT32_ATI 0x8815 +#define GL_ALPHA_FLOAT32_ATI 0x8816 +#define GL_INTENSITY_FLOAT32_ATI 0x8817 +#define GL_LUMINANCE_FLOAT32_ATI 0x8818 +#define GL_LUMINANCE_ALPHA_FLOAT32_ATI 0x8819 +#define GL_RGBA_FLOAT16_ATI 0x881A +#define GL_RGB_FLOAT16_ATI 0x881B +#define GL_ALPHA_FLOAT16_ATI 0x881C +#define GL_INTENSITY_FLOAT16_ATI 0x881D +#define GL_LUMINANCE_FLOAT16_ATI 0x881E +#define GL_LUMINANCE_ALPHA_FLOAT16_ATI 0x881F +#endif /* GL_ATI_texture_float */ + +#ifndef GL_ATI_texture_mirror_once +#define GL_ATI_texture_mirror_once 1 +#define GL_MIRROR_CLAMP_ATI 0x8742 +#define GL_MIRROR_CLAMP_TO_EDGE_ATI 0x8743 +#endif /* GL_ATI_texture_mirror_once */ + +#ifndef GL_ATI_vertex_array_object +#define GL_ATI_vertex_array_object 1 +#define GL_STATIC_ATI 0x8760 +#define GL_DYNAMIC_ATI 0x8761 +#define GL_PRESERVE_ATI 0x8762 +#define GL_DISCARD_ATI 0x8763 +#define GL_OBJECT_BUFFER_SIZE_ATI 0x8764 +#define GL_OBJECT_BUFFER_USAGE_ATI 0x8765 +#define GL_ARRAY_OBJECT_BUFFER_ATI 0x8766 +#define GL_ARRAY_OBJECT_OFFSET_ATI 0x8767 +typedef GLuint (APIENTRYP PFNGLNEWOBJECTBUFFERATIPROC) (GLsizei size, const void *pointer, GLenum usage); +typedef GLboolean (APIENTRYP PFNGLISOBJECTBUFFERATIPROC) (GLuint buffer); +typedef void (APIENTRYP PFNGLUPDATEOBJECTBUFFERATIPROC) (GLuint buffer, GLuint offset, GLsizei size, const void *pointer, GLenum preserve); +typedef void (APIENTRYP PFNGLGETOBJECTBUFFERFVATIPROC) (GLuint buffer, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETOBJECTBUFFERIVATIPROC) (GLuint buffer, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLFREEOBJECTBUFFERATIPROC) (GLuint buffer); +typedef void (APIENTRYP PFNGLARRAYOBJECTATIPROC) (GLenum array, GLint size, GLenum type, GLsizei stride, GLuint buffer, GLuint offset); +typedef void (APIENTRYP PFNGLGETARRAYOBJECTFVATIPROC) (GLenum array, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETARRAYOBJECTIVATIPROC) (GLenum array, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLVARIANTARRAYOBJECTATIPROC) (GLuint id, GLenum type, GLsizei stride, GLuint buffer, GLuint offset); +typedef void (APIENTRYP PFNGLGETVARIANTARRAYOBJECTFVATIPROC) (GLuint id, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETVARIANTARRAYOBJECTIVATIPROC) (GLuint id, GLenum pname, GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLuint APIENTRY glNewObjectBufferATI (GLsizei size, const void *pointer, GLenum usage); +GLAPI GLboolean APIENTRY glIsObjectBufferATI (GLuint buffer); +GLAPI void APIENTRY glUpdateObjectBufferATI (GLuint buffer, GLuint offset, GLsizei size, const void *pointer, GLenum preserve); +GLAPI void APIENTRY glGetObjectBufferfvATI (GLuint buffer, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetObjectBufferivATI (GLuint buffer, GLenum pname, GLint *params); +GLAPI void APIENTRY glFreeObjectBufferATI (GLuint buffer); +GLAPI void APIENTRY glArrayObjectATI (GLenum array, GLint size, GLenum type, GLsizei stride, GLuint buffer, GLuint offset); +GLAPI void APIENTRY glGetArrayObjectfvATI (GLenum array, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetArrayObjectivATI (GLenum array, GLenum pname, GLint *params); +GLAPI void APIENTRY glVariantArrayObjectATI (GLuint id, GLenum type, GLsizei stride, GLuint buffer, GLuint offset); +GLAPI void APIENTRY glGetVariantArrayObjectfvATI (GLuint id, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetVariantArrayObjectivATI (GLuint id, GLenum pname, GLint *params); +#endif +#endif /* GL_ATI_vertex_array_object */ + +#ifndef GL_ATI_vertex_attrib_array_object +#define GL_ATI_vertex_attrib_array_object 1 +typedef void (APIENTRYP PFNGLVERTEXATTRIBARRAYOBJECTATIPROC) (GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride, GLuint buffer, GLuint offset); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBARRAYOBJECTFVATIPROC) (GLuint index, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBARRAYOBJECTIVATIPROC) (GLuint index, GLenum pname, GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexAttribArrayObjectATI (GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride, GLuint buffer, GLuint offset); +GLAPI void APIENTRY glGetVertexAttribArrayObjectfvATI (GLuint index, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetVertexAttribArrayObjectivATI (GLuint index, GLenum pname, GLint *params); +#endif +#endif /* GL_ATI_vertex_attrib_array_object */ + +#ifndef GL_ATI_vertex_streams +#define GL_ATI_vertex_streams 1 +#define GL_MAX_VERTEX_STREAMS_ATI 0x876B +#define GL_VERTEX_STREAM0_ATI 0x876C +#define GL_VERTEX_STREAM1_ATI 0x876D +#define GL_VERTEX_STREAM2_ATI 0x876E +#define GL_VERTEX_STREAM3_ATI 0x876F +#define GL_VERTEX_STREAM4_ATI 0x8770 +#define GL_VERTEX_STREAM5_ATI 0x8771 +#define GL_VERTEX_STREAM6_ATI 0x8772 +#define GL_VERTEX_STREAM7_ATI 0x8773 +#define GL_VERTEX_SOURCE_ATI 0x8774 +typedef void (APIENTRYP PFNGLVERTEXSTREAM1SATIPROC) (GLenum stream, GLshort x); +typedef void (APIENTRYP PFNGLVERTEXSTREAM1SVATIPROC) (GLenum stream, const GLshort *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM1IATIPROC) (GLenum stream, GLint x); +typedef void (APIENTRYP PFNGLVERTEXSTREAM1IVATIPROC) (GLenum stream, const GLint *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM1FATIPROC) (GLenum stream, GLfloat x); +typedef void (APIENTRYP PFNGLVERTEXSTREAM1FVATIPROC) (GLenum stream, const GLfloat *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM1DATIPROC) (GLenum stream, GLdouble x); +typedef void (APIENTRYP PFNGLVERTEXSTREAM1DVATIPROC) (GLenum stream, const GLdouble *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2SATIPROC) (GLenum stream, GLshort x, GLshort y); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2SVATIPROC) (GLenum stream, const GLshort *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2IATIPROC) (GLenum stream, GLint x, GLint y); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2IVATIPROC) (GLenum stream, const GLint *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2FATIPROC) (GLenum stream, GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2FVATIPROC) (GLenum stream, const GLfloat *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2DATIPROC) (GLenum stream, GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2DVATIPROC) (GLenum stream, const GLdouble *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3SATIPROC) (GLenum stream, GLshort x, GLshort y, GLshort z); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3SVATIPROC) (GLenum stream, const GLshort *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3IATIPROC) (GLenum stream, GLint x, GLint y, GLint z); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3IVATIPROC) (GLenum stream, const GLint *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3FATIPROC) (GLenum stream, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3FVATIPROC) (GLenum stream, const GLfloat *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3DATIPROC) (GLenum stream, GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3DVATIPROC) (GLenum stream, const GLdouble *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4SATIPROC) (GLenum stream, GLshort x, GLshort y, GLshort z, GLshort w); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4SVATIPROC) (GLenum stream, const GLshort *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4IATIPROC) (GLenum stream, GLint x, GLint y, GLint z, GLint w); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4IVATIPROC) (GLenum stream, const GLint *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4FATIPROC) (GLenum stream, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4FVATIPROC) (GLenum stream, const GLfloat *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4DATIPROC) (GLenum stream, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4DVATIPROC) (GLenum stream, const GLdouble *coords); +typedef void (APIENTRYP PFNGLNORMALSTREAM3BATIPROC) (GLenum stream, GLbyte nx, GLbyte ny, GLbyte nz); +typedef void (APIENTRYP PFNGLNORMALSTREAM3BVATIPROC) (GLenum stream, const GLbyte *coords); +typedef void (APIENTRYP PFNGLNORMALSTREAM3SATIPROC) (GLenum stream, GLshort nx, GLshort ny, GLshort nz); +typedef void (APIENTRYP PFNGLNORMALSTREAM3SVATIPROC) (GLenum stream, const GLshort *coords); +typedef void (APIENTRYP PFNGLNORMALSTREAM3IATIPROC) (GLenum stream, GLint nx, GLint ny, GLint nz); +typedef void (APIENTRYP PFNGLNORMALSTREAM3IVATIPROC) (GLenum stream, const GLint *coords); +typedef void (APIENTRYP PFNGLNORMALSTREAM3FATIPROC) (GLenum stream, GLfloat nx, GLfloat ny, GLfloat nz); +typedef void (APIENTRYP PFNGLNORMALSTREAM3FVATIPROC) (GLenum stream, const GLfloat *coords); +typedef void (APIENTRYP PFNGLNORMALSTREAM3DATIPROC) (GLenum stream, GLdouble nx, GLdouble ny, GLdouble nz); +typedef void (APIENTRYP PFNGLNORMALSTREAM3DVATIPROC) (GLenum stream, const GLdouble *coords); +typedef void (APIENTRYP PFNGLCLIENTACTIVEVERTEXSTREAMATIPROC) (GLenum stream); +typedef void (APIENTRYP PFNGLVERTEXBLENDENVIATIPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLVERTEXBLENDENVFATIPROC) (GLenum pname, GLfloat param); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexStream1sATI (GLenum stream, GLshort x); +GLAPI void APIENTRY glVertexStream1svATI (GLenum stream, const GLshort *coords); +GLAPI void APIENTRY glVertexStream1iATI (GLenum stream, GLint x); +GLAPI void APIENTRY glVertexStream1ivATI (GLenum stream, const GLint *coords); +GLAPI void APIENTRY glVertexStream1fATI (GLenum stream, GLfloat x); +GLAPI void APIENTRY glVertexStream1fvATI (GLenum stream, const GLfloat *coords); +GLAPI void APIENTRY glVertexStream1dATI (GLenum stream, GLdouble x); +GLAPI void APIENTRY glVertexStream1dvATI (GLenum stream, const GLdouble *coords); +GLAPI void APIENTRY glVertexStream2sATI (GLenum stream, GLshort x, GLshort y); +GLAPI void APIENTRY glVertexStream2svATI (GLenum stream, const GLshort *coords); +GLAPI void APIENTRY glVertexStream2iATI (GLenum stream, GLint x, GLint y); +GLAPI void APIENTRY glVertexStream2ivATI (GLenum stream, const GLint *coords); +GLAPI void APIENTRY glVertexStream2fATI (GLenum stream, GLfloat x, GLfloat y); +GLAPI void APIENTRY glVertexStream2fvATI (GLenum stream, const GLfloat *coords); +GLAPI void APIENTRY glVertexStream2dATI (GLenum stream, GLdouble x, GLdouble y); +GLAPI void APIENTRY glVertexStream2dvATI (GLenum stream, const GLdouble *coords); +GLAPI void APIENTRY glVertexStream3sATI (GLenum stream, GLshort x, GLshort y, GLshort z); +GLAPI void APIENTRY glVertexStream3svATI (GLenum stream, const GLshort *coords); +GLAPI void APIENTRY glVertexStream3iATI (GLenum stream, GLint x, GLint y, GLint z); +GLAPI void APIENTRY glVertexStream3ivATI (GLenum stream, const GLint *coords); +GLAPI void APIENTRY glVertexStream3fATI (GLenum stream, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glVertexStream3fvATI (GLenum stream, const GLfloat *coords); +GLAPI void APIENTRY glVertexStream3dATI (GLenum stream, GLdouble x, GLdouble y, GLdouble z); +GLAPI void APIENTRY glVertexStream3dvATI (GLenum stream, const GLdouble *coords); +GLAPI void APIENTRY glVertexStream4sATI (GLenum stream, GLshort x, GLshort y, GLshort z, GLshort w); +GLAPI void APIENTRY glVertexStream4svATI (GLenum stream, const GLshort *coords); +GLAPI void APIENTRY glVertexStream4iATI (GLenum stream, GLint x, GLint y, GLint z, GLint w); +GLAPI void APIENTRY glVertexStream4ivATI (GLenum stream, const GLint *coords); +GLAPI void APIENTRY glVertexStream4fATI (GLenum stream, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +GLAPI void APIENTRY glVertexStream4fvATI (GLenum stream, const GLfloat *coords); +GLAPI void APIENTRY glVertexStream4dATI (GLenum stream, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +GLAPI void APIENTRY glVertexStream4dvATI (GLenum stream, const GLdouble *coords); +GLAPI void APIENTRY glNormalStream3bATI (GLenum stream, GLbyte nx, GLbyte ny, GLbyte nz); +GLAPI void APIENTRY glNormalStream3bvATI (GLenum stream, const GLbyte *coords); +GLAPI void APIENTRY glNormalStream3sATI (GLenum stream, GLshort nx, GLshort ny, GLshort nz); +GLAPI void APIENTRY glNormalStream3svATI (GLenum stream, const GLshort *coords); +GLAPI void APIENTRY glNormalStream3iATI (GLenum stream, GLint nx, GLint ny, GLint nz); +GLAPI void APIENTRY glNormalStream3ivATI (GLenum stream, const GLint *coords); +GLAPI void APIENTRY glNormalStream3fATI (GLenum stream, GLfloat nx, GLfloat ny, GLfloat nz); +GLAPI void APIENTRY glNormalStream3fvATI (GLenum stream, const GLfloat *coords); +GLAPI void APIENTRY glNormalStream3dATI (GLenum stream, GLdouble nx, GLdouble ny, GLdouble nz); +GLAPI void APIENTRY glNormalStream3dvATI (GLenum stream, const GLdouble *coords); +GLAPI void APIENTRY glClientActiveVertexStreamATI (GLenum stream); +GLAPI void APIENTRY glVertexBlendEnviATI (GLenum pname, GLint param); +GLAPI void APIENTRY glVertexBlendEnvfATI (GLenum pname, GLfloat param); +#endif +#endif /* GL_ATI_vertex_streams */ + +#ifndef GL_EXT_422_pixels +#define GL_EXT_422_pixels 1 +#define GL_422_EXT 0x80CC +#define GL_422_REV_EXT 0x80CD +#define GL_422_AVERAGE_EXT 0x80CE +#define GL_422_REV_AVERAGE_EXT 0x80CF +#endif /* GL_EXT_422_pixels */ + +#ifndef GL_EXT_EGL_image_storage +#define GL_EXT_EGL_image_storage 1 +typedef void *GLeglImageOES; +typedef void (APIENTRYP PFNGLEGLIMAGETARGETTEXSTORAGEEXTPROC) (GLenum target, GLeglImageOES image, const GLint* attrib_list); +typedef void (APIENTRYP PFNGLEGLIMAGETARGETTEXTURESTORAGEEXTPROC) (GLuint texture, GLeglImageOES image, const GLint* attrib_list); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glEGLImageTargetTexStorageEXT (GLenum target, GLeglImageOES image, const GLint* attrib_list); +GLAPI void APIENTRY glEGLImageTargetTextureStorageEXT (GLuint texture, GLeglImageOES image, const GLint* attrib_list); +#endif +#endif /* GL_EXT_EGL_image_storage */ + +#ifndef GL_EXT_EGL_sync +#define GL_EXT_EGL_sync 1 +#endif /* GL_EXT_EGL_sync */ + +#ifndef GL_EXT_abgr +#define GL_EXT_abgr 1 +#define GL_ABGR_EXT 0x8000 +#endif /* GL_EXT_abgr */ + +#ifndef GL_EXT_bgra +#define GL_EXT_bgra 1 +#define GL_BGR_EXT 0x80E0 +#define GL_BGRA_EXT 0x80E1 +#endif /* GL_EXT_bgra */ + +#ifndef GL_EXT_bindable_uniform +#define GL_EXT_bindable_uniform 1 +#define GL_MAX_VERTEX_BINDABLE_UNIFORMS_EXT 0x8DE2 +#define GL_MAX_FRAGMENT_BINDABLE_UNIFORMS_EXT 0x8DE3 +#define GL_MAX_GEOMETRY_BINDABLE_UNIFORMS_EXT 0x8DE4 +#define GL_MAX_BINDABLE_UNIFORM_SIZE_EXT 0x8DED +#define GL_UNIFORM_BUFFER_EXT 0x8DEE +#define GL_UNIFORM_BUFFER_BINDING_EXT 0x8DEF +typedef void (APIENTRYP PFNGLUNIFORMBUFFEREXTPROC) (GLuint program, GLint location, GLuint buffer); +typedef GLint (APIENTRYP PFNGLGETUNIFORMBUFFERSIZEEXTPROC) (GLuint program, GLint location); +typedef GLintptr (APIENTRYP PFNGLGETUNIFORMOFFSETEXTPROC) (GLuint program, GLint location); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glUniformBufferEXT (GLuint program, GLint location, GLuint buffer); +GLAPI GLint APIENTRY glGetUniformBufferSizeEXT (GLuint program, GLint location); +GLAPI GLintptr APIENTRY glGetUniformOffsetEXT (GLuint program, GLint location); +#endif +#endif /* GL_EXT_bindable_uniform */ + +#ifndef GL_EXT_blend_color +#define GL_EXT_blend_color 1 +#define GL_CONSTANT_COLOR_EXT 0x8001 +#define GL_ONE_MINUS_CONSTANT_COLOR_EXT 0x8002 +#define GL_CONSTANT_ALPHA_EXT 0x8003 +#define GL_ONE_MINUS_CONSTANT_ALPHA_EXT 0x8004 +#define GL_BLEND_COLOR_EXT 0x8005 +typedef void (APIENTRYP PFNGLBLENDCOLOREXTPROC) (GLfloat red, GLfloat green, GLfloat blue, GLfloat alpha); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendColorEXT (GLfloat red, GLfloat green, GLfloat blue, GLfloat alpha); +#endif +#endif /* GL_EXT_blend_color */ + +#ifndef GL_EXT_blend_equation_separate +#define GL_EXT_blend_equation_separate 1 +#define GL_BLEND_EQUATION_RGB_EXT 0x8009 +#define GL_BLEND_EQUATION_ALPHA_EXT 0x883D +typedef void (APIENTRYP PFNGLBLENDEQUATIONSEPARATEEXTPROC) (GLenum modeRGB, GLenum modeAlpha); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendEquationSeparateEXT (GLenum modeRGB, GLenum modeAlpha); +#endif +#endif /* GL_EXT_blend_equation_separate */ + +#ifndef GL_EXT_blend_func_separate +#define GL_EXT_blend_func_separate 1 +#define GL_BLEND_DST_RGB_EXT 0x80C8 +#define GL_BLEND_SRC_RGB_EXT 0x80C9 +#define GL_BLEND_DST_ALPHA_EXT 0x80CA +#define GL_BLEND_SRC_ALPHA_EXT 0x80CB +typedef void (APIENTRYP PFNGLBLENDFUNCSEPARATEEXTPROC) (GLenum sfactorRGB, GLenum dfactorRGB, GLenum sfactorAlpha, GLenum dfactorAlpha); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendFuncSeparateEXT (GLenum sfactorRGB, GLenum dfactorRGB, GLenum sfactorAlpha, GLenum dfactorAlpha); +#endif +#endif /* GL_EXT_blend_func_separate */ + +#ifndef GL_EXT_blend_logic_op +#define GL_EXT_blend_logic_op 1 +#endif /* GL_EXT_blend_logic_op */ + +#ifndef GL_EXT_blend_minmax +#define GL_EXT_blend_minmax 1 +#define GL_MIN_EXT 0x8007 +#define GL_MAX_EXT 0x8008 +#define GL_FUNC_ADD_EXT 0x8006 +#define GL_BLEND_EQUATION_EXT 0x8009 +typedef void (APIENTRYP PFNGLBLENDEQUATIONEXTPROC) (GLenum mode); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendEquationEXT (GLenum mode); +#endif +#endif /* GL_EXT_blend_minmax */ + +#ifndef GL_EXT_blend_subtract +#define GL_EXT_blend_subtract 1 +#define GL_FUNC_SUBTRACT_EXT 0x800A +#define GL_FUNC_REVERSE_SUBTRACT_EXT 0x800B +#endif /* GL_EXT_blend_subtract */ + +#ifndef GL_EXT_clip_volume_hint +#define GL_EXT_clip_volume_hint 1 +#define GL_CLIP_VOLUME_CLIPPING_HINT_EXT 0x80F0 +#endif /* GL_EXT_clip_volume_hint */ + +#ifndef GL_EXT_cmyka +#define GL_EXT_cmyka 1 +#define GL_CMYK_EXT 0x800C +#define GL_CMYKA_EXT 0x800D +#define GL_PACK_CMYK_HINT_EXT 0x800E +#define GL_UNPACK_CMYK_HINT_EXT 0x800F +#endif /* GL_EXT_cmyka */ + +#ifndef GL_EXT_color_subtable +#define GL_EXT_color_subtable 1 +typedef void (APIENTRYP PFNGLCOLORSUBTABLEEXTPROC) (GLenum target, GLsizei start, GLsizei count, GLenum format, GLenum type, const void *data); +typedef void (APIENTRYP PFNGLCOPYCOLORSUBTABLEEXTPROC) (GLenum target, GLsizei start, GLint x, GLint y, GLsizei width); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glColorSubTableEXT (GLenum target, GLsizei start, GLsizei count, GLenum format, GLenum type, const void *data); +GLAPI void APIENTRY glCopyColorSubTableEXT (GLenum target, GLsizei start, GLint x, GLint y, GLsizei width); +#endif +#endif /* GL_EXT_color_subtable */ + +#ifndef GL_EXT_compiled_vertex_array +#define GL_EXT_compiled_vertex_array 1 +#define GL_ARRAY_ELEMENT_LOCK_FIRST_EXT 0x81A8 +#define GL_ARRAY_ELEMENT_LOCK_COUNT_EXT 0x81A9 +typedef void (APIENTRYP PFNGLLOCKARRAYSEXTPROC) (GLint first, GLsizei count); +typedef void (APIENTRYP PFNGLUNLOCKARRAYSEXTPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glLockArraysEXT (GLint first, GLsizei count); +GLAPI void APIENTRY glUnlockArraysEXT (void); +#endif +#endif /* GL_EXT_compiled_vertex_array */ + +#ifndef GL_EXT_convolution +#define GL_EXT_convolution 1 +#define GL_CONVOLUTION_1D_EXT 0x8010 +#define GL_CONVOLUTION_2D_EXT 0x8011 +#define GL_SEPARABLE_2D_EXT 0x8012 +#define GL_CONVOLUTION_BORDER_MODE_EXT 0x8013 +#define GL_CONVOLUTION_FILTER_SCALE_EXT 0x8014 +#define GL_CONVOLUTION_FILTER_BIAS_EXT 0x8015 +#define GL_REDUCE_EXT 0x8016 +#define GL_CONVOLUTION_FORMAT_EXT 0x8017 +#define GL_CONVOLUTION_WIDTH_EXT 0x8018 +#define GL_CONVOLUTION_HEIGHT_EXT 0x8019 +#define GL_MAX_CONVOLUTION_WIDTH_EXT 0x801A +#define GL_MAX_CONVOLUTION_HEIGHT_EXT 0x801B +#define GL_POST_CONVOLUTION_RED_SCALE_EXT 0x801C +#define GL_POST_CONVOLUTION_GREEN_SCALE_EXT 0x801D +#define GL_POST_CONVOLUTION_BLUE_SCALE_EXT 0x801E +#define GL_POST_CONVOLUTION_ALPHA_SCALE_EXT 0x801F +#define GL_POST_CONVOLUTION_RED_BIAS_EXT 0x8020 +#define GL_POST_CONVOLUTION_GREEN_BIAS_EXT 0x8021 +#define GL_POST_CONVOLUTION_BLUE_BIAS_EXT 0x8022 +#define GL_POST_CONVOLUTION_ALPHA_BIAS_EXT 0x8023 +typedef void (APIENTRYP PFNGLCONVOLUTIONFILTER1DEXTPROC) (GLenum target, GLenum internalformat, GLsizei width, GLenum format, GLenum type, const void *image); +typedef void (APIENTRYP PFNGLCONVOLUTIONFILTER2DEXTPROC) (GLenum target, GLenum internalformat, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *image); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERFEXTPROC) (GLenum target, GLenum pname, GLfloat params); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERFVEXTPROC) (GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERIEXTPROC) (GLenum target, GLenum pname, GLint params); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERIVEXTPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLCOPYCONVOLUTIONFILTER1DEXTPROC) (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width); +typedef void (APIENTRYP PFNGLCOPYCONVOLUTIONFILTER2DEXTPROC) (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLGETCONVOLUTIONFILTEREXTPROC) (GLenum target, GLenum format, GLenum type, void *image); +typedef void (APIENTRYP PFNGLGETCONVOLUTIONPARAMETERFVEXTPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETCONVOLUTIONPARAMETERIVEXTPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETSEPARABLEFILTEREXTPROC) (GLenum target, GLenum format, GLenum type, void *row, void *column, void *span); +typedef void (APIENTRYP PFNGLSEPARABLEFILTER2DEXTPROC) (GLenum target, GLenum internalformat, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *row, const void *column); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glConvolutionFilter1DEXT (GLenum target, GLenum internalformat, GLsizei width, GLenum format, GLenum type, const void *image); +GLAPI void APIENTRY glConvolutionFilter2DEXT (GLenum target, GLenum internalformat, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *image); +GLAPI void APIENTRY glConvolutionParameterfEXT (GLenum target, GLenum pname, GLfloat params); +GLAPI void APIENTRY glConvolutionParameterfvEXT (GLenum target, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glConvolutionParameteriEXT (GLenum target, GLenum pname, GLint params); +GLAPI void APIENTRY glConvolutionParameterivEXT (GLenum target, GLenum pname, const GLint *params); +GLAPI void APIENTRY glCopyConvolutionFilter1DEXT (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width); +GLAPI void APIENTRY glCopyConvolutionFilter2DEXT (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height); +GLAPI void APIENTRY glGetConvolutionFilterEXT (GLenum target, GLenum format, GLenum type, void *image); +GLAPI void APIENTRY glGetConvolutionParameterfvEXT (GLenum target, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetConvolutionParameterivEXT (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetSeparableFilterEXT (GLenum target, GLenum format, GLenum type, void *row, void *column, void *span); +GLAPI void APIENTRY glSeparableFilter2DEXT (GLenum target, GLenum internalformat, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *row, const void *column); +#endif +#endif /* GL_EXT_convolution */ + +#ifndef GL_EXT_coordinate_frame +#define GL_EXT_coordinate_frame 1 +#define GL_TANGENT_ARRAY_EXT 0x8439 +#define GL_BINORMAL_ARRAY_EXT 0x843A +#define GL_CURRENT_TANGENT_EXT 0x843B +#define GL_CURRENT_BINORMAL_EXT 0x843C +#define GL_TANGENT_ARRAY_TYPE_EXT 0x843E +#define GL_TANGENT_ARRAY_STRIDE_EXT 0x843F +#define GL_BINORMAL_ARRAY_TYPE_EXT 0x8440 +#define GL_BINORMAL_ARRAY_STRIDE_EXT 0x8441 +#define GL_TANGENT_ARRAY_POINTER_EXT 0x8442 +#define GL_BINORMAL_ARRAY_POINTER_EXT 0x8443 +#define GL_MAP1_TANGENT_EXT 0x8444 +#define GL_MAP2_TANGENT_EXT 0x8445 +#define GL_MAP1_BINORMAL_EXT 0x8446 +#define GL_MAP2_BINORMAL_EXT 0x8447 +typedef void (APIENTRYP PFNGLTANGENT3BEXTPROC) (GLbyte tx, GLbyte ty, GLbyte tz); +typedef void (APIENTRYP PFNGLTANGENT3BVEXTPROC) (const GLbyte *v); +typedef void (APIENTRYP PFNGLTANGENT3DEXTPROC) (GLdouble tx, GLdouble ty, GLdouble tz); +typedef void (APIENTRYP PFNGLTANGENT3DVEXTPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLTANGENT3FEXTPROC) (GLfloat tx, GLfloat ty, GLfloat tz); +typedef void (APIENTRYP PFNGLTANGENT3FVEXTPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLTANGENT3IEXTPROC) (GLint tx, GLint ty, GLint tz); +typedef void (APIENTRYP PFNGLTANGENT3IVEXTPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLTANGENT3SEXTPROC) (GLshort tx, GLshort ty, GLshort tz); +typedef void (APIENTRYP PFNGLTANGENT3SVEXTPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLBINORMAL3BEXTPROC) (GLbyte bx, GLbyte by, GLbyte bz); +typedef void (APIENTRYP PFNGLBINORMAL3BVEXTPROC) (const GLbyte *v); +typedef void (APIENTRYP PFNGLBINORMAL3DEXTPROC) (GLdouble bx, GLdouble by, GLdouble bz); +typedef void (APIENTRYP PFNGLBINORMAL3DVEXTPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLBINORMAL3FEXTPROC) (GLfloat bx, GLfloat by, GLfloat bz); +typedef void (APIENTRYP PFNGLBINORMAL3FVEXTPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLBINORMAL3IEXTPROC) (GLint bx, GLint by, GLint bz); +typedef void (APIENTRYP PFNGLBINORMAL3IVEXTPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLBINORMAL3SEXTPROC) (GLshort bx, GLshort by, GLshort bz); +typedef void (APIENTRYP PFNGLBINORMAL3SVEXTPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLTANGENTPOINTEREXTPROC) (GLenum type, GLsizei stride, const void *pointer); +typedef void (APIENTRYP PFNGLBINORMALPOINTEREXTPROC) (GLenum type, GLsizei stride, const void *pointer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTangent3bEXT (GLbyte tx, GLbyte ty, GLbyte tz); +GLAPI void APIENTRY glTangent3bvEXT (const GLbyte *v); +GLAPI void APIENTRY glTangent3dEXT (GLdouble tx, GLdouble ty, GLdouble tz); +GLAPI void APIENTRY glTangent3dvEXT (const GLdouble *v); +GLAPI void APIENTRY glTangent3fEXT (GLfloat tx, GLfloat ty, GLfloat tz); +GLAPI void APIENTRY glTangent3fvEXT (const GLfloat *v); +GLAPI void APIENTRY glTangent3iEXT (GLint tx, GLint ty, GLint tz); +GLAPI void APIENTRY glTangent3ivEXT (const GLint *v); +GLAPI void APIENTRY glTangent3sEXT (GLshort tx, GLshort ty, GLshort tz); +GLAPI void APIENTRY glTangent3svEXT (const GLshort *v); +GLAPI void APIENTRY glBinormal3bEXT (GLbyte bx, GLbyte by, GLbyte bz); +GLAPI void APIENTRY glBinormal3bvEXT (const GLbyte *v); +GLAPI void APIENTRY glBinormal3dEXT (GLdouble bx, GLdouble by, GLdouble bz); +GLAPI void APIENTRY glBinormal3dvEXT (const GLdouble *v); +GLAPI void APIENTRY glBinormal3fEXT (GLfloat bx, GLfloat by, GLfloat bz); +GLAPI void APIENTRY glBinormal3fvEXT (const GLfloat *v); +GLAPI void APIENTRY glBinormal3iEXT (GLint bx, GLint by, GLint bz); +GLAPI void APIENTRY glBinormal3ivEXT (const GLint *v); +GLAPI void APIENTRY glBinormal3sEXT (GLshort bx, GLshort by, GLshort bz); +GLAPI void APIENTRY glBinormal3svEXT (const GLshort *v); +GLAPI void APIENTRY glTangentPointerEXT (GLenum type, GLsizei stride, const void *pointer); +GLAPI void APIENTRY glBinormalPointerEXT (GLenum type, GLsizei stride, const void *pointer); +#endif +#endif /* GL_EXT_coordinate_frame */ + +#ifndef GL_EXT_copy_texture +#define GL_EXT_copy_texture 1 +typedef void (APIENTRYP PFNGLCOPYTEXIMAGE1DEXTPROC) (GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLint border); +typedef void (APIENTRYP PFNGLCOPYTEXIMAGE2DEXTPROC) (GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height, GLint border); +typedef void (APIENTRYP PFNGLCOPYTEXSUBIMAGE1DEXTPROC) (GLenum target, GLint level, GLint xoffset, GLint x, GLint y, GLsizei width); +typedef void (APIENTRYP PFNGLCOPYTEXSUBIMAGE2DEXTPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLCOPYTEXSUBIMAGE3DEXTPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint x, GLint y, GLsizei width, GLsizei height); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCopyTexImage1DEXT (GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLint border); +GLAPI void APIENTRY glCopyTexImage2DEXT (GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height, GLint border); +GLAPI void APIENTRY glCopyTexSubImage1DEXT (GLenum target, GLint level, GLint xoffset, GLint x, GLint y, GLsizei width); +GLAPI void APIENTRY glCopyTexSubImage2DEXT (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint x, GLint y, GLsizei width, GLsizei height); +GLAPI void APIENTRY glCopyTexSubImage3DEXT (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint x, GLint y, GLsizei width, GLsizei height); +#endif +#endif /* GL_EXT_copy_texture */ + +#ifndef GL_EXT_cull_vertex +#define GL_EXT_cull_vertex 1 +#define GL_CULL_VERTEX_EXT 0x81AA +#define GL_CULL_VERTEX_EYE_POSITION_EXT 0x81AB +#define GL_CULL_VERTEX_OBJECT_POSITION_EXT 0x81AC +typedef void (APIENTRYP PFNGLCULLPARAMETERDVEXTPROC) (GLenum pname, GLdouble *params); +typedef void (APIENTRYP PFNGLCULLPARAMETERFVEXTPROC) (GLenum pname, GLfloat *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCullParameterdvEXT (GLenum pname, GLdouble *params); +GLAPI void APIENTRY glCullParameterfvEXT (GLenum pname, GLfloat *params); +#endif +#endif /* GL_EXT_cull_vertex */ + +#ifndef GL_EXT_debug_label +#define GL_EXT_debug_label 1 +#define GL_PROGRAM_PIPELINE_OBJECT_EXT 0x8A4F +#define GL_PROGRAM_OBJECT_EXT 0x8B40 +#define GL_SHADER_OBJECT_EXT 0x8B48 +#define GL_BUFFER_OBJECT_EXT 0x9151 +#define GL_QUERY_OBJECT_EXT 0x9153 +#define GL_VERTEX_ARRAY_OBJECT_EXT 0x9154 +typedef void (APIENTRYP PFNGLLABELOBJECTEXTPROC) (GLenum type, GLuint object, GLsizei length, const GLchar *label); +typedef void (APIENTRYP PFNGLGETOBJECTLABELEXTPROC) (GLenum type, GLuint object, GLsizei bufSize, GLsizei *length, GLchar *label); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glLabelObjectEXT (GLenum type, GLuint object, GLsizei length, const GLchar *label); +GLAPI void APIENTRY glGetObjectLabelEXT (GLenum type, GLuint object, GLsizei bufSize, GLsizei *length, GLchar *label); +#endif +#endif /* GL_EXT_debug_label */ + +#ifndef GL_EXT_debug_marker +#define GL_EXT_debug_marker 1 +typedef void (APIENTRYP PFNGLINSERTEVENTMARKEREXTPROC) (GLsizei length, const GLchar *marker); +typedef void (APIENTRYP PFNGLPUSHGROUPMARKEREXTPROC) (GLsizei length, const GLchar *marker); +typedef void (APIENTRYP PFNGLPOPGROUPMARKEREXTPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glInsertEventMarkerEXT (GLsizei length, const GLchar *marker); +GLAPI void APIENTRY glPushGroupMarkerEXT (GLsizei length, const GLchar *marker); +GLAPI void APIENTRY glPopGroupMarkerEXT (void); +#endif +#endif /* GL_EXT_debug_marker */ + +#ifndef GL_EXT_depth_bounds_test +#define GL_EXT_depth_bounds_test 1 +#define GL_DEPTH_BOUNDS_TEST_EXT 0x8890 +#define GL_DEPTH_BOUNDS_EXT 0x8891 +typedef void (APIENTRYP PFNGLDEPTHBOUNDSEXTPROC) (GLclampd zmin, GLclampd zmax); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDepthBoundsEXT (GLclampd zmin, GLclampd zmax); +#endif +#endif /* GL_EXT_depth_bounds_test */ + +#ifndef GL_EXT_direct_state_access +#define GL_EXT_direct_state_access 1 +#define GL_PROGRAM_MATRIX_EXT 0x8E2D +#define GL_TRANSPOSE_PROGRAM_MATRIX_EXT 0x8E2E +#define GL_PROGRAM_MATRIX_STACK_DEPTH_EXT 0x8E2F +typedef void (APIENTRYP PFNGLMATRIXLOADFEXTPROC) (GLenum mode, const GLfloat *m); +typedef void (APIENTRYP PFNGLMATRIXLOADDEXTPROC) (GLenum mode, const GLdouble *m); +typedef void (APIENTRYP PFNGLMATRIXMULTFEXTPROC) (GLenum mode, const GLfloat *m); +typedef void (APIENTRYP PFNGLMATRIXMULTDEXTPROC) (GLenum mode, const GLdouble *m); +typedef void (APIENTRYP PFNGLMATRIXLOADIDENTITYEXTPROC) (GLenum mode); +typedef void (APIENTRYP PFNGLMATRIXROTATEFEXTPROC) (GLenum mode, GLfloat angle, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLMATRIXROTATEDEXTPROC) (GLenum mode, GLdouble angle, GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLMATRIXSCALEFEXTPROC) (GLenum mode, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLMATRIXSCALEDEXTPROC) (GLenum mode, GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLMATRIXTRANSLATEFEXTPROC) (GLenum mode, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLMATRIXTRANSLATEDEXTPROC) (GLenum mode, GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLMATRIXFRUSTUMEXTPROC) (GLenum mode, GLdouble left, GLdouble right, GLdouble bottom, GLdouble top, GLdouble zNear, GLdouble zFar); +typedef void (APIENTRYP PFNGLMATRIXORTHOEXTPROC) (GLenum mode, GLdouble left, GLdouble right, GLdouble bottom, GLdouble top, GLdouble zNear, GLdouble zFar); +typedef void (APIENTRYP PFNGLMATRIXPOPEXTPROC) (GLenum mode); +typedef void (APIENTRYP PFNGLMATRIXPUSHEXTPROC) (GLenum mode); +typedef void (APIENTRYP PFNGLCLIENTATTRIBDEFAULTEXTPROC) (GLbitfield mask); +typedef void (APIENTRYP PFNGLPUSHCLIENTATTRIBDEFAULTEXTPROC) (GLbitfield mask); +typedef void (APIENTRYP PFNGLTEXTUREPARAMETERFEXTPROC) (GLuint texture, GLenum target, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLTEXTUREPARAMETERFVEXTPROC) (GLuint texture, GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLTEXTUREPARAMETERIEXTPROC) (GLuint texture, GLenum target, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLTEXTUREPARAMETERIVEXTPROC) (GLuint texture, GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLTEXTUREIMAGE1DEXTPROC) (GLuint texture, GLenum target, GLint level, GLint internalformat, GLsizei width, GLint border, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLTEXTUREIMAGE2DEXTPROC) (GLuint texture, GLenum target, GLint level, GLint internalformat, GLsizei width, GLsizei height, GLint border, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLTEXTURESUBIMAGE1DEXTPROC) (GLuint texture, GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLTEXTURESUBIMAGE2DEXTPROC) (GLuint texture, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLCOPYTEXTUREIMAGE1DEXTPROC) (GLuint texture, GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLint border); +typedef void (APIENTRYP PFNGLCOPYTEXTUREIMAGE2DEXTPROC) (GLuint texture, GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height, GLint border); +typedef void (APIENTRYP PFNGLCOPYTEXTURESUBIMAGE1DEXTPROC) (GLuint texture, GLenum target, GLint level, GLint xoffset, GLint x, GLint y, GLsizei width); +typedef void (APIENTRYP PFNGLCOPYTEXTURESUBIMAGE2DEXTPROC) (GLuint texture, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLGETTEXTUREIMAGEEXTPROC) (GLuint texture, GLenum target, GLint level, GLenum format, GLenum type, void *pixels); +typedef void (APIENTRYP PFNGLGETTEXTUREPARAMETERFVEXTPROC) (GLuint texture, GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETTEXTUREPARAMETERIVEXTPROC) (GLuint texture, GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETTEXTURELEVELPARAMETERFVEXTPROC) (GLuint texture, GLenum target, GLint level, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETTEXTURELEVELPARAMETERIVEXTPROC) (GLuint texture, GLenum target, GLint level, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLTEXTUREIMAGE3DEXTPROC) (GLuint texture, GLenum target, GLint level, GLint internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLTEXTURESUBIMAGE3DEXTPROC) (GLuint texture, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLCOPYTEXTURESUBIMAGE3DEXTPROC) (GLuint texture, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLBINDMULTITEXTUREEXTPROC) (GLenum texunit, GLenum target, GLuint texture); +typedef void (APIENTRYP PFNGLMULTITEXCOORDPOINTEREXTPROC) (GLenum texunit, GLint size, GLenum type, GLsizei stride, const void *pointer); +typedef void (APIENTRYP PFNGLMULTITEXENVFEXTPROC) (GLenum texunit, GLenum target, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLMULTITEXENVFVEXTPROC) (GLenum texunit, GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLMULTITEXENVIEXTPROC) (GLenum texunit, GLenum target, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLMULTITEXENVIVEXTPROC) (GLenum texunit, GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLMULTITEXGENDEXTPROC) (GLenum texunit, GLenum coord, GLenum pname, GLdouble param); +typedef void (APIENTRYP PFNGLMULTITEXGENDVEXTPROC) (GLenum texunit, GLenum coord, GLenum pname, const GLdouble *params); +typedef void (APIENTRYP PFNGLMULTITEXGENFEXTPROC) (GLenum texunit, GLenum coord, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLMULTITEXGENFVEXTPROC) (GLenum texunit, GLenum coord, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLMULTITEXGENIEXTPROC) (GLenum texunit, GLenum coord, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLMULTITEXGENIVEXTPROC) (GLenum texunit, GLenum coord, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLGETMULTITEXENVFVEXTPROC) (GLenum texunit, GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETMULTITEXENVIVEXTPROC) (GLenum texunit, GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETMULTITEXGENDVEXTPROC) (GLenum texunit, GLenum coord, GLenum pname, GLdouble *params); +typedef void (APIENTRYP PFNGLGETMULTITEXGENFVEXTPROC) (GLenum texunit, GLenum coord, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETMULTITEXGENIVEXTPROC) (GLenum texunit, GLenum coord, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLMULTITEXPARAMETERIEXTPROC) (GLenum texunit, GLenum target, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLMULTITEXPARAMETERIVEXTPROC) (GLenum texunit, GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLMULTITEXPARAMETERFEXTPROC) (GLenum texunit, GLenum target, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLMULTITEXPARAMETERFVEXTPROC) (GLenum texunit, GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLMULTITEXIMAGE1DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLint internalformat, GLsizei width, GLint border, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLMULTITEXIMAGE2DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLint internalformat, GLsizei width, GLsizei height, GLint border, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLMULTITEXSUBIMAGE1DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLMULTITEXSUBIMAGE2DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLCOPYMULTITEXIMAGE1DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLint border); +typedef void (APIENTRYP PFNGLCOPYMULTITEXIMAGE2DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height, GLint border); +typedef void (APIENTRYP PFNGLCOPYMULTITEXSUBIMAGE1DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLint x, GLint y, GLsizei width); +typedef void (APIENTRYP PFNGLCOPYMULTITEXSUBIMAGE2DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLGETMULTITEXIMAGEEXTPROC) (GLenum texunit, GLenum target, GLint level, GLenum format, GLenum type, void *pixels); +typedef void (APIENTRYP PFNGLGETMULTITEXPARAMETERFVEXTPROC) (GLenum texunit, GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETMULTITEXPARAMETERIVEXTPROC) (GLenum texunit, GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETMULTITEXLEVELPARAMETERFVEXTPROC) (GLenum texunit, GLenum target, GLint level, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETMULTITEXLEVELPARAMETERIVEXTPROC) (GLenum texunit, GLenum target, GLint level, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLMULTITEXIMAGE3DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLint internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLMULTITEXSUBIMAGE3DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLCOPYMULTITEXSUBIMAGE3DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLENABLECLIENTSTATEINDEXEDEXTPROC) (GLenum array, GLuint index); +typedef void (APIENTRYP PFNGLDISABLECLIENTSTATEINDEXEDEXTPROC) (GLenum array, GLuint index); +typedef void (APIENTRYP PFNGLGETFLOATINDEXEDVEXTPROC) (GLenum target, GLuint index, GLfloat *data); +typedef void (APIENTRYP PFNGLGETDOUBLEINDEXEDVEXTPROC) (GLenum target, GLuint index, GLdouble *data); +typedef void (APIENTRYP PFNGLGETPOINTERINDEXEDVEXTPROC) (GLenum target, GLuint index, void **data); +typedef void (APIENTRYP PFNGLENABLEINDEXEDEXTPROC) (GLenum target, GLuint index); +typedef void (APIENTRYP PFNGLDISABLEINDEXEDEXTPROC) (GLenum target, GLuint index); +typedef GLboolean (APIENTRYP PFNGLISENABLEDINDEXEDEXTPROC) (GLenum target, GLuint index); +typedef void (APIENTRYP PFNGLGETINTEGERINDEXEDVEXTPROC) (GLenum target, GLuint index, GLint *data); +typedef void (APIENTRYP PFNGLGETBOOLEANINDEXEDVEXTPROC) (GLenum target, GLuint index, GLboolean *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXTUREIMAGE3DEXTPROC) (GLuint texture, GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLsizei imageSize, const void *bits); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXTUREIMAGE2DEXTPROC) (GLuint texture, GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLsizei imageSize, const void *bits); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXTUREIMAGE1DEXTPROC) (GLuint texture, GLenum target, GLint level, GLenum internalformat, GLsizei width, GLint border, GLsizei imageSize, const void *bits); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXTURESUBIMAGE3DEXTPROC) (GLuint texture, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const void *bits); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXTURESUBIMAGE2DEXTPROC) (GLuint texture, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const void *bits); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXTURESUBIMAGE1DEXTPROC) (GLuint texture, GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLsizei imageSize, const void *bits); +typedef void (APIENTRYP PFNGLGETCOMPRESSEDTEXTUREIMAGEEXTPROC) (GLuint texture, GLenum target, GLint lod, void *img); +typedef void (APIENTRYP PFNGLCOMPRESSEDMULTITEXIMAGE3DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLsizei imageSize, const void *bits); +typedef void (APIENTRYP PFNGLCOMPRESSEDMULTITEXIMAGE2DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLsizei imageSize, const void *bits); +typedef void (APIENTRYP PFNGLCOMPRESSEDMULTITEXIMAGE1DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLenum internalformat, GLsizei width, GLint border, GLsizei imageSize, const void *bits); +typedef void (APIENTRYP PFNGLCOMPRESSEDMULTITEXSUBIMAGE3DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const void *bits); +typedef void (APIENTRYP PFNGLCOMPRESSEDMULTITEXSUBIMAGE2DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const void *bits); +typedef void (APIENTRYP PFNGLCOMPRESSEDMULTITEXSUBIMAGE1DEXTPROC) (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLsizei imageSize, const void *bits); +typedef void (APIENTRYP PFNGLGETCOMPRESSEDMULTITEXIMAGEEXTPROC) (GLenum texunit, GLenum target, GLint lod, void *img); +typedef void (APIENTRYP PFNGLMATRIXLOADTRANSPOSEFEXTPROC) (GLenum mode, const GLfloat *m); +typedef void (APIENTRYP PFNGLMATRIXLOADTRANSPOSEDEXTPROC) (GLenum mode, const GLdouble *m); +typedef void (APIENTRYP PFNGLMATRIXMULTTRANSPOSEFEXTPROC) (GLenum mode, const GLfloat *m); +typedef void (APIENTRYP PFNGLMATRIXMULTTRANSPOSEDEXTPROC) (GLenum mode, const GLdouble *m); +typedef void (APIENTRYP PFNGLNAMEDBUFFERDATAEXTPROC) (GLuint buffer, GLsizeiptr size, const void *data, GLenum usage); +typedef void (APIENTRYP PFNGLNAMEDBUFFERSUBDATAEXTPROC) (GLuint buffer, GLintptr offset, GLsizeiptr size, const void *data); +typedef void *(APIENTRYP PFNGLMAPNAMEDBUFFEREXTPROC) (GLuint buffer, GLenum access); +typedef GLboolean (APIENTRYP PFNGLUNMAPNAMEDBUFFEREXTPROC) (GLuint buffer); +typedef void (APIENTRYP PFNGLGETNAMEDBUFFERPARAMETERIVEXTPROC) (GLuint buffer, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETNAMEDBUFFERPOINTERVEXTPROC) (GLuint buffer, GLenum pname, void **params); +typedef void (APIENTRYP PFNGLGETNAMEDBUFFERSUBDATAEXTPROC) (GLuint buffer, GLintptr offset, GLsizeiptr size, void *data); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1FEXTPROC) (GLuint program, GLint location, GLfloat v0); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2FEXTPROC) (GLuint program, GLint location, GLfloat v0, GLfloat v1); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3FEXTPROC) (GLuint program, GLint location, GLfloat v0, GLfloat v1, GLfloat v2); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4FEXTPROC) (GLuint program, GLint location, GLfloat v0, GLfloat v1, GLfloat v2, GLfloat v3); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1IEXTPROC) (GLuint program, GLint location, GLint v0); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2IEXTPROC) (GLuint program, GLint location, GLint v0, GLint v1); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3IEXTPROC) (GLuint program, GLint location, GLint v0, GLint v1, GLint v2); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4IEXTPROC) (GLuint program, GLint location, GLint v0, GLint v1, GLint v2, GLint v3); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1FVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2FVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3FVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4FVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1IVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2IVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3IVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4IVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX2FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX3FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX4FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX2X3FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX3X2FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX2X4FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX4X2FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX3X4FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX4X3FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLTEXTUREBUFFEREXTPROC) (GLuint texture, GLenum target, GLenum internalformat, GLuint buffer); +typedef void (APIENTRYP PFNGLMULTITEXBUFFEREXTPROC) (GLenum texunit, GLenum target, GLenum internalformat, GLuint buffer); +typedef void (APIENTRYP PFNGLTEXTUREPARAMETERIIVEXTPROC) (GLuint texture, GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLTEXTUREPARAMETERIUIVEXTPROC) (GLuint texture, GLenum target, GLenum pname, const GLuint *params); +typedef void (APIENTRYP PFNGLGETTEXTUREPARAMETERIIVEXTPROC) (GLuint texture, GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETTEXTUREPARAMETERIUIVEXTPROC) (GLuint texture, GLenum target, GLenum pname, GLuint *params); +typedef void (APIENTRYP PFNGLMULTITEXPARAMETERIIVEXTPROC) (GLenum texunit, GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLMULTITEXPARAMETERIUIVEXTPROC) (GLenum texunit, GLenum target, GLenum pname, const GLuint *params); +typedef void (APIENTRYP PFNGLGETMULTITEXPARAMETERIIVEXTPROC) (GLenum texunit, GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETMULTITEXPARAMETERIUIVEXTPROC) (GLenum texunit, GLenum target, GLenum pname, GLuint *params); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1UIEXTPROC) (GLuint program, GLint location, GLuint v0); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2UIEXTPROC) (GLuint program, GLint location, GLuint v0, GLuint v1); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3UIEXTPROC) (GLuint program, GLint location, GLuint v0, GLuint v1, GLuint v2); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4UIEXTPROC) (GLuint program, GLint location, GLuint v0, GLuint v1, GLuint v2, GLuint v3); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1UIVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2UIVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3UIVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4UIVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLNAMEDPROGRAMLOCALPARAMETERS4FVEXTPROC) (GLuint program, GLenum target, GLuint index, GLsizei count, const GLfloat *params); +typedef void (APIENTRYP PFNGLNAMEDPROGRAMLOCALPARAMETERI4IEXTPROC) (GLuint program, GLenum target, GLuint index, GLint x, GLint y, GLint z, GLint w); +typedef void (APIENTRYP PFNGLNAMEDPROGRAMLOCALPARAMETERI4IVEXTPROC) (GLuint program, GLenum target, GLuint index, const GLint *params); +typedef void (APIENTRYP PFNGLNAMEDPROGRAMLOCALPARAMETERSI4IVEXTPROC) (GLuint program, GLenum target, GLuint index, GLsizei count, const GLint *params); +typedef void (APIENTRYP PFNGLNAMEDPROGRAMLOCALPARAMETERI4UIEXTPROC) (GLuint program, GLenum target, GLuint index, GLuint x, GLuint y, GLuint z, GLuint w); +typedef void (APIENTRYP PFNGLNAMEDPROGRAMLOCALPARAMETERI4UIVEXTPROC) (GLuint program, GLenum target, GLuint index, const GLuint *params); +typedef void (APIENTRYP PFNGLNAMEDPROGRAMLOCALPARAMETERSI4UIVEXTPROC) (GLuint program, GLenum target, GLuint index, GLsizei count, const GLuint *params); +typedef void (APIENTRYP PFNGLGETNAMEDPROGRAMLOCALPARAMETERIIVEXTPROC) (GLuint program, GLenum target, GLuint index, GLint *params); +typedef void (APIENTRYP PFNGLGETNAMEDPROGRAMLOCALPARAMETERIUIVEXTPROC) (GLuint program, GLenum target, GLuint index, GLuint *params); +typedef void (APIENTRYP PFNGLENABLECLIENTSTATEIEXTPROC) (GLenum array, GLuint index); +typedef void (APIENTRYP PFNGLDISABLECLIENTSTATEIEXTPROC) (GLenum array, GLuint index); +typedef void (APIENTRYP PFNGLGETFLOATI_VEXTPROC) (GLenum pname, GLuint index, GLfloat *params); +typedef void (APIENTRYP PFNGLGETDOUBLEI_VEXTPROC) (GLenum pname, GLuint index, GLdouble *params); +typedef void (APIENTRYP PFNGLGETPOINTERI_VEXTPROC) (GLenum pname, GLuint index, void **params); +typedef void (APIENTRYP PFNGLNAMEDPROGRAMSTRINGEXTPROC) (GLuint program, GLenum target, GLenum format, GLsizei len, const void *string); +typedef void (APIENTRYP PFNGLNAMEDPROGRAMLOCALPARAMETER4DEXTPROC) (GLuint program, GLenum target, GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLNAMEDPROGRAMLOCALPARAMETER4DVEXTPROC) (GLuint program, GLenum target, GLuint index, const GLdouble *params); +typedef void (APIENTRYP PFNGLNAMEDPROGRAMLOCALPARAMETER4FEXTPROC) (GLuint program, GLenum target, GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLNAMEDPROGRAMLOCALPARAMETER4FVEXTPROC) (GLuint program, GLenum target, GLuint index, const GLfloat *params); +typedef void (APIENTRYP PFNGLGETNAMEDPROGRAMLOCALPARAMETERDVEXTPROC) (GLuint program, GLenum target, GLuint index, GLdouble *params); +typedef void (APIENTRYP PFNGLGETNAMEDPROGRAMLOCALPARAMETERFVEXTPROC) (GLuint program, GLenum target, GLuint index, GLfloat *params); +typedef void (APIENTRYP PFNGLGETNAMEDPROGRAMIVEXTPROC) (GLuint program, GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETNAMEDPROGRAMSTRINGEXTPROC) (GLuint program, GLenum target, GLenum pname, void *string); +typedef void (APIENTRYP PFNGLNAMEDRENDERBUFFERSTORAGEEXTPROC) (GLuint renderbuffer, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLGETNAMEDRENDERBUFFERPARAMETERIVEXTPROC) (GLuint renderbuffer, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLNAMEDRENDERBUFFERSTORAGEMULTISAMPLEEXTPROC) (GLuint renderbuffer, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLNAMEDRENDERBUFFERSTORAGEMULTISAMPLECOVERAGEEXTPROC) (GLuint renderbuffer, GLsizei coverageSamples, GLsizei colorSamples, GLenum internalformat, GLsizei width, GLsizei height); +typedef GLenum (APIENTRYP PFNGLCHECKNAMEDFRAMEBUFFERSTATUSEXTPROC) (GLuint framebuffer, GLenum target); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERTEXTURE1DEXTPROC) (GLuint framebuffer, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERTEXTURE2DEXTPROC) (GLuint framebuffer, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERTEXTURE3DEXTPROC) (GLuint framebuffer, GLenum attachment, GLenum textarget, GLuint texture, GLint level, GLint zoffset); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERRENDERBUFFEREXTPROC) (GLuint framebuffer, GLenum attachment, GLenum renderbuffertarget, GLuint renderbuffer); +typedef void (APIENTRYP PFNGLGETNAMEDFRAMEBUFFERATTACHMENTPARAMETERIVEXTPROC) (GLuint framebuffer, GLenum attachment, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGENERATETEXTUREMIPMAPEXTPROC) (GLuint texture, GLenum target); +typedef void (APIENTRYP PFNGLGENERATEMULTITEXMIPMAPEXTPROC) (GLenum texunit, GLenum target); +typedef void (APIENTRYP PFNGLFRAMEBUFFERDRAWBUFFEREXTPROC) (GLuint framebuffer, GLenum mode); +typedef void (APIENTRYP PFNGLFRAMEBUFFERDRAWBUFFERSEXTPROC) (GLuint framebuffer, GLsizei n, const GLenum *bufs); +typedef void (APIENTRYP PFNGLFRAMEBUFFERREADBUFFEREXTPROC) (GLuint framebuffer, GLenum mode); +typedef void (APIENTRYP PFNGLGETFRAMEBUFFERPARAMETERIVEXTPROC) (GLuint framebuffer, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLNAMEDCOPYBUFFERSUBDATAEXTPROC) (GLuint readBuffer, GLuint writeBuffer, GLintptr readOffset, GLintptr writeOffset, GLsizeiptr size); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERTEXTUREEXTPROC) (GLuint framebuffer, GLenum attachment, GLuint texture, GLint level); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERTEXTURELAYEREXTPROC) (GLuint framebuffer, GLenum attachment, GLuint texture, GLint level, GLint layer); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERTEXTUREFACEEXTPROC) (GLuint framebuffer, GLenum attachment, GLuint texture, GLint level, GLenum face); +typedef void (APIENTRYP PFNGLTEXTURERENDERBUFFEREXTPROC) (GLuint texture, GLenum target, GLuint renderbuffer); +typedef void (APIENTRYP PFNGLMULTITEXRENDERBUFFEREXTPROC) (GLenum texunit, GLenum target, GLuint renderbuffer); +typedef void (APIENTRYP PFNGLVERTEXARRAYVERTEXOFFSETEXTPROC) (GLuint vaobj, GLuint buffer, GLint size, GLenum type, GLsizei stride, GLintptr offset); +typedef void (APIENTRYP PFNGLVERTEXARRAYCOLOROFFSETEXTPROC) (GLuint vaobj, GLuint buffer, GLint size, GLenum type, GLsizei stride, GLintptr offset); +typedef void (APIENTRYP PFNGLVERTEXARRAYEDGEFLAGOFFSETEXTPROC) (GLuint vaobj, GLuint buffer, GLsizei stride, GLintptr offset); +typedef void (APIENTRYP PFNGLVERTEXARRAYINDEXOFFSETEXTPROC) (GLuint vaobj, GLuint buffer, GLenum type, GLsizei stride, GLintptr offset); +typedef void (APIENTRYP PFNGLVERTEXARRAYNORMALOFFSETEXTPROC) (GLuint vaobj, GLuint buffer, GLenum type, GLsizei stride, GLintptr offset); +typedef void (APIENTRYP PFNGLVERTEXARRAYTEXCOORDOFFSETEXTPROC) (GLuint vaobj, GLuint buffer, GLint size, GLenum type, GLsizei stride, GLintptr offset); +typedef void (APIENTRYP PFNGLVERTEXARRAYMULTITEXCOORDOFFSETEXTPROC) (GLuint vaobj, GLuint buffer, GLenum texunit, GLint size, GLenum type, GLsizei stride, GLintptr offset); +typedef void (APIENTRYP PFNGLVERTEXARRAYFOGCOORDOFFSETEXTPROC) (GLuint vaobj, GLuint buffer, GLenum type, GLsizei stride, GLintptr offset); +typedef void (APIENTRYP PFNGLVERTEXARRAYSECONDARYCOLOROFFSETEXTPROC) (GLuint vaobj, GLuint buffer, GLint size, GLenum type, GLsizei stride, GLintptr offset); +typedef void (APIENTRYP PFNGLVERTEXARRAYVERTEXATTRIBOFFSETEXTPROC) (GLuint vaobj, GLuint buffer, GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride, GLintptr offset); +typedef void (APIENTRYP PFNGLVERTEXARRAYVERTEXATTRIBIOFFSETEXTPROC) (GLuint vaobj, GLuint buffer, GLuint index, GLint size, GLenum type, GLsizei stride, GLintptr offset); +typedef void (APIENTRYP PFNGLENABLEVERTEXARRAYEXTPROC) (GLuint vaobj, GLenum array); +typedef void (APIENTRYP PFNGLDISABLEVERTEXARRAYEXTPROC) (GLuint vaobj, GLenum array); +typedef void (APIENTRYP PFNGLENABLEVERTEXARRAYATTRIBEXTPROC) (GLuint vaobj, GLuint index); +typedef void (APIENTRYP PFNGLDISABLEVERTEXARRAYATTRIBEXTPROC) (GLuint vaobj, GLuint index); +typedef void (APIENTRYP PFNGLGETVERTEXARRAYINTEGERVEXTPROC) (GLuint vaobj, GLenum pname, GLint *param); +typedef void (APIENTRYP PFNGLGETVERTEXARRAYPOINTERVEXTPROC) (GLuint vaobj, GLenum pname, void **param); +typedef void (APIENTRYP PFNGLGETVERTEXARRAYINTEGERI_VEXTPROC) (GLuint vaobj, GLuint index, GLenum pname, GLint *param); +typedef void (APIENTRYP PFNGLGETVERTEXARRAYPOINTERI_VEXTPROC) (GLuint vaobj, GLuint index, GLenum pname, void **param); +typedef void *(APIENTRYP PFNGLMAPNAMEDBUFFERRANGEEXTPROC) (GLuint buffer, GLintptr offset, GLsizeiptr length, GLbitfield access); +typedef void (APIENTRYP PFNGLFLUSHMAPPEDNAMEDBUFFERRANGEEXTPROC) (GLuint buffer, GLintptr offset, GLsizeiptr length); +typedef void (APIENTRYP PFNGLNAMEDBUFFERSTORAGEEXTPROC) (GLuint buffer, GLsizeiptr size, const void *data, GLbitfield flags); +typedef void (APIENTRYP PFNGLCLEARNAMEDBUFFERDATAEXTPROC) (GLuint buffer, GLenum internalformat, GLenum format, GLenum type, const void *data); +typedef void (APIENTRYP PFNGLCLEARNAMEDBUFFERSUBDATAEXTPROC) (GLuint buffer, GLenum internalformat, GLsizeiptr offset, GLsizeiptr size, GLenum format, GLenum type, const void *data); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERPARAMETERIEXTPROC) (GLuint framebuffer, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLGETNAMEDFRAMEBUFFERPARAMETERIVEXTPROC) (GLuint framebuffer, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1DEXTPROC) (GLuint program, GLint location, GLdouble x); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2DEXTPROC) (GLuint program, GLint location, GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3DEXTPROC) (GLuint program, GLint location, GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4DEXTPROC) (GLuint program, GLint location, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM1DVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM2DVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM3DVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORM4DVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX2DVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX3DVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX4DVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX2X3DVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX2X4DVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX3X2DVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX3X4DVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX4X2DVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMMATRIX4X3DVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +typedef void (APIENTRYP PFNGLTEXTUREBUFFERRANGEEXTPROC) (GLuint texture, GLenum target, GLenum internalformat, GLuint buffer, GLintptr offset, GLsizeiptr size); +typedef void (APIENTRYP PFNGLTEXTURESTORAGE1DEXTPROC) (GLuint texture, GLenum target, GLsizei levels, GLenum internalformat, GLsizei width); +typedef void (APIENTRYP PFNGLTEXTURESTORAGE2DEXTPROC) (GLuint texture, GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLTEXTURESTORAGE3DEXTPROC) (GLuint texture, GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth); +typedef void (APIENTRYP PFNGLTEXTURESTORAGE2DMULTISAMPLEEXTPROC) (GLuint texture, GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLboolean fixedsamplelocations); +typedef void (APIENTRYP PFNGLTEXTURESTORAGE3DMULTISAMPLEEXTPROC) (GLuint texture, GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedsamplelocations); +typedef void (APIENTRYP PFNGLVERTEXARRAYBINDVERTEXBUFFEREXTPROC) (GLuint vaobj, GLuint bindingindex, GLuint buffer, GLintptr offset, GLsizei stride); +typedef void (APIENTRYP PFNGLVERTEXARRAYVERTEXATTRIBFORMATEXTPROC) (GLuint vaobj, GLuint attribindex, GLint size, GLenum type, GLboolean normalized, GLuint relativeoffset); +typedef void (APIENTRYP PFNGLVERTEXARRAYVERTEXATTRIBIFORMATEXTPROC) (GLuint vaobj, GLuint attribindex, GLint size, GLenum type, GLuint relativeoffset); +typedef void (APIENTRYP PFNGLVERTEXARRAYVERTEXATTRIBLFORMATEXTPROC) (GLuint vaobj, GLuint attribindex, GLint size, GLenum type, GLuint relativeoffset); +typedef void (APIENTRYP PFNGLVERTEXARRAYVERTEXATTRIBBINDINGEXTPROC) (GLuint vaobj, GLuint attribindex, GLuint bindingindex); +typedef void (APIENTRYP PFNGLVERTEXARRAYVERTEXBINDINGDIVISOREXTPROC) (GLuint vaobj, GLuint bindingindex, GLuint divisor); +typedef void (APIENTRYP PFNGLVERTEXARRAYVERTEXATTRIBLOFFSETEXTPROC) (GLuint vaobj, GLuint buffer, GLuint index, GLint size, GLenum type, GLsizei stride, GLintptr offset); +typedef void (APIENTRYP PFNGLTEXTUREPAGECOMMITMENTEXTPROC) (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLboolean commit); +typedef void (APIENTRYP PFNGLVERTEXARRAYVERTEXATTRIBDIVISOREXTPROC) (GLuint vaobj, GLuint index, GLuint divisor); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMatrixLoadfEXT (GLenum mode, const GLfloat *m); +GLAPI void APIENTRY glMatrixLoaddEXT (GLenum mode, const GLdouble *m); +GLAPI void APIENTRY glMatrixMultfEXT (GLenum mode, const GLfloat *m); +GLAPI void APIENTRY glMatrixMultdEXT (GLenum mode, const GLdouble *m); +GLAPI void APIENTRY glMatrixLoadIdentityEXT (GLenum mode); +GLAPI void APIENTRY glMatrixRotatefEXT (GLenum mode, GLfloat angle, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glMatrixRotatedEXT (GLenum mode, GLdouble angle, GLdouble x, GLdouble y, GLdouble z); +GLAPI void APIENTRY glMatrixScalefEXT (GLenum mode, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glMatrixScaledEXT (GLenum mode, GLdouble x, GLdouble y, GLdouble z); +GLAPI void APIENTRY glMatrixTranslatefEXT (GLenum mode, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glMatrixTranslatedEXT (GLenum mode, GLdouble x, GLdouble y, GLdouble z); +GLAPI void APIENTRY glMatrixFrustumEXT (GLenum mode, GLdouble left, GLdouble right, GLdouble bottom, GLdouble top, GLdouble zNear, GLdouble zFar); +GLAPI void APIENTRY glMatrixOrthoEXT (GLenum mode, GLdouble left, GLdouble right, GLdouble bottom, GLdouble top, GLdouble zNear, GLdouble zFar); +GLAPI void APIENTRY glMatrixPopEXT (GLenum mode); +GLAPI void APIENTRY glMatrixPushEXT (GLenum mode); +GLAPI void APIENTRY glClientAttribDefaultEXT (GLbitfield mask); +GLAPI void APIENTRY glPushClientAttribDefaultEXT (GLbitfield mask); +GLAPI void APIENTRY glTextureParameterfEXT (GLuint texture, GLenum target, GLenum pname, GLfloat param); +GLAPI void APIENTRY glTextureParameterfvEXT (GLuint texture, GLenum target, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glTextureParameteriEXT (GLuint texture, GLenum target, GLenum pname, GLint param); +GLAPI void APIENTRY glTextureParameterivEXT (GLuint texture, GLenum target, GLenum pname, const GLint *params); +GLAPI void APIENTRY glTextureImage1DEXT (GLuint texture, GLenum target, GLint level, GLint internalformat, GLsizei width, GLint border, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glTextureImage2DEXT (GLuint texture, GLenum target, GLint level, GLint internalformat, GLsizei width, GLsizei height, GLint border, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glTextureSubImage1DEXT (GLuint texture, GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glTextureSubImage2DEXT (GLuint texture, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glCopyTextureImage1DEXT (GLuint texture, GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLint border); +GLAPI void APIENTRY glCopyTextureImage2DEXT (GLuint texture, GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height, GLint border); +GLAPI void APIENTRY glCopyTextureSubImage1DEXT (GLuint texture, GLenum target, GLint level, GLint xoffset, GLint x, GLint y, GLsizei width); +GLAPI void APIENTRY glCopyTextureSubImage2DEXT (GLuint texture, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint x, GLint y, GLsizei width, GLsizei height); +GLAPI void APIENTRY glGetTextureImageEXT (GLuint texture, GLenum target, GLint level, GLenum format, GLenum type, void *pixels); +GLAPI void APIENTRY glGetTextureParameterfvEXT (GLuint texture, GLenum target, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetTextureParameterivEXT (GLuint texture, GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetTextureLevelParameterfvEXT (GLuint texture, GLenum target, GLint level, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetTextureLevelParameterivEXT (GLuint texture, GLenum target, GLint level, GLenum pname, GLint *params); +GLAPI void APIENTRY glTextureImage3DEXT (GLuint texture, GLenum target, GLint level, GLint internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glTextureSubImage3DEXT (GLuint texture, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glCopyTextureSubImage3DEXT (GLuint texture, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint x, GLint y, GLsizei width, GLsizei height); +GLAPI void APIENTRY glBindMultiTextureEXT (GLenum texunit, GLenum target, GLuint texture); +GLAPI void APIENTRY glMultiTexCoordPointerEXT (GLenum texunit, GLint size, GLenum type, GLsizei stride, const void *pointer); +GLAPI void APIENTRY glMultiTexEnvfEXT (GLenum texunit, GLenum target, GLenum pname, GLfloat param); +GLAPI void APIENTRY glMultiTexEnvfvEXT (GLenum texunit, GLenum target, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glMultiTexEnviEXT (GLenum texunit, GLenum target, GLenum pname, GLint param); +GLAPI void APIENTRY glMultiTexEnvivEXT (GLenum texunit, GLenum target, GLenum pname, const GLint *params); +GLAPI void APIENTRY glMultiTexGendEXT (GLenum texunit, GLenum coord, GLenum pname, GLdouble param); +GLAPI void APIENTRY glMultiTexGendvEXT (GLenum texunit, GLenum coord, GLenum pname, const GLdouble *params); +GLAPI void APIENTRY glMultiTexGenfEXT (GLenum texunit, GLenum coord, GLenum pname, GLfloat param); +GLAPI void APIENTRY glMultiTexGenfvEXT (GLenum texunit, GLenum coord, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glMultiTexGeniEXT (GLenum texunit, GLenum coord, GLenum pname, GLint param); +GLAPI void APIENTRY glMultiTexGenivEXT (GLenum texunit, GLenum coord, GLenum pname, const GLint *params); +GLAPI void APIENTRY glGetMultiTexEnvfvEXT (GLenum texunit, GLenum target, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetMultiTexEnvivEXT (GLenum texunit, GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetMultiTexGendvEXT (GLenum texunit, GLenum coord, GLenum pname, GLdouble *params); +GLAPI void APIENTRY glGetMultiTexGenfvEXT (GLenum texunit, GLenum coord, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetMultiTexGenivEXT (GLenum texunit, GLenum coord, GLenum pname, GLint *params); +GLAPI void APIENTRY glMultiTexParameteriEXT (GLenum texunit, GLenum target, GLenum pname, GLint param); +GLAPI void APIENTRY glMultiTexParameterivEXT (GLenum texunit, GLenum target, GLenum pname, const GLint *params); +GLAPI void APIENTRY glMultiTexParameterfEXT (GLenum texunit, GLenum target, GLenum pname, GLfloat param); +GLAPI void APIENTRY glMultiTexParameterfvEXT (GLenum texunit, GLenum target, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glMultiTexImage1DEXT (GLenum texunit, GLenum target, GLint level, GLint internalformat, GLsizei width, GLint border, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glMultiTexImage2DEXT (GLenum texunit, GLenum target, GLint level, GLint internalformat, GLsizei width, GLsizei height, GLint border, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glMultiTexSubImage1DEXT (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glMultiTexSubImage2DEXT (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glCopyMultiTexImage1DEXT (GLenum texunit, GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLint border); +GLAPI void APIENTRY glCopyMultiTexImage2DEXT (GLenum texunit, GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height, GLint border); +GLAPI void APIENTRY glCopyMultiTexSubImage1DEXT (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLint x, GLint y, GLsizei width); +GLAPI void APIENTRY glCopyMultiTexSubImage2DEXT (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint x, GLint y, GLsizei width, GLsizei height); +GLAPI void APIENTRY glGetMultiTexImageEXT (GLenum texunit, GLenum target, GLint level, GLenum format, GLenum type, void *pixels); +GLAPI void APIENTRY glGetMultiTexParameterfvEXT (GLenum texunit, GLenum target, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetMultiTexParameterivEXT (GLenum texunit, GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetMultiTexLevelParameterfvEXT (GLenum texunit, GLenum target, GLint level, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetMultiTexLevelParameterivEXT (GLenum texunit, GLenum target, GLint level, GLenum pname, GLint *params); +GLAPI void APIENTRY glMultiTexImage3DEXT (GLenum texunit, GLenum target, GLint level, GLint internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glMultiTexSubImage3DEXT (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glCopyMultiTexSubImage3DEXT (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint x, GLint y, GLsizei width, GLsizei height); +GLAPI void APIENTRY glEnableClientStateIndexedEXT (GLenum array, GLuint index); +GLAPI void APIENTRY glDisableClientStateIndexedEXT (GLenum array, GLuint index); +GLAPI void APIENTRY glGetFloatIndexedvEXT (GLenum target, GLuint index, GLfloat *data); +GLAPI void APIENTRY glGetDoubleIndexedvEXT (GLenum target, GLuint index, GLdouble *data); +GLAPI void APIENTRY glGetPointerIndexedvEXT (GLenum target, GLuint index, void **data); +GLAPI void APIENTRY glEnableIndexedEXT (GLenum target, GLuint index); +GLAPI void APIENTRY glDisableIndexedEXT (GLenum target, GLuint index); +GLAPI GLboolean APIENTRY glIsEnabledIndexedEXT (GLenum target, GLuint index); +GLAPI void APIENTRY glGetIntegerIndexedvEXT (GLenum target, GLuint index, GLint *data); +GLAPI void APIENTRY glGetBooleanIndexedvEXT (GLenum target, GLuint index, GLboolean *data); +GLAPI void APIENTRY glCompressedTextureImage3DEXT (GLuint texture, GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLsizei imageSize, const void *bits); +GLAPI void APIENTRY glCompressedTextureImage2DEXT (GLuint texture, GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLsizei imageSize, const void *bits); +GLAPI void APIENTRY glCompressedTextureImage1DEXT (GLuint texture, GLenum target, GLint level, GLenum internalformat, GLsizei width, GLint border, GLsizei imageSize, const void *bits); +GLAPI void APIENTRY glCompressedTextureSubImage3DEXT (GLuint texture, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const void *bits); +GLAPI void APIENTRY glCompressedTextureSubImage2DEXT (GLuint texture, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const void *bits); +GLAPI void APIENTRY glCompressedTextureSubImage1DEXT (GLuint texture, GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLsizei imageSize, const void *bits); +GLAPI void APIENTRY glGetCompressedTextureImageEXT (GLuint texture, GLenum target, GLint lod, void *img); +GLAPI void APIENTRY glCompressedMultiTexImage3DEXT (GLenum texunit, GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLsizei imageSize, const void *bits); +GLAPI void APIENTRY glCompressedMultiTexImage2DEXT (GLenum texunit, GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLsizei imageSize, const void *bits); +GLAPI void APIENTRY glCompressedMultiTexImage1DEXT (GLenum texunit, GLenum target, GLint level, GLenum internalformat, GLsizei width, GLint border, GLsizei imageSize, const void *bits); +GLAPI void APIENTRY glCompressedMultiTexSubImage3DEXT (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const void *bits); +GLAPI void APIENTRY glCompressedMultiTexSubImage2DEXT (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const void *bits); +GLAPI void APIENTRY glCompressedMultiTexSubImage1DEXT (GLenum texunit, GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLsizei imageSize, const void *bits); +GLAPI void APIENTRY glGetCompressedMultiTexImageEXT (GLenum texunit, GLenum target, GLint lod, void *img); +GLAPI void APIENTRY glMatrixLoadTransposefEXT (GLenum mode, const GLfloat *m); +GLAPI void APIENTRY glMatrixLoadTransposedEXT (GLenum mode, const GLdouble *m); +GLAPI void APIENTRY glMatrixMultTransposefEXT (GLenum mode, const GLfloat *m); +GLAPI void APIENTRY glMatrixMultTransposedEXT (GLenum mode, const GLdouble *m); +GLAPI void APIENTRY glNamedBufferDataEXT (GLuint buffer, GLsizeiptr size, const void *data, GLenum usage); +GLAPI void APIENTRY glNamedBufferSubDataEXT (GLuint buffer, GLintptr offset, GLsizeiptr size, const void *data); +GLAPI void *APIENTRY glMapNamedBufferEXT (GLuint buffer, GLenum access); +GLAPI GLboolean APIENTRY glUnmapNamedBufferEXT (GLuint buffer); +GLAPI void APIENTRY glGetNamedBufferParameterivEXT (GLuint buffer, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetNamedBufferPointervEXT (GLuint buffer, GLenum pname, void **params); +GLAPI void APIENTRY glGetNamedBufferSubDataEXT (GLuint buffer, GLintptr offset, GLsizeiptr size, void *data); +GLAPI void APIENTRY glProgramUniform1fEXT (GLuint program, GLint location, GLfloat v0); +GLAPI void APIENTRY glProgramUniform2fEXT (GLuint program, GLint location, GLfloat v0, GLfloat v1); +GLAPI void APIENTRY glProgramUniform3fEXT (GLuint program, GLint location, GLfloat v0, GLfloat v1, GLfloat v2); +GLAPI void APIENTRY glProgramUniform4fEXT (GLuint program, GLint location, GLfloat v0, GLfloat v1, GLfloat v2, GLfloat v3); +GLAPI void APIENTRY glProgramUniform1iEXT (GLuint program, GLint location, GLint v0); +GLAPI void APIENTRY glProgramUniform2iEXT (GLuint program, GLint location, GLint v0, GLint v1); +GLAPI void APIENTRY glProgramUniform3iEXT (GLuint program, GLint location, GLint v0, GLint v1, GLint v2); +GLAPI void APIENTRY glProgramUniform4iEXT (GLuint program, GLint location, GLint v0, GLint v1, GLint v2, GLint v3); +GLAPI void APIENTRY glProgramUniform1fvEXT (GLuint program, GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glProgramUniform2fvEXT (GLuint program, GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glProgramUniform3fvEXT (GLuint program, GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glProgramUniform4fvEXT (GLuint program, GLint location, GLsizei count, const GLfloat *value); +GLAPI void APIENTRY glProgramUniform1ivEXT (GLuint program, GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glProgramUniform2ivEXT (GLuint program, GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glProgramUniform3ivEXT (GLuint program, GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glProgramUniform4ivEXT (GLuint program, GLint location, GLsizei count, const GLint *value); +GLAPI void APIENTRY glProgramUniformMatrix2fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix3fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix4fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix2x3fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix3x2fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix2x4fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix4x2fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix3x4fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glProgramUniformMatrix4x3fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GLAPI void APIENTRY glTextureBufferEXT (GLuint texture, GLenum target, GLenum internalformat, GLuint buffer); +GLAPI void APIENTRY glMultiTexBufferEXT (GLenum texunit, GLenum target, GLenum internalformat, GLuint buffer); +GLAPI void APIENTRY glTextureParameterIivEXT (GLuint texture, GLenum target, GLenum pname, const GLint *params); +GLAPI void APIENTRY glTextureParameterIuivEXT (GLuint texture, GLenum target, GLenum pname, const GLuint *params); +GLAPI void APIENTRY glGetTextureParameterIivEXT (GLuint texture, GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetTextureParameterIuivEXT (GLuint texture, GLenum target, GLenum pname, GLuint *params); +GLAPI void APIENTRY glMultiTexParameterIivEXT (GLenum texunit, GLenum target, GLenum pname, const GLint *params); +GLAPI void APIENTRY glMultiTexParameterIuivEXT (GLenum texunit, GLenum target, GLenum pname, const GLuint *params); +GLAPI void APIENTRY glGetMultiTexParameterIivEXT (GLenum texunit, GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetMultiTexParameterIuivEXT (GLenum texunit, GLenum target, GLenum pname, GLuint *params); +GLAPI void APIENTRY glProgramUniform1uiEXT (GLuint program, GLint location, GLuint v0); +GLAPI void APIENTRY glProgramUniform2uiEXT (GLuint program, GLint location, GLuint v0, GLuint v1); +GLAPI void APIENTRY glProgramUniform3uiEXT (GLuint program, GLint location, GLuint v0, GLuint v1, GLuint v2); +GLAPI void APIENTRY glProgramUniform4uiEXT (GLuint program, GLint location, GLuint v0, GLuint v1, GLuint v2, GLuint v3); +GLAPI void APIENTRY glProgramUniform1uivEXT (GLuint program, GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glProgramUniform2uivEXT (GLuint program, GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glProgramUniform3uivEXT (GLuint program, GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glProgramUniform4uivEXT (GLuint program, GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glNamedProgramLocalParameters4fvEXT (GLuint program, GLenum target, GLuint index, GLsizei count, const GLfloat *params); +GLAPI void APIENTRY glNamedProgramLocalParameterI4iEXT (GLuint program, GLenum target, GLuint index, GLint x, GLint y, GLint z, GLint w); +GLAPI void APIENTRY glNamedProgramLocalParameterI4ivEXT (GLuint program, GLenum target, GLuint index, const GLint *params); +GLAPI void APIENTRY glNamedProgramLocalParametersI4ivEXT (GLuint program, GLenum target, GLuint index, GLsizei count, const GLint *params); +GLAPI void APIENTRY glNamedProgramLocalParameterI4uiEXT (GLuint program, GLenum target, GLuint index, GLuint x, GLuint y, GLuint z, GLuint w); +GLAPI void APIENTRY glNamedProgramLocalParameterI4uivEXT (GLuint program, GLenum target, GLuint index, const GLuint *params); +GLAPI void APIENTRY glNamedProgramLocalParametersI4uivEXT (GLuint program, GLenum target, GLuint index, GLsizei count, const GLuint *params); +GLAPI void APIENTRY glGetNamedProgramLocalParameterIivEXT (GLuint program, GLenum target, GLuint index, GLint *params); +GLAPI void APIENTRY glGetNamedProgramLocalParameterIuivEXT (GLuint program, GLenum target, GLuint index, GLuint *params); +GLAPI void APIENTRY glEnableClientStateiEXT (GLenum array, GLuint index); +GLAPI void APIENTRY glDisableClientStateiEXT (GLenum array, GLuint index); +GLAPI void APIENTRY glGetFloati_vEXT (GLenum pname, GLuint index, GLfloat *params); +GLAPI void APIENTRY glGetDoublei_vEXT (GLenum pname, GLuint index, GLdouble *params); +GLAPI void APIENTRY glGetPointeri_vEXT (GLenum pname, GLuint index, void **params); +GLAPI void APIENTRY glNamedProgramStringEXT (GLuint program, GLenum target, GLenum format, GLsizei len, const void *string); +GLAPI void APIENTRY glNamedProgramLocalParameter4dEXT (GLuint program, GLenum target, GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +GLAPI void APIENTRY glNamedProgramLocalParameter4dvEXT (GLuint program, GLenum target, GLuint index, const GLdouble *params); +GLAPI void APIENTRY glNamedProgramLocalParameter4fEXT (GLuint program, GLenum target, GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +GLAPI void APIENTRY glNamedProgramLocalParameter4fvEXT (GLuint program, GLenum target, GLuint index, const GLfloat *params); +GLAPI void APIENTRY glGetNamedProgramLocalParameterdvEXT (GLuint program, GLenum target, GLuint index, GLdouble *params); +GLAPI void APIENTRY glGetNamedProgramLocalParameterfvEXT (GLuint program, GLenum target, GLuint index, GLfloat *params); +GLAPI void APIENTRY glGetNamedProgramivEXT (GLuint program, GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetNamedProgramStringEXT (GLuint program, GLenum target, GLenum pname, void *string); +GLAPI void APIENTRY glNamedRenderbufferStorageEXT (GLuint renderbuffer, GLenum internalformat, GLsizei width, GLsizei height); +GLAPI void APIENTRY glGetNamedRenderbufferParameterivEXT (GLuint renderbuffer, GLenum pname, GLint *params); +GLAPI void APIENTRY glNamedRenderbufferStorageMultisampleEXT (GLuint renderbuffer, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +GLAPI void APIENTRY glNamedRenderbufferStorageMultisampleCoverageEXT (GLuint renderbuffer, GLsizei coverageSamples, GLsizei colorSamples, GLenum internalformat, GLsizei width, GLsizei height); +GLAPI GLenum APIENTRY glCheckNamedFramebufferStatusEXT (GLuint framebuffer, GLenum target); +GLAPI void APIENTRY glNamedFramebufferTexture1DEXT (GLuint framebuffer, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +GLAPI void APIENTRY glNamedFramebufferTexture2DEXT (GLuint framebuffer, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +GLAPI void APIENTRY glNamedFramebufferTexture3DEXT (GLuint framebuffer, GLenum attachment, GLenum textarget, GLuint texture, GLint level, GLint zoffset); +GLAPI void APIENTRY glNamedFramebufferRenderbufferEXT (GLuint framebuffer, GLenum attachment, GLenum renderbuffertarget, GLuint renderbuffer); +GLAPI void APIENTRY glGetNamedFramebufferAttachmentParameterivEXT (GLuint framebuffer, GLenum attachment, GLenum pname, GLint *params); +GLAPI void APIENTRY glGenerateTextureMipmapEXT (GLuint texture, GLenum target); +GLAPI void APIENTRY glGenerateMultiTexMipmapEXT (GLenum texunit, GLenum target); +GLAPI void APIENTRY glFramebufferDrawBufferEXT (GLuint framebuffer, GLenum mode); +GLAPI void APIENTRY glFramebufferDrawBuffersEXT (GLuint framebuffer, GLsizei n, const GLenum *bufs); +GLAPI void APIENTRY glFramebufferReadBufferEXT (GLuint framebuffer, GLenum mode); +GLAPI void APIENTRY glGetFramebufferParameterivEXT (GLuint framebuffer, GLenum pname, GLint *params); +GLAPI void APIENTRY glNamedCopyBufferSubDataEXT (GLuint readBuffer, GLuint writeBuffer, GLintptr readOffset, GLintptr writeOffset, GLsizeiptr size); +GLAPI void APIENTRY glNamedFramebufferTextureEXT (GLuint framebuffer, GLenum attachment, GLuint texture, GLint level); +GLAPI void APIENTRY glNamedFramebufferTextureLayerEXT (GLuint framebuffer, GLenum attachment, GLuint texture, GLint level, GLint layer); +GLAPI void APIENTRY glNamedFramebufferTextureFaceEXT (GLuint framebuffer, GLenum attachment, GLuint texture, GLint level, GLenum face); +GLAPI void APIENTRY glTextureRenderbufferEXT (GLuint texture, GLenum target, GLuint renderbuffer); +GLAPI void APIENTRY glMultiTexRenderbufferEXT (GLenum texunit, GLenum target, GLuint renderbuffer); +GLAPI void APIENTRY glVertexArrayVertexOffsetEXT (GLuint vaobj, GLuint buffer, GLint size, GLenum type, GLsizei stride, GLintptr offset); +GLAPI void APIENTRY glVertexArrayColorOffsetEXT (GLuint vaobj, GLuint buffer, GLint size, GLenum type, GLsizei stride, GLintptr offset); +GLAPI void APIENTRY glVertexArrayEdgeFlagOffsetEXT (GLuint vaobj, GLuint buffer, GLsizei stride, GLintptr offset); +GLAPI void APIENTRY glVertexArrayIndexOffsetEXT (GLuint vaobj, GLuint buffer, GLenum type, GLsizei stride, GLintptr offset); +GLAPI void APIENTRY glVertexArrayNormalOffsetEXT (GLuint vaobj, GLuint buffer, GLenum type, GLsizei stride, GLintptr offset); +GLAPI void APIENTRY glVertexArrayTexCoordOffsetEXT (GLuint vaobj, GLuint buffer, GLint size, GLenum type, GLsizei stride, GLintptr offset); +GLAPI void APIENTRY glVertexArrayMultiTexCoordOffsetEXT (GLuint vaobj, GLuint buffer, GLenum texunit, GLint size, GLenum type, GLsizei stride, GLintptr offset); +GLAPI void APIENTRY glVertexArrayFogCoordOffsetEXT (GLuint vaobj, GLuint buffer, GLenum type, GLsizei stride, GLintptr offset); +GLAPI void APIENTRY glVertexArraySecondaryColorOffsetEXT (GLuint vaobj, GLuint buffer, GLint size, GLenum type, GLsizei stride, GLintptr offset); +GLAPI void APIENTRY glVertexArrayVertexAttribOffsetEXT (GLuint vaobj, GLuint buffer, GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride, GLintptr offset); +GLAPI void APIENTRY glVertexArrayVertexAttribIOffsetEXT (GLuint vaobj, GLuint buffer, GLuint index, GLint size, GLenum type, GLsizei stride, GLintptr offset); +GLAPI void APIENTRY glEnableVertexArrayEXT (GLuint vaobj, GLenum array); +GLAPI void APIENTRY glDisableVertexArrayEXT (GLuint vaobj, GLenum array); +GLAPI void APIENTRY glEnableVertexArrayAttribEXT (GLuint vaobj, GLuint index); +GLAPI void APIENTRY glDisableVertexArrayAttribEXT (GLuint vaobj, GLuint index); +GLAPI void APIENTRY glGetVertexArrayIntegervEXT (GLuint vaobj, GLenum pname, GLint *param); +GLAPI void APIENTRY glGetVertexArrayPointervEXT (GLuint vaobj, GLenum pname, void **param); +GLAPI void APIENTRY glGetVertexArrayIntegeri_vEXT (GLuint vaobj, GLuint index, GLenum pname, GLint *param); +GLAPI void APIENTRY glGetVertexArrayPointeri_vEXT (GLuint vaobj, GLuint index, GLenum pname, void **param); +GLAPI void *APIENTRY glMapNamedBufferRangeEXT (GLuint buffer, GLintptr offset, GLsizeiptr length, GLbitfield access); +GLAPI void APIENTRY glFlushMappedNamedBufferRangeEXT (GLuint buffer, GLintptr offset, GLsizeiptr length); +GLAPI void APIENTRY glNamedBufferStorageEXT (GLuint buffer, GLsizeiptr size, const void *data, GLbitfield flags); +GLAPI void APIENTRY glClearNamedBufferDataEXT (GLuint buffer, GLenum internalformat, GLenum format, GLenum type, const void *data); +GLAPI void APIENTRY glClearNamedBufferSubDataEXT (GLuint buffer, GLenum internalformat, GLsizeiptr offset, GLsizeiptr size, GLenum format, GLenum type, const void *data); +GLAPI void APIENTRY glNamedFramebufferParameteriEXT (GLuint framebuffer, GLenum pname, GLint param); +GLAPI void APIENTRY glGetNamedFramebufferParameterivEXT (GLuint framebuffer, GLenum pname, GLint *params); +GLAPI void APIENTRY glProgramUniform1dEXT (GLuint program, GLint location, GLdouble x); +GLAPI void APIENTRY glProgramUniform2dEXT (GLuint program, GLint location, GLdouble x, GLdouble y); +GLAPI void APIENTRY glProgramUniform3dEXT (GLuint program, GLint location, GLdouble x, GLdouble y, GLdouble z); +GLAPI void APIENTRY glProgramUniform4dEXT (GLuint program, GLint location, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +GLAPI void APIENTRY glProgramUniform1dvEXT (GLuint program, GLint location, GLsizei count, const GLdouble *value); +GLAPI void APIENTRY glProgramUniform2dvEXT (GLuint program, GLint location, GLsizei count, const GLdouble *value); +GLAPI void APIENTRY glProgramUniform3dvEXT (GLuint program, GLint location, GLsizei count, const GLdouble *value); +GLAPI void APIENTRY glProgramUniform4dvEXT (GLuint program, GLint location, GLsizei count, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix2dvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix3dvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix4dvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix2x3dvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix2x4dvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix3x2dvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix3x4dvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix4x2dvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glProgramUniformMatrix4x3dvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLdouble *value); +GLAPI void APIENTRY glTextureBufferRangeEXT (GLuint texture, GLenum target, GLenum internalformat, GLuint buffer, GLintptr offset, GLsizeiptr size); +GLAPI void APIENTRY glTextureStorage1DEXT (GLuint texture, GLenum target, GLsizei levels, GLenum internalformat, GLsizei width); +GLAPI void APIENTRY glTextureStorage2DEXT (GLuint texture, GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height); +GLAPI void APIENTRY glTextureStorage3DEXT (GLuint texture, GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth); +GLAPI void APIENTRY glTextureStorage2DMultisampleEXT (GLuint texture, GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLboolean fixedsamplelocations); +GLAPI void APIENTRY glTextureStorage3DMultisampleEXT (GLuint texture, GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedsamplelocations); +GLAPI void APIENTRY glVertexArrayBindVertexBufferEXT (GLuint vaobj, GLuint bindingindex, GLuint buffer, GLintptr offset, GLsizei stride); +GLAPI void APIENTRY glVertexArrayVertexAttribFormatEXT (GLuint vaobj, GLuint attribindex, GLint size, GLenum type, GLboolean normalized, GLuint relativeoffset); +GLAPI void APIENTRY glVertexArrayVertexAttribIFormatEXT (GLuint vaobj, GLuint attribindex, GLint size, GLenum type, GLuint relativeoffset); +GLAPI void APIENTRY glVertexArrayVertexAttribLFormatEXT (GLuint vaobj, GLuint attribindex, GLint size, GLenum type, GLuint relativeoffset); +GLAPI void APIENTRY glVertexArrayVertexAttribBindingEXT (GLuint vaobj, GLuint attribindex, GLuint bindingindex); +GLAPI void APIENTRY glVertexArrayVertexBindingDivisorEXT (GLuint vaobj, GLuint bindingindex, GLuint divisor); +GLAPI void APIENTRY glVertexArrayVertexAttribLOffsetEXT (GLuint vaobj, GLuint buffer, GLuint index, GLint size, GLenum type, GLsizei stride, GLintptr offset); +GLAPI void APIENTRY glTexturePageCommitmentEXT (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLboolean commit); +GLAPI void APIENTRY glVertexArrayVertexAttribDivisorEXT (GLuint vaobj, GLuint index, GLuint divisor); +#endif +#endif /* GL_EXT_direct_state_access */ + +#ifndef GL_EXT_draw_buffers2 +#define GL_EXT_draw_buffers2 1 +typedef void (APIENTRYP PFNGLCOLORMASKINDEXEDEXTPROC) (GLuint index, GLboolean r, GLboolean g, GLboolean b, GLboolean a); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glColorMaskIndexedEXT (GLuint index, GLboolean r, GLboolean g, GLboolean b, GLboolean a); +#endif +#endif /* GL_EXT_draw_buffers2 */ + +#ifndef GL_EXT_draw_instanced +#define GL_EXT_draw_instanced 1 +typedef void (APIENTRYP PFNGLDRAWARRAYSINSTANCEDEXTPROC) (GLenum mode, GLint start, GLsizei count, GLsizei primcount); +typedef void (APIENTRYP PFNGLDRAWELEMENTSINSTANCEDEXTPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei primcount); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawArraysInstancedEXT (GLenum mode, GLint start, GLsizei count, GLsizei primcount); +GLAPI void APIENTRY glDrawElementsInstancedEXT (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei primcount); +#endif +#endif /* GL_EXT_draw_instanced */ + +#ifndef GL_EXT_draw_range_elements +#define GL_EXT_draw_range_elements 1 +#define GL_MAX_ELEMENTS_VERTICES_EXT 0x80E8 +#define GL_MAX_ELEMENTS_INDICES_EXT 0x80E9 +typedef void (APIENTRYP PFNGLDRAWRANGEELEMENTSEXTPROC) (GLenum mode, GLuint start, GLuint end, GLsizei count, GLenum type, const void *indices); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawRangeElementsEXT (GLenum mode, GLuint start, GLuint end, GLsizei count, GLenum type, const void *indices); +#endif +#endif /* GL_EXT_draw_range_elements */ + +#ifndef GL_EXT_external_buffer +#define GL_EXT_external_buffer 1 +typedef void *GLeglClientBufferEXT; +typedef void (APIENTRYP PFNGLBUFFERSTORAGEEXTERNALEXTPROC) (GLenum target, GLintptr offset, GLsizeiptr size, GLeglClientBufferEXT clientBuffer, GLbitfield flags); +typedef void (APIENTRYP PFNGLNAMEDBUFFERSTORAGEEXTERNALEXTPROC) (GLuint buffer, GLintptr offset, GLsizeiptr size, GLeglClientBufferEXT clientBuffer, GLbitfield flags); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBufferStorageExternalEXT (GLenum target, GLintptr offset, GLsizeiptr size, GLeglClientBufferEXT clientBuffer, GLbitfield flags); +GLAPI void APIENTRY glNamedBufferStorageExternalEXT (GLuint buffer, GLintptr offset, GLsizeiptr size, GLeglClientBufferEXT clientBuffer, GLbitfield flags); +#endif +#endif /* GL_EXT_external_buffer */ + +#ifndef GL_EXT_fog_coord +#define GL_EXT_fog_coord 1 +#define GL_FOG_COORDINATE_SOURCE_EXT 0x8450 +#define GL_FOG_COORDINATE_EXT 0x8451 +#define GL_FRAGMENT_DEPTH_EXT 0x8452 +#define GL_CURRENT_FOG_COORDINATE_EXT 0x8453 +#define GL_FOG_COORDINATE_ARRAY_TYPE_EXT 0x8454 +#define GL_FOG_COORDINATE_ARRAY_STRIDE_EXT 0x8455 +#define GL_FOG_COORDINATE_ARRAY_POINTER_EXT 0x8456 +#define GL_FOG_COORDINATE_ARRAY_EXT 0x8457 +typedef void (APIENTRYP PFNGLFOGCOORDFEXTPROC) (GLfloat coord); +typedef void (APIENTRYP PFNGLFOGCOORDFVEXTPROC) (const GLfloat *coord); +typedef void (APIENTRYP PFNGLFOGCOORDDEXTPROC) (GLdouble coord); +typedef void (APIENTRYP PFNGLFOGCOORDDVEXTPROC) (const GLdouble *coord); +typedef void (APIENTRYP PFNGLFOGCOORDPOINTEREXTPROC) (GLenum type, GLsizei stride, const void *pointer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFogCoordfEXT (GLfloat coord); +GLAPI void APIENTRY glFogCoordfvEXT (const GLfloat *coord); +GLAPI void APIENTRY glFogCoorddEXT (GLdouble coord); +GLAPI void APIENTRY glFogCoorddvEXT (const GLdouble *coord); +GLAPI void APIENTRY glFogCoordPointerEXT (GLenum type, GLsizei stride, const void *pointer); +#endif +#endif /* GL_EXT_fog_coord */ + +#ifndef GL_EXT_framebuffer_blit +#define GL_EXT_framebuffer_blit 1 +#define GL_READ_FRAMEBUFFER_EXT 0x8CA8 +#define GL_DRAW_FRAMEBUFFER_EXT 0x8CA9 +#define GL_DRAW_FRAMEBUFFER_BINDING_EXT 0x8CA6 +#define GL_READ_FRAMEBUFFER_BINDING_EXT 0x8CAA +typedef void (APIENTRYP PFNGLBLITFRAMEBUFFEREXTPROC) (GLint srcX0, GLint srcY0, GLint srcX1, GLint srcY1, GLint dstX0, GLint dstY0, GLint dstX1, GLint dstY1, GLbitfield mask, GLenum filter); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlitFramebufferEXT (GLint srcX0, GLint srcY0, GLint srcX1, GLint srcY1, GLint dstX0, GLint dstY0, GLint dstX1, GLint dstY1, GLbitfield mask, GLenum filter); +#endif +#endif /* GL_EXT_framebuffer_blit */ + +#ifndef GL_EXT_framebuffer_multisample +#define GL_EXT_framebuffer_multisample 1 +#define GL_RENDERBUFFER_SAMPLES_EXT 0x8CAB +#define GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE_EXT 0x8D56 +#define GL_MAX_SAMPLES_EXT 0x8D57 +typedef void (APIENTRYP PFNGLRENDERBUFFERSTORAGEMULTISAMPLEEXTPROC) (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glRenderbufferStorageMultisampleEXT (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +#endif +#endif /* GL_EXT_framebuffer_multisample */ + +#ifndef GL_EXT_framebuffer_multisample_blit_scaled +#define GL_EXT_framebuffer_multisample_blit_scaled 1 +#define GL_SCALED_RESOLVE_FASTEST_EXT 0x90BA +#define GL_SCALED_RESOLVE_NICEST_EXT 0x90BB +#endif /* GL_EXT_framebuffer_multisample_blit_scaled */ + +#ifndef GL_EXT_framebuffer_object +#define GL_EXT_framebuffer_object 1 +#define GL_INVALID_FRAMEBUFFER_OPERATION_EXT 0x0506 +#define GL_MAX_RENDERBUFFER_SIZE_EXT 0x84E8 +#define GL_FRAMEBUFFER_BINDING_EXT 0x8CA6 +#define GL_RENDERBUFFER_BINDING_EXT 0x8CA7 +#define GL_FRAMEBUFFER_ATTACHMENT_OBJECT_TYPE_EXT 0x8CD0 +#define GL_FRAMEBUFFER_ATTACHMENT_OBJECT_NAME_EXT 0x8CD1 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_LEVEL_EXT 0x8CD2 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_CUBE_MAP_FACE_EXT 0x8CD3 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_3D_ZOFFSET_EXT 0x8CD4 +#define GL_FRAMEBUFFER_COMPLETE_EXT 0x8CD5 +#define GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT_EXT 0x8CD6 +#define GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT_EXT 0x8CD7 +#define GL_FRAMEBUFFER_INCOMPLETE_DIMENSIONS_EXT 0x8CD9 +#define GL_FRAMEBUFFER_INCOMPLETE_FORMATS_EXT 0x8CDA +#define GL_FRAMEBUFFER_INCOMPLETE_DRAW_BUFFER_EXT 0x8CDB +#define GL_FRAMEBUFFER_INCOMPLETE_READ_BUFFER_EXT 0x8CDC +#define GL_FRAMEBUFFER_UNSUPPORTED_EXT 0x8CDD +#define GL_MAX_COLOR_ATTACHMENTS_EXT 0x8CDF +#define GL_COLOR_ATTACHMENT0_EXT 0x8CE0 +#define GL_COLOR_ATTACHMENT1_EXT 0x8CE1 +#define GL_COLOR_ATTACHMENT2_EXT 0x8CE2 +#define GL_COLOR_ATTACHMENT3_EXT 0x8CE3 +#define GL_COLOR_ATTACHMENT4_EXT 0x8CE4 +#define GL_COLOR_ATTACHMENT5_EXT 0x8CE5 +#define GL_COLOR_ATTACHMENT6_EXT 0x8CE6 +#define GL_COLOR_ATTACHMENT7_EXT 0x8CE7 +#define GL_COLOR_ATTACHMENT8_EXT 0x8CE8 +#define GL_COLOR_ATTACHMENT9_EXT 0x8CE9 +#define GL_COLOR_ATTACHMENT10_EXT 0x8CEA +#define GL_COLOR_ATTACHMENT11_EXT 0x8CEB +#define GL_COLOR_ATTACHMENT12_EXT 0x8CEC +#define GL_COLOR_ATTACHMENT13_EXT 0x8CED +#define GL_COLOR_ATTACHMENT14_EXT 0x8CEE +#define GL_COLOR_ATTACHMENT15_EXT 0x8CEF +#define GL_DEPTH_ATTACHMENT_EXT 0x8D00 +#define GL_STENCIL_ATTACHMENT_EXT 0x8D20 +#define GL_FRAMEBUFFER_EXT 0x8D40 +#define GL_RENDERBUFFER_EXT 0x8D41 +#define GL_RENDERBUFFER_WIDTH_EXT 0x8D42 +#define GL_RENDERBUFFER_HEIGHT_EXT 0x8D43 +#define GL_RENDERBUFFER_INTERNAL_FORMAT_EXT 0x8D44 +#define GL_STENCIL_INDEX1_EXT 0x8D46 +#define GL_STENCIL_INDEX4_EXT 0x8D47 +#define GL_STENCIL_INDEX8_EXT 0x8D48 +#define GL_STENCIL_INDEX16_EXT 0x8D49 +#define GL_RENDERBUFFER_RED_SIZE_EXT 0x8D50 +#define GL_RENDERBUFFER_GREEN_SIZE_EXT 0x8D51 +#define GL_RENDERBUFFER_BLUE_SIZE_EXT 0x8D52 +#define GL_RENDERBUFFER_ALPHA_SIZE_EXT 0x8D53 +#define GL_RENDERBUFFER_DEPTH_SIZE_EXT 0x8D54 +#define GL_RENDERBUFFER_STENCIL_SIZE_EXT 0x8D55 +typedef GLboolean (APIENTRYP PFNGLISRENDERBUFFEREXTPROC) (GLuint renderbuffer); +typedef void (APIENTRYP PFNGLBINDRENDERBUFFEREXTPROC) (GLenum target, GLuint renderbuffer); +typedef void (APIENTRYP PFNGLDELETERENDERBUFFERSEXTPROC) (GLsizei n, const GLuint *renderbuffers); +typedef void (APIENTRYP PFNGLGENRENDERBUFFERSEXTPROC) (GLsizei n, GLuint *renderbuffers); +typedef void (APIENTRYP PFNGLRENDERBUFFERSTORAGEEXTPROC) (GLenum target, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLGETRENDERBUFFERPARAMETERIVEXTPROC) (GLenum target, GLenum pname, GLint *params); +typedef GLboolean (APIENTRYP PFNGLISFRAMEBUFFEREXTPROC) (GLuint framebuffer); +typedef void (APIENTRYP PFNGLBINDFRAMEBUFFEREXTPROC) (GLenum target, GLuint framebuffer); +typedef void (APIENTRYP PFNGLDELETEFRAMEBUFFERSEXTPROC) (GLsizei n, const GLuint *framebuffers); +typedef void (APIENTRYP PFNGLGENFRAMEBUFFERSEXTPROC) (GLsizei n, GLuint *framebuffers); +typedef GLenum (APIENTRYP PFNGLCHECKFRAMEBUFFERSTATUSEXTPROC) (GLenum target); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTURE1DEXTPROC) (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTURE2DEXTPROC) (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTURE3DEXTPROC) (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level, GLint zoffset); +typedef void (APIENTRYP PFNGLFRAMEBUFFERRENDERBUFFEREXTPROC) (GLenum target, GLenum attachment, GLenum renderbuffertarget, GLuint renderbuffer); +typedef void (APIENTRYP PFNGLGETFRAMEBUFFERATTACHMENTPARAMETERIVEXTPROC) (GLenum target, GLenum attachment, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGENERATEMIPMAPEXTPROC) (GLenum target); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLboolean APIENTRY glIsRenderbufferEXT (GLuint renderbuffer); +GLAPI void APIENTRY glBindRenderbufferEXT (GLenum target, GLuint renderbuffer); +GLAPI void APIENTRY glDeleteRenderbuffersEXT (GLsizei n, const GLuint *renderbuffers); +GLAPI void APIENTRY glGenRenderbuffersEXT (GLsizei n, GLuint *renderbuffers); +GLAPI void APIENTRY glRenderbufferStorageEXT (GLenum target, GLenum internalformat, GLsizei width, GLsizei height); +GLAPI void APIENTRY glGetRenderbufferParameterivEXT (GLenum target, GLenum pname, GLint *params); +GLAPI GLboolean APIENTRY glIsFramebufferEXT (GLuint framebuffer); +GLAPI void APIENTRY glBindFramebufferEXT (GLenum target, GLuint framebuffer); +GLAPI void APIENTRY glDeleteFramebuffersEXT (GLsizei n, const GLuint *framebuffers); +GLAPI void APIENTRY glGenFramebuffersEXT (GLsizei n, GLuint *framebuffers); +GLAPI GLenum APIENTRY glCheckFramebufferStatusEXT (GLenum target); +GLAPI void APIENTRY glFramebufferTexture1DEXT (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +GLAPI void APIENTRY glFramebufferTexture2DEXT (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +GLAPI void APIENTRY glFramebufferTexture3DEXT (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level, GLint zoffset); +GLAPI void APIENTRY glFramebufferRenderbufferEXT (GLenum target, GLenum attachment, GLenum renderbuffertarget, GLuint renderbuffer); +GLAPI void APIENTRY glGetFramebufferAttachmentParameterivEXT (GLenum target, GLenum attachment, GLenum pname, GLint *params); +GLAPI void APIENTRY glGenerateMipmapEXT (GLenum target); +#endif +#endif /* GL_EXT_framebuffer_object */ + +#ifndef GL_EXT_framebuffer_sRGB +#define GL_EXT_framebuffer_sRGB 1 +#define GL_FRAMEBUFFER_SRGB_EXT 0x8DB9 +#define GL_FRAMEBUFFER_SRGB_CAPABLE_EXT 0x8DBA +#endif /* GL_EXT_framebuffer_sRGB */ + +#ifndef GL_EXT_geometry_shader4 +#define GL_EXT_geometry_shader4 1 +#define GL_GEOMETRY_SHADER_EXT 0x8DD9 +#define GL_GEOMETRY_VERTICES_OUT_EXT 0x8DDA +#define GL_GEOMETRY_INPUT_TYPE_EXT 0x8DDB +#define GL_GEOMETRY_OUTPUT_TYPE_EXT 0x8DDC +#define GL_MAX_GEOMETRY_TEXTURE_IMAGE_UNITS_EXT 0x8C29 +#define GL_MAX_GEOMETRY_VARYING_COMPONENTS_EXT 0x8DDD +#define GL_MAX_VERTEX_VARYING_COMPONENTS_EXT 0x8DDE +#define GL_MAX_VARYING_COMPONENTS_EXT 0x8B4B +#define GL_MAX_GEOMETRY_UNIFORM_COMPONENTS_EXT 0x8DDF +#define GL_MAX_GEOMETRY_OUTPUT_VERTICES_EXT 0x8DE0 +#define GL_MAX_GEOMETRY_TOTAL_OUTPUT_COMPONENTS_EXT 0x8DE1 +#define GL_LINES_ADJACENCY_EXT 0x000A +#define GL_LINE_STRIP_ADJACENCY_EXT 0x000B +#define GL_TRIANGLES_ADJACENCY_EXT 0x000C +#define GL_TRIANGLE_STRIP_ADJACENCY_EXT 0x000D +#define GL_FRAMEBUFFER_INCOMPLETE_LAYER_TARGETS_EXT 0x8DA8 +#define GL_FRAMEBUFFER_INCOMPLETE_LAYER_COUNT_EXT 0x8DA9 +#define GL_FRAMEBUFFER_ATTACHMENT_LAYERED_EXT 0x8DA7 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_LAYER_EXT 0x8CD4 +#define GL_PROGRAM_POINT_SIZE_EXT 0x8642 +typedef void (APIENTRYP PFNGLPROGRAMPARAMETERIEXTPROC) (GLuint program, GLenum pname, GLint value); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glProgramParameteriEXT (GLuint program, GLenum pname, GLint value); +#endif +#endif /* GL_EXT_geometry_shader4 */ + +#ifndef GL_EXT_gpu_program_parameters +#define GL_EXT_gpu_program_parameters 1 +typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETERS4FVEXTPROC) (GLenum target, GLuint index, GLsizei count, const GLfloat *params); +typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETERS4FVEXTPROC) (GLenum target, GLuint index, GLsizei count, const GLfloat *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glProgramEnvParameters4fvEXT (GLenum target, GLuint index, GLsizei count, const GLfloat *params); +GLAPI void APIENTRY glProgramLocalParameters4fvEXT (GLenum target, GLuint index, GLsizei count, const GLfloat *params); +#endif +#endif /* GL_EXT_gpu_program_parameters */ + +#ifndef GL_EXT_gpu_shader4 +#define GL_EXT_gpu_shader4 1 +#define GL_SAMPLER_1D_ARRAY_EXT 0x8DC0 +#define GL_SAMPLER_2D_ARRAY_EXT 0x8DC1 +#define GL_SAMPLER_BUFFER_EXT 0x8DC2 +#define GL_SAMPLER_1D_ARRAY_SHADOW_EXT 0x8DC3 +#define GL_SAMPLER_2D_ARRAY_SHADOW_EXT 0x8DC4 +#define GL_SAMPLER_CUBE_SHADOW_EXT 0x8DC5 +#define GL_UNSIGNED_INT_VEC2_EXT 0x8DC6 +#define GL_UNSIGNED_INT_VEC3_EXT 0x8DC7 +#define GL_UNSIGNED_INT_VEC4_EXT 0x8DC8 +#define GL_INT_SAMPLER_1D_EXT 0x8DC9 +#define GL_INT_SAMPLER_2D_EXT 0x8DCA +#define GL_INT_SAMPLER_3D_EXT 0x8DCB +#define GL_INT_SAMPLER_CUBE_EXT 0x8DCC +#define GL_INT_SAMPLER_2D_RECT_EXT 0x8DCD +#define GL_INT_SAMPLER_1D_ARRAY_EXT 0x8DCE +#define GL_INT_SAMPLER_2D_ARRAY_EXT 0x8DCF +#define GL_INT_SAMPLER_BUFFER_EXT 0x8DD0 +#define GL_UNSIGNED_INT_SAMPLER_1D_EXT 0x8DD1 +#define GL_UNSIGNED_INT_SAMPLER_2D_EXT 0x8DD2 +#define GL_UNSIGNED_INT_SAMPLER_3D_EXT 0x8DD3 +#define GL_UNSIGNED_INT_SAMPLER_CUBE_EXT 0x8DD4 +#define GL_UNSIGNED_INT_SAMPLER_2D_RECT_EXT 0x8DD5 +#define GL_UNSIGNED_INT_SAMPLER_1D_ARRAY_EXT 0x8DD6 +#define GL_UNSIGNED_INT_SAMPLER_2D_ARRAY_EXT 0x8DD7 +#define GL_UNSIGNED_INT_SAMPLER_BUFFER_EXT 0x8DD8 +#define GL_MIN_PROGRAM_TEXEL_OFFSET_EXT 0x8904 +#define GL_MAX_PROGRAM_TEXEL_OFFSET_EXT 0x8905 +#define GL_VERTEX_ATTRIB_ARRAY_INTEGER_EXT 0x88FD +typedef void (APIENTRYP PFNGLGETUNIFORMUIVEXTPROC) (GLuint program, GLint location, GLuint *params); +typedef void (APIENTRYP PFNGLBINDFRAGDATALOCATIONEXTPROC) (GLuint program, GLuint color, const GLchar *name); +typedef GLint (APIENTRYP PFNGLGETFRAGDATALOCATIONEXTPROC) (GLuint program, const GLchar *name); +typedef void (APIENTRYP PFNGLUNIFORM1UIEXTPROC) (GLint location, GLuint v0); +typedef void (APIENTRYP PFNGLUNIFORM2UIEXTPROC) (GLint location, GLuint v0, GLuint v1); +typedef void (APIENTRYP PFNGLUNIFORM3UIEXTPROC) (GLint location, GLuint v0, GLuint v1, GLuint v2); +typedef void (APIENTRYP PFNGLUNIFORM4UIEXTPROC) (GLint location, GLuint v0, GLuint v1, GLuint v2, GLuint v3); +typedef void (APIENTRYP PFNGLUNIFORM1UIVEXTPROC) (GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLUNIFORM2UIVEXTPROC) (GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLUNIFORM3UIVEXTPROC) (GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLUNIFORM4UIVEXTPROC) (GLint location, GLsizei count, const GLuint *value); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI1IEXTPROC) (GLuint index, GLint x); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI2IEXTPROC) (GLuint index, GLint x, GLint y); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI3IEXTPROC) (GLuint index, GLint x, GLint y, GLint z); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4IEXTPROC) (GLuint index, GLint x, GLint y, GLint z, GLint w); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI1UIEXTPROC) (GLuint index, GLuint x); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI2UIEXTPROC) (GLuint index, GLuint x, GLuint y); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI3UIEXTPROC) (GLuint index, GLuint x, GLuint y, GLuint z); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4UIEXTPROC) (GLuint index, GLuint x, GLuint y, GLuint z, GLuint w); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI1IVEXTPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI2IVEXTPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI3IVEXTPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4IVEXTPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI1UIVEXTPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI2UIVEXTPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI3UIVEXTPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4UIVEXTPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4BVEXTPROC) (GLuint index, const GLbyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4SVEXTPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4UBVEXTPROC) (GLuint index, const GLubyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBI4USVEXTPROC) (GLuint index, const GLushort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBIPOINTEREXTPROC) (GLuint index, GLint size, GLenum type, GLsizei stride, const void *pointer); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBIIVEXTPROC) (GLuint index, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBIUIVEXTPROC) (GLuint index, GLenum pname, GLuint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGetUniformuivEXT (GLuint program, GLint location, GLuint *params); +GLAPI void APIENTRY glBindFragDataLocationEXT (GLuint program, GLuint color, const GLchar *name); +GLAPI GLint APIENTRY glGetFragDataLocationEXT (GLuint program, const GLchar *name); +GLAPI void APIENTRY glUniform1uiEXT (GLint location, GLuint v0); +GLAPI void APIENTRY glUniform2uiEXT (GLint location, GLuint v0, GLuint v1); +GLAPI void APIENTRY glUniform3uiEXT (GLint location, GLuint v0, GLuint v1, GLuint v2); +GLAPI void APIENTRY glUniform4uiEXT (GLint location, GLuint v0, GLuint v1, GLuint v2, GLuint v3); +GLAPI void APIENTRY glUniform1uivEXT (GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glUniform2uivEXT (GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glUniform3uivEXT (GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glUniform4uivEXT (GLint location, GLsizei count, const GLuint *value); +GLAPI void APIENTRY glVertexAttribI1iEXT (GLuint index, GLint x); +GLAPI void APIENTRY glVertexAttribI2iEXT (GLuint index, GLint x, GLint y); +GLAPI void APIENTRY glVertexAttribI3iEXT (GLuint index, GLint x, GLint y, GLint z); +GLAPI void APIENTRY glVertexAttribI4iEXT (GLuint index, GLint x, GLint y, GLint z, GLint w); +GLAPI void APIENTRY glVertexAttribI1uiEXT (GLuint index, GLuint x); +GLAPI void APIENTRY glVertexAttribI2uiEXT (GLuint index, GLuint x, GLuint y); +GLAPI void APIENTRY glVertexAttribI3uiEXT (GLuint index, GLuint x, GLuint y, GLuint z); +GLAPI void APIENTRY glVertexAttribI4uiEXT (GLuint index, GLuint x, GLuint y, GLuint z, GLuint w); +GLAPI void APIENTRY glVertexAttribI1ivEXT (GLuint index, const GLint *v); +GLAPI void APIENTRY glVertexAttribI2ivEXT (GLuint index, const GLint *v); +GLAPI void APIENTRY glVertexAttribI3ivEXT (GLuint index, const GLint *v); +GLAPI void APIENTRY glVertexAttribI4ivEXT (GLuint index, const GLint *v); +GLAPI void APIENTRY glVertexAttribI1uivEXT (GLuint index, const GLuint *v); +GLAPI void APIENTRY glVertexAttribI2uivEXT (GLuint index, const GLuint *v); +GLAPI void APIENTRY glVertexAttribI3uivEXT (GLuint index, const GLuint *v); +GLAPI void APIENTRY glVertexAttribI4uivEXT (GLuint index, const GLuint *v); +GLAPI void APIENTRY glVertexAttribI4bvEXT (GLuint index, const GLbyte *v); +GLAPI void APIENTRY glVertexAttribI4svEXT (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttribI4ubvEXT (GLuint index, const GLubyte *v); +GLAPI void APIENTRY glVertexAttribI4usvEXT (GLuint index, const GLushort *v); +GLAPI void APIENTRY glVertexAttribIPointerEXT (GLuint index, GLint size, GLenum type, GLsizei stride, const void *pointer); +GLAPI void APIENTRY glGetVertexAttribIivEXT (GLuint index, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetVertexAttribIuivEXT (GLuint index, GLenum pname, GLuint *params); +#endif +#endif /* GL_EXT_gpu_shader4 */ + +#ifndef GL_EXT_histogram +#define GL_EXT_histogram 1 +#define GL_HISTOGRAM_EXT 0x8024 +#define GL_PROXY_HISTOGRAM_EXT 0x8025 +#define GL_HISTOGRAM_WIDTH_EXT 0x8026 +#define GL_HISTOGRAM_FORMAT_EXT 0x8027 +#define GL_HISTOGRAM_RED_SIZE_EXT 0x8028 +#define GL_HISTOGRAM_GREEN_SIZE_EXT 0x8029 +#define GL_HISTOGRAM_BLUE_SIZE_EXT 0x802A +#define GL_HISTOGRAM_ALPHA_SIZE_EXT 0x802B +#define GL_HISTOGRAM_LUMINANCE_SIZE_EXT 0x802C +#define GL_HISTOGRAM_SINK_EXT 0x802D +#define GL_MINMAX_EXT 0x802E +#define GL_MINMAX_FORMAT_EXT 0x802F +#define GL_MINMAX_SINK_EXT 0x8030 +#define GL_TABLE_TOO_LARGE_EXT 0x8031 +typedef void (APIENTRYP PFNGLGETHISTOGRAMEXTPROC) (GLenum target, GLboolean reset, GLenum format, GLenum type, void *values); +typedef void (APIENTRYP PFNGLGETHISTOGRAMPARAMETERFVEXTPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETHISTOGRAMPARAMETERIVEXTPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETMINMAXEXTPROC) (GLenum target, GLboolean reset, GLenum format, GLenum type, void *values); +typedef void (APIENTRYP PFNGLGETMINMAXPARAMETERFVEXTPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETMINMAXPARAMETERIVEXTPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLHISTOGRAMEXTPROC) (GLenum target, GLsizei width, GLenum internalformat, GLboolean sink); +typedef void (APIENTRYP PFNGLMINMAXEXTPROC) (GLenum target, GLenum internalformat, GLboolean sink); +typedef void (APIENTRYP PFNGLRESETHISTOGRAMEXTPROC) (GLenum target); +typedef void (APIENTRYP PFNGLRESETMINMAXEXTPROC) (GLenum target); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGetHistogramEXT (GLenum target, GLboolean reset, GLenum format, GLenum type, void *values); +GLAPI void APIENTRY glGetHistogramParameterfvEXT (GLenum target, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetHistogramParameterivEXT (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetMinmaxEXT (GLenum target, GLboolean reset, GLenum format, GLenum type, void *values); +GLAPI void APIENTRY glGetMinmaxParameterfvEXT (GLenum target, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetMinmaxParameterivEXT (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glHistogramEXT (GLenum target, GLsizei width, GLenum internalformat, GLboolean sink); +GLAPI void APIENTRY glMinmaxEXT (GLenum target, GLenum internalformat, GLboolean sink); +GLAPI void APIENTRY glResetHistogramEXT (GLenum target); +GLAPI void APIENTRY glResetMinmaxEXT (GLenum target); +#endif +#endif /* GL_EXT_histogram */ + +#ifndef GL_EXT_index_array_formats +#define GL_EXT_index_array_formats 1 +#define GL_IUI_V2F_EXT 0x81AD +#define GL_IUI_V3F_EXT 0x81AE +#define GL_IUI_N3F_V2F_EXT 0x81AF +#define GL_IUI_N3F_V3F_EXT 0x81B0 +#define GL_T2F_IUI_V2F_EXT 0x81B1 +#define GL_T2F_IUI_V3F_EXT 0x81B2 +#define GL_T2F_IUI_N3F_V2F_EXT 0x81B3 +#define GL_T2F_IUI_N3F_V3F_EXT 0x81B4 +#endif /* GL_EXT_index_array_formats */ + +#ifndef GL_EXT_index_func +#define GL_EXT_index_func 1 +#define GL_INDEX_TEST_EXT 0x81B5 +#define GL_INDEX_TEST_FUNC_EXT 0x81B6 +#define GL_INDEX_TEST_REF_EXT 0x81B7 +typedef void (APIENTRYP PFNGLINDEXFUNCEXTPROC) (GLenum func, GLclampf ref); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glIndexFuncEXT (GLenum func, GLclampf ref); +#endif +#endif /* GL_EXT_index_func */ + +#ifndef GL_EXT_index_material +#define GL_EXT_index_material 1 +#define GL_INDEX_MATERIAL_EXT 0x81B8 +#define GL_INDEX_MATERIAL_PARAMETER_EXT 0x81B9 +#define GL_INDEX_MATERIAL_FACE_EXT 0x81BA +typedef void (APIENTRYP PFNGLINDEXMATERIALEXTPROC) (GLenum face, GLenum mode); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glIndexMaterialEXT (GLenum face, GLenum mode); +#endif +#endif /* GL_EXT_index_material */ + +#ifndef GL_EXT_index_texture +#define GL_EXT_index_texture 1 +#endif /* GL_EXT_index_texture */ + +#ifndef GL_EXT_light_texture +#define GL_EXT_light_texture 1 +#define GL_FRAGMENT_MATERIAL_EXT 0x8349 +#define GL_FRAGMENT_NORMAL_EXT 0x834A +#define GL_FRAGMENT_COLOR_EXT 0x834C +#define GL_ATTENUATION_EXT 0x834D +#define GL_SHADOW_ATTENUATION_EXT 0x834E +#define GL_TEXTURE_APPLICATION_MODE_EXT 0x834F +#define GL_TEXTURE_LIGHT_EXT 0x8350 +#define GL_TEXTURE_MATERIAL_FACE_EXT 0x8351 +#define GL_TEXTURE_MATERIAL_PARAMETER_EXT 0x8352 +typedef void (APIENTRYP PFNGLAPPLYTEXTUREEXTPROC) (GLenum mode); +typedef void (APIENTRYP PFNGLTEXTURELIGHTEXTPROC) (GLenum pname); +typedef void (APIENTRYP PFNGLTEXTUREMATERIALEXTPROC) (GLenum face, GLenum mode); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glApplyTextureEXT (GLenum mode); +GLAPI void APIENTRY glTextureLightEXT (GLenum pname); +GLAPI void APIENTRY glTextureMaterialEXT (GLenum face, GLenum mode); +#endif +#endif /* GL_EXT_light_texture */ + +#ifndef GL_EXT_memory_object +#define GL_EXT_memory_object 1 +#define GL_TEXTURE_TILING_EXT 0x9580 +#define GL_DEDICATED_MEMORY_OBJECT_EXT 0x9581 +#define GL_PROTECTED_MEMORY_OBJECT_EXT 0x959B +#define GL_NUM_TILING_TYPES_EXT 0x9582 +#define GL_TILING_TYPES_EXT 0x9583 +#define GL_OPTIMAL_TILING_EXT 0x9584 +#define GL_LINEAR_TILING_EXT 0x9585 +#define GL_NUM_DEVICE_UUIDS_EXT 0x9596 +#define GL_DEVICE_UUID_EXT 0x9597 +#define GL_DRIVER_UUID_EXT 0x9598 +#define GL_UUID_SIZE_EXT 16 +typedef void (APIENTRYP PFNGLGETUNSIGNEDBYTEVEXTPROC) (GLenum pname, GLubyte *data); +typedef void (APIENTRYP PFNGLGETUNSIGNEDBYTEI_VEXTPROC) (GLenum target, GLuint index, GLubyte *data); +typedef void (APIENTRYP PFNGLDELETEMEMORYOBJECTSEXTPROC) (GLsizei n, const GLuint *memoryObjects); +typedef GLboolean (APIENTRYP PFNGLISMEMORYOBJECTEXTPROC) (GLuint memoryObject); +typedef void (APIENTRYP PFNGLCREATEMEMORYOBJECTSEXTPROC) (GLsizei n, GLuint *memoryObjects); +typedef void (APIENTRYP PFNGLMEMORYOBJECTPARAMETERIVEXTPROC) (GLuint memoryObject, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLGETMEMORYOBJECTPARAMETERIVEXTPROC) (GLuint memoryObject, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLTEXSTORAGEMEM2DEXTPROC) (GLenum target, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLuint memory, GLuint64 offset); +typedef void (APIENTRYP PFNGLTEXSTORAGEMEM2DMULTISAMPLEEXTPROC) (GLenum target, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +typedef void (APIENTRYP PFNGLTEXSTORAGEMEM3DEXTPROC) (GLenum target, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset); +typedef void (APIENTRYP PFNGLTEXSTORAGEMEM3DMULTISAMPLEEXTPROC) (GLenum target, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +typedef void (APIENTRYP PFNGLBUFFERSTORAGEMEMEXTPROC) (GLenum target, GLsizeiptr size, GLuint memory, GLuint64 offset); +typedef void (APIENTRYP PFNGLTEXTURESTORAGEMEM2DEXTPROC) (GLuint texture, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLuint memory, GLuint64 offset); +typedef void (APIENTRYP PFNGLTEXTURESTORAGEMEM2DMULTISAMPLEEXTPROC) (GLuint texture, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +typedef void (APIENTRYP PFNGLTEXTURESTORAGEMEM3DEXTPROC) (GLuint texture, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset); +typedef void (APIENTRYP PFNGLTEXTURESTORAGEMEM3DMULTISAMPLEEXTPROC) (GLuint texture, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +typedef void (APIENTRYP PFNGLNAMEDBUFFERSTORAGEMEMEXTPROC) (GLuint buffer, GLsizeiptr size, GLuint memory, GLuint64 offset); +typedef void (APIENTRYP PFNGLTEXSTORAGEMEM1DEXTPROC) (GLenum target, GLsizei levels, GLenum internalFormat, GLsizei width, GLuint memory, GLuint64 offset); +typedef void (APIENTRYP PFNGLTEXTURESTORAGEMEM1DEXTPROC) (GLuint texture, GLsizei levels, GLenum internalFormat, GLsizei width, GLuint memory, GLuint64 offset); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGetUnsignedBytevEXT (GLenum pname, GLubyte *data); +GLAPI void APIENTRY glGetUnsignedBytei_vEXT (GLenum target, GLuint index, GLubyte *data); +GLAPI void APIENTRY glDeleteMemoryObjectsEXT (GLsizei n, const GLuint *memoryObjects); +GLAPI GLboolean APIENTRY glIsMemoryObjectEXT (GLuint memoryObject); +GLAPI void APIENTRY glCreateMemoryObjectsEXT (GLsizei n, GLuint *memoryObjects); +GLAPI void APIENTRY glMemoryObjectParameterivEXT (GLuint memoryObject, GLenum pname, const GLint *params); +GLAPI void APIENTRY glGetMemoryObjectParameterivEXT (GLuint memoryObject, GLenum pname, GLint *params); +GLAPI void APIENTRY glTexStorageMem2DEXT (GLenum target, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLuint memory, GLuint64 offset); +GLAPI void APIENTRY glTexStorageMem2DMultisampleEXT (GLenum target, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +GLAPI void APIENTRY glTexStorageMem3DEXT (GLenum target, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset); +GLAPI void APIENTRY glTexStorageMem3DMultisampleEXT (GLenum target, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +GLAPI void APIENTRY glBufferStorageMemEXT (GLenum target, GLsizeiptr size, GLuint memory, GLuint64 offset); +GLAPI void APIENTRY glTextureStorageMem2DEXT (GLuint texture, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLuint memory, GLuint64 offset); +GLAPI void APIENTRY glTextureStorageMem2DMultisampleEXT (GLuint texture, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +GLAPI void APIENTRY glTextureStorageMem3DEXT (GLuint texture, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset); +GLAPI void APIENTRY glTextureStorageMem3DMultisampleEXT (GLuint texture, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +GLAPI void APIENTRY glNamedBufferStorageMemEXT (GLuint buffer, GLsizeiptr size, GLuint memory, GLuint64 offset); +GLAPI void APIENTRY glTexStorageMem1DEXT (GLenum target, GLsizei levels, GLenum internalFormat, GLsizei width, GLuint memory, GLuint64 offset); +GLAPI void APIENTRY glTextureStorageMem1DEXT (GLuint texture, GLsizei levels, GLenum internalFormat, GLsizei width, GLuint memory, GLuint64 offset); +#endif +#endif /* GL_EXT_memory_object */ + +#ifndef GL_EXT_memory_object_fd +#define GL_EXT_memory_object_fd 1 +#define GL_HANDLE_TYPE_OPAQUE_FD_EXT 0x9586 +typedef void (APIENTRYP PFNGLIMPORTMEMORYFDEXTPROC) (GLuint memory, GLuint64 size, GLenum handleType, GLint fd); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glImportMemoryFdEXT (GLuint memory, GLuint64 size, GLenum handleType, GLint fd); +#endif +#endif /* GL_EXT_memory_object_fd */ + +#ifndef GL_EXT_memory_object_win32 +#define GL_EXT_memory_object_win32 1 +#define GL_HANDLE_TYPE_OPAQUE_WIN32_EXT 0x9587 +#define GL_HANDLE_TYPE_OPAQUE_WIN32_KMT_EXT 0x9588 +#define GL_DEVICE_LUID_EXT 0x9599 +#define GL_DEVICE_NODE_MASK_EXT 0x959A +#define GL_LUID_SIZE_EXT 8 +#define GL_HANDLE_TYPE_D3D12_TILEPOOL_EXT 0x9589 +#define GL_HANDLE_TYPE_D3D12_RESOURCE_EXT 0x958A +#define GL_HANDLE_TYPE_D3D11_IMAGE_EXT 0x958B +#define GL_HANDLE_TYPE_D3D11_IMAGE_KMT_EXT 0x958C +typedef void (APIENTRYP PFNGLIMPORTMEMORYWIN32HANDLEEXTPROC) (GLuint memory, GLuint64 size, GLenum handleType, void *handle); +typedef void (APIENTRYP PFNGLIMPORTMEMORYWIN32NAMEEXTPROC) (GLuint memory, GLuint64 size, GLenum handleType, const void *name); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glImportMemoryWin32HandleEXT (GLuint memory, GLuint64 size, GLenum handleType, void *handle); +GLAPI void APIENTRY glImportMemoryWin32NameEXT (GLuint memory, GLuint64 size, GLenum handleType, const void *name); +#endif +#endif /* GL_EXT_memory_object_win32 */ + +#ifndef GL_EXT_misc_attribute +#define GL_EXT_misc_attribute 1 +#endif /* GL_EXT_misc_attribute */ + +#ifndef GL_EXT_multi_draw_arrays +#define GL_EXT_multi_draw_arrays 1 +typedef void (APIENTRYP PFNGLMULTIDRAWARRAYSEXTPROC) (GLenum mode, const GLint *first, const GLsizei *count, GLsizei primcount); +typedef void (APIENTRYP PFNGLMULTIDRAWELEMENTSEXTPROC) (GLenum mode, const GLsizei *count, GLenum type, const void *const*indices, GLsizei primcount); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMultiDrawArraysEXT (GLenum mode, const GLint *first, const GLsizei *count, GLsizei primcount); +GLAPI void APIENTRY glMultiDrawElementsEXT (GLenum mode, const GLsizei *count, GLenum type, const void *const*indices, GLsizei primcount); +#endif +#endif /* GL_EXT_multi_draw_arrays */ + +#ifndef GL_EXT_multisample +#define GL_EXT_multisample 1 +#define GL_MULTISAMPLE_EXT 0x809D +#define GL_SAMPLE_ALPHA_TO_MASK_EXT 0x809E +#define GL_SAMPLE_ALPHA_TO_ONE_EXT 0x809F +#define GL_SAMPLE_MASK_EXT 0x80A0 +#define GL_1PASS_EXT 0x80A1 +#define GL_2PASS_0_EXT 0x80A2 +#define GL_2PASS_1_EXT 0x80A3 +#define GL_4PASS_0_EXT 0x80A4 +#define GL_4PASS_1_EXT 0x80A5 +#define GL_4PASS_2_EXT 0x80A6 +#define GL_4PASS_3_EXT 0x80A7 +#define GL_SAMPLE_BUFFERS_EXT 0x80A8 +#define GL_SAMPLES_EXT 0x80A9 +#define GL_SAMPLE_MASK_VALUE_EXT 0x80AA +#define GL_SAMPLE_MASK_INVERT_EXT 0x80AB +#define GL_SAMPLE_PATTERN_EXT 0x80AC +#define GL_MULTISAMPLE_BIT_EXT 0x20000000 +typedef void (APIENTRYP PFNGLSAMPLEMASKEXTPROC) (GLclampf value, GLboolean invert); +typedef void (APIENTRYP PFNGLSAMPLEPATTERNEXTPROC) (GLenum pattern); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSampleMaskEXT (GLclampf value, GLboolean invert); +GLAPI void APIENTRY glSamplePatternEXT (GLenum pattern); +#endif +#endif /* GL_EXT_multisample */ + +#ifndef GL_EXT_multiview_tessellation_geometry_shader +#define GL_EXT_multiview_tessellation_geometry_shader 1 +#endif /* GL_EXT_multiview_tessellation_geometry_shader */ + +#ifndef GL_EXT_multiview_texture_multisample +#define GL_EXT_multiview_texture_multisample 1 +#endif /* GL_EXT_multiview_texture_multisample */ + +#ifndef GL_EXT_multiview_timer_query +#define GL_EXT_multiview_timer_query 1 +#endif /* GL_EXT_multiview_timer_query */ + +#ifndef GL_EXT_packed_depth_stencil +#define GL_EXT_packed_depth_stencil 1 +#define GL_DEPTH_STENCIL_EXT 0x84F9 +#define GL_UNSIGNED_INT_24_8_EXT 0x84FA +#define GL_DEPTH24_STENCIL8_EXT 0x88F0 +#define GL_TEXTURE_STENCIL_SIZE_EXT 0x88F1 +#endif /* GL_EXT_packed_depth_stencil */ + +#ifndef GL_EXT_packed_float +#define GL_EXT_packed_float 1 +#define GL_R11F_G11F_B10F_EXT 0x8C3A +#define GL_UNSIGNED_INT_10F_11F_11F_REV_EXT 0x8C3B +#define GL_RGBA_SIGNED_COMPONENTS_EXT 0x8C3C +#endif /* GL_EXT_packed_float */ + +#ifndef GL_EXT_packed_pixels +#define GL_EXT_packed_pixels 1 +#define GL_UNSIGNED_BYTE_3_3_2_EXT 0x8032 +#define GL_UNSIGNED_SHORT_4_4_4_4_EXT 0x8033 +#define GL_UNSIGNED_SHORT_5_5_5_1_EXT 0x8034 +#define GL_UNSIGNED_INT_8_8_8_8_EXT 0x8035 +#define GL_UNSIGNED_INT_10_10_10_2_EXT 0x8036 +#endif /* GL_EXT_packed_pixels */ + +#ifndef GL_EXT_paletted_texture +#define GL_EXT_paletted_texture 1 +#define GL_COLOR_INDEX1_EXT 0x80E2 +#define GL_COLOR_INDEX2_EXT 0x80E3 +#define GL_COLOR_INDEX4_EXT 0x80E4 +#define GL_COLOR_INDEX8_EXT 0x80E5 +#define GL_COLOR_INDEX12_EXT 0x80E6 +#define GL_COLOR_INDEX16_EXT 0x80E7 +#define GL_TEXTURE_INDEX_SIZE_EXT 0x80ED +typedef void (APIENTRYP PFNGLCOLORTABLEEXTPROC) (GLenum target, GLenum internalFormat, GLsizei width, GLenum format, GLenum type, const void *table); +typedef void (APIENTRYP PFNGLGETCOLORTABLEEXTPROC) (GLenum target, GLenum format, GLenum type, void *data); +typedef void (APIENTRYP PFNGLGETCOLORTABLEPARAMETERIVEXTPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETCOLORTABLEPARAMETERFVEXTPROC) (GLenum target, GLenum pname, GLfloat *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glColorTableEXT (GLenum target, GLenum internalFormat, GLsizei width, GLenum format, GLenum type, const void *table); +GLAPI void APIENTRY glGetColorTableEXT (GLenum target, GLenum format, GLenum type, void *data); +GLAPI void APIENTRY glGetColorTableParameterivEXT (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetColorTableParameterfvEXT (GLenum target, GLenum pname, GLfloat *params); +#endif +#endif /* GL_EXT_paletted_texture */ + +#ifndef GL_EXT_pixel_buffer_object +#define GL_EXT_pixel_buffer_object 1 +#define GL_PIXEL_PACK_BUFFER_EXT 0x88EB +#define GL_PIXEL_UNPACK_BUFFER_EXT 0x88EC +#define GL_PIXEL_PACK_BUFFER_BINDING_EXT 0x88ED +#define GL_PIXEL_UNPACK_BUFFER_BINDING_EXT 0x88EF +#endif /* GL_EXT_pixel_buffer_object */ + +#ifndef GL_EXT_pixel_transform +#define GL_EXT_pixel_transform 1 +#define GL_PIXEL_TRANSFORM_2D_EXT 0x8330 +#define GL_PIXEL_MAG_FILTER_EXT 0x8331 +#define GL_PIXEL_MIN_FILTER_EXT 0x8332 +#define GL_PIXEL_CUBIC_WEIGHT_EXT 0x8333 +#define GL_CUBIC_EXT 0x8334 +#define GL_AVERAGE_EXT 0x8335 +#define GL_PIXEL_TRANSFORM_2D_STACK_DEPTH_EXT 0x8336 +#define GL_MAX_PIXEL_TRANSFORM_2D_STACK_DEPTH_EXT 0x8337 +#define GL_PIXEL_TRANSFORM_2D_MATRIX_EXT 0x8338 +typedef void (APIENTRYP PFNGLPIXELTRANSFORMPARAMETERIEXTPROC) (GLenum target, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLPIXELTRANSFORMPARAMETERFEXTPROC) (GLenum target, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLPIXELTRANSFORMPARAMETERIVEXTPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLPIXELTRANSFORMPARAMETERFVEXTPROC) (GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLGETPIXELTRANSFORMPARAMETERIVEXTPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETPIXELTRANSFORMPARAMETERFVEXTPROC) (GLenum target, GLenum pname, GLfloat *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPixelTransformParameteriEXT (GLenum target, GLenum pname, GLint param); +GLAPI void APIENTRY glPixelTransformParameterfEXT (GLenum target, GLenum pname, GLfloat param); +GLAPI void APIENTRY glPixelTransformParameterivEXT (GLenum target, GLenum pname, const GLint *params); +GLAPI void APIENTRY glPixelTransformParameterfvEXT (GLenum target, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glGetPixelTransformParameterivEXT (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetPixelTransformParameterfvEXT (GLenum target, GLenum pname, GLfloat *params); +#endif +#endif /* GL_EXT_pixel_transform */ + +#ifndef GL_EXT_pixel_transform_color_table +#define GL_EXT_pixel_transform_color_table 1 +#endif /* GL_EXT_pixel_transform_color_table */ + +#ifndef GL_EXT_point_parameters +#define GL_EXT_point_parameters 1 +#define GL_POINT_SIZE_MIN_EXT 0x8126 +#define GL_POINT_SIZE_MAX_EXT 0x8127 +#define GL_POINT_FADE_THRESHOLD_SIZE_EXT 0x8128 +#define GL_DISTANCE_ATTENUATION_EXT 0x8129 +typedef void (APIENTRYP PFNGLPOINTPARAMETERFEXTPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLPOINTPARAMETERFVEXTPROC) (GLenum pname, const GLfloat *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPointParameterfEXT (GLenum pname, GLfloat param); +GLAPI void APIENTRY glPointParameterfvEXT (GLenum pname, const GLfloat *params); +#endif +#endif /* GL_EXT_point_parameters */ + +#ifndef GL_EXT_polygon_offset +#define GL_EXT_polygon_offset 1 +#define GL_POLYGON_OFFSET_EXT 0x8037 +#define GL_POLYGON_OFFSET_FACTOR_EXT 0x8038 +#define GL_POLYGON_OFFSET_BIAS_EXT 0x8039 +typedef void (APIENTRYP PFNGLPOLYGONOFFSETEXTPROC) (GLfloat factor, GLfloat bias); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPolygonOffsetEXT (GLfloat factor, GLfloat bias); +#endif +#endif /* GL_EXT_polygon_offset */ + +#ifndef GL_EXT_polygon_offset_clamp +#define GL_EXT_polygon_offset_clamp 1 +#define GL_POLYGON_OFFSET_CLAMP_EXT 0x8E1B +typedef void (APIENTRYP PFNGLPOLYGONOFFSETCLAMPEXTPROC) (GLfloat factor, GLfloat units, GLfloat clamp); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPolygonOffsetClampEXT (GLfloat factor, GLfloat units, GLfloat clamp); +#endif +#endif /* GL_EXT_polygon_offset_clamp */ + +#ifndef GL_EXT_post_depth_coverage +#define GL_EXT_post_depth_coverage 1 +#endif /* GL_EXT_post_depth_coverage */ + +#ifndef GL_EXT_provoking_vertex +#define GL_EXT_provoking_vertex 1 +#define GL_QUADS_FOLLOW_PROVOKING_VERTEX_CONVENTION_EXT 0x8E4C +#define GL_FIRST_VERTEX_CONVENTION_EXT 0x8E4D +#define GL_LAST_VERTEX_CONVENTION_EXT 0x8E4E +#define GL_PROVOKING_VERTEX_EXT 0x8E4F +typedef void (APIENTRYP PFNGLPROVOKINGVERTEXEXTPROC) (GLenum mode); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glProvokingVertexEXT (GLenum mode); +#endif +#endif /* GL_EXT_provoking_vertex */ + +#ifndef GL_EXT_raster_multisample +#define GL_EXT_raster_multisample 1 +#define GL_RASTER_MULTISAMPLE_EXT 0x9327 +#define GL_RASTER_SAMPLES_EXT 0x9328 +#define GL_MAX_RASTER_SAMPLES_EXT 0x9329 +#define GL_RASTER_FIXED_SAMPLE_LOCATIONS_EXT 0x932A +#define GL_MULTISAMPLE_RASTERIZATION_ALLOWED_EXT 0x932B +#define GL_EFFECTIVE_RASTER_SAMPLES_EXT 0x932C +typedef void (APIENTRYP PFNGLRASTERSAMPLESEXTPROC) (GLuint samples, GLboolean fixedsamplelocations); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glRasterSamplesEXT (GLuint samples, GLboolean fixedsamplelocations); +#endif +#endif /* GL_EXT_raster_multisample */ + +#ifndef GL_EXT_rescale_normal +#define GL_EXT_rescale_normal 1 +#define GL_RESCALE_NORMAL_EXT 0x803A +#endif /* GL_EXT_rescale_normal */ + +#ifndef GL_EXT_secondary_color +#define GL_EXT_secondary_color 1 +#define GL_COLOR_SUM_EXT 0x8458 +#define GL_CURRENT_SECONDARY_COLOR_EXT 0x8459 +#define GL_SECONDARY_COLOR_ARRAY_SIZE_EXT 0x845A +#define GL_SECONDARY_COLOR_ARRAY_TYPE_EXT 0x845B +#define GL_SECONDARY_COLOR_ARRAY_STRIDE_EXT 0x845C +#define GL_SECONDARY_COLOR_ARRAY_POINTER_EXT 0x845D +#define GL_SECONDARY_COLOR_ARRAY_EXT 0x845E +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3BEXTPROC) (GLbyte red, GLbyte green, GLbyte blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3BVEXTPROC) (const GLbyte *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3DEXTPROC) (GLdouble red, GLdouble green, GLdouble blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3DVEXTPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3FEXTPROC) (GLfloat red, GLfloat green, GLfloat blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3FVEXTPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3IEXTPROC) (GLint red, GLint green, GLint blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3IVEXTPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3SEXTPROC) (GLshort red, GLshort green, GLshort blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3SVEXTPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UBEXTPROC) (GLubyte red, GLubyte green, GLubyte blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UBVEXTPROC) (const GLubyte *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UIEXTPROC) (GLuint red, GLuint green, GLuint blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UIVEXTPROC) (const GLuint *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3USEXTPROC) (GLushort red, GLushort green, GLushort blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3USVEXTPROC) (const GLushort *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLORPOINTEREXTPROC) (GLint size, GLenum type, GLsizei stride, const void *pointer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSecondaryColor3bEXT (GLbyte red, GLbyte green, GLbyte blue); +GLAPI void APIENTRY glSecondaryColor3bvEXT (const GLbyte *v); +GLAPI void APIENTRY glSecondaryColor3dEXT (GLdouble red, GLdouble green, GLdouble blue); +GLAPI void APIENTRY glSecondaryColor3dvEXT (const GLdouble *v); +GLAPI void APIENTRY glSecondaryColor3fEXT (GLfloat red, GLfloat green, GLfloat blue); +GLAPI void APIENTRY glSecondaryColor3fvEXT (const GLfloat *v); +GLAPI void APIENTRY glSecondaryColor3iEXT (GLint red, GLint green, GLint blue); +GLAPI void APIENTRY glSecondaryColor3ivEXT (const GLint *v); +GLAPI void APIENTRY glSecondaryColor3sEXT (GLshort red, GLshort green, GLshort blue); +GLAPI void APIENTRY glSecondaryColor3svEXT (const GLshort *v); +GLAPI void APIENTRY glSecondaryColor3ubEXT (GLubyte red, GLubyte green, GLubyte blue); +GLAPI void APIENTRY glSecondaryColor3ubvEXT (const GLubyte *v); +GLAPI void APIENTRY glSecondaryColor3uiEXT (GLuint red, GLuint green, GLuint blue); +GLAPI void APIENTRY glSecondaryColor3uivEXT (const GLuint *v); +GLAPI void APIENTRY glSecondaryColor3usEXT (GLushort red, GLushort green, GLushort blue); +GLAPI void APIENTRY glSecondaryColor3usvEXT (const GLushort *v); +GLAPI void APIENTRY glSecondaryColorPointerEXT (GLint size, GLenum type, GLsizei stride, const void *pointer); +#endif +#endif /* GL_EXT_secondary_color */ + +#ifndef GL_EXT_semaphore +#define GL_EXT_semaphore 1 +#define GL_LAYOUT_GENERAL_EXT 0x958D +#define GL_LAYOUT_COLOR_ATTACHMENT_EXT 0x958E +#define GL_LAYOUT_DEPTH_STENCIL_ATTACHMENT_EXT 0x958F +#define GL_LAYOUT_DEPTH_STENCIL_READ_ONLY_EXT 0x9590 +#define GL_LAYOUT_SHADER_READ_ONLY_EXT 0x9591 +#define GL_LAYOUT_TRANSFER_SRC_EXT 0x9592 +#define GL_LAYOUT_TRANSFER_DST_EXT 0x9593 +#define GL_LAYOUT_DEPTH_READ_ONLY_STENCIL_ATTACHMENT_EXT 0x9530 +#define GL_LAYOUT_DEPTH_ATTACHMENT_STENCIL_READ_ONLY_EXT 0x9531 +typedef void (APIENTRYP PFNGLGENSEMAPHORESEXTPROC) (GLsizei n, GLuint *semaphores); +typedef void (APIENTRYP PFNGLDELETESEMAPHORESEXTPROC) (GLsizei n, const GLuint *semaphores); +typedef GLboolean (APIENTRYP PFNGLISSEMAPHOREEXTPROC) (GLuint semaphore); +typedef void (APIENTRYP PFNGLSEMAPHOREPARAMETERUI64VEXTPROC) (GLuint semaphore, GLenum pname, const GLuint64 *params); +typedef void (APIENTRYP PFNGLGETSEMAPHOREPARAMETERUI64VEXTPROC) (GLuint semaphore, GLenum pname, GLuint64 *params); +typedef void (APIENTRYP PFNGLWAITSEMAPHOREEXTPROC) (GLuint semaphore, GLuint numBufferBarriers, const GLuint *buffers, GLuint numTextureBarriers, const GLuint *textures, const GLenum *srcLayouts); +typedef void (APIENTRYP PFNGLSIGNALSEMAPHOREEXTPROC) (GLuint semaphore, GLuint numBufferBarriers, const GLuint *buffers, GLuint numTextureBarriers, const GLuint *textures, const GLenum *dstLayouts); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGenSemaphoresEXT (GLsizei n, GLuint *semaphores); +GLAPI void APIENTRY glDeleteSemaphoresEXT (GLsizei n, const GLuint *semaphores); +GLAPI GLboolean APIENTRY glIsSemaphoreEXT (GLuint semaphore); +GLAPI void APIENTRY glSemaphoreParameterui64vEXT (GLuint semaphore, GLenum pname, const GLuint64 *params); +GLAPI void APIENTRY glGetSemaphoreParameterui64vEXT (GLuint semaphore, GLenum pname, GLuint64 *params); +GLAPI void APIENTRY glWaitSemaphoreEXT (GLuint semaphore, GLuint numBufferBarriers, const GLuint *buffers, GLuint numTextureBarriers, const GLuint *textures, const GLenum *srcLayouts); +GLAPI void APIENTRY glSignalSemaphoreEXT (GLuint semaphore, GLuint numBufferBarriers, const GLuint *buffers, GLuint numTextureBarriers, const GLuint *textures, const GLenum *dstLayouts); +#endif +#endif /* GL_EXT_semaphore */ + +#ifndef GL_EXT_semaphore_fd +#define GL_EXT_semaphore_fd 1 +typedef void (APIENTRYP PFNGLIMPORTSEMAPHOREFDEXTPROC) (GLuint semaphore, GLenum handleType, GLint fd); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glImportSemaphoreFdEXT (GLuint semaphore, GLenum handleType, GLint fd); +#endif +#endif /* GL_EXT_semaphore_fd */ + +#ifndef GL_EXT_semaphore_win32 +#define GL_EXT_semaphore_win32 1 +#define GL_HANDLE_TYPE_D3D12_FENCE_EXT 0x9594 +#define GL_D3D12_FENCE_VALUE_EXT 0x9595 +typedef void (APIENTRYP PFNGLIMPORTSEMAPHOREWIN32HANDLEEXTPROC) (GLuint semaphore, GLenum handleType, void *handle); +typedef void (APIENTRYP PFNGLIMPORTSEMAPHOREWIN32NAMEEXTPROC) (GLuint semaphore, GLenum handleType, const void *name); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glImportSemaphoreWin32HandleEXT (GLuint semaphore, GLenum handleType, void *handle); +GLAPI void APIENTRY glImportSemaphoreWin32NameEXT (GLuint semaphore, GLenum handleType, const void *name); +#endif +#endif /* GL_EXT_semaphore_win32 */ + +#ifndef GL_EXT_separate_shader_objects +#define GL_EXT_separate_shader_objects 1 +#define GL_ACTIVE_PROGRAM_EXT 0x8B8D +typedef void (APIENTRYP PFNGLUSESHADERPROGRAMEXTPROC) (GLenum type, GLuint program); +typedef void (APIENTRYP PFNGLACTIVEPROGRAMEXTPROC) (GLuint program); +typedef GLuint (APIENTRYP PFNGLCREATESHADERPROGRAMEXTPROC) (GLenum type, const GLchar *string); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glUseShaderProgramEXT (GLenum type, GLuint program); +GLAPI void APIENTRY glActiveProgramEXT (GLuint program); +GLAPI GLuint APIENTRY glCreateShaderProgramEXT (GLenum type, const GLchar *string); +#endif +#endif /* GL_EXT_separate_shader_objects */ + +#ifndef GL_EXT_separate_specular_color +#define GL_EXT_separate_specular_color 1 +#define GL_LIGHT_MODEL_COLOR_CONTROL_EXT 0x81F8 +#define GL_SINGLE_COLOR_EXT 0x81F9 +#define GL_SEPARATE_SPECULAR_COLOR_EXT 0x81FA +#endif /* GL_EXT_separate_specular_color */ + +#ifndef GL_EXT_shader_framebuffer_fetch +#define GL_EXT_shader_framebuffer_fetch 1 +#define GL_FRAGMENT_SHADER_DISCARDS_SAMPLES_EXT 0x8A52 +#endif /* GL_EXT_shader_framebuffer_fetch */ + +#ifndef GL_EXT_shader_framebuffer_fetch_non_coherent +#define GL_EXT_shader_framebuffer_fetch_non_coherent 1 +typedef void (APIENTRYP PFNGLFRAMEBUFFERFETCHBARRIEREXTPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFramebufferFetchBarrierEXT (void); +#endif +#endif /* GL_EXT_shader_framebuffer_fetch_non_coherent */ + +#ifndef GL_EXT_shader_image_load_formatted +#define GL_EXT_shader_image_load_formatted 1 +#endif /* GL_EXT_shader_image_load_formatted */ + +#ifndef GL_EXT_shader_image_load_store +#define GL_EXT_shader_image_load_store 1 +#define GL_MAX_IMAGE_UNITS_EXT 0x8F38 +#define GL_MAX_COMBINED_IMAGE_UNITS_AND_FRAGMENT_OUTPUTS_EXT 0x8F39 +#define GL_IMAGE_BINDING_NAME_EXT 0x8F3A +#define GL_IMAGE_BINDING_LEVEL_EXT 0x8F3B +#define GL_IMAGE_BINDING_LAYERED_EXT 0x8F3C +#define GL_IMAGE_BINDING_LAYER_EXT 0x8F3D +#define GL_IMAGE_BINDING_ACCESS_EXT 0x8F3E +#define GL_IMAGE_1D_EXT 0x904C +#define GL_IMAGE_2D_EXT 0x904D +#define GL_IMAGE_3D_EXT 0x904E +#define GL_IMAGE_2D_RECT_EXT 0x904F +#define GL_IMAGE_CUBE_EXT 0x9050 +#define GL_IMAGE_BUFFER_EXT 0x9051 +#define GL_IMAGE_1D_ARRAY_EXT 0x9052 +#define GL_IMAGE_2D_ARRAY_EXT 0x9053 +#define GL_IMAGE_CUBE_MAP_ARRAY_EXT 0x9054 +#define GL_IMAGE_2D_MULTISAMPLE_EXT 0x9055 +#define GL_IMAGE_2D_MULTISAMPLE_ARRAY_EXT 0x9056 +#define GL_INT_IMAGE_1D_EXT 0x9057 +#define GL_INT_IMAGE_2D_EXT 0x9058 +#define GL_INT_IMAGE_3D_EXT 0x9059 +#define GL_INT_IMAGE_2D_RECT_EXT 0x905A +#define GL_INT_IMAGE_CUBE_EXT 0x905B +#define GL_INT_IMAGE_BUFFER_EXT 0x905C +#define GL_INT_IMAGE_1D_ARRAY_EXT 0x905D +#define GL_INT_IMAGE_2D_ARRAY_EXT 0x905E +#define GL_INT_IMAGE_CUBE_MAP_ARRAY_EXT 0x905F +#define GL_INT_IMAGE_2D_MULTISAMPLE_EXT 0x9060 +#define GL_INT_IMAGE_2D_MULTISAMPLE_ARRAY_EXT 0x9061 +#define GL_UNSIGNED_INT_IMAGE_1D_EXT 0x9062 +#define GL_UNSIGNED_INT_IMAGE_2D_EXT 0x9063 +#define GL_UNSIGNED_INT_IMAGE_3D_EXT 0x9064 +#define GL_UNSIGNED_INT_IMAGE_2D_RECT_EXT 0x9065 +#define GL_UNSIGNED_INT_IMAGE_CUBE_EXT 0x9066 +#define GL_UNSIGNED_INT_IMAGE_BUFFER_EXT 0x9067 +#define GL_UNSIGNED_INT_IMAGE_1D_ARRAY_EXT 0x9068 +#define GL_UNSIGNED_INT_IMAGE_2D_ARRAY_EXT 0x9069 +#define GL_UNSIGNED_INT_IMAGE_CUBE_MAP_ARRAY_EXT 0x906A +#define GL_UNSIGNED_INT_IMAGE_2D_MULTISAMPLE_EXT 0x906B +#define GL_UNSIGNED_INT_IMAGE_2D_MULTISAMPLE_ARRAY_EXT 0x906C +#define GL_MAX_IMAGE_SAMPLES_EXT 0x906D +#define GL_IMAGE_BINDING_FORMAT_EXT 0x906E +#define GL_VERTEX_ATTRIB_ARRAY_BARRIER_BIT_EXT 0x00000001 +#define GL_ELEMENT_ARRAY_BARRIER_BIT_EXT 0x00000002 +#define GL_UNIFORM_BARRIER_BIT_EXT 0x00000004 +#define GL_TEXTURE_FETCH_BARRIER_BIT_EXT 0x00000008 +#define GL_SHADER_IMAGE_ACCESS_BARRIER_BIT_EXT 0x00000020 +#define GL_COMMAND_BARRIER_BIT_EXT 0x00000040 +#define GL_PIXEL_BUFFER_BARRIER_BIT_EXT 0x00000080 +#define GL_TEXTURE_UPDATE_BARRIER_BIT_EXT 0x00000100 +#define GL_BUFFER_UPDATE_BARRIER_BIT_EXT 0x00000200 +#define GL_FRAMEBUFFER_BARRIER_BIT_EXT 0x00000400 +#define GL_TRANSFORM_FEEDBACK_BARRIER_BIT_EXT 0x00000800 +#define GL_ATOMIC_COUNTER_BARRIER_BIT_EXT 0x00001000 +#define GL_ALL_BARRIER_BITS_EXT 0xFFFFFFFF +typedef void (APIENTRYP PFNGLBINDIMAGETEXTUREEXTPROC) (GLuint index, GLuint texture, GLint level, GLboolean layered, GLint layer, GLenum access, GLint format); +typedef void (APIENTRYP PFNGLMEMORYBARRIEREXTPROC) (GLbitfield barriers); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBindImageTextureEXT (GLuint index, GLuint texture, GLint level, GLboolean layered, GLint layer, GLenum access, GLint format); +GLAPI void APIENTRY glMemoryBarrierEXT (GLbitfield barriers); +#endif +#endif /* GL_EXT_shader_image_load_store */ + +#ifndef GL_EXT_shader_integer_mix +#define GL_EXT_shader_integer_mix 1 +#endif /* GL_EXT_shader_integer_mix */ + +#ifndef GL_EXT_shader_samples_identical +#define GL_EXT_shader_samples_identical 1 +#endif /* GL_EXT_shader_samples_identical */ + +#ifndef GL_EXT_shadow_funcs +#define GL_EXT_shadow_funcs 1 +#endif /* GL_EXT_shadow_funcs */ + +#ifndef GL_EXT_shared_texture_palette +#define GL_EXT_shared_texture_palette 1 +#define GL_SHARED_TEXTURE_PALETTE_EXT 0x81FB +#endif /* GL_EXT_shared_texture_palette */ + +#ifndef GL_EXT_sparse_texture2 +#define GL_EXT_sparse_texture2 1 +#endif /* GL_EXT_sparse_texture2 */ + +#ifndef GL_EXT_stencil_clear_tag +#define GL_EXT_stencil_clear_tag 1 +#define GL_STENCIL_TAG_BITS_EXT 0x88F2 +#define GL_STENCIL_CLEAR_TAG_VALUE_EXT 0x88F3 +typedef void (APIENTRYP PFNGLSTENCILCLEARTAGEXTPROC) (GLsizei stencilTagBits, GLuint stencilClearTag); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glStencilClearTagEXT (GLsizei stencilTagBits, GLuint stencilClearTag); +#endif +#endif /* GL_EXT_stencil_clear_tag */ + +#ifndef GL_EXT_stencil_two_side +#define GL_EXT_stencil_two_side 1 +#define GL_STENCIL_TEST_TWO_SIDE_EXT 0x8910 +#define GL_ACTIVE_STENCIL_FACE_EXT 0x8911 +typedef void (APIENTRYP PFNGLACTIVESTENCILFACEEXTPROC) (GLenum face); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glActiveStencilFaceEXT (GLenum face); +#endif +#endif /* GL_EXT_stencil_two_side */ + +#ifndef GL_EXT_stencil_wrap +#define GL_EXT_stencil_wrap 1 +#define GL_INCR_WRAP_EXT 0x8507 +#define GL_DECR_WRAP_EXT 0x8508 +#endif /* GL_EXT_stencil_wrap */ + +#ifndef GL_EXT_subtexture +#define GL_EXT_subtexture 1 +typedef void (APIENTRYP PFNGLTEXSUBIMAGE1DEXTPROC) (GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLTEXSUBIMAGE2DEXTPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *pixels); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTexSubImage1DEXT (GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glTexSubImage2DEXT (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *pixels); +#endif +#endif /* GL_EXT_subtexture */ + +#ifndef GL_EXT_texture +#define GL_EXT_texture 1 +#define GL_ALPHA4_EXT 0x803B +#define GL_ALPHA8_EXT 0x803C +#define GL_ALPHA12_EXT 0x803D +#define GL_ALPHA16_EXT 0x803E +#define GL_LUMINANCE4_EXT 0x803F +#define GL_LUMINANCE8_EXT 0x8040 +#define GL_LUMINANCE12_EXT 0x8041 +#define GL_LUMINANCE16_EXT 0x8042 +#define GL_LUMINANCE4_ALPHA4_EXT 0x8043 +#define GL_LUMINANCE6_ALPHA2_EXT 0x8044 +#define GL_LUMINANCE8_ALPHA8_EXT 0x8045 +#define GL_LUMINANCE12_ALPHA4_EXT 0x8046 +#define GL_LUMINANCE12_ALPHA12_EXT 0x8047 +#define GL_LUMINANCE16_ALPHA16_EXT 0x8048 +#define GL_INTENSITY_EXT 0x8049 +#define GL_INTENSITY4_EXT 0x804A +#define GL_INTENSITY8_EXT 0x804B +#define GL_INTENSITY12_EXT 0x804C +#define GL_INTENSITY16_EXT 0x804D +#define GL_RGB2_EXT 0x804E +#define GL_RGB4_EXT 0x804F +#define GL_RGB5_EXT 0x8050 +#define GL_RGB8_EXT 0x8051 +#define GL_RGB10_EXT 0x8052 +#define GL_RGB12_EXT 0x8053 +#define GL_RGB16_EXT 0x8054 +#define GL_RGBA2_EXT 0x8055 +#define GL_RGBA4_EXT 0x8056 +#define GL_RGB5_A1_EXT 0x8057 +#define GL_RGBA8_EXT 0x8058 +#define GL_RGB10_A2_EXT 0x8059 +#define GL_RGBA12_EXT 0x805A +#define GL_RGBA16_EXT 0x805B +#define GL_TEXTURE_RED_SIZE_EXT 0x805C +#define GL_TEXTURE_GREEN_SIZE_EXT 0x805D +#define GL_TEXTURE_BLUE_SIZE_EXT 0x805E +#define GL_TEXTURE_ALPHA_SIZE_EXT 0x805F +#define GL_TEXTURE_LUMINANCE_SIZE_EXT 0x8060 +#define GL_TEXTURE_INTENSITY_SIZE_EXT 0x8061 +#define GL_REPLACE_EXT 0x8062 +#define GL_PROXY_TEXTURE_1D_EXT 0x8063 +#define GL_PROXY_TEXTURE_2D_EXT 0x8064 +#define GL_TEXTURE_TOO_LARGE_EXT 0x8065 +#endif /* GL_EXT_texture */ + +#ifndef GL_EXT_texture3D +#define GL_EXT_texture3D 1 +#define GL_PACK_SKIP_IMAGES_EXT 0x806B +#define GL_PACK_IMAGE_HEIGHT_EXT 0x806C +#define GL_UNPACK_SKIP_IMAGES_EXT 0x806D +#define GL_UNPACK_IMAGE_HEIGHT_EXT 0x806E +#define GL_TEXTURE_3D_EXT 0x806F +#define GL_PROXY_TEXTURE_3D_EXT 0x8070 +#define GL_TEXTURE_DEPTH_EXT 0x8071 +#define GL_TEXTURE_WRAP_R_EXT 0x8072 +#define GL_MAX_3D_TEXTURE_SIZE_EXT 0x8073 +typedef void (APIENTRYP PFNGLTEXIMAGE3DEXTPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLTEXSUBIMAGE3DEXTPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *pixels); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTexImage3DEXT (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glTexSubImage3DEXT (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *pixels); +#endif +#endif /* GL_EXT_texture3D */ + +#ifndef GL_EXT_texture_array +#define GL_EXT_texture_array 1 +#define GL_TEXTURE_1D_ARRAY_EXT 0x8C18 +#define GL_PROXY_TEXTURE_1D_ARRAY_EXT 0x8C19 +#define GL_TEXTURE_2D_ARRAY_EXT 0x8C1A +#define GL_PROXY_TEXTURE_2D_ARRAY_EXT 0x8C1B +#define GL_TEXTURE_BINDING_1D_ARRAY_EXT 0x8C1C +#define GL_TEXTURE_BINDING_2D_ARRAY_EXT 0x8C1D +#define GL_MAX_ARRAY_TEXTURE_LAYERS_EXT 0x88FF +#define GL_COMPARE_REF_DEPTH_TO_TEXTURE_EXT 0x884E +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTURELAYEREXTPROC) (GLenum target, GLenum attachment, GLuint texture, GLint level, GLint layer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFramebufferTextureLayerEXT (GLenum target, GLenum attachment, GLuint texture, GLint level, GLint layer); +#endif +#endif /* GL_EXT_texture_array */ + +#ifndef GL_EXT_texture_buffer_object +#define GL_EXT_texture_buffer_object 1 +#define GL_TEXTURE_BUFFER_EXT 0x8C2A +#define GL_MAX_TEXTURE_BUFFER_SIZE_EXT 0x8C2B +#define GL_TEXTURE_BINDING_BUFFER_EXT 0x8C2C +#define GL_TEXTURE_BUFFER_DATA_STORE_BINDING_EXT 0x8C2D +#define GL_TEXTURE_BUFFER_FORMAT_EXT 0x8C2E +typedef void (APIENTRYP PFNGLTEXBUFFEREXTPROC) (GLenum target, GLenum internalformat, GLuint buffer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTexBufferEXT (GLenum target, GLenum internalformat, GLuint buffer); +#endif +#endif /* GL_EXT_texture_buffer_object */ + +#ifndef GL_EXT_texture_compression_latc +#define GL_EXT_texture_compression_latc 1 +#define GL_COMPRESSED_LUMINANCE_LATC1_EXT 0x8C70 +#define GL_COMPRESSED_SIGNED_LUMINANCE_LATC1_EXT 0x8C71 +#define GL_COMPRESSED_LUMINANCE_ALPHA_LATC2_EXT 0x8C72 +#define GL_COMPRESSED_SIGNED_LUMINANCE_ALPHA_LATC2_EXT 0x8C73 +#endif /* GL_EXT_texture_compression_latc */ + +#ifndef GL_EXT_texture_compression_rgtc +#define GL_EXT_texture_compression_rgtc 1 +#define GL_COMPRESSED_RED_RGTC1_EXT 0x8DBB +#define GL_COMPRESSED_SIGNED_RED_RGTC1_EXT 0x8DBC +#define GL_COMPRESSED_RED_GREEN_RGTC2_EXT 0x8DBD +#define GL_COMPRESSED_SIGNED_RED_GREEN_RGTC2_EXT 0x8DBE +#endif /* GL_EXT_texture_compression_rgtc */ + +#ifndef GL_EXT_texture_compression_s3tc +#define GL_EXT_texture_compression_s3tc 1 +#define GL_COMPRESSED_RGB_S3TC_DXT1_EXT 0x83F0 +#define GL_COMPRESSED_RGBA_S3TC_DXT1_EXT 0x83F1 +#define GL_COMPRESSED_RGBA_S3TC_DXT3_EXT 0x83F2 +#define GL_COMPRESSED_RGBA_S3TC_DXT5_EXT 0x83F3 +#endif /* GL_EXT_texture_compression_s3tc */ + +#ifndef GL_EXT_texture_cube_map +#define GL_EXT_texture_cube_map 1 +#define GL_NORMAL_MAP_EXT 0x8511 +#define GL_REFLECTION_MAP_EXT 0x8512 +#define GL_TEXTURE_CUBE_MAP_EXT 0x8513 +#define GL_TEXTURE_BINDING_CUBE_MAP_EXT 0x8514 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_X_EXT 0x8515 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_X_EXT 0x8516 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Y_EXT 0x8517 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Y_EXT 0x8518 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Z_EXT 0x8519 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Z_EXT 0x851A +#define GL_PROXY_TEXTURE_CUBE_MAP_EXT 0x851B +#define GL_MAX_CUBE_MAP_TEXTURE_SIZE_EXT 0x851C +#endif /* GL_EXT_texture_cube_map */ + +#ifndef GL_EXT_texture_env_add +#define GL_EXT_texture_env_add 1 +#endif /* GL_EXT_texture_env_add */ + +#ifndef GL_EXT_texture_env_combine +#define GL_EXT_texture_env_combine 1 +#define GL_COMBINE_EXT 0x8570 +#define GL_COMBINE_RGB_EXT 0x8571 +#define GL_COMBINE_ALPHA_EXT 0x8572 +#define GL_RGB_SCALE_EXT 0x8573 +#define GL_ADD_SIGNED_EXT 0x8574 +#define GL_INTERPOLATE_EXT 0x8575 +#define GL_CONSTANT_EXT 0x8576 +#define GL_PRIMARY_COLOR_EXT 0x8577 +#define GL_PREVIOUS_EXT 0x8578 +#define GL_SOURCE0_RGB_EXT 0x8580 +#define GL_SOURCE1_RGB_EXT 0x8581 +#define GL_SOURCE2_RGB_EXT 0x8582 +#define GL_SOURCE0_ALPHA_EXT 0x8588 +#define GL_SOURCE1_ALPHA_EXT 0x8589 +#define GL_SOURCE2_ALPHA_EXT 0x858A +#define GL_OPERAND0_RGB_EXT 0x8590 +#define GL_OPERAND1_RGB_EXT 0x8591 +#define GL_OPERAND2_RGB_EXT 0x8592 +#define GL_OPERAND0_ALPHA_EXT 0x8598 +#define GL_OPERAND1_ALPHA_EXT 0x8599 +#define GL_OPERAND2_ALPHA_EXT 0x859A +#endif /* GL_EXT_texture_env_combine */ + +#ifndef GL_EXT_texture_env_dot3 +#define GL_EXT_texture_env_dot3 1 +#define GL_DOT3_RGB_EXT 0x8740 +#define GL_DOT3_RGBA_EXT 0x8741 +#endif /* GL_EXT_texture_env_dot3 */ + +#ifndef GL_EXT_texture_filter_anisotropic +#define GL_EXT_texture_filter_anisotropic 1 +#define GL_TEXTURE_MAX_ANISOTROPY_EXT 0x84FE +#define GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT 0x84FF +#endif /* GL_EXT_texture_filter_anisotropic */ + +#ifndef GL_EXT_texture_filter_minmax +#define GL_EXT_texture_filter_minmax 1 +#define GL_TEXTURE_REDUCTION_MODE_EXT 0x9366 +#define GL_WEIGHTED_AVERAGE_EXT 0x9367 +#endif /* GL_EXT_texture_filter_minmax */ + +#ifndef GL_EXT_texture_integer +#define GL_EXT_texture_integer 1 +#define GL_RGBA32UI_EXT 0x8D70 +#define GL_RGB32UI_EXT 0x8D71 +#define GL_ALPHA32UI_EXT 0x8D72 +#define GL_INTENSITY32UI_EXT 0x8D73 +#define GL_LUMINANCE32UI_EXT 0x8D74 +#define GL_LUMINANCE_ALPHA32UI_EXT 0x8D75 +#define GL_RGBA16UI_EXT 0x8D76 +#define GL_RGB16UI_EXT 0x8D77 +#define GL_ALPHA16UI_EXT 0x8D78 +#define GL_INTENSITY16UI_EXT 0x8D79 +#define GL_LUMINANCE16UI_EXT 0x8D7A +#define GL_LUMINANCE_ALPHA16UI_EXT 0x8D7B +#define GL_RGBA8UI_EXT 0x8D7C +#define GL_RGB8UI_EXT 0x8D7D +#define GL_ALPHA8UI_EXT 0x8D7E +#define GL_INTENSITY8UI_EXT 0x8D7F +#define GL_LUMINANCE8UI_EXT 0x8D80 +#define GL_LUMINANCE_ALPHA8UI_EXT 0x8D81 +#define GL_RGBA32I_EXT 0x8D82 +#define GL_RGB32I_EXT 0x8D83 +#define GL_ALPHA32I_EXT 0x8D84 +#define GL_INTENSITY32I_EXT 0x8D85 +#define GL_LUMINANCE32I_EXT 0x8D86 +#define GL_LUMINANCE_ALPHA32I_EXT 0x8D87 +#define GL_RGBA16I_EXT 0x8D88 +#define GL_RGB16I_EXT 0x8D89 +#define GL_ALPHA16I_EXT 0x8D8A +#define GL_INTENSITY16I_EXT 0x8D8B +#define GL_LUMINANCE16I_EXT 0x8D8C +#define GL_LUMINANCE_ALPHA16I_EXT 0x8D8D +#define GL_RGBA8I_EXT 0x8D8E +#define GL_RGB8I_EXT 0x8D8F +#define GL_ALPHA8I_EXT 0x8D90 +#define GL_INTENSITY8I_EXT 0x8D91 +#define GL_LUMINANCE8I_EXT 0x8D92 +#define GL_LUMINANCE_ALPHA8I_EXT 0x8D93 +#define GL_RED_INTEGER_EXT 0x8D94 +#define GL_GREEN_INTEGER_EXT 0x8D95 +#define GL_BLUE_INTEGER_EXT 0x8D96 +#define GL_ALPHA_INTEGER_EXT 0x8D97 +#define GL_RGB_INTEGER_EXT 0x8D98 +#define GL_RGBA_INTEGER_EXT 0x8D99 +#define GL_BGR_INTEGER_EXT 0x8D9A +#define GL_BGRA_INTEGER_EXT 0x8D9B +#define GL_LUMINANCE_INTEGER_EXT 0x8D9C +#define GL_LUMINANCE_ALPHA_INTEGER_EXT 0x8D9D +#define GL_RGBA_INTEGER_MODE_EXT 0x8D9E +typedef void (APIENTRYP PFNGLTEXPARAMETERIIVEXTPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLTEXPARAMETERIUIVEXTPROC) (GLenum target, GLenum pname, const GLuint *params); +typedef void (APIENTRYP PFNGLGETTEXPARAMETERIIVEXTPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETTEXPARAMETERIUIVEXTPROC) (GLenum target, GLenum pname, GLuint *params); +typedef void (APIENTRYP PFNGLCLEARCOLORIIEXTPROC) (GLint red, GLint green, GLint blue, GLint alpha); +typedef void (APIENTRYP PFNGLCLEARCOLORIUIEXTPROC) (GLuint red, GLuint green, GLuint blue, GLuint alpha); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTexParameterIivEXT (GLenum target, GLenum pname, const GLint *params); +GLAPI void APIENTRY glTexParameterIuivEXT (GLenum target, GLenum pname, const GLuint *params); +GLAPI void APIENTRY glGetTexParameterIivEXT (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetTexParameterIuivEXT (GLenum target, GLenum pname, GLuint *params); +GLAPI void APIENTRY glClearColorIiEXT (GLint red, GLint green, GLint blue, GLint alpha); +GLAPI void APIENTRY glClearColorIuiEXT (GLuint red, GLuint green, GLuint blue, GLuint alpha); +#endif +#endif /* GL_EXT_texture_integer */ + +#ifndef GL_EXT_texture_lod_bias +#define GL_EXT_texture_lod_bias 1 +#define GL_MAX_TEXTURE_LOD_BIAS_EXT 0x84FD +#define GL_TEXTURE_FILTER_CONTROL_EXT 0x8500 +#define GL_TEXTURE_LOD_BIAS_EXT 0x8501 +#endif /* GL_EXT_texture_lod_bias */ + +#ifndef GL_EXT_texture_mirror_clamp +#define GL_EXT_texture_mirror_clamp 1 +#define GL_MIRROR_CLAMP_EXT 0x8742 +#define GL_MIRROR_CLAMP_TO_EDGE_EXT 0x8743 +#define GL_MIRROR_CLAMP_TO_BORDER_EXT 0x8912 +#endif /* GL_EXT_texture_mirror_clamp */ + +#ifndef GL_EXT_texture_object +#define GL_EXT_texture_object 1 +#define GL_TEXTURE_PRIORITY_EXT 0x8066 +#define GL_TEXTURE_RESIDENT_EXT 0x8067 +#define GL_TEXTURE_1D_BINDING_EXT 0x8068 +#define GL_TEXTURE_2D_BINDING_EXT 0x8069 +#define GL_TEXTURE_3D_BINDING_EXT 0x806A +typedef GLboolean (APIENTRYP PFNGLARETEXTURESRESIDENTEXTPROC) (GLsizei n, const GLuint *textures, GLboolean *residences); +typedef void (APIENTRYP PFNGLBINDTEXTUREEXTPROC) (GLenum target, GLuint texture); +typedef void (APIENTRYP PFNGLDELETETEXTURESEXTPROC) (GLsizei n, const GLuint *textures); +typedef void (APIENTRYP PFNGLGENTEXTURESEXTPROC) (GLsizei n, GLuint *textures); +typedef GLboolean (APIENTRYP PFNGLISTEXTUREEXTPROC) (GLuint texture); +typedef void (APIENTRYP PFNGLPRIORITIZETEXTURESEXTPROC) (GLsizei n, const GLuint *textures, const GLclampf *priorities); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLboolean APIENTRY glAreTexturesResidentEXT (GLsizei n, const GLuint *textures, GLboolean *residences); +GLAPI void APIENTRY glBindTextureEXT (GLenum target, GLuint texture); +GLAPI void APIENTRY glDeleteTexturesEXT (GLsizei n, const GLuint *textures); +GLAPI void APIENTRY glGenTexturesEXT (GLsizei n, GLuint *textures); +GLAPI GLboolean APIENTRY glIsTextureEXT (GLuint texture); +GLAPI void APIENTRY glPrioritizeTexturesEXT (GLsizei n, const GLuint *textures, const GLclampf *priorities); +#endif +#endif /* GL_EXT_texture_object */ + +#ifndef GL_EXT_texture_perturb_normal +#define GL_EXT_texture_perturb_normal 1 +#define GL_PERTURB_EXT 0x85AE +#define GL_TEXTURE_NORMAL_EXT 0x85AF +typedef void (APIENTRYP PFNGLTEXTURENORMALEXTPROC) (GLenum mode); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTextureNormalEXT (GLenum mode); +#endif +#endif /* GL_EXT_texture_perturb_normal */ + +#ifndef GL_EXT_texture_sRGB +#define GL_EXT_texture_sRGB 1 +#define GL_SRGB_EXT 0x8C40 +#define GL_SRGB8_EXT 0x8C41 +#define GL_SRGB_ALPHA_EXT 0x8C42 +#define GL_SRGB8_ALPHA8_EXT 0x8C43 +#define GL_SLUMINANCE_ALPHA_EXT 0x8C44 +#define GL_SLUMINANCE8_ALPHA8_EXT 0x8C45 +#define GL_SLUMINANCE_EXT 0x8C46 +#define GL_SLUMINANCE8_EXT 0x8C47 +#define GL_COMPRESSED_SRGB_EXT 0x8C48 +#define GL_COMPRESSED_SRGB_ALPHA_EXT 0x8C49 +#define GL_COMPRESSED_SLUMINANCE_EXT 0x8C4A +#define GL_COMPRESSED_SLUMINANCE_ALPHA_EXT 0x8C4B +#define GL_COMPRESSED_SRGB_S3TC_DXT1_EXT 0x8C4C +#define GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT 0x8C4D +#define GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT3_EXT 0x8C4E +#define GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT 0x8C4F +#endif /* GL_EXT_texture_sRGB */ + +#ifndef GL_EXT_texture_sRGB_R8 +#define GL_EXT_texture_sRGB_R8 1 +#define GL_SR8_EXT 0x8FBD +#endif /* GL_EXT_texture_sRGB_R8 */ + +#ifndef GL_EXT_texture_sRGB_RG8 +#define GL_EXT_texture_sRGB_RG8 1 +#define GL_SRG8_EXT 0x8FBE +#endif /* GL_EXT_texture_sRGB_RG8 */ + +#ifndef GL_EXT_texture_sRGB_decode +#define GL_EXT_texture_sRGB_decode 1 +#define GL_TEXTURE_SRGB_DECODE_EXT 0x8A48 +#define GL_DECODE_EXT 0x8A49 +#define GL_SKIP_DECODE_EXT 0x8A4A +#endif /* GL_EXT_texture_sRGB_decode */ + +#ifndef GL_EXT_texture_shadow_lod +#define GL_EXT_texture_shadow_lod 1 +#endif /* GL_EXT_texture_shadow_lod */ + +#ifndef GL_EXT_texture_shared_exponent +#define GL_EXT_texture_shared_exponent 1 +#define GL_RGB9_E5_EXT 0x8C3D +#define GL_UNSIGNED_INT_5_9_9_9_REV_EXT 0x8C3E +#define GL_TEXTURE_SHARED_SIZE_EXT 0x8C3F +#endif /* GL_EXT_texture_shared_exponent */ + +#ifndef GL_EXT_texture_snorm +#define GL_EXT_texture_snorm 1 +#define GL_ALPHA_SNORM 0x9010 +#define GL_LUMINANCE_SNORM 0x9011 +#define GL_LUMINANCE_ALPHA_SNORM 0x9012 +#define GL_INTENSITY_SNORM 0x9013 +#define GL_ALPHA8_SNORM 0x9014 +#define GL_LUMINANCE8_SNORM 0x9015 +#define GL_LUMINANCE8_ALPHA8_SNORM 0x9016 +#define GL_INTENSITY8_SNORM 0x9017 +#define GL_ALPHA16_SNORM 0x9018 +#define GL_LUMINANCE16_SNORM 0x9019 +#define GL_LUMINANCE16_ALPHA16_SNORM 0x901A +#define GL_INTENSITY16_SNORM 0x901B +#define GL_RED_SNORM 0x8F90 +#define GL_RG_SNORM 0x8F91 +#define GL_RGB_SNORM 0x8F92 +#define GL_RGBA_SNORM 0x8F93 +#endif /* GL_EXT_texture_snorm */ + +#ifndef GL_EXT_texture_storage +#define GL_EXT_texture_storage 1 +#define GL_TEXTURE_IMMUTABLE_FORMAT_EXT 0x912F +#define GL_RGBA32F_EXT 0x8814 +#define GL_RGB32F_EXT 0x8815 +#define GL_ALPHA32F_EXT 0x8816 +#define GL_LUMINANCE32F_EXT 0x8818 +#define GL_LUMINANCE_ALPHA32F_EXT 0x8819 +#define GL_RGBA16F_EXT 0x881A +#define GL_RGB16F_EXT 0x881B +#define GL_ALPHA16F_EXT 0x881C +#define GL_LUMINANCE16F_EXT 0x881E +#define GL_LUMINANCE_ALPHA16F_EXT 0x881F +#define GL_BGRA8_EXT 0x93A1 +#define GL_R8_EXT 0x8229 +#define GL_RG8_EXT 0x822B +#define GL_R32F_EXT 0x822E +#define GL_RG32F_EXT 0x8230 +#define GL_R16F_EXT 0x822D +#define GL_RG16F_EXT 0x822F +typedef void (APIENTRYP PFNGLTEXSTORAGE1DEXTPROC) (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width); +typedef void (APIENTRYP PFNGLTEXSTORAGE2DEXTPROC) (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLTEXSTORAGE3DEXTPROC) (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTexStorage1DEXT (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width); +GLAPI void APIENTRY glTexStorage2DEXT (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height); +GLAPI void APIENTRY glTexStorage3DEXT (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth); +#endif +#endif /* GL_EXT_texture_storage */ + +#ifndef GL_EXT_texture_swizzle +#define GL_EXT_texture_swizzle 1 +#define GL_TEXTURE_SWIZZLE_R_EXT 0x8E42 +#define GL_TEXTURE_SWIZZLE_G_EXT 0x8E43 +#define GL_TEXTURE_SWIZZLE_B_EXT 0x8E44 +#define GL_TEXTURE_SWIZZLE_A_EXT 0x8E45 +#define GL_TEXTURE_SWIZZLE_RGBA_EXT 0x8E46 +#endif /* GL_EXT_texture_swizzle */ + +#ifndef GL_EXT_timer_query +#define GL_EXT_timer_query 1 +#define GL_TIME_ELAPSED_EXT 0x88BF +typedef void (APIENTRYP PFNGLGETQUERYOBJECTI64VEXTPROC) (GLuint id, GLenum pname, GLint64 *params); +typedef void (APIENTRYP PFNGLGETQUERYOBJECTUI64VEXTPROC) (GLuint id, GLenum pname, GLuint64 *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGetQueryObjecti64vEXT (GLuint id, GLenum pname, GLint64 *params); +GLAPI void APIENTRY glGetQueryObjectui64vEXT (GLuint id, GLenum pname, GLuint64 *params); +#endif +#endif /* GL_EXT_timer_query */ + +#ifndef GL_EXT_transform_feedback +#define GL_EXT_transform_feedback 1 +#define GL_TRANSFORM_FEEDBACK_BUFFER_EXT 0x8C8E +#define GL_TRANSFORM_FEEDBACK_BUFFER_START_EXT 0x8C84 +#define GL_TRANSFORM_FEEDBACK_BUFFER_SIZE_EXT 0x8C85 +#define GL_TRANSFORM_FEEDBACK_BUFFER_BINDING_EXT 0x8C8F +#define GL_INTERLEAVED_ATTRIBS_EXT 0x8C8C +#define GL_SEPARATE_ATTRIBS_EXT 0x8C8D +#define GL_PRIMITIVES_GENERATED_EXT 0x8C87 +#define GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN_EXT 0x8C88 +#define GL_RASTERIZER_DISCARD_EXT 0x8C89 +#define GL_MAX_TRANSFORM_FEEDBACK_INTERLEAVED_COMPONENTS_EXT 0x8C8A +#define GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_ATTRIBS_EXT 0x8C8B +#define GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_COMPONENTS_EXT 0x8C80 +#define GL_TRANSFORM_FEEDBACK_VARYINGS_EXT 0x8C83 +#define GL_TRANSFORM_FEEDBACK_BUFFER_MODE_EXT 0x8C7F +#define GL_TRANSFORM_FEEDBACK_VARYING_MAX_LENGTH_EXT 0x8C76 +typedef void (APIENTRYP PFNGLBEGINTRANSFORMFEEDBACKEXTPROC) (GLenum primitiveMode); +typedef void (APIENTRYP PFNGLENDTRANSFORMFEEDBACKEXTPROC) (void); +typedef void (APIENTRYP PFNGLBINDBUFFERRANGEEXTPROC) (GLenum target, GLuint index, GLuint buffer, GLintptr offset, GLsizeiptr size); +typedef void (APIENTRYP PFNGLBINDBUFFEROFFSETEXTPROC) (GLenum target, GLuint index, GLuint buffer, GLintptr offset); +typedef void (APIENTRYP PFNGLBINDBUFFERBASEEXTPROC) (GLenum target, GLuint index, GLuint buffer); +typedef void (APIENTRYP PFNGLTRANSFORMFEEDBACKVARYINGSEXTPROC) (GLuint program, GLsizei count, const GLchar *const*varyings, GLenum bufferMode); +typedef void (APIENTRYP PFNGLGETTRANSFORMFEEDBACKVARYINGEXTPROC) (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLsizei *size, GLenum *type, GLchar *name); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBeginTransformFeedbackEXT (GLenum primitiveMode); +GLAPI void APIENTRY glEndTransformFeedbackEXT (void); +GLAPI void APIENTRY glBindBufferRangeEXT (GLenum target, GLuint index, GLuint buffer, GLintptr offset, GLsizeiptr size); +GLAPI void APIENTRY glBindBufferOffsetEXT (GLenum target, GLuint index, GLuint buffer, GLintptr offset); +GLAPI void APIENTRY glBindBufferBaseEXT (GLenum target, GLuint index, GLuint buffer); +GLAPI void APIENTRY glTransformFeedbackVaryingsEXT (GLuint program, GLsizei count, const GLchar *const*varyings, GLenum bufferMode); +GLAPI void APIENTRY glGetTransformFeedbackVaryingEXT (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLsizei *size, GLenum *type, GLchar *name); +#endif +#endif /* GL_EXT_transform_feedback */ + +#ifndef GL_EXT_vertex_array +#define GL_EXT_vertex_array 1 +#define GL_VERTEX_ARRAY_EXT 0x8074 +#define GL_NORMAL_ARRAY_EXT 0x8075 +#define GL_COLOR_ARRAY_EXT 0x8076 +#define GL_INDEX_ARRAY_EXT 0x8077 +#define GL_TEXTURE_COORD_ARRAY_EXT 0x8078 +#define GL_EDGE_FLAG_ARRAY_EXT 0x8079 +#define GL_VERTEX_ARRAY_SIZE_EXT 0x807A +#define GL_VERTEX_ARRAY_TYPE_EXT 0x807B +#define GL_VERTEX_ARRAY_STRIDE_EXT 0x807C +#define GL_VERTEX_ARRAY_COUNT_EXT 0x807D +#define GL_NORMAL_ARRAY_TYPE_EXT 0x807E +#define GL_NORMAL_ARRAY_STRIDE_EXT 0x807F +#define GL_NORMAL_ARRAY_COUNT_EXT 0x8080 +#define GL_COLOR_ARRAY_SIZE_EXT 0x8081 +#define GL_COLOR_ARRAY_TYPE_EXT 0x8082 +#define GL_COLOR_ARRAY_STRIDE_EXT 0x8083 +#define GL_COLOR_ARRAY_COUNT_EXT 0x8084 +#define GL_INDEX_ARRAY_TYPE_EXT 0x8085 +#define GL_INDEX_ARRAY_STRIDE_EXT 0x8086 +#define GL_INDEX_ARRAY_COUNT_EXT 0x8087 +#define GL_TEXTURE_COORD_ARRAY_SIZE_EXT 0x8088 +#define GL_TEXTURE_COORD_ARRAY_TYPE_EXT 0x8089 +#define GL_TEXTURE_COORD_ARRAY_STRIDE_EXT 0x808A +#define GL_TEXTURE_COORD_ARRAY_COUNT_EXT 0x808B +#define GL_EDGE_FLAG_ARRAY_STRIDE_EXT 0x808C +#define GL_EDGE_FLAG_ARRAY_COUNT_EXT 0x808D +#define GL_VERTEX_ARRAY_POINTER_EXT 0x808E +#define GL_NORMAL_ARRAY_POINTER_EXT 0x808F +#define GL_COLOR_ARRAY_POINTER_EXT 0x8090 +#define GL_INDEX_ARRAY_POINTER_EXT 0x8091 +#define GL_TEXTURE_COORD_ARRAY_POINTER_EXT 0x8092 +#define GL_EDGE_FLAG_ARRAY_POINTER_EXT 0x8093 +typedef void (APIENTRYP PFNGLARRAYELEMENTEXTPROC) (GLint i); +typedef void (APIENTRYP PFNGLCOLORPOINTEREXTPROC) (GLint size, GLenum type, GLsizei stride, GLsizei count, const void *pointer); +typedef void (APIENTRYP PFNGLDRAWARRAYSEXTPROC) (GLenum mode, GLint first, GLsizei count); +typedef void (APIENTRYP PFNGLEDGEFLAGPOINTEREXTPROC) (GLsizei stride, GLsizei count, const GLboolean *pointer); +typedef void (APIENTRYP PFNGLGETPOINTERVEXTPROC) (GLenum pname, void **params); +typedef void (APIENTRYP PFNGLINDEXPOINTEREXTPROC) (GLenum type, GLsizei stride, GLsizei count, const void *pointer); +typedef void (APIENTRYP PFNGLNORMALPOINTEREXTPROC) (GLenum type, GLsizei stride, GLsizei count, const void *pointer); +typedef void (APIENTRYP PFNGLTEXCOORDPOINTEREXTPROC) (GLint size, GLenum type, GLsizei stride, GLsizei count, const void *pointer); +typedef void (APIENTRYP PFNGLVERTEXPOINTEREXTPROC) (GLint size, GLenum type, GLsizei stride, GLsizei count, const void *pointer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glArrayElementEXT (GLint i); +GLAPI void APIENTRY glColorPointerEXT (GLint size, GLenum type, GLsizei stride, GLsizei count, const void *pointer); +GLAPI void APIENTRY glDrawArraysEXT (GLenum mode, GLint first, GLsizei count); +GLAPI void APIENTRY glEdgeFlagPointerEXT (GLsizei stride, GLsizei count, const GLboolean *pointer); +GLAPI void APIENTRY glGetPointervEXT (GLenum pname, void **params); +GLAPI void APIENTRY glIndexPointerEXT (GLenum type, GLsizei stride, GLsizei count, const void *pointer); +GLAPI void APIENTRY glNormalPointerEXT (GLenum type, GLsizei stride, GLsizei count, const void *pointer); +GLAPI void APIENTRY glTexCoordPointerEXT (GLint size, GLenum type, GLsizei stride, GLsizei count, const void *pointer); +GLAPI void APIENTRY glVertexPointerEXT (GLint size, GLenum type, GLsizei stride, GLsizei count, const void *pointer); +#endif +#endif /* GL_EXT_vertex_array */ + +#ifndef GL_EXT_vertex_array_bgra +#define GL_EXT_vertex_array_bgra 1 +#endif /* GL_EXT_vertex_array_bgra */ + +#ifndef GL_EXT_vertex_attrib_64bit +#define GL_EXT_vertex_attrib_64bit 1 +#define GL_DOUBLE_VEC2_EXT 0x8FFC +#define GL_DOUBLE_VEC3_EXT 0x8FFD +#define GL_DOUBLE_VEC4_EXT 0x8FFE +#define GL_DOUBLE_MAT2_EXT 0x8F46 +#define GL_DOUBLE_MAT3_EXT 0x8F47 +#define GL_DOUBLE_MAT4_EXT 0x8F48 +#define GL_DOUBLE_MAT2x3_EXT 0x8F49 +#define GL_DOUBLE_MAT2x4_EXT 0x8F4A +#define GL_DOUBLE_MAT3x2_EXT 0x8F4B +#define GL_DOUBLE_MAT3x4_EXT 0x8F4C +#define GL_DOUBLE_MAT4x2_EXT 0x8F4D +#define GL_DOUBLE_MAT4x3_EXT 0x8F4E +typedef void (APIENTRYP PFNGLVERTEXATTRIBL1DEXTPROC) (GLuint index, GLdouble x); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL2DEXTPROC) (GLuint index, GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL3DEXTPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL4DEXTPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL1DVEXTPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL2DVEXTPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL3DVEXTPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL4DVEXTPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBLPOINTEREXTPROC) (GLuint index, GLint size, GLenum type, GLsizei stride, const void *pointer); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBLDVEXTPROC) (GLuint index, GLenum pname, GLdouble *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexAttribL1dEXT (GLuint index, GLdouble x); +GLAPI void APIENTRY glVertexAttribL2dEXT (GLuint index, GLdouble x, GLdouble y); +GLAPI void APIENTRY glVertexAttribL3dEXT (GLuint index, GLdouble x, GLdouble y, GLdouble z); +GLAPI void APIENTRY glVertexAttribL4dEXT (GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +GLAPI void APIENTRY glVertexAttribL1dvEXT (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttribL2dvEXT (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttribL3dvEXT (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttribL4dvEXT (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttribLPointerEXT (GLuint index, GLint size, GLenum type, GLsizei stride, const void *pointer); +GLAPI void APIENTRY glGetVertexAttribLdvEXT (GLuint index, GLenum pname, GLdouble *params); +#endif +#endif /* GL_EXT_vertex_attrib_64bit */ + +#ifndef GL_EXT_vertex_shader +#define GL_EXT_vertex_shader 1 +#define GL_VERTEX_SHADER_EXT 0x8780 +#define GL_VERTEX_SHADER_BINDING_EXT 0x8781 +#define GL_OP_INDEX_EXT 0x8782 +#define GL_OP_NEGATE_EXT 0x8783 +#define GL_OP_DOT3_EXT 0x8784 +#define GL_OP_DOT4_EXT 0x8785 +#define GL_OP_MUL_EXT 0x8786 +#define GL_OP_ADD_EXT 0x8787 +#define GL_OP_MADD_EXT 0x8788 +#define GL_OP_FRAC_EXT 0x8789 +#define GL_OP_MAX_EXT 0x878A +#define GL_OP_MIN_EXT 0x878B +#define GL_OP_SET_GE_EXT 0x878C +#define GL_OP_SET_LT_EXT 0x878D +#define GL_OP_CLAMP_EXT 0x878E +#define GL_OP_FLOOR_EXT 0x878F +#define GL_OP_ROUND_EXT 0x8790 +#define GL_OP_EXP_BASE_2_EXT 0x8791 +#define GL_OP_LOG_BASE_2_EXT 0x8792 +#define GL_OP_POWER_EXT 0x8793 +#define GL_OP_RECIP_EXT 0x8794 +#define GL_OP_RECIP_SQRT_EXT 0x8795 +#define GL_OP_SUB_EXT 0x8796 +#define GL_OP_CROSS_PRODUCT_EXT 0x8797 +#define GL_OP_MULTIPLY_MATRIX_EXT 0x8798 +#define GL_OP_MOV_EXT 0x8799 +#define GL_OUTPUT_VERTEX_EXT 0x879A +#define GL_OUTPUT_COLOR0_EXT 0x879B +#define GL_OUTPUT_COLOR1_EXT 0x879C +#define GL_OUTPUT_TEXTURE_COORD0_EXT 0x879D +#define GL_OUTPUT_TEXTURE_COORD1_EXT 0x879E +#define GL_OUTPUT_TEXTURE_COORD2_EXT 0x879F +#define GL_OUTPUT_TEXTURE_COORD3_EXT 0x87A0 +#define GL_OUTPUT_TEXTURE_COORD4_EXT 0x87A1 +#define GL_OUTPUT_TEXTURE_COORD5_EXT 0x87A2 +#define GL_OUTPUT_TEXTURE_COORD6_EXT 0x87A3 +#define GL_OUTPUT_TEXTURE_COORD7_EXT 0x87A4 +#define GL_OUTPUT_TEXTURE_COORD8_EXT 0x87A5 +#define GL_OUTPUT_TEXTURE_COORD9_EXT 0x87A6 +#define GL_OUTPUT_TEXTURE_COORD10_EXT 0x87A7 +#define GL_OUTPUT_TEXTURE_COORD11_EXT 0x87A8 +#define GL_OUTPUT_TEXTURE_COORD12_EXT 0x87A9 +#define GL_OUTPUT_TEXTURE_COORD13_EXT 0x87AA +#define GL_OUTPUT_TEXTURE_COORD14_EXT 0x87AB +#define GL_OUTPUT_TEXTURE_COORD15_EXT 0x87AC +#define GL_OUTPUT_TEXTURE_COORD16_EXT 0x87AD +#define GL_OUTPUT_TEXTURE_COORD17_EXT 0x87AE +#define GL_OUTPUT_TEXTURE_COORD18_EXT 0x87AF +#define GL_OUTPUT_TEXTURE_COORD19_EXT 0x87B0 +#define GL_OUTPUT_TEXTURE_COORD20_EXT 0x87B1 +#define GL_OUTPUT_TEXTURE_COORD21_EXT 0x87B2 +#define GL_OUTPUT_TEXTURE_COORD22_EXT 0x87B3 +#define GL_OUTPUT_TEXTURE_COORD23_EXT 0x87B4 +#define GL_OUTPUT_TEXTURE_COORD24_EXT 0x87B5 +#define GL_OUTPUT_TEXTURE_COORD25_EXT 0x87B6 +#define GL_OUTPUT_TEXTURE_COORD26_EXT 0x87B7 +#define GL_OUTPUT_TEXTURE_COORD27_EXT 0x87B8 +#define GL_OUTPUT_TEXTURE_COORD28_EXT 0x87B9 +#define GL_OUTPUT_TEXTURE_COORD29_EXT 0x87BA +#define GL_OUTPUT_TEXTURE_COORD30_EXT 0x87BB +#define GL_OUTPUT_TEXTURE_COORD31_EXT 0x87BC +#define GL_OUTPUT_FOG_EXT 0x87BD +#define GL_SCALAR_EXT 0x87BE +#define GL_VECTOR_EXT 0x87BF +#define GL_MATRIX_EXT 0x87C0 +#define GL_VARIANT_EXT 0x87C1 +#define GL_INVARIANT_EXT 0x87C2 +#define GL_LOCAL_CONSTANT_EXT 0x87C3 +#define GL_LOCAL_EXT 0x87C4 +#define GL_MAX_VERTEX_SHADER_INSTRUCTIONS_EXT 0x87C5 +#define GL_MAX_VERTEX_SHADER_VARIANTS_EXT 0x87C6 +#define GL_MAX_VERTEX_SHADER_INVARIANTS_EXT 0x87C7 +#define GL_MAX_VERTEX_SHADER_LOCAL_CONSTANTS_EXT 0x87C8 +#define GL_MAX_VERTEX_SHADER_LOCALS_EXT 0x87C9 +#define GL_MAX_OPTIMIZED_VERTEX_SHADER_INSTRUCTIONS_EXT 0x87CA +#define GL_MAX_OPTIMIZED_VERTEX_SHADER_VARIANTS_EXT 0x87CB +#define GL_MAX_OPTIMIZED_VERTEX_SHADER_LOCAL_CONSTANTS_EXT 0x87CC +#define GL_MAX_OPTIMIZED_VERTEX_SHADER_INVARIANTS_EXT 0x87CD +#define GL_MAX_OPTIMIZED_VERTEX_SHADER_LOCALS_EXT 0x87CE +#define GL_VERTEX_SHADER_INSTRUCTIONS_EXT 0x87CF +#define GL_VERTEX_SHADER_VARIANTS_EXT 0x87D0 +#define GL_VERTEX_SHADER_INVARIANTS_EXT 0x87D1 +#define GL_VERTEX_SHADER_LOCAL_CONSTANTS_EXT 0x87D2 +#define GL_VERTEX_SHADER_LOCALS_EXT 0x87D3 +#define GL_VERTEX_SHADER_OPTIMIZED_EXT 0x87D4 +#define GL_X_EXT 0x87D5 +#define GL_Y_EXT 0x87D6 +#define GL_Z_EXT 0x87D7 +#define GL_W_EXT 0x87D8 +#define GL_NEGATIVE_X_EXT 0x87D9 +#define GL_NEGATIVE_Y_EXT 0x87DA +#define GL_NEGATIVE_Z_EXT 0x87DB +#define GL_NEGATIVE_W_EXT 0x87DC +#define GL_ZERO_EXT 0x87DD +#define GL_ONE_EXT 0x87DE +#define GL_NEGATIVE_ONE_EXT 0x87DF +#define GL_NORMALIZED_RANGE_EXT 0x87E0 +#define GL_FULL_RANGE_EXT 0x87E1 +#define GL_CURRENT_VERTEX_EXT 0x87E2 +#define GL_MVP_MATRIX_EXT 0x87E3 +#define GL_VARIANT_VALUE_EXT 0x87E4 +#define GL_VARIANT_DATATYPE_EXT 0x87E5 +#define GL_VARIANT_ARRAY_STRIDE_EXT 0x87E6 +#define GL_VARIANT_ARRAY_TYPE_EXT 0x87E7 +#define GL_VARIANT_ARRAY_EXT 0x87E8 +#define GL_VARIANT_ARRAY_POINTER_EXT 0x87E9 +#define GL_INVARIANT_VALUE_EXT 0x87EA +#define GL_INVARIANT_DATATYPE_EXT 0x87EB +#define GL_LOCAL_CONSTANT_VALUE_EXT 0x87EC +#define GL_LOCAL_CONSTANT_DATATYPE_EXT 0x87ED +typedef void (APIENTRYP PFNGLBEGINVERTEXSHADEREXTPROC) (void); +typedef void (APIENTRYP PFNGLENDVERTEXSHADEREXTPROC) (void); +typedef void (APIENTRYP PFNGLBINDVERTEXSHADEREXTPROC) (GLuint id); +typedef GLuint (APIENTRYP PFNGLGENVERTEXSHADERSEXTPROC) (GLuint range); +typedef void (APIENTRYP PFNGLDELETEVERTEXSHADEREXTPROC) (GLuint id); +typedef void (APIENTRYP PFNGLSHADEROP1EXTPROC) (GLenum op, GLuint res, GLuint arg1); +typedef void (APIENTRYP PFNGLSHADEROP2EXTPROC) (GLenum op, GLuint res, GLuint arg1, GLuint arg2); +typedef void (APIENTRYP PFNGLSHADEROP3EXTPROC) (GLenum op, GLuint res, GLuint arg1, GLuint arg2, GLuint arg3); +typedef void (APIENTRYP PFNGLSWIZZLEEXTPROC) (GLuint res, GLuint in, GLenum outX, GLenum outY, GLenum outZ, GLenum outW); +typedef void (APIENTRYP PFNGLWRITEMASKEXTPROC) (GLuint res, GLuint in, GLenum outX, GLenum outY, GLenum outZ, GLenum outW); +typedef void (APIENTRYP PFNGLINSERTCOMPONENTEXTPROC) (GLuint res, GLuint src, GLuint num); +typedef void (APIENTRYP PFNGLEXTRACTCOMPONENTEXTPROC) (GLuint res, GLuint src, GLuint num); +typedef GLuint (APIENTRYP PFNGLGENSYMBOLSEXTPROC) (GLenum datatype, GLenum storagetype, GLenum range, GLuint components); +typedef void (APIENTRYP PFNGLSETINVARIANTEXTPROC) (GLuint id, GLenum type, const void *addr); +typedef void (APIENTRYP PFNGLSETLOCALCONSTANTEXTPROC) (GLuint id, GLenum type, const void *addr); +typedef void (APIENTRYP PFNGLVARIANTBVEXTPROC) (GLuint id, const GLbyte *addr); +typedef void (APIENTRYP PFNGLVARIANTSVEXTPROC) (GLuint id, const GLshort *addr); +typedef void (APIENTRYP PFNGLVARIANTIVEXTPROC) (GLuint id, const GLint *addr); +typedef void (APIENTRYP PFNGLVARIANTFVEXTPROC) (GLuint id, const GLfloat *addr); +typedef void (APIENTRYP PFNGLVARIANTDVEXTPROC) (GLuint id, const GLdouble *addr); +typedef void (APIENTRYP PFNGLVARIANTUBVEXTPROC) (GLuint id, const GLubyte *addr); +typedef void (APIENTRYP PFNGLVARIANTUSVEXTPROC) (GLuint id, const GLushort *addr); +typedef void (APIENTRYP PFNGLVARIANTUIVEXTPROC) (GLuint id, const GLuint *addr); +typedef void (APIENTRYP PFNGLVARIANTPOINTEREXTPROC) (GLuint id, GLenum type, GLuint stride, const void *addr); +typedef void (APIENTRYP PFNGLENABLEVARIANTCLIENTSTATEEXTPROC) (GLuint id); +typedef void (APIENTRYP PFNGLDISABLEVARIANTCLIENTSTATEEXTPROC) (GLuint id); +typedef GLuint (APIENTRYP PFNGLBINDLIGHTPARAMETEREXTPROC) (GLenum light, GLenum value); +typedef GLuint (APIENTRYP PFNGLBINDMATERIALPARAMETEREXTPROC) (GLenum face, GLenum value); +typedef GLuint (APIENTRYP PFNGLBINDTEXGENPARAMETEREXTPROC) (GLenum unit, GLenum coord, GLenum value); +typedef GLuint (APIENTRYP PFNGLBINDTEXTUREUNITPARAMETEREXTPROC) (GLenum unit, GLenum value); +typedef GLuint (APIENTRYP PFNGLBINDPARAMETEREXTPROC) (GLenum value); +typedef GLboolean (APIENTRYP PFNGLISVARIANTENABLEDEXTPROC) (GLuint id, GLenum cap); +typedef void (APIENTRYP PFNGLGETVARIANTBOOLEANVEXTPROC) (GLuint id, GLenum value, GLboolean *data); +typedef void (APIENTRYP PFNGLGETVARIANTINTEGERVEXTPROC) (GLuint id, GLenum value, GLint *data); +typedef void (APIENTRYP PFNGLGETVARIANTFLOATVEXTPROC) (GLuint id, GLenum value, GLfloat *data); +typedef void (APIENTRYP PFNGLGETVARIANTPOINTERVEXTPROC) (GLuint id, GLenum value, void **data); +typedef void (APIENTRYP PFNGLGETINVARIANTBOOLEANVEXTPROC) (GLuint id, GLenum value, GLboolean *data); +typedef void (APIENTRYP PFNGLGETINVARIANTINTEGERVEXTPROC) (GLuint id, GLenum value, GLint *data); +typedef void (APIENTRYP PFNGLGETINVARIANTFLOATVEXTPROC) (GLuint id, GLenum value, GLfloat *data); +typedef void (APIENTRYP PFNGLGETLOCALCONSTANTBOOLEANVEXTPROC) (GLuint id, GLenum value, GLboolean *data); +typedef void (APIENTRYP PFNGLGETLOCALCONSTANTINTEGERVEXTPROC) (GLuint id, GLenum value, GLint *data); +typedef void (APIENTRYP PFNGLGETLOCALCONSTANTFLOATVEXTPROC) (GLuint id, GLenum value, GLfloat *data); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBeginVertexShaderEXT (void); +GLAPI void APIENTRY glEndVertexShaderEXT (void); +GLAPI void APIENTRY glBindVertexShaderEXT (GLuint id); +GLAPI GLuint APIENTRY glGenVertexShadersEXT (GLuint range); +GLAPI void APIENTRY glDeleteVertexShaderEXT (GLuint id); +GLAPI void APIENTRY glShaderOp1EXT (GLenum op, GLuint res, GLuint arg1); +GLAPI void APIENTRY glShaderOp2EXT (GLenum op, GLuint res, GLuint arg1, GLuint arg2); +GLAPI void APIENTRY glShaderOp3EXT (GLenum op, GLuint res, GLuint arg1, GLuint arg2, GLuint arg3); +GLAPI void APIENTRY glSwizzleEXT (GLuint res, GLuint in, GLenum outX, GLenum outY, GLenum outZ, GLenum outW); +GLAPI void APIENTRY glWriteMaskEXT (GLuint res, GLuint in, GLenum outX, GLenum outY, GLenum outZ, GLenum outW); +GLAPI void APIENTRY glInsertComponentEXT (GLuint res, GLuint src, GLuint num); +GLAPI void APIENTRY glExtractComponentEXT (GLuint res, GLuint src, GLuint num); +GLAPI GLuint APIENTRY glGenSymbolsEXT (GLenum datatype, GLenum storagetype, GLenum range, GLuint components); +GLAPI void APIENTRY glSetInvariantEXT (GLuint id, GLenum type, const void *addr); +GLAPI void APIENTRY glSetLocalConstantEXT (GLuint id, GLenum type, const void *addr); +GLAPI void APIENTRY glVariantbvEXT (GLuint id, const GLbyte *addr); +GLAPI void APIENTRY glVariantsvEXT (GLuint id, const GLshort *addr); +GLAPI void APIENTRY glVariantivEXT (GLuint id, const GLint *addr); +GLAPI void APIENTRY glVariantfvEXT (GLuint id, const GLfloat *addr); +GLAPI void APIENTRY glVariantdvEXT (GLuint id, const GLdouble *addr); +GLAPI void APIENTRY glVariantubvEXT (GLuint id, const GLubyte *addr); +GLAPI void APIENTRY glVariantusvEXT (GLuint id, const GLushort *addr); +GLAPI void APIENTRY glVariantuivEXT (GLuint id, const GLuint *addr); +GLAPI void APIENTRY glVariantPointerEXT (GLuint id, GLenum type, GLuint stride, const void *addr); +GLAPI void APIENTRY glEnableVariantClientStateEXT (GLuint id); +GLAPI void APIENTRY glDisableVariantClientStateEXT (GLuint id); +GLAPI GLuint APIENTRY glBindLightParameterEXT (GLenum light, GLenum value); +GLAPI GLuint APIENTRY glBindMaterialParameterEXT (GLenum face, GLenum value); +GLAPI GLuint APIENTRY glBindTexGenParameterEXT (GLenum unit, GLenum coord, GLenum value); +GLAPI GLuint APIENTRY glBindTextureUnitParameterEXT (GLenum unit, GLenum value); +GLAPI GLuint APIENTRY glBindParameterEXT (GLenum value); +GLAPI GLboolean APIENTRY glIsVariantEnabledEXT (GLuint id, GLenum cap); +GLAPI void APIENTRY glGetVariantBooleanvEXT (GLuint id, GLenum value, GLboolean *data); +GLAPI void APIENTRY glGetVariantIntegervEXT (GLuint id, GLenum value, GLint *data); +GLAPI void APIENTRY glGetVariantFloatvEXT (GLuint id, GLenum value, GLfloat *data); +GLAPI void APIENTRY glGetVariantPointervEXT (GLuint id, GLenum value, void **data); +GLAPI void APIENTRY glGetInvariantBooleanvEXT (GLuint id, GLenum value, GLboolean *data); +GLAPI void APIENTRY glGetInvariantIntegervEXT (GLuint id, GLenum value, GLint *data); +GLAPI void APIENTRY glGetInvariantFloatvEXT (GLuint id, GLenum value, GLfloat *data); +GLAPI void APIENTRY glGetLocalConstantBooleanvEXT (GLuint id, GLenum value, GLboolean *data); +GLAPI void APIENTRY glGetLocalConstantIntegervEXT (GLuint id, GLenum value, GLint *data); +GLAPI void APIENTRY glGetLocalConstantFloatvEXT (GLuint id, GLenum value, GLfloat *data); +#endif +#endif /* GL_EXT_vertex_shader */ + +#ifndef GL_EXT_vertex_weighting +#define GL_EXT_vertex_weighting 1 +#define GL_MODELVIEW0_STACK_DEPTH_EXT 0x0BA3 +#define GL_MODELVIEW1_STACK_DEPTH_EXT 0x8502 +#define GL_MODELVIEW0_MATRIX_EXT 0x0BA6 +#define GL_MODELVIEW1_MATRIX_EXT 0x8506 +#define GL_VERTEX_WEIGHTING_EXT 0x8509 +#define GL_MODELVIEW0_EXT 0x1700 +#define GL_MODELVIEW1_EXT 0x850A +#define GL_CURRENT_VERTEX_WEIGHT_EXT 0x850B +#define GL_VERTEX_WEIGHT_ARRAY_EXT 0x850C +#define GL_VERTEX_WEIGHT_ARRAY_SIZE_EXT 0x850D +#define GL_VERTEX_WEIGHT_ARRAY_TYPE_EXT 0x850E +#define GL_VERTEX_WEIGHT_ARRAY_STRIDE_EXT 0x850F +#define GL_VERTEX_WEIGHT_ARRAY_POINTER_EXT 0x8510 +typedef void (APIENTRYP PFNGLVERTEXWEIGHTFEXTPROC) (GLfloat weight); +typedef void (APIENTRYP PFNGLVERTEXWEIGHTFVEXTPROC) (const GLfloat *weight); +typedef void (APIENTRYP PFNGLVERTEXWEIGHTPOINTEREXTPROC) (GLint size, GLenum type, GLsizei stride, const void *pointer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexWeightfEXT (GLfloat weight); +GLAPI void APIENTRY glVertexWeightfvEXT (const GLfloat *weight); +GLAPI void APIENTRY glVertexWeightPointerEXT (GLint size, GLenum type, GLsizei stride, const void *pointer); +#endif +#endif /* GL_EXT_vertex_weighting */ + +#ifndef GL_EXT_win32_keyed_mutex +#define GL_EXT_win32_keyed_mutex 1 +typedef GLboolean (APIENTRYP PFNGLACQUIREKEYEDMUTEXWIN32EXTPROC) (GLuint memory, GLuint64 key, GLuint timeout); +typedef GLboolean (APIENTRYP PFNGLRELEASEKEYEDMUTEXWIN32EXTPROC) (GLuint memory, GLuint64 key); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLboolean APIENTRY glAcquireKeyedMutexWin32EXT (GLuint memory, GLuint64 key, GLuint timeout); +GLAPI GLboolean APIENTRY glReleaseKeyedMutexWin32EXT (GLuint memory, GLuint64 key); +#endif +#endif /* GL_EXT_win32_keyed_mutex */ + +#ifndef GL_EXT_window_rectangles +#define GL_EXT_window_rectangles 1 +#define GL_INCLUSIVE_EXT 0x8F10 +#define GL_EXCLUSIVE_EXT 0x8F11 +#define GL_WINDOW_RECTANGLE_EXT 0x8F12 +#define GL_WINDOW_RECTANGLE_MODE_EXT 0x8F13 +#define GL_MAX_WINDOW_RECTANGLES_EXT 0x8F14 +#define GL_NUM_WINDOW_RECTANGLES_EXT 0x8F15 +typedef void (APIENTRYP PFNGLWINDOWRECTANGLESEXTPROC) (GLenum mode, GLsizei count, const GLint *box); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glWindowRectanglesEXT (GLenum mode, GLsizei count, const GLint *box); +#endif +#endif /* GL_EXT_window_rectangles */ + +#ifndef GL_EXT_x11_sync_object +#define GL_EXT_x11_sync_object 1 +#define GL_SYNC_X11_FENCE_EXT 0x90E1 +typedef GLsync (APIENTRYP PFNGLIMPORTSYNCEXTPROC) (GLenum external_sync_type, GLintptr external_sync, GLbitfield flags); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLsync APIENTRY glImportSyncEXT (GLenum external_sync_type, GLintptr external_sync, GLbitfield flags); +#endif +#endif /* GL_EXT_x11_sync_object */ + +#ifndef GL_GREMEDY_frame_terminator +#define GL_GREMEDY_frame_terminator 1 +typedef void (APIENTRYP PFNGLFRAMETERMINATORGREMEDYPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFrameTerminatorGREMEDY (void); +#endif +#endif /* GL_GREMEDY_frame_terminator */ + +#ifndef GL_GREMEDY_string_marker +#define GL_GREMEDY_string_marker 1 +typedef void (APIENTRYP PFNGLSTRINGMARKERGREMEDYPROC) (GLsizei len, const void *string); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glStringMarkerGREMEDY (GLsizei len, const void *string); +#endif +#endif /* GL_GREMEDY_string_marker */ + +#ifndef GL_HP_convolution_border_modes +#define GL_HP_convolution_border_modes 1 +#define GL_IGNORE_BORDER_HP 0x8150 +#define GL_CONSTANT_BORDER_HP 0x8151 +#define GL_REPLICATE_BORDER_HP 0x8153 +#define GL_CONVOLUTION_BORDER_COLOR_HP 0x8154 +#endif /* GL_HP_convolution_border_modes */ + +#ifndef GL_HP_image_transform +#define GL_HP_image_transform 1 +#define GL_IMAGE_SCALE_X_HP 0x8155 +#define GL_IMAGE_SCALE_Y_HP 0x8156 +#define GL_IMAGE_TRANSLATE_X_HP 0x8157 +#define GL_IMAGE_TRANSLATE_Y_HP 0x8158 +#define GL_IMAGE_ROTATE_ANGLE_HP 0x8159 +#define GL_IMAGE_ROTATE_ORIGIN_X_HP 0x815A +#define GL_IMAGE_ROTATE_ORIGIN_Y_HP 0x815B +#define GL_IMAGE_MAG_FILTER_HP 0x815C +#define GL_IMAGE_MIN_FILTER_HP 0x815D +#define GL_IMAGE_CUBIC_WEIGHT_HP 0x815E +#define GL_CUBIC_HP 0x815F +#define GL_AVERAGE_HP 0x8160 +#define GL_IMAGE_TRANSFORM_2D_HP 0x8161 +#define GL_POST_IMAGE_TRANSFORM_COLOR_TABLE_HP 0x8162 +#define GL_PROXY_POST_IMAGE_TRANSFORM_COLOR_TABLE_HP 0x8163 +typedef void (APIENTRYP PFNGLIMAGETRANSFORMPARAMETERIHPPROC) (GLenum target, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLIMAGETRANSFORMPARAMETERFHPPROC) (GLenum target, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLIMAGETRANSFORMPARAMETERIVHPPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLIMAGETRANSFORMPARAMETERFVHPPROC) (GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLGETIMAGETRANSFORMPARAMETERIVHPPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETIMAGETRANSFORMPARAMETERFVHPPROC) (GLenum target, GLenum pname, GLfloat *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glImageTransformParameteriHP (GLenum target, GLenum pname, GLint param); +GLAPI void APIENTRY glImageTransformParameterfHP (GLenum target, GLenum pname, GLfloat param); +GLAPI void APIENTRY glImageTransformParameterivHP (GLenum target, GLenum pname, const GLint *params); +GLAPI void APIENTRY glImageTransformParameterfvHP (GLenum target, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glGetImageTransformParameterivHP (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetImageTransformParameterfvHP (GLenum target, GLenum pname, GLfloat *params); +#endif +#endif /* GL_HP_image_transform */ + +#ifndef GL_HP_occlusion_test +#define GL_HP_occlusion_test 1 +#define GL_OCCLUSION_TEST_HP 0x8165 +#define GL_OCCLUSION_TEST_RESULT_HP 0x8166 +#endif /* GL_HP_occlusion_test */ + +#ifndef GL_HP_texture_lighting +#define GL_HP_texture_lighting 1 +#define GL_TEXTURE_LIGHTING_MODE_HP 0x8167 +#define GL_TEXTURE_POST_SPECULAR_HP 0x8168 +#define GL_TEXTURE_PRE_SPECULAR_HP 0x8169 +#endif /* GL_HP_texture_lighting */ + +#ifndef GL_IBM_cull_vertex +#define GL_IBM_cull_vertex 1 +#define GL_CULL_VERTEX_IBM 103050 +#endif /* GL_IBM_cull_vertex */ + +#ifndef GL_IBM_multimode_draw_arrays +#define GL_IBM_multimode_draw_arrays 1 +typedef void (APIENTRYP PFNGLMULTIMODEDRAWARRAYSIBMPROC) (const GLenum *mode, const GLint *first, const GLsizei *count, GLsizei primcount, GLint modestride); +typedef void (APIENTRYP PFNGLMULTIMODEDRAWELEMENTSIBMPROC) (const GLenum *mode, const GLsizei *count, GLenum type, const void *const*indices, GLsizei primcount, GLint modestride); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMultiModeDrawArraysIBM (const GLenum *mode, const GLint *first, const GLsizei *count, GLsizei primcount, GLint modestride); +GLAPI void APIENTRY glMultiModeDrawElementsIBM (const GLenum *mode, const GLsizei *count, GLenum type, const void *const*indices, GLsizei primcount, GLint modestride); +#endif +#endif /* GL_IBM_multimode_draw_arrays */ + +#ifndef GL_IBM_rasterpos_clip +#define GL_IBM_rasterpos_clip 1 +#define GL_RASTER_POSITION_UNCLIPPED_IBM 0x19262 +#endif /* GL_IBM_rasterpos_clip */ + +#ifndef GL_IBM_static_data +#define GL_IBM_static_data 1 +#define GL_ALL_STATIC_DATA_IBM 103060 +#define GL_STATIC_VERTEX_ARRAY_IBM 103061 +typedef void (APIENTRYP PFNGLFLUSHSTATICDATAIBMPROC) (GLenum target); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFlushStaticDataIBM (GLenum target); +#endif +#endif /* GL_IBM_static_data */ + +#ifndef GL_IBM_texture_mirrored_repeat +#define GL_IBM_texture_mirrored_repeat 1 +#define GL_MIRRORED_REPEAT_IBM 0x8370 +#endif /* GL_IBM_texture_mirrored_repeat */ + +#ifndef GL_IBM_vertex_array_lists +#define GL_IBM_vertex_array_lists 1 +#define GL_VERTEX_ARRAY_LIST_IBM 103070 +#define GL_NORMAL_ARRAY_LIST_IBM 103071 +#define GL_COLOR_ARRAY_LIST_IBM 103072 +#define GL_INDEX_ARRAY_LIST_IBM 103073 +#define GL_TEXTURE_COORD_ARRAY_LIST_IBM 103074 +#define GL_EDGE_FLAG_ARRAY_LIST_IBM 103075 +#define GL_FOG_COORDINATE_ARRAY_LIST_IBM 103076 +#define GL_SECONDARY_COLOR_ARRAY_LIST_IBM 103077 +#define GL_VERTEX_ARRAY_LIST_STRIDE_IBM 103080 +#define GL_NORMAL_ARRAY_LIST_STRIDE_IBM 103081 +#define GL_COLOR_ARRAY_LIST_STRIDE_IBM 103082 +#define GL_INDEX_ARRAY_LIST_STRIDE_IBM 103083 +#define GL_TEXTURE_COORD_ARRAY_LIST_STRIDE_IBM 103084 +#define GL_EDGE_FLAG_ARRAY_LIST_STRIDE_IBM 103085 +#define GL_FOG_COORDINATE_ARRAY_LIST_STRIDE_IBM 103086 +#define GL_SECONDARY_COLOR_ARRAY_LIST_STRIDE_IBM 103087 +typedef void (APIENTRYP PFNGLCOLORPOINTERLISTIBMPROC) (GLint size, GLenum type, GLint stride, const void **pointer, GLint ptrstride); +typedef void (APIENTRYP PFNGLSECONDARYCOLORPOINTERLISTIBMPROC) (GLint size, GLenum type, GLint stride, const void **pointer, GLint ptrstride); +typedef void (APIENTRYP PFNGLEDGEFLAGPOINTERLISTIBMPROC) (GLint stride, const GLboolean **pointer, GLint ptrstride); +typedef void (APIENTRYP PFNGLFOGCOORDPOINTERLISTIBMPROC) (GLenum type, GLint stride, const void **pointer, GLint ptrstride); +typedef void (APIENTRYP PFNGLINDEXPOINTERLISTIBMPROC) (GLenum type, GLint stride, const void **pointer, GLint ptrstride); +typedef void (APIENTRYP PFNGLNORMALPOINTERLISTIBMPROC) (GLenum type, GLint stride, const void **pointer, GLint ptrstride); +typedef void (APIENTRYP PFNGLTEXCOORDPOINTERLISTIBMPROC) (GLint size, GLenum type, GLint stride, const void **pointer, GLint ptrstride); +typedef void (APIENTRYP PFNGLVERTEXPOINTERLISTIBMPROC) (GLint size, GLenum type, GLint stride, const void **pointer, GLint ptrstride); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glColorPointerListIBM (GLint size, GLenum type, GLint stride, const void **pointer, GLint ptrstride); +GLAPI void APIENTRY glSecondaryColorPointerListIBM (GLint size, GLenum type, GLint stride, const void **pointer, GLint ptrstride); +GLAPI void APIENTRY glEdgeFlagPointerListIBM (GLint stride, const GLboolean **pointer, GLint ptrstride); +GLAPI void APIENTRY glFogCoordPointerListIBM (GLenum type, GLint stride, const void **pointer, GLint ptrstride); +GLAPI void APIENTRY glIndexPointerListIBM (GLenum type, GLint stride, const void **pointer, GLint ptrstride); +GLAPI void APIENTRY glNormalPointerListIBM (GLenum type, GLint stride, const void **pointer, GLint ptrstride); +GLAPI void APIENTRY glTexCoordPointerListIBM (GLint size, GLenum type, GLint stride, const void **pointer, GLint ptrstride); +GLAPI void APIENTRY glVertexPointerListIBM (GLint size, GLenum type, GLint stride, const void **pointer, GLint ptrstride); +#endif +#endif /* GL_IBM_vertex_array_lists */ + +#ifndef GL_INGR_blend_func_separate +#define GL_INGR_blend_func_separate 1 +typedef void (APIENTRYP PFNGLBLENDFUNCSEPARATEINGRPROC) (GLenum sfactorRGB, GLenum dfactorRGB, GLenum sfactorAlpha, GLenum dfactorAlpha); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendFuncSeparateINGR (GLenum sfactorRGB, GLenum dfactorRGB, GLenum sfactorAlpha, GLenum dfactorAlpha); +#endif +#endif /* GL_INGR_blend_func_separate */ + +#ifndef GL_INGR_color_clamp +#define GL_INGR_color_clamp 1 +#define GL_RED_MIN_CLAMP_INGR 0x8560 +#define GL_GREEN_MIN_CLAMP_INGR 0x8561 +#define GL_BLUE_MIN_CLAMP_INGR 0x8562 +#define GL_ALPHA_MIN_CLAMP_INGR 0x8563 +#define GL_RED_MAX_CLAMP_INGR 0x8564 +#define GL_GREEN_MAX_CLAMP_INGR 0x8565 +#define GL_BLUE_MAX_CLAMP_INGR 0x8566 +#define GL_ALPHA_MAX_CLAMP_INGR 0x8567 +#endif /* GL_INGR_color_clamp */ + +#ifndef GL_INGR_interlace_read +#define GL_INGR_interlace_read 1 +#define GL_INTERLACE_READ_INGR 0x8568 +#endif /* GL_INGR_interlace_read */ + +#ifndef GL_INTEL_blackhole_render +#define GL_INTEL_blackhole_render 1 +#define GL_BLACKHOLE_RENDER_INTEL 0x83FC +#endif /* GL_INTEL_blackhole_render */ + +#ifndef GL_INTEL_conservative_rasterization +#define GL_INTEL_conservative_rasterization 1 +#define GL_CONSERVATIVE_RASTERIZATION_INTEL 0x83FE +#endif /* GL_INTEL_conservative_rasterization */ + +#ifndef GL_INTEL_fragment_shader_ordering +#define GL_INTEL_fragment_shader_ordering 1 +#endif /* GL_INTEL_fragment_shader_ordering */ + +#ifndef GL_INTEL_framebuffer_CMAA +#define GL_INTEL_framebuffer_CMAA 1 +typedef void (APIENTRYP PFNGLAPPLYFRAMEBUFFERATTACHMENTCMAAINTELPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glApplyFramebufferAttachmentCMAAINTEL (void); +#endif +#endif /* GL_INTEL_framebuffer_CMAA */ + +#ifndef GL_INTEL_map_texture +#define GL_INTEL_map_texture 1 +#define GL_TEXTURE_MEMORY_LAYOUT_INTEL 0x83FF +#define GL_LAYOUT_DEFAULT_INTEL 0 +#define GL_LAYOUT_LINEAR_INTEL 1 +#define GL_LAYOUT_LINEAR_CPU_CACHED_INTEL 2 +typedef void (APIENTRYP PFNGLSYNCTEXTUREINTELPROC) (GLuint texture); +typedef void (APIENTRYP PFNGLUNMAPTEXTURE2DINTELPROC) (GLuint texture, GLint level); +typedef void *(APIENTRYP PFNGLMAPTEXTURE2DINTELPROC) (GLuint texture, GLint level, GLbitfield access, GLint *stride, GLenum *layout); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSyncTextureINTEL (GLuint texture); +GLAPI void APIENTRY glUnmapTexture2DINTEL (GLuint texture, GLint level); +GLAPI void *APIENTRY glMapTexture2DINTEL (GLuint texture, GLint level, GLbitfield access, GLint *stride, GLenum *layout); +#endif +#endif /* GL_INTEL_map_texture */ + +#ifndef GL_INTEL_parallel_arrays +#define GL_INTEL_parallel_arrays 1 +#define GL_PARALLEL_ARRAYS_INTEL 0x83F4 +#define GL_VERTEX_ARRAY_PARALLEL_POINTERS_INTEL 0x83F5 +#define GL_NORMAL_ARRAY_PARALLEL_POINTERS_INTEL 0x83F6 +#define GL_COLOR_ARRAY_PARALLEL_POINTERS_INTEL 0x83F7 +#define GL_TEXTURE_COORD_ARRAY_PARALLEL_POINTERS_INTEL 0x83F8 +typedef void (APIENTRYP PFNGLVERTEXPOINTERVINTELPROC) (GLint size, GLenum type, const void **pointer); +typedef void (APIENTRYP PFNGLNORMALPOINTERVINTELPROC) (GLenum type, const void **pointer); +typedef void (APIENTRYP PFNGLCOLORPOINTERVINTELPROC) (GLint size, GLenum type, const void **pointer); +typedef void (APIENTRYP PFNGLTEXCOORDPOINTERVINTELPROC) (GLint size, GLenum type, const void **pointer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexPointervINTEL (GLint size, GLenum type, const void **pointer); +GLAPI void APIENTRY glNormalPointervINTEL (GLenum type, const void **pointer); +GLAPI void APIENTRY glColorPointervINTEL (GLint size, GLenum type, const void **pointer); +GLAPI void APIENTRY glTexCoordPointervINTEL (GLint size, GLenum type, const void **pointer); +#endif +#endif /* GL_INTEL_parallel_arrays */ + +#ifndef GL_INTEL_performance_query +#define GL_INTEL_performance_query 1 +#define GL_PERFQUERY_SINGLE_CONTEXT_INTEL 0x00000000 +#define GL_PERFQUERY_GLOBAL_CONTEXT_INTEL 0x00000001 +#define GL_PERFQUERY_WAIT_INTEL 0x83FB +#define GL_PERFQUERY_FLUSH_INTEL 0x83FA +#define GL_PERFQUERY_DONOT_FLUSH_INTEL 0x83F9 +#define GL_PERFQUERY_COUNTER_EVENT_INTEL 0x94F0 +#define GL_PERFQUERY_COUNTER_DURATION_NORM_INTEL 0x94F1 +#define GL_PERFQUERY_COUNTER_DURATION_RAW_INTEL 0x94F2 +#define GL_PERFQUERY_COUNTER_THROUGHPUT_INTEL 0x94F3 +#define GL_PERFQUERY_COUNTER_RAW_INTEL 0x94F4 +#define GL_PERFQUERY_COUNTER_TIMESTAMP_INTEL 0x94F5 +#define GL_PERFQUERY_COUNTER_DATA_UINT32_INTEL 0x94F8 +#define GL_PERFQUERY_COUNTER_DATA_UINT64_INTEL 0x94F9 +#define GL_PERFQUERY_COUNTER_DATA_FLOAT_INTEL 0x94FA +#define GL_PERFQUERY_COUNTER_DATA_DOUBLE_INTEL 0x94FB +#define GL_PERFQUERY_COUNTER_DATA_BOOL32_INTEL 0x94FC +#define GL_PERFQUERY_QUERY_NAME_LENGTH_MAX_INTEL 0x94FD +#define GL_PERFQUERY_COUNTER_NAME_LENGTH_MAX_INTEL 0x94FE +#define GL_PERFQUERY_COUNTER_DESC_LENGTH_MAX_INTEL 0x94FF +#define GL_PERFQUERY_GPA_EXTENDED_COUNTERS_INTEL 0x9500 +typedef void (APIENTRYP PFNGLBEGINPERFQUERYINTELPROC) (GLuint queryHandle); +typedef void (APIENTRYP PFNGLCREATEPERFQUERYINTELPROC) (GLuint queryId, GLuint *queryHandle); +typedef void (APIENTRYP PFNGLDELETEPERFQUERYINTELPROC) (GLuint queryHandle); +typedef void (APIENTRYP PFNGLENDPERFQUERYINTELPROC) (GLuint queryHandle); +typedef void (APIENTRYP PFNGLGETFIRSTPERFQUERYIDINTELPROC) (GLuint *queryId); +typedef void (APIENTRYP PFNGLGETNEXTPERFQUERYIDINTELPROC) (GLuint queryId, GLuint *nextQueryId); +typedef void (APIENTRYP PFNGLGETPERFCOUNTERINFOINTELPROC) (GLuint queryId, GLuint counterId, GLuint counterNameLength, GLchar *counterName, GLuint counterDescLength, GLchar *counterDesc, GLuint *counterOffset, GLuint *counterDataSize, GLuint *counterTypeEnum, GLuint *counterDataTypeEnum, GLuint64 *rawCounterMaxValue); +typedef void (APIENTRYP PFNGLGETPERFQUERYDATAINTELPROC) (GLuint queryHandle, GLuint flags, GLsizei dataSize, void *data, GLuint *bytesWritten); +typedef void (APIENTRYP PFNGLGETPERFQUERYIDBYNAMEINTELPROC) (GLchar *queryName, GLuint *queryId); +typedef void (APIENTRYP PFNGLGETPERFQUERYINFOINTELPROC) (GLuint queryId, GLuint queryNameLength, GLchar *queryName, GLuint *dataSize, GLuint *noCounters, GLuint *noInstances, GLuint *capsMask); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBeginPerfQueryINTEL (GLuint queryHandle); +GLAPI void APIENTRY glCreatePerfQueryINTEL (GLuint queryId, GLuint *queryHandle); +GLAPI void APIENTRY glDeletePerfQueryINTEL (GLuint queryHandle); +GLAPI void APIENTRY glEndPerfQueryINTEL (GLuint queryHandle); +GLAPI void APIENTRY glGetFirstPerfQueryIdINTEL (GLuint *queryId); +GLAPI void APIENTRY glGetNextPerfQueryIdINTEL (GLuint queryId, GLuint *nextQueryId); +GLAPI void APIENTRY glGetPerfCounterInfoINTEL (GLuint queryId, GLuint counterId, GLuint counterNameLength, GLchar *counterName, GLuint counterDescLength, GLchar *counterDesc, GLuint *counterOffset, GLuint *counterDataSize, GLuint *counterTypeEnum, GLuint *counterDataTypeEnum, GLuint64 *rawCounterMaxValue); +GLAPI void APIENTRY glGetPerfQueryDataINTEL (GLuint queryHandle, GLuint flags, GLsizei dataSize, void *data, GLuint *bytesWritten); +GLAPI void APIENTRY glGetPerfQueryIdByNameINTEL (GLchar *queryName, GLuint *queryId); +GLAPI void APIENTRY glGetPerfQueryInfoINTEL (GLuint queryId, GLuint queryNameLength, GLchar *queryName, GLuint *dataSize, GLuint *noCounters, GLuint *noInstances, GLuint *capsMask); +#endif +#endif /* GL_INTEL_performance_query */ + +#ifndef GL_MESAX_texture_stack +#define GL_MESAX_texture_stack 1 +#define GL_TEXTURE_1D_STACK_MESAX 0x8759 +#define GL_TEXTURE_2D_STACK_MESAX 0x875A +#define GL_PROXY_TEXTURE_1D_STACK_MESAX 0x875B +#define GL_PROXY_TEXTURE_2D_STACK_MESAX 0x875C +#define GL_TEXTURE_1D_STACK_BINDING_MESAX 0x875D +#define GL_TEXTURE_2D_STACK_BINDING_MESAX 0x875E +#endif /* GL_MESAX_texture_stack */ + +#ifndef GL_MESA_framebuffer_flip_x +#define GL_MESA_framebuffer_flip_x 1 +#define GL_FRAMEBUFFER_FLIP_X_MESA 0x8BBC +#endif /* GL_MESA_framebuffer_flip_x */ + +#ifndef GL_MESA_framebuffer_flip_y +#define GL_MESA_framebuffer_flip_y 1 +#define GL_FRAMEBUFFER_FLIP_Y_MESA 0x8BBB +typedef void (APIENTRYP PFNGLFRAMEBUFFERPARAMETERIMESAPROC) (GLenum target, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLGETFRAMEBUFFERPARAMETERIVMESAPROC) (GLenum target, GLenum pname, GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFramebufferParameteriMESA (GLenum target, GLenum pname, GLint param); +GLAPI void APIENTRY glGetFramebufferParameterivMESA (GLenum target, GLenum pname, GLint *params); +#endif +#endif /* GL_MESA_framebuffer_flip_y */ + +#ifndef GL_MESA_framebuffer_swap_xy +#define GL_MESA_framebuffer_swap_xy 1 +#define GL_FRAMEBUFFER_SWAP_XY_MESA 0x8BBD +#endif /* GL_MESA_framebuffer_swap_xy */ + +#ifndef GL_MESA_pack_invert +#define GL_MESA_pack_invert 1 +#define GL_PACK_INVERT_MESA 0x8758 +#endif /* GL_MESA_pack_invert */ + +#ifndef GL_MESA_program_binary_formats +#define GL_MESA_program_binary_formats 1 +#define GL_PROGRAM_BINARY_FORMAT_MESA 0x875F +#endif /* GL_MESA_program_binary_formats */ + +#ifndef GL_MESA_resize_buffers +#define GL_MESA_resize_buffers 1 +typedef void (APIENTRYP PFNGLRESIZEBUFFERSMESAPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glResizeBuffersMESA (void); +#endif +#endif /* GL_MESA_resize_buffers */ + +#ifndef GL_MESA_shader_integer_functions +#define GL_MESA_shader_integer_functions 1 +#endif /* GL_MESA_shader_integer_functions */ + +#ifndef GL_MESA_tile_raster_order +#define GL_MESA_tile_raster_order 1 +#define GL_TILE_RASTER_ORDER_FIXED_MESA 0x8BB8 +#define GL_TILE_RASTER_ORDER_INCREASING_X_MESA 0x8BB9 +#define GL_TILE_RASTER_ORDER_INCREASING_Y_MESA 0x8BBA +#endif /* GL_MESA_tile_raster_order */ + +#ifndef GL_MESA_window_pos +#define GL_MESA_window_pos 1 +typedef void (APIENTRYP PFNGLWINDOWPOS2DMESAPROC) (GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLWINDOWPOS2DVMESAPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2FMESAPROC) (GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLWINDOWPOS2FVMESAPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2IMESAPROC) (GLint x, GLint y); +typedef void (APIENTRYP PFNGLWINDOWPOS2IVMESAPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2SMESAPROC) (GLshort x, GLshort y); +typedef void (APIENTRYP PFNGLWINDOWPOS2SVMESAPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3DMESAPROC) (GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLWINDOWPOS3DVMESAPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3FMESAPROC) (GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLWINDOWPOS3FVMESAPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3IMESAPROC) (GLint x, GLint y, GLint z); +typedef void (APIENTRYP PFNGLWINDOWPOS3IVMESAPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3SMESAPROC) (GLshort x, GLshort y, GLshort z); +typedef void (APIENTRYP PFNGLWINDOWPOS3SVMESAPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLWINDOWPOS4DMESAPROC) (GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLWINDOWPOS4DVMESAPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLWINDOWPOS4FMESAPROC) (GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLWINDOWPOS4FVMESAPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLWINDOWPOS4IMESAPROC) (GLint x, GLint y, GLint z, GLint w); +typedef void (APIENTRYP PFNGLWINDOWPOS4IVMESAPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLWINDOWPOS4SMESAPROC) (GLshort x, GLshort y, GLshort z, GLshort w); +typedef void (APIENTRYP PFNGLWINDOWPOS4SVMESAPROC) (const GLshort *v); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glWindowPos2dMESA (GLdouble x, GLdouble y); +GLAPI void APIENTRY glWindowPos2dvMESA (const GLdouble *v); +GLAPI void APIENTRY glWindowPos2fMESA (GLfloat x, GLfloat y); +GLAPI void APIENTRY glWindowPos2fvMESA (const GLfloat *v); +GLAPI void APIENTRY glWindowPos2iMESA (GLint x, GLint y); +GLAPI void APIENTRY glWindowPos2ivMESA (const GLint *v); +GLAPI void APIENTRY glWindowPos2sMESA (GLshort x, GLshort y); +GLAPI void APIENTRY glWindowPos2svMESA (const GLshort *v); +GLAPI void APIENTRY glWindowPos3dMESA (GLdouble x, GLdouble y, GLdouble z); +GLAPI void APIENTRY glWindowPos3dvMESA (const GLdouble *v); +GLAPI void APIENTRY glWindowPos3fMESA (GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glWindowPos3fvMESA (const GLfloat *v); +GLAPI void APIENTRY glWindowPos3iMESA (GLint x, GLint y, GLint z); +GLAPI void APIENTRY glWindowPos3ivMESA (const GLint *v); +GLAPI void APIENTRY glWindowPos3sMESA (GLshort x, GLshort y, GLshort z); +GLAPI void APIENTRY glWindowPos3svMESA (const GLshort *v); +GLAPI void APIENTRY glWindowPos4dMESA (GLdouble x, GLdouble y, GLdouble z, GLdouble w); +GLAPI void APIENTRY glWindowPos4dvMESA (const GLdouble *v); +GLAPI void APIENTRY glWindowPos4fMESA (GLfloat x, GLfloat y, GLfloat z, GLfloat w); +GLAPI void APIENTRY glWindowPos4fvMESA (const GLfloat *v); +GLAPI void APIENTRY glWindowPos4iMESA (GLint x, GLint y, GLint z, GLint w); +GLAPI void APIENTRY glWindowPos4ivMESA (const GLint *v); +GLAPI void APIENTRY glWindowPos4sMESA (GLshort x, GLshort y, GLshort z, GLshort w); +GLAPI void APIENTRY glWindowPos4svMESA (const GLshort *v); +#endif +#endif /* GL_MESA_window_pos */ + +#ifndef GL_MESA_ycbcr_texture +#define GL_MESA_ycbcr_texture 1 +#define GL_UNSIGNED_SHORT_8_8_MESA 0x85BA +#define GL_UNSIGNED_SHORT_8_8_REV_MESA 0x85BB +#define GL_YCBCR_MESA 0x8757 +#endif /* GL_MESA_ycbcr_texture */ + +#ifndef GL_NVX_blend_equation_advanced_multi_draw_buffers +#define GL_NVX_blend_equation_advanced_multi_draw_buffers 1 +#endif /* GL_NVX_blend_equation_advanced_multi_draw_buffers */ + +#ifndef GL_NVX_conditional_render +#define GL_NVX_conditional_render 1 +typedef void (APIENTRYP PFNGLBEGINCONDITIONALRENDERNVXPROC) (GLuint id); +typedef void (APIENTRYP PFNGLENDCONDITIONALRENDERNVXPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBeginConditionalRenderNVX (GLuint id); +GLAPI void APIENTRY glEndConditionalRenderNVX (void); +#endif +#endif /* GL_NVX_conditional_render */ + +#ifndef GL_NVX_gpu_memory_info +#define GL_NVX_gpu_memory_info 1 +#define GL_GPU_MEMORY_INFO_DEDICATED_VIDMEM_NVX 0x9047 +#define GL_GPU_MEMORY_INFO_TOTAL_AVAILABLE_MEMORY_NVX 0x9048 +#define GL_GPU_MEMORY_INFO_CURRENT_AVAILABLE_VIDMEM_NVX 0x9049 +#define GL_GPU_MEMORY_INFO_EVICTION_COUNT_NVX 0x904A +#define GL_GPU_MEMORY_INFO_EVICTED_MEMORY_NVX 0x904B +#endif /* GL_NVX_gpu_memory_info */ + +#ifndef GL_NVX_gpu_multicast2 +#define GL_NVX_gpu_multicast2 1 +#define GL_UPLOAD_GPU_MASK_NVX 0x954A +typedef void (APIENTRYP PFNGLUPLOADGPUMASKNVXPROC) (GLbitfield mask); +typedef void (APIENTRYP PFNGLMULTICASTVIEWPORTARRAYVNVXPROC) (GLuint gpu, GLuint first, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTICASTVIEWPORTPOSITIONWSCALENVXPROC) (GLuint gpu, GLuint index, GLfloat xcoeff, GLfloat ycoeff); +typedef void (APIENTRYP PFNGLMULTICASTSCISSORARRAYVNVXPROC) (GLuint gpu, GLuint first, GLsizei count, const GLint *v); +typedef GLuint (APIENTRYP PFNGLASYNCCOPYBUFFERSUBDATANVXPROC) (GLsizei waitSemaphoreCount, const GLuint *waitSemaphoreArray, const GLuint64 *fenceValueArray, GLuint readGpu, GLbitfield writeGpuMask, GLuint readBuffer, GLuint writeBuffer, GLintptr readOffset, GLintptr writeOffset, GLsizeiptr size, GLsizei signalSemaphoreCount, const GLuint *signalSemaphoreArray, const GLuint64 *signalValueArray); +typedef GLuint (APIENTRYP PFNGLASYNCCOPYIMAGESUBDATANVXPROC) (GLsizei waitSemaphoreCount, const GLuint *waitSemaphoreArray, const GLuint64 *waitValueArray, GLuint srcGpu, GLbitfield dstGpuMask, GLuint srcName, GLenum srcTarget, GLint srcLevel, GLint srcX, GLint srcY, GLint srcZ, GLuint dstName, GLenum dstTarget, GLint dstLevel, GLint dstX, GLint dstY, GLint dstZ, GLsizei srcWidth, GLsizei srcHeight, GLsizei srcDepth, GLsizei signalSemaphoreCount, const GLuint *signalSemaphoreArray, const GLuint64 *signalValueArray); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glUploadGpuMaskNVX (GLbitfield mask); +GLAPI void APIENTRY glMulticastViewportArrayvNVX (GLuint gpu, GLuint first, GLsizei count, const GLfloat *v); +GLAPI void APIENTRY glMulticastViewportPositionWScaleNVX (GLuint gpu, GLuint index, GLfloat xcoeff, GLfloat ycoeff); +GLAPI void APIENTRY glMulticastScissorArrayvNVX (GLuint gpu, GLuint first, GLsizei count, const GLint *v); +GLAPI GLuint APIENTRY glAsyncCopyBufferSubDataNVX (GLsizei waitSemaphoreCount, const GLuint *waitSemaphoreArray, const GLuint64 *fenceValueArray, GLuint readGpu, GLbitfield writeGpuMask, GLuint readBuffer, GLuint writeBuffer, GLintptr readOffset, GLintptr writeOffset, GLsizeiptr size, GLsizei signalSemaphoreCount, const GLuint *signalSemaphoreArray, const GLuint64 *signalValueArray); +GLAPI GLuint APIENTRY glAsyncCopyImageSubDataNVX (GLsizei waitSemaphoreCount, const GLuint *waitSemaphoreArray, const GLuint64 *waitValueArray, GLuint srcGpu, GLbitfield dstGpuMask, GLuint srcName, GLenum srcTarget, GLint srcLevel, GLint srcX, GLint srcY, GLint srcZ, GLuint dstName, GLenum dstTarget, GLint dstLevel, GLint dstX, GLint dstY, GLint dstZ, GLsizei srcWidth, GLsizei srcHeight, GLsizei srcDepth, GLsizei signalSemaphoreCount, const GLuint *signalSemaphoreArray, const GLuint64 *signalValueArray); +#endif +#endif /* GL_NVX_gpu_multicast2 */ + +#ifndef GL_NVX_linked_gpu_multicast +#define GL_NVX_linked_gpu_multicast 1 +#define GL_LGPU_SEPARATE_STORAGE_BIT_NVX 0x0800 +#define GL_MAX_LGPU_GPUS_NVX 0x92BA +typedef void (APIENTRYP PFNGLLGPUNAMEDBUFFERSUBDATANVXPROC) (GLbitfield gpuMask, GLuint buffer, GLintptr offset, GLsizeiptr size, const void *data); +typedef void (APIENTRYP PFNGLLGPUCOPYIMAGESUBDATANVXPROC) (GLuint sourceGpu, GLbitfield destinationGpuMask, GLuint srcName, GLenum srcTarget, GLint srcLevel, GLint srcX, GLint srxY, GLint srcZ, GLuint dstName, GLenum dstTarget, GLint dstLevel, GLint dstX, GLint dstY, GLint dstZ, GLsizei width, GLsizei height, GLsizei depth); +typedef void (APIENTRYP PFNGLLGPUINTERLOCKNVXPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glLGPUNamedBufferSubDataNVX (GLbitfield gpuMask, GLuint buffer, GLintptr offset, GLsizeiptr size, const void *data); +GLAPI void APIENTRY glLGPUCopyImageSubDataNVX (GLuint sourceGpu, GLbitfield destinationGpuMask, GLuint srcName, GLenum srcTarget, GLint srcLevel, GLint srcX, GLint srxY, GLint srcZ, GLuint dstName, GLenum dstTarget, GLint dstLevel, GLint dstX, GLint dstY, GLint dstZ, GLsizei width, GLsizei height, GLsizei depth); +GLAPI void APIENTRY glLGPUInterlockNVX (void); +#endif +#endif /* GL_NVX_linked_gpu_multicast */ + +#ifndef GL_NVX_progress_fence +#define GL_NVX_progress_fence 1 +typedef GLuint (APIENTRYP PFNGLCREATEPROGRESSFENCENVXPROC) (void); +typedef void (APIENTRYP PFNGLSIGNALSEMAPHOREUI64NVXPROC) (GLuint signalGpu, GLsizei fenceObjectCount, const GLuint *semaphoreArray, const GLuint64 *fenceValueArray); +typedef void (APIENTRYP PFNGLWAITSEMAPHOREUI64NVXPROC) (GLuint waitGpu, GLsizei fenceObjectCount, const GLuint *semaphoreArray, const GLuint64 *fenceValueArray); +typedef void (APIENTRYP PFNGLCLIENTWAITSEMAPHOREUI64NVXPROC) (GLsizei fenceObjectCount, const GLuint *semaphoreArray, const GLuint64 *fenceValueArray); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLuint APIENTRY glCreateProgressFenceNVX (void); +GLAPI void APIENTRY glSignalSemaphoreui64NVX (GLuint signalGpu, GLsizei fenceObjectCount, const GLuint *semaphoreArray, const GLuint64 *fenceValueArray); +GLAPI void APIENTRY glWaitSemaphoreui64NVX (GLuint waitGpu, GLsizei fenceObjectCount, const GLuint *semaphoreArray, const GLuint64 *fenceValueArray); +GLAPI void APIENTRY glClientWaitSemaphoreui64NVX (GLsizei fenceObjectCount, const GLuint *semaphoreArray, const GLuint64 *fenceValueArray); +#endif +#endif /* GL_NVX_progress_fence */ + +#ifndef GL_NV_alpha_to_coverage_dither_control +#define GL_NV_alpha_to_coverage_dither_control 1 +#define GL_ALPHA_TO_COVERAGE_DITHER_DEFAULT_NV 0x934D +#define GL_ALPHA_TO_COVERAGE_DITHER_ENABLE_NV 0x934E +#define GL_ALPHA_TO_COVERAGE_DITHER_DISABLE_NV 0x934F +#define GL_ALPHA_TO_COVERAGE_DITHER_MODE_NV 0x92BF +typedef void (APIENTRYP PFNGLALPHATOCOVERAGEDITHERCONTROLNVPROC) (GLenum mode); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glAlphaToCoverageDitherControlNV (GLenum mode); +#endif +#endif /* GL_NV_alpha_to_coverage_dither_control */ + +#ifndef GL_NV_bindless_multi_draw_indirect +#define GL_NV_bindless_multi_draw_indirect 1 +typedef void (APIENTRYP PFNGLMULTIDRAWARRAYSINDIRECTBINDLESSNVPROC) (GLenum mode, const void *indirect, GLsizei drawCount, GLsizei stride, GLint vertexBufferCount); +typedef void (APIENTRYP PFNGLMULTIDRAWELEMENTSINDIRECTBINDLESSNVPROC) (GLenum mode, GLenum type, const void *indirect, GLsizei drawCount, GLsizei stride, GLint vertexBufferCount); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMultiDrawArraysIndirectBindlessNV (GLenum mode, const void *indirect, GLsizei drawCount, GLsizei stride, GLint vertexBufferCount); +GLAPI void APIENTRY glMultiDrawElementsIndirectBindlessNV (GLenum mode, GLenum type, const void *indirect, GLsizei drawCount, GLsizei stride, GLint vertexBufferCount); +#endif +#endif /* GL_NV_bindless_multi_draw_indirect */ + +#ifndef GL_NV_bindless_multi_draw_indirect_count +#define GL_NV_bindless_multi_draw_indirect_count 1 +typedef void (APIENTRYP PFNGLMULTIDRAWARRAYSINDIRECTBINDLESSCOUNTNVPROC) (GLenum mode, const void *indirect, GLsizei drawCount, GLsizei maxDrawCount, GLsizei stride, GLint vertexBufferCount); +typedef void (APIENTRYP PFNGLMULTIDRAWELEMENTSINDIRECTBINDLESSCOUNTNVPROC) (GLenum mode, GLenum type, const void *indirect, GLsizei drawCount, GLsizei maxDrawCount, GLsizei stride, GLint vertexBufferCount); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMultiDrawArraysIndirectBindlessCountNV (GLenum mode, const void *indirect, GLsizei drawCount, GLsizei maxDrawCount, GLsizei stride, GLint vertexBufferCount); +GLAPI void APIENTRY glMultiDrawElementsIndirectBindlessCountNV (GLenum mode, GLenum type, const void *indirect, GLsizei drawCount, GLsizei maxDrawCount, GLsizei stride, GLint vertexBufferCount); +#endif +#endif /* GL_NV_bindless_multi_draw_indirect_count */ + +#ifndef GL_NV_bindless_texture +#define GL_NV_bindless_texture 1 +typedef GLuint64 (APIENTRYP PFNGLGETTEXTUREHANDLENVPROC) (GLuint texture); +typedef GLuint64 (APIENTRYP PFNGLGETTEXTURESAMPLERHANDLENVPROC) (GLuint texture, GLuint sampler); +typedef void (APIENTRYP PFNGLMAKETEXTUREHANDLERESIDENTNVPROC) (GLuint64 handle); +typedef void (APIENTRYP PFNGLMAKETEXTUREHANDLENONRESIDENTNVPROC) (GLuint64 handle); +typedef GLuint64 (APIENTRYP PFNGLGETIMAGEHANDLENVPROC) (GLuint texture, GLint level, GLboolean layered, GLint layer, GLenum format); +typedef void (APIENTRYP PFNGLMAKEIMAGEHANDLERESIDENTNVPROC) (GLuint64 handle, GLenum access); +typedef void (APIENTRYP PFNGLMAKEIMAGEHANDLENONRESIDENTNVPROC) (GLuint64 handle); +typedef void (APIENTRYP PFNGLUNIFORMHANDLEUI64NVPROC) (GLint location, GLuint64 value); +typedef void (APIENTRYP PFNGLUNIFORMHANDLEUI64VNVPROC) (GLint location, GLsizei count, const GLuint64 *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMHANDLEUI64NVPROC) (GLuint program, GLint location, GLuint64 value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMHANDLEUI64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLuint64 *values); +typedef GLboolean (APIENTRYP PFNGLISTEXTUREHANDLERESIDENTNVPROC) (GLuint64 handle); +typedef GLboolean (APIENTRYP PFNGLISIMAGEHANDLERESIDENTNVPROC) (GLuint64 handle); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLuint64 APIENTRY glGetTextureHandleNV (GLuint texture); +GLAPI GLuint64 APIENTRY glGetTextureSamplerHandleNV (GLuint texture, GLuint sampler); +GLAPI void APIENTRY glMakeTextureHandleResidentNV (GLuint64 handle); +GLAPI void APIENTRY glMakeTextureHandleNonResidentNV (GLuint64 handle); +GLAPI GLuint64 APIENTRY glGetImageHandleNV (GLuint texture, GLint level, GLboolean layered, GLint layer, GLenum format); +GLAPI void APIENTRY glMakeImageHandleResidentNV (GLuint64 handle, GLenum access); +GLAPI void APIENTRY glMakeImageHandleNonResidentNV (GLuint64 handle); +GLAPI void APIENTRY glUniformHandleui64NV (GLint location, GLuint64 value); +GLAPI void APIENTRY glUniformHandleui64vNV (GLint location, GLsizei count, const GLuint64 *value); +GLAPI void APIENTRY glProgramUniformHandleui64NV (GLuint program, GLint location, GLuint64 value); +GLAPI void APIENTRY glProgramUniformHandleui64vNV (GLuint program, GLint location, GLsizei count, const GLuint64 *values); +GLAPI GLboolean APIENTRY glIsTextureHandleResidentNV (GLuint64 handle); +GLAPI GLboolean APIENTRY glIsImageHandleResidentNV (GLuint64 handle); +#endif +#endif /* GL_NV_bindless_texture */ + +#ifndef GL_NV_blend_equation_advanced +#define GL_NV_blend_equation_advanced 1 +#define GL_BLEND_OVERLAP_NV 0x9281 +#define GL_BLEND_PREMULTIPLIED_SRC_NV 0x9280 +#define GL_BLUE_NV 0x1905 +#define GL_COLORBURN_NV 0x929A +#define GL_COLORDODGE_NV 0x9299 +#define GL_CONJOINT_NV 0x9284 +#define GL_CONTRAST_NV 0x92A1 +#define GL_DARKEN_NV 0x9297 +#define GL_DIFFERENCE_NV 0x929E +#define GL_DISJOINT_NV 0x9283 +#define GL_DST_ATOP_NV 0x928F +#define GL_DST_IN_NV 0x928B +#define GL_DST_NV 0x9287 +#define GL_DST_OUT_NV 0x928D +#define GL_DST_OVER_NV 0x9289 +#define GL_EXCLUSION_NV 0x92A0 +#define GL_GREEN_NV 0x1904 +#define GL_HARDLIGHT_NV 0x929B +#define GL_HARDMIX_NV 0x92A9 +#define GL_HSL_COLOR_NV 0x92AF +#define GL_HSL_HUE_NV 0x92AD +#define GL_HSL_LUMINOSITY_NV 0x92B0 +#define GL_HSL_SATURATION_NV 0x92AE +#define GL_INVERT_OVG_NV 0x92B4 +#define GL_INVERT_RGB_NV 0x92A3 +#define GL_LIGHTEN_NV 0x9298 +#define GL_LINEARBURN_NV 0x92A5 +#define GL_LINEARDODGE_NV 0x92A4 +#define GL_LINEARLIGHT_NV 0x92A7 +#define GL_MINUS_CLAMPED_NV 0x92B3 +#define GL_MINUS_NV 0x929F +#define GL_MULTIPLY_NV 0x9294 +#define GL_OVERLAY_NV 0x9296 +#define GL_PINLIGHT_NV 0x92A8 +#define GL_PLUS_CLAMPED_ALPHA_NV 0x92B2 +#define GL_PLUS_CLAMPED_NV 0x92B1 +#define GL_PLUS_DARKER_NV 0x9292 +#define GL_PLUS_NV 0x9291 +#define GL_RED_NV 0x1903 +#define GL_SCREEN_NV 0x9295 +#define GL_SOFTLIGHT_NV 0x929C +#define GL_SRC_ATOP_NV 0x928E +#define GL_SRC_IN_NV 0x928A +#define GL_SRC_NV 0x9286 +#define GL_SRC_OUT_NV 0x928C +#define GL_SRC_OVER_NV 0x9288 +#define GL_UNCORRELATED_NV 0x9282 +#define GL_VIVIDLIGHT_NV 0x92A6 +#define GL_XOR_NV 0x1506 +typedef void (APIENTRYP PFNGLBLENDPARAMETERINVPROC) (GLenum pname, GLint value); +typedef void (APIENTRYP PFNGLBLENDBARRIERNVPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendParameteriNV (GLenum pname, GLint value); +GLAPI void APIENTRY glBlendBarrierNV (void); +#endif +#endif /* GL_NV_blend_equation_advanced */ + +#ifndef GL_NV_blend_equation_advanced_coherent +#define GL_NV_blend_equation_advanced_coherent 1 +#define GL_BLEND_ADVANCED_COHERENT_NV 0x9285 +#endif /* GL_NV_blend_equation_advanced_coherent */ + +#ifndef GL_NV_blend_minmax_factor +#define GL_NV_blend_minmax_factor 1 +#endif /* GL_NV_blend_minmax_factor */ + +#ifndef GL_NV_blend_square +#define GL_NV_blend_square 1 +#endif /* GL_NV_blend_square */ + +#ifndef GL_NV_clip_space_w_scaling +#define GL_NV_clip_space_w_scaling 1 +#define GL_VIEWPORT_POSITION_W_SCALE_NV 0x937C +#define GL_VIEWPORT_POSITION_W_SCALE_X_COEFF_NV 0x937D +#define GL_VIEWPORT_POSITION_W_SCALE_Y_COEFF_NV 0x937E +typedef void (APIENTRYP PFNGLVIEWPORTPOSITIONWSCALENVPROC) (GLuint index, GLfloat xcoeff, GLfloat ycoeff); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glViewportPositionWScaleNV (GLuint index, GLfloat xcoeff, GLfloat ycoeff); +#endif +#endif /* GL_NV_clip_space_w_scaling */ + +#ifndef GL_NV_command_list +#define GL_NV_command_list 1 +#define GL_TERMINATE_SEQUENCE_COMMAND_NV 0x0000 +#define GL_NOP_COMMAND_NV 0x0001 +#define GL_DRAW_ELEMENTS_COMMAND_NV 0x0002 +#define GL_DRAW_ARRAYS_COMMAND_NV 0x0003 +#define GL_DRAW_ELEMENTS_STRIP_COMMAND_NV 0x0004 +#define GL_DRAW_ARRAYS_STRIP_COMMAND_NV 0x0005 +#define GL_DRAW_ELEMENTS_INSTANCED_COMMAND_NV 0x0006 +#define GL_DRAW_ARRAYS_INSTANCED_COMMAND_NV 0x0007 +#define GL_ELEMENT_ADDRESS_COMMAND_NV 0x0008 +#define GL_ATTRIBUTE_ADDRESS_COMMAND_NV 0x0009 +#define GL_UNIFORM_ADDRESS_COMMAND_NV 0x000A +#define GL_BLEND_COLOR_COMMAND_NV 0x000B +#define GL_STENCIL_REF_COMMAND_NV 0x000C +#define GL_LINE_WIDTH_COMMAND_NV 0x000D +#define GL_POLYGON_OFFSET_COMMAND_NV 0x000E +#define GL_ALPHA_REF_COMMAND_NV 0x000F +#define GL_VIEWPORT_COMMAND_NV 0x0010 +#define GL_SCISSOR_COMMAND_NV 0x0011 +#define GL_FRONT_FACE_COMMAND_NV 0x0012 +typedef void (APIENTRYP PFNGLCREATESTATESNVPROC) (GLsizei n, GLuint *states); +typedef void (APIENTRYP PFNGLDELETESTATESNVPROC) (GLsizei n, const GLuint *states); +typedef GLboolean (APIENTRYP PFNGLISSTATENVPROC) (GLuint state); +typedef void (APIENTRYP PFNGLSTATECAPTURENVPROC) (GLuint state, GLenum mode); +typedef GLuint (APIENTRYP PFNGLGETCOMMANDHEADERNVPROC) (GLenum tokenID, GLuint size); +typedef GLushort (APIENTRYP PFNGLGETSTAGEINDEXNVPROC) (GLenum shadertype); +typedef void (APIENTRYP PFNGLDRAWCOMMANDSNVPROC) (GLenum primitiveMode, GLuint buffer, const GLintptr *indirects, const GLsizei *sizes, GLuint count); +typedef void (APIENTRYP PFNGLDRAWCOMMANDSADDRESSNVPROC) (GLenum primitiveMode, const GLuint64 *indirects, const GLsizei *sizes, GLuint count); +typedef void (APIENTRYP PFNGLDRAWCOMMANDSSTATESNVPROC) (GLuint buffer, const GLintptr *indirects, const GLsizei *sizes, const GLuint *states, const GLuint *fbos, GLuint count); +typedef void (APIENTRYP PFNGLDRAWCOMMANDSSTATESADDRESSNVPROC) (const GLuint64 *indirects, const GLsizei *sizes, const GLuint *states, const GLuint *fbos, GLuint count); +typedef void (APIENTRYP PFNGLCREATECOMMANDLISTSNVPROC) (GLsizei n, GLuint *lists); +typedef void (APIENTRYP PFNGLDELETECOMMANDLISTSNVPROC) (GLsizei n, const GLuint *lists); +typedef GLboolean (APIENTRYP PFNGLISCOMMANDLISTNVPROC) (GLuint list); +typedef void (APIENTRYP PFNGLLISTDRAWCOMMANDSSTATESCLIENTNVPROC) (GLuint list, GLuint segment, const void **indirects, const GLsizei *sizes, const GLuint *states, const GLuint *fbos, GLuint count); +typedef void (APIENTRYP PFNGLCOMMANDLISTSEGMENTSNVPROC) (GLuint list, GLuint segments); +typedef void (APIENTRYP PFNGLCOMPILECOMMANDLISTNVPROC) (GLuint list); +typedef void (APIENTRYP PFNGLCALLCOMMANDLISTNVPROC) (GLuint list); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCreateStatesNV (GLsizei n, GLuint *states); +GLAPI void APIENTRY glDeleteStatesNV (GLsizei n, const GLuint *states); +GLAPI GLboolean APIENTRY glIsStateNV (GLuint state); +GLAPI void APIENTRY glStateCaptureNV (GLuint state, GLenum mode); +GLAPI GLuint APIENTRY glGetCommandHeaderNV (GLenum tokenID, GLuint size); +GLAPI GLushort APIENTRY glGetStageIndexNV (GLenum shadertype); +GLAPI void APIENTRY glDrawCommandsNV (GLenum primitiveMode, GLuint buffer, const GLintptr *indirects, const GLsizei *sizes, GLuint count); +GLAPI void APIENTRY glDrawCommandsAddressNV (GLenum primitiveMode, const GLuint64 *indirects, const GLsizei *sizes, GLuint count); +GLAPI void APIENTRY glDrawCommandsStatesNV (GLuint buffer, const GLintptr *indirects, const GLsizei *sizes, const GLuint *states, const GLuint *fbos, GLuint count); +GLAPI void APIENTRY glDrawCommandsStatesAddressNV (const GLuint64 *indirects, const GLsizei *sizes, const GLuint *states, const GLuint *fbos, GLuint count); +GLAPI void APIENTRY glCreateCommandListsNV (GLsizei n, GLuint *lists); +GLAPI void APIENTRY glDeleteCommandListsNV (GLsizei n, const GLuint *lists); +GLAPI GLboolean APIENTRY glIsCommandListNV (GLuint list); +GLAPI void APIENTRY glListDrawCommandsStatesClientNV (GLuint list, GLuint segment, const void **indirects, const GLsizei *sizes, const GLuint *states, const GLuint *fbos, GLuint count); +GLAPI void APIENTRY glCommandListSegmentsNV (GLuint list, GLuint segments); +GLAPI void APIENTRY glCompileCommandListNV (GLuint list); +GLAPI void APIENTRY glCallCommandListNV (GLuint list); +#endif +#endif /* GL_NV_command_list */ + +#ifndef GL_NV_compute_program5 +#define GL_NV_compute_program5 1 +#define GL_COMPUTE_PROGRAM_NV 0x90FB +#define GL_COMPUTE_PROGRAM_PARAMETER_BUFFER_NV 0x90FC +#endif /* GL_NV_compute_program5 */ + +#ifndef GL_NV_compute_shader_derivatives +#define GL_NV_compute_shader_derivatives 1 +#endif /* GL_NV_compute_shader_derivatives */ + +#ifndef GL_NV_conditional_render +#define GL_NV_conditional_render 1 +#define GL_QUERY_WAIT_NV 0x8E13 +#define GL_QUERY_NO_WAIT_NV 0x8E14 +#define GL_QUERY_BY_REGION_WAIT_NV 0x8E15 +#define GL_QUERY_BY_REGION_NO_WAIT_NV 0x8E16 +typedef void (APIENTRYP PFNGLBEGINCONDITIONALRENDERNVPROC) (GLuint id, GLenum mode); +typedef void (APIENTRYP PFNGLENDCONDITIONALRENDERNVPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBeginConditionalRenderNV (GLuint id, GLenum mode); +GLAPI void APIENTRY glEndConditionalRenderNV (void); +#endif +#endif /* GL_NV_conditional_render */ + +#ifndef GL_NV_conservative_raster +#define GL_NV_conservative_raster 1 +#define GL_CONSERVATIVE_RASTERIZATION_NV 0x9346 +#define GL_SUBPIXEL_PRECISION_BIAS_X_BITS_NV 0x9347 +#define GL_SUBPIXEL_PRECISION_BIAS_Y_BITS_NV 0x9348 +#define GL_MAX_SUBPIXEL_PRECISION_BIAS_BITS_NV 0x9349 +typedef void (APIENTRYP PFNGLSUBPIXELPRECISIONBIASNVPROC) (GLuint xbits, GLuint ybits); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSubpixelPrecisionBiasNV (GLuint xbits, GLuint ybits); +#endif +#endif /* GL_NV_conservative_raster */ + +#ifndef GL_NV_conservative_raster_dilate +#define GL_NV_conservative_raster_dilate 1 +#define GL_CONSERVATIVE_RASTER_DILATE_NV 0x9379 +#define GL_CONSERVATIVE_RASTER_DILATE_RANGE_NV 0x937A +#define GL_CONSERVATIVE_RASTER_DILATE_GRANULARITY_NV 0x937B +typedef void (APIENTRYP PFNGLCONSERVATIVERASTERPARAMETERFNVPROC) (GLenum pname, GLfloat value); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glConservativeRasterParameterfNV (GLenum pname, GLfloat value); +#endif +#endif /* GL_NV_conservative_raster_dilate */ + +#ifndef GL_NV_conservative_raster_pre_snap +#define GL_NV_conservative_raster_pre_snap 1 +#define GL_CONSERVATIVE_RASTER_MODE_PRE_SNAP_NV 0x9550 +#endif /* GL_NV_conservative_raster_pre_snap */ + +#ifndef GL_NV_conservative_raster_pre_snap_triangles +#define GL_NV_conservative_raster_pre_snap_triangles 1 +#define GL_CONSERVATIVE_RASTER_MODE_NV 0x954D +#define GL_CONSERVATIVE_RASTER_MODE_POST_SNAP_NV 0x954E +#define GL_CONSERVATIVE_RASTER_MODE_PRE_SNAP_TRIANGLES_NV 0x954F +typedef void (APIENTRYP PFNGLCONSERVATIVERASTERPARAMETERINVPROC) (GLenum pname, GLint param); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glConservativeRasterParameteriNV (GLenum pname, GLint param); +#endif +#endif /* GL_NV_conservative_raster_pre_snap_triangles */ + +#ifndef GL_NV_conservative_raster_underestimation +#define GL_NV_conservative_raster_underestimation 1 +#endif /* GL_NV_conservative_raster_underestimation */ + +#ifndef GL_NV_copy_depth_to_color +#define GL_NV_copy_depth_to_color 1 +#define GL_DEPTH_STENCIL_TO_RGBA_NV 0x886E +#define GL_DEPTH_STENCIL_TO_BGRA_NV 0x886F +#endif /* GL_NV_copy_depth_to_color */ + +#ifndef GL_NV_copy_image +#define GL_NV_copy_image 1 +typedef void (APIENTRYP PFNGLCOPYIMAGESUBDATANVPROC) (GLuint srcName, GLenum srcTarget, GLint srcLevel, GLint srcX, GLint srcY, GLint srcZ, GLuint dstName, GLenum dstTarget, GLint dstLevel, GLint dstX, GLint dstY, GLint dstZ, GLsizei width, GLsizei height, GLsizei depth); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCopyImageSubDataNV (GLuint srcName, GLenum srcTarget, GLint srcLevel, GLint srcX, GLint srcY, GLint srcZ, GLuint dstName, GLenum dstTarget, GLint dstLevel, GLint dstX, GLint dstY, GLint dstZ, GLsizei width, GLsizei height, GLsizei depth); +#endif +#endif /* GL_NV_copy_image */ + +#ifndef GL_NV_deep_texture3D +#define GL_NV_deep_texture3D 1 +#define GL_MAX_DEEP_3D_TEXTURE_WIDTH_HEIGHT_NV 0x90D0 +#define GL_MAX_DEEP_3D_TEXTURE_DEPTH_NV 0x90D1 +#endif /* GL_NV_deep_texture3D */ + +#ifndef GL_NV_depth_buffer_float +#define GL_NV_depth_buffer_float 1 +#define GL_DEPTH_COMPONENT32F_NV 0x8DAB +#define GL_DEPTH32F_STENCIL8_NV 0x8DAC +#define GL_FLOAT_32_UNSIGNED_INT_24_8_REV_NV 0x8DAD +#define GL_DEPTH_BUFFER_FLOAT_MODE_NV 0x8DAF +typedef void (APIENTRYP PFNGLDEPTHRANGEDNVPROC) (GLdouble zNear, GLdouble zFar); +typedef void (APIENTRYP PFNGLCLEARDEPTHDNVPROC) (GLdouble depth); +typedef void (APIENTRYP PFNGLDEPTHBOUNDSDNVPROC) (GLdouble zmin, GLdouble zmax); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDepthRangedNV (GLdouble zNear, GLdouble zFar); +GLAPI void APIENTRY glClearDepthdNV (GLdouble depth); +GLAPI void APIENTRY glDepthBoundsdNV (GLdouble zmin, GLdouble zmax); +#endif +#endif /* GL_NV_depth_buffer_float */ + +#ifndef GL_NV_depth_clamp +#define GL_NV_depth_clamp 1 +#define GL_DEPTH_CLAMP_NV 0x864F +#endif /* GL_NV_depth_clamp */ + +#ifndef GL_NV_draw_texture +#define GL_NV_draw_texture 1 +typedef void (APIENTRYP PFNGLDRAWTEXTURENVPROC) (GLuint texture, GLuint sampler, GLfloat x0, GLfloat y0, GLfloat x1, GLfloat y1, GLfloat z, GLfloat s0, GLfloat t0, GLfloat s1, GLfloat t1); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawTextureNV (GLuint texture, GLuint sampler, GLfloat x0, GLfloat y0, GLfloat x1, GLfloat y1, GLfloat z, GLfloat s0, GLfloat t0, GLfloat s1, GLfloat t1); +#endif +#endif /* GL_NV_draw_texture */ + +#ifndef GL_NV_draw_vulkan_image +#define GL_NV_draw_vulkan_image 1 +typedef void (APIENTRY *GLVULKANPROCNV)(void); +typedef void (APIENTRYP PFNGLDRAWVKIMAGENVPROC) (GLuint64 vkImage, GLuint sampler, GLfloat x0, GLfloat y0, GLfloat x1, GLfloat y1, GLfloat z, GLfloat s0, GLfloat t0, GLfloat s1, GLfloat t1); +typedef GLVULKANPROCNV (APIENTRYP PFNGLGETVKPROCADDRNVPROC) (const GLchar *name); +typedef void (APIENTRYP PFNGLWAITVKSEMAPHORENVPROC) (GLuint64 vkSemaphore); +typedef void (APIENTRYP PFNGLSIGNALVKSEMAPHORENVPROC) (GLuint64 vkSemaphore); +typedef void (APIENTRYP PFNGLSIGNALVKFENCENVPROC) (GLuint64 vkFence); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawVkImageNV (GLuint64 vkImage, GLuint sampler, GLfloat x0, GLfloat y0, GLfloat x1, GLfloat y1, GLfloat z, GLfloat s0, GLfloat t0, GLfloat s1, GLfloat t1); +GLAPI GLVULKANPROCNV APIENTRY glGetVkProcAddrNV (const GLchar *name); +GLAPI void APIENTRY glWaitVkSemaphoreNV (GLuint64 vkSemaphore); +GLAPI void APIENTRY glSignalVkSemaphoreNV (GLuint64 vkSemaphore); +GLAPI void APIENTRY glSignalVkFenceNV (GLuint64 vkFence); +#endif +#endif /* GL_NV_draw_vulkan_image */ + +#ifndef GL_NV_evaluators +#define GL_NV_evaluators 1 +#define GL_EVAL_2D_NV 0x86C0 +#define GL_EVAL_TRIANGULAR_2D_NV 0x86C1 +#define GL_MAP_TESSELLATION_NV 0x86C2 +#define GL_MAP_ATTRIB_U_ORDER_NV 0x86C3 +#define GL_MAP_ATTRIB_V_ORDER_NV 0x86C4 +#define GL_EVAL_FRACTIONAL_TESSELLATION_NV 0x86C5 +#define GL_EVAL_VERTEX_ATTRIB0_NV 0x86C6 +#define GL_EVAL_VERTEX_ATTRIB1_NV 0x86C7 +#define GL_EVAL_VERTEX_ATTRIB2_NV 0x86C8 +#define GL_EVAL_VERTEX_ATTRIB3_NV 0x86C9 +#define GL_EVAL_VERTEX_ATTRIB4_NV 0x86CA +#define GL_EVAL_VERTEX_ATTRIB5_NV 0x86CB +#define GL_EVAL_VERTEX_ATTRIB6_NV 0x86CC +#define GL_EVAL_VERTEX_ATTRIB7_NV 0x86CD +#define GL_EVAL_VERTEX_ATTRIB8_NV 0x86CE +#define GL_EVAL_VERTEX_ATTRIB9_NV 0x86CF +#define GL_EVAL_VERTEX_ATTRIB10_NV 0x86D0 +#define GL_EVAL_VERTEX_ATTRIB11_NV 0x86D1 +#define GL_EVAL_VERTEX_ATTRIB12_NV 0x86D2 +#define GL_EVAL_VERTEX_ATTRIB13_NV 0x86D3 +#define GL_EVAL_VERTEX_ATTRIB14_NV 0x86D4 +#define GL_EVAL_VERTEX_ATTRIB15_NV 0x86D5 +#define GL_MAX_MAP_TESSELLATION_NV 0x86D6 +#define GL_MAX_RATIONAL_EVAL_ORDER_NV 0x86D7 +typedef void (APIENTRYP PFNGLMAPCONTROLPOINTSNVPROC) (GLenum target, GLuint index, GLenum type, GLsizei ustride, GLsizei vstride, GLint uorder, GLint vorder, GLboolean packed, const void *points); +typedef void (APIENTRYP PFNGLMAPPARAMETERIVNVPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLMAPPARAMETERFVNVPROC) (GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLGETMAPCONTROLPOINTSNVPROC) (GLenum target, GLuint index, GLenum type, GLsizei ustride, GLsizei vstride, GLboolean packed, void *points); +typedef void (APIENTRYP PFNGLGETMAPPARAMETERIVNVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETMAPPARAMETERFVNVPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETMAPATTRIBPARAMETERIVNVPROC) (GLenum target, GLuint index, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETMAPATTRIBPARAMETERFVNVPROC) (GLenum target, GLuint index, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLEVALMAPSNVPROC) (GLenum target, GLenum mode); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMapControlPointsNV (GLenum target, GLuint index, GLenum type, GLsizei ustride, GLsizei vstride, GLint uorder, GLint vorder, GLboolean packed, const void *points); +GLAPI void APIENTRY glMapParameterivNV (GLenum target, GLenum pname, const GLint *params); +GLAPI void APIENTRY glMapParameterfvNV (GLenum target, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glGetMapControlPointsNV (GLenum target, GLuint index, GLenum type, GLsizei ustride, GLsizei vstride, GLboolean packed, void *points); +GLAPI void APIENTRY glGetMapParameterivNV (GLenum target, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetMapParameterfvNV (GLenum target, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetMapAttribParameterivNV (GLenum target, GLuint index, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetMapAttribParameterfvNV (GLenum target, GLuint index, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glEvalMapsNV (GLenum target, GLenum mode); +#endif +#endif /* GL_NV_evaluators */ + +#ifndef GL_NV_explicit_multisample +#define GL_NV_explicit_multisample 1 +#define GL_SAMPLE_POSITION_NV 0x8E50 +#define GL_SAMPLE_MASK_NV 0x8E51 +#define GL_SAMPLE_MASK_VALUE_NV 0x8E52 +#define GL_TEXTURE_BINDING_RENDERBUFFER_NV 0x8E53 +#define GL_TEXTURE_RENDERBUFFER_DATA_STORE_BINDING_NV 0x8E54 +#define GL_TEXTURE_RENDERBUFFER_NV 0x8E55 +#define GL_SAMPLER_RENDERBUFFER_NV 0x8E56 +#define GL_INT_SAMPLER_RENDERBUFFER_NV 0x8E57 +#define GL_UNSIGNED_INT_SAMPLER_RENDERBUFFER_NV 0x8E58 +#define GL_MAX_SAMPLE_MASK_WORDS_NV 0x8E59 +typedef void (APIENTRYP PFNGLGETMULTISAMPLEFVNVPROC) (GLenum pname, GLuint index, GLfloat *val); +typedef void (APIENTRYP PFNGLSAMPLEMASKINDEXEDNVPROC) (GLuint index, GLbitfield mask); +typedef void (APIENTRYP PFNGLTEXRENDERBUFFERNVPROC) (GLenum target, GLuint renderbuffer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGetMultisamplefvNV (GLenum pname, GLuint index, GLfloat *val); +GLAPI void APIENTRY glSampleMaskIndexedNV (GLuint index, GLbitfield mask); +GLAPI void APIENTRY glTexRenderbufferNV (GLenum target, GLuint renderbuffer); +#endif +#endif /* GL_NV_explicit_multisample */ + +#ifndef GL_NV_fence +#define GL_NV_fence 1 +#define GL_ALL_COMPLETED_NV 0x84F2 +#define GL_FENCE_STATUS_NV 0x84F3 +#define GL_FENCE_CONDITION_NV 0x84F4 +typedef void (APIENTRYP PFNGLDELETEFENCESNVPROC) (GLsizei n, const GLuint *fences); +typedef void (APIENTRYP PFNGLGENFENCESNVPROC) (GLsizei n, GLuint *fences); +typedef GLboolean (APIENTRYP PFNGLISFENCENVPROC) (GLuint fence); +typedef GLboolean (APIENTRYP PFNGLTESTFENCENVPROC) (GLuint fence); +typedef void (APIENTRYP PFNGLGETFENCEIVNVPROC) (GLuint fence, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLFINISHFENCENVPROC) (GLuint fence); +typedef void (APIENTRYP PFNGLSETFENCENVPROC) (GLuint fence, GLenum condition); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDeleteFencesNV (GLsizei n, const GLuint *fences); +GLAPI void APIENTRY glGenFencesNV (GLsizei n, GLuint *fences); +GLAPI GLboolean APIENTRY glIsFenceNV (GLuint fence); +GLAPI GLboolean APIENTRY glTestFenceNV (GLuint fence); +GLAPI void APIENTRY glGetFenceivNV (GLuint fence, GLenum pname, GLint *params); +GLAPI void APIENTRY glFinishFenceNV (GLuint fence); +GLAPI void APIENTRY glSetFenceNV (GLuint fence, GLenum condition); +#endif +#endif /* GL_NV_fence */ + +#ifndef GL_NV_fill_rectangle +#define GL_NV_fill_rectangle 1 +#define GL_FILL_RECTANGLE_NV 0x933C +#endif /* GL_NV_fill_rectangle */ + +#ifndef GL_NV_float_buffer +#define GL_NV_float_buffer 1 +#define GL_FLOAT_R_NV 0x8880 +#define GL_FLOAT_RG_NV 0x8881 +#define GL_FLOAT_RGB_NV 0x8882 +#define GL_FLOAT_RGBA_NV 0x8883 +#define GL_FLOAT_R16_NV 0x8884 +#define GL_FLOAT_R32_NV 0x8885 +#define GL_FLOAT_RG16_NV 0x8886 +#define GL_FLOAT_RG32_NV 0x8887 +#define GL_FLOAT_RGB16_NV 0x8888 +#define GL_FLOAT_RGB32_NV 0x8889 +#define GL_FLOAT_RGBA16_NV 0x888A +#define GL_FLOAT_RGBA32_NV 0x888B +#define GL_TEXTURE_FLOAT_COMPONENTS_NV 0x888C +#define GL_FLOAT_CLEAR_COLOR_VALUE_NV 0x888D +#define GL_FLOAT_RGBA_MODE_NV 0x888E +#endif /* GL_NV_float_buffer */ + +#ifndef GL_NV_fog_distance +#define GL_NV_fog_distance 1 +#define GL_FOG_DISTANCE_MODE_NV 0x855A +#define GL_EYE_RADIAL_NV 0x855B +#define GL_EYE_PLANE_ABSOLUTE_NV 0x855C +#endif /* GL_NV_fog_distance */ + +#ifndef GL_NV_fragment_coverage_to_color +#define GL_NV_fragment_coverage_to_color 1 +#define GL_FRAGMENT_COVERAGE_TO_COLOR_NV 0x92DD +#define GL_FRAGMENT_COVERAGE_COLOR_NV 0x92DE +typedef void (APIENTRYP PFNGLFRAGMENTCOVERAGECOLORNVPROC) (GLuint color); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFragmentCoverageColorNV (GLuint color); +#endif +#endif /* GL_NV_fragment_coverage_to_color */ + +#ifndef GL_NV_fragment_program +#define GL_NV_fragment_program 1 +#define GL_MAX_FRAGMENT_PROGRAM_LOCAL_PARAMETERS_NV 0x8868 +#define GL_FRAGMENT_PROGRAM_NV 0x8870 +#define GL_MAX_TEXTURE_COORDS_NV 0x8871 +#define GL_MAX_TEXTURE_IMAGE_UNITS_NV 0x8872 +#define GL_FRAGMENT_PROGRAM_BINDING_NV 0x8873 +#define GL_PROGRAM_ERROR_STRING_NV 0x8874 +typedef void (APIENTRYP PFNGLPROGRAMNAMEDPARAMETER4FNVPROC) (GLuint id, GLsizei len, const GLubyte *name, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLPROGRAMNAMEDPARAMETER4FVNVPROC) (GLuint id, GLsizei len, const GLubyte *name, const GLfloat *v); +typedef void (APIENTRYP PFNGLPROGRAMNAMEDPARAMETER4DNVPROC) (GLuint id, GLsizei len, const GLubyte *name, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLPROGRAMNAMEDPARAMETER4DVNVPROC) (GLuint id, GLsizei len, const GLubyte *name, const GLdouble *v); +typedef void (APIENTRYP PFNGLGETPROGRAMNAMEDPARAMETERFVNVPROC) (GLuint id, GLsizei len, const GLubyte *name, GLfloat *params); +typedef void (APIENTRYP PFNGLGETPROGRAMNAMEDPARAMETERDVNVPROC) (GLuint id, GLsizei len, const GLubyte *name, GLdouble *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glProgramNamedParameter4fNV (GLuint id, GLsizei len, const GLubyte *name, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +GLAPI void APIENTRY glProgramNamedParameter4fvNV (GLuint id, GLsizei len, const GLubyte *name, const GLfloat *v); +GLAPI void APIENTRY glProgramNamedParameter4dNV (GLuint id, GLsizei len, const GLubyte *name, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +GLAPI void APIENTRY glProgramNamedParameter4dvNV (GLuint id, GLsizei len, const GLubyte *name, const GLdouble *v); +GLAPI void APIENTRY glGetProgramNamedParameterfvNV (GLuint id, GLsizei len, const GLubyte *name, GLfloat *params); +GLAPI void APIENTRY glGetProgramNamedParameterdvNV (GLuint id, GLsizei len, const GLubyte *name, GLdouble *params); +#endif +#endif /* GL_NV_fragment_program */ + +#ifndef GL_NV_fragment_program2 +#define GL_NV_fragment_program2 1 +#define GL_MAX_PROGRAM_EXEC_INSTRUCTIONS_NV 0x88F4 +#define GL_MAX_PROGRAM_CALL_DEPTH_NV 0x88F5 +#define GL_MAX_PROGRAM_IF_DEPTH_NV 0x88F6 +#define GL_MAX_PROGRAM_LOOP_DEPTH_NV 0x88F7 +#define GL_MAX_PROGRAM_LOOP_COUNT_NV 0x88F8 +#endif /* GL_NV_fragment_program2 */ + +#ifndef GL_NV_fragment_program4 +#define GL_NV_fragment_program4 1 +#endif /* GL_NV_fragment_program4 */ + +#ifndef GL_NV_fragment_program_option +#define GL_NV_fragment_program_option 1 +#endif /* GL_NV_fragment_program_option */ + +#ifndef GL_NV_fragment_shader_barycentric +#define GL_NV_fragment_shader_barycentric 1 +#endif /* GL_NV_fragment_shader_barycentric */ + +#ifndef GL_NV_fragment_shader_interlock +#define GL_NV_fragment_shader_interlock 1 +#endif /* GL_NV_fragment_shader_interlock */ + +#ifndef GL_NV_framebuffer_mixed_samples +#define GL_NV_framebuffer_mixed_samples 1 +#define GL_COVERAGE_MODULATION_TABLE_NV 0x9331 +#define GL_COLOR_SAMPLES_NV 0x8E20 +#define GL_DEPTH_SAMPLES_NV 0x932D +#define GL_STENCIL_SAMPLES_NV 0x932E +#define GL_MIXED_DEPTH_SAMPLES_SUPPORTED_NV 0x932F +#define GL_MIXED_STENCIL_SAMPLES_SUPPORTED_NV 0x9330 +#define GL_COVERAGE_MODULATION_NV 0x9332 +#define GL_COVERAGE_MODULATION_TABLE_SIZE_NV 0x9333 +typedef void (APIENTRYP PFNGLCOVERAGEMODULATIONTABLENVPROC) (GLsizei n, const GLfloat *v); +typedef void (APIENTRYP PFNGLGETCOVERAGEMODULATIONTABLENVPROC) (GLsizei bufSize, GLfloat *v); +typedef void (APIENTRYP PFNGLCOVERAGEMODULATIONNVPROC) (GLenum components); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCoverageModulationTableNV (GLsizei n, const GLfloat *v); +GLAPI void APIENTRY glGetCoverageModulationTableNV (GLsizei bufSize, GLfloat *v); +GLAPI void APIENTRY glCoverageModulationNV (GLenum components); +#endif +#endif /* GL_NV_framebuffer_mixed_samples */ + +#ifndef GL_NV_framebuffer_multisample_coverage +#define GL_NV_framebuffer_multisample_coverage 1 +#define GL_RENDERBUFFER_COVERAGE_SAMPLES_NV 0x8CAB +#define GL_RENDERBUFFER_COLOR_SAMPLES_NV 0x8E10 +#define GL_MAX_MULTISAMPLE_COVERAGE_MODES_NV 0x8E11 +#define GL_MULTISAMPLE_COVERAGE_MODES_NV 0x8E12 +typedef void (APIENTRYP PFNGLRENDERBUFFERSTORAGEMULTISAMPLECOVERAGENVPROC) (GLenum target, GLsizei coverageSamples, GLsizei colorSamples, GLenum internalformat, GLsizei width, GLsizei height); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glRenderbufferStorageMultisampleCoverageNV (GLenum target, GLsizei coverageSamples, GLsizei colorSamples, GLenum internalformat, GLsizei width, GLsizei height); +#endif +#endif /* GL_NV_framebuffer_multisample_coverage */ + +#ifndef GL_NV_geometry_program4 +#define GL_NV_geometry_program4 1 +#define GL_GEOMETRY_PROGRAM_NV 0x8C26 +#define GL_MAX_PROGRAM_OUTPUT_VERTICES_NV 0x8C27 +#define GL_MAX_PROGRAM_TOTAL_OUTPUT_COMPONENTS_NV 0x8C28 +typedef void (APIENTRYP PFNGLPROGRAMVERTEXLIMITNVPROC) (GLenum target, GLint limit); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTUREEXTPROC) (GLenum target, GLenum attachment, GLuint texture, GLint level); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTUREFACEEXTPROC) (GLenum target, GLenum attachment, GLuint texture, GLint level, GLenum face); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glProgramVertexLimitNV (GLenum target, GLint limit); +GLAPI void APIENTRY glFramebufferTextureEXT (GLenum target, GLenum attachment, GLuint texture, GLint level); +GLAPI void APIENTRY glFramebufferTextureFaceEXT (GLenum target, GLenum attachment, GLuint texture, GLint level, GLenum face); +#endif +#endif /* GL_NV_geometry_program4 */ + +#ifndef GL_NV_geometry_shader4 +#define GL_NV_geometry_shader4 1 +#endif /* GL_NV_geometry_shader4 */ + +#ifndef GL_NV_geometry_shader_passthrough +#define GL_NV_geometry_shader_passthrough 1 +#endif /* GL_NV_geometry_shader_passthrough */ + +#ifndef GL_NV_gpu_multicast +#define GL_NV_gpu_multicast 1 +#define GL_PER_GPU_STORAGE_BIT_NV 0x0800 +#define GL_MULTICAST_GPUS_NV 0x92BA +#define GL_RENDER_GPU_MASK_NV 0x9558 +#define GL_PER_GPU_STORAGE_NV 0x9548 +#define GL_MULTICAST_PROGRAMMABLE_SAMPLE_LOCATION_NV 0x9549 +typedef void (APIENTRYP PFNGLRENDERGPUMASKNVPROC) (GLbitfield mask); +typedef void (APIENTRYP PFNGLMULTICASTBUFFERSUBDATANVPROC) (GLbitfield gpuMask, GLuint buffer, GLintptr offset, GLsizeiptr size, const void *data); +typedef void (APIENTRYP PFNGLMULTICASTCOPYBUFFERSUBDATANVPROC) (GLuint readGpu, GLbitfield writeGpuMask, GLuint readBuffer, GLuint writeBuffer, GLintptr readOffset, GLintptr writeOffset, GLsizeiptr size); +typedef void (APIENTRYP PFNGLMULTICASTCOPYIMAGESUBDATANVPROC) (GLuint srcGpu, GLbitfield dstGpuMask, GLuint srcName, GLenum srcTarget, GLint srcLevel, GLint srcX, GLint srcY, GLint srcZ, GLuint dstName, GLenum dstTarget, GLint dstLevel, GLint dstX, GLint dstY, GLint dstZ, GLsizei srcWidth, GLsizei srcHeight, GLsizei srcDepth); +typedef void (APIENTRYP PFNGLMULTICASTBLITFRAMEBUFFERNVPROC) (GLuint srcGpu, GLuint dstGpu, GLint srcX0, GLint srcY0, GLint srcX1, GLint srcY1, GLint dstX0, GLint dstY0, GLint dstX1, GLint dstY1, GLbitfield mask, GLenum filter); +typedef void (APIENTRYP PFNGLMULTICASTFRAMEBUFFERSAMPLELOCATIONSFVNVPROC) (GLuint gpu, GLuint framebuffer, GLuint start, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTICASTBARRIERNVPROC) (void); +typedef void (APIENTRYP PFNGLMULTICASTWAITSYNCNVPROC) (GLuint signalGpu, GLbitfield waitGpuMask); +typedef void (APIENTRYP PFNGLMULTICASTGETQUERYOBJECTIVNVPROC) (GLuint gpu, GLuint id, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLMULTICASTGETQUERYOBJECTUIVNVPROC) (GLuint gpu, GLuint id, GLenum pname, GLuint *params); +typedef void (APIENTRYP PFNGLMULTICASTGETQUERYOBJECTI64VNVPROC) (GLuint gpu, GLuint id, GLenum pname, GLint64 *params); +typedef void (APIENTRYP PFNGLMULTICASTGETQUERYOBJECTUI64VNVPROC) (GLuint gpu, GLuint id, GLenum pname, GLuint64 *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glRenderGpuMaskNV (GLbitfield mask); +GLAPI void APIENTRY glMulticastBufferSubDataNV (GLbitfield gpuMask, GLuint buffer, GLintptr offset, GLsizeiptr size, const void *data); +GLAPI void APIENTRY glMulticastCopyBufferSubDataNV (GLuint readGpu, GLbitfield writeGpuMask, GLuint readBuffer, GLuint writeBuffer, GLintptr readOffset, GLintptr writeOffset, GLsizeiptr size); +GLAPI void APIENTRY glMulticastCopyImageSubDataNV (GLuint srcGpu, GLbitfield dstGpuMask, GLuint srcName, GLenum srcTarget, GLint srcLevel, GLint srcX, GLint srcY, GLint srcZ, GLuint dstName, GLenum dstTarget, GLint dstLevel, GLint dstX, GLint dstY, GLint dstZ, GLsizei srcWidth, GLsizei srcHeight, GLsizei srcDepth); +GLAPI void APIENTRY glMulticastBlitFramebufferNV (GLuint srcGpu, GLuint dstGpu, GLint srcX0, GLint srcY0, GLint srcX1, GLint srcY1, GLint dstX0, GLint dstY0, GLint dstX1, GLint dstY1, GLbitfield mask, GLenum filter); +GLAPI void APIENTRY glMulticastFramebufferSampleLocationsfvNV (GLuint gpu, GLuint framebuffer, GLuint start, GLsizei count, const GLfloat *v); +GLAPI void APIENTRY glMulticastBarrierNV (void); +GLAPI void APIENTRY glMulticastWaitSyncNV (GLuint signalGpu, GLbitfield waitGpuMask); +GLAPI void APIENTRY glMulticastGetQueryObjectivNV (GLuint gpu, GLuint id, GLenum pname, GLint *params); +GLAPI void APIENTRY glMulticastGetQueryObjectuivNV (GLuint gpu, GLuint id, GLenum pname, GLuint *params); +GLAPI void APIENTRY glMulticastGetQueryObjecti64vNV (GLuint gpu, GLuint id, GLenum pname, GLint64 *params); +GLAPI void APIENTRY glMulticastGetQueryObjectui64vNV (GLuint gpu, GLuint id, GLenum pname, GLuint64 *params); +#endif +#endif /* GL_NV_gpu_multicast */ + +#ifndef GL_NV_gpu_program4 +#define GL_NV_gpu_program4 1 +#define GL_MIN_PROGRAM_TEXEL_OFFSET_NV 0x8904 +#define GL_MAX_PROGRAM_TEXEL_OFFSET_NV 0x8905 +#define GL_PROGRAM_ATTRIB_COMPONENTS_NV 0x8906 +#define GL_PROGRAM_RESULT_COMPONENTS_NV 0x8907 +#define GL_MAX_PROGRAM_ATTRIB_COMPONENTS_NV 0x8908 +#define GL_MAX_PROGRAM_RESULT_COMPONENTS_NV 0x8909 +#define GL_MAX_PROGRAM_GENERIC_ATTRIBS_NV 0x8DA5 +#define GL_MAX_PROGRAM_GENERIC_RESULTS_NV 0x8DA6 +typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETERI4INVPROC) (GLenum target, GLuint index, GLint x, GLint y, GLint z, GLint w); +typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETERI4IVNVPROC) (GLenum target, GLuint index, const GLint *params); +typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETERSI4IVNVPROC) (GLenum target, GLuint index, GLsizei count, const GLint *params); +typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETERI4UINVPROC) (GLenum target, GLuint index, GLuint x, GLuint y, GLuint z, GLuint w); +typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETERI4UIVNVPROC) (GLenum target, GLuint index, const GLuint *params); +typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETERSI4UIVNVPROC) (GLenum target, GLuint index, GLsizei count, const GLuint *params); +typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETERI4INVPROC) (GLenum target, GLuint index, GLint x, GLint y, GLint z, GLint w); +typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETERI4IVNVPROC) (GLenum target, GLuint index, const GLint *params); +typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETERSI4IVNVPROC) (GLenum target, GLuint index, GLsizei count, const GLint *params); +typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETERI4UINVPROC) (GLenum target, GLuint index, GLuint x, GLuint y, GLuint z, GLuint w); +typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETERI4UIVNVPROC) (GLenum target, GLuint index, const GLuint *params); +typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETERSI4UIVNVPROC) (GLenum target, GLuint index, GLsizei count, const GLuint *params); +typedef void (APIENTRYP PFNGLGETPROGRAMLOCALPARAMETERIIVNVPROC) (GLenum target, GLuint index, GLint *params); +typedef void (APIENTRYP PFNGLGETPROGRAMLOCALPARAMETERIUIVNVPROC) (GLenum target, GLuint index, GLuint *params); +typedef void (APIENTRYP PFNGLGETPROGRAMENVPARAMETERIIVNVPROC) (GLenum target, GLuint index, GLint *params); +typedef void (APIENTRYP PFNGLGETPROGRAMENVPARAMETERIUIVNVPROC) (GLenum target, GLuint index, GLuint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glProgramLocalParameterI4iNV (GLenum target, GLuint index, GLint x, GLint y, GLint z, GLint w); +GLAPI void APIENTRY glProgramLocalParameterI4ivNV (GLenum target, GLuint index, const GLint *params); +GLAPI void APIENTRY glProgramLocalParametersI4ivNV (GLenum target, GLuint index, GLsizei count, const GLint *params); +GLAPI void APIENTRY glProgramLocalParameterI4uiNV (GLenum target, GLuint index, GLuint x, GLuint y, GLuint z, GLuint w); +GLAPI void APIENTRY glProgramLocalParameterI4uivNV (GLenum target, GLuint index, const GLuint *params); +GLAPI void APIENTRY glProgramLocalParametersI4uivNV (GLenum target, GLuint index, GLsizei count, const GLuint *params); +GLAPI void APIENTRY glProgramEnvParameterI4iNV (GLenum target, GLuint index, GLint x, GLint y, GLint z, GLint w); +GLAPI void APIENTRY glProgramEnvParameterI4ivNV (GLenum target, GLuint index, const GLint *params); +GLAPI void APIENTRY glProgramEnvParametersI4ivNV (GLenum target, GLuint index, GLsizei count, const GLint *params); +GLAPI void APIENTRY glProgramEnvParameterI4uiNV (GLenum target, GLuint index, GLuint x, GLuint y, GLuint z, GLuint w); +GLAPI void APIENTRY glProgramEnvParameterI4uivNV (GLenum target, GLuint index, const GLuint *params); +GLAPI void APIENTRY glProgramEnvParametersI4uivNV (GLenum target, GLuint index, GLsizei count, const GLuint *params); +GLAPI void APIENTRY glGetProgramLocalParameterIivNV (GLenum target, GLuint index, GLint *params); +GLAPI void APIENTRY glGetProgramLocalParameterIuivNV (GLenum target, GLuint index, GLuint *params); +GLAPI void APIENTRY glGetProgramEnvParameterIivNV (GLenum target, GLuint index, GLint *params); +GLAPI void APIENTRY glGetProgramEnvParameterIuivNV (GLenum target, GLuint index, GLuint *params); +#endif +#endif /* GL_NV_gpu_program4 */ + +#ifndef GL_NV_gpu_program5 +#define GL_NV_gpu_program5 1 +#define GL_MAX_GEOMETRY_PROGRAM_INVOCATIONS_NV 0x8E5A +#define GL_MIN_FRAGMENT_INTERPOLATION_OFFSET_NV 0x8E5B +#define GL_MAX_FRAGMENT_INTERPOLATION_OFFSET_NV 0x8E5C +#define GL_FRAGMENT_PROGRAM_INTERPOLATION_OFFSET_BITS_NV 0x8E5D +#define GL_MIN_PROGRAM_TEXTURE_GATHER_OFFSET_NV 0x8E5E +#define GL_MAX_PROGRAM_TEXTURE_GATHER_OFFSET_NV 0x8E5F +#define GL_MAX_PROGRAM_SUBROUTINE_PARAMETERS_NV 0x8F44 +#define GL_MAX_PROGRAM_SUBROUTINE_NUM_NV 0x8F45 +typedef void (APIENTRYP PFNGLPROGRAMSUBROUTINEPARAMETERSUIVNVPROC) (GLenum target, GLsizei count, const GLuint *params); +typedef void (APIENTRYP PFNGLGETPROGRAMSUBROUTINEPARAMETERUIVNVPROC) (GLenum target, GLuint index, GLuint *param); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glProgramSubroutineParametersuivNV (GLenum target, GLsizei count, const GLuint *params); +GLAPI void APIENTRY glGetProgramSubroutineParameteruivNV (GLenum target, GLuint index, GLuint *param); +#endif +#endif /* GL_NV_gpu_program5 */ + +#ifndef GL_NV_gpu_program5_mem_extended +#define GL_NV_gpu_program5_mem_extended 1 +#endif /* GL_NV_gpu_program5_mem_extended */ + +#ifndef GL_NV_gpu_shader5 +#define GL_NV_gpu_shader5 1 +#endif /* GL_NV_gpu_shader5 */ + +#ifndef GL_NV_half_float +#define GL_NV_half_float 1 +typedef unsigned short GLhalfNV; +#define GL_HALF_FLOAT_NV 0x140B +typedef void (APIENTRYP PFNGLVERTEX2HNVPROC) (GLhalfNV x, GLhalfNV y); +typedef void (APIENTRYP PFNGLVERTEX2HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEX3HNVPROC) (GLhalfNV x, GLhalfNV y, GLhalfNV z); +typedef void (APIENTRYP PFNGLVERTEX3HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEX4HNVPROC) (GLhalfNV x, GLhalfNV y, GLhalfNV z, GLhalfNV w); +typedef void (APIENTRYP PFNGLVERTEX4HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLNORMAL3HNVPROC) (GLhalfNV nx, GLhalfNV ny, GLhalfNV nz); +typedef void (APIENTRYP PFNGLNORMAL3HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLCOLOR3HNVPROC) (GLhalfNV red, GLhalfNV green, GLhalfNV blue); +typedef void (APIENTRYP PFNGLCOLOR3HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLCOLOR4HNVPROC) (GLhalfNV red, GLhalfNV green, GLhalfNV blue, GLhalfNV alpha); +typedef void (APIENTRYP PFNGLCOLOR4HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLTEXCOORD1HNVPROC) (GLhalfNV s); +typedef void (APIENTRYP PFNGLTEXCOORD1HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLTEXCOORD2HNVPROC) (GLhalfNV s, GLhalfNV t); +typedef void (APIENTRYP PFNGLTEXCOORD2HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLTEXCOORD3HNVPROC) (GLhalfNV s, GLhalfNV t, GLhalfNV r); +typedef void (APIENTRYP PFNGLTEXCOORD3HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLTEXCOORD4HNVPROC) (GLhalfNV s, GLhalfNV t, GLhalfNV r, GLhalfNV q); +typedef void (APIENTRYP PFNGLTEXCOORD4HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1HNVPROC) (GLenum target, GLhalfNV s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1HVNVPROC) (GLenum target, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2HNVPROC) (GLenum target, GLhalfNV s, GLhalfNV t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2HVNVPROC) (GLenum target, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3HNVPROC) (GLenum target, GLhalfNV s, GLhalfNV t, GLhalfNV r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3HVNVPROC) (GLenum target, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4HNVPROC) (GLenum target, GLhalfNV s, GLhalfNV t, GLhalfNV r, GLhalfNV q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4HVNVPROC) (GLenum target, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLFOGCOORDHNVPROC) (GLhalfNV fog); +typedef void (APIENTRYP PFNGLFOGCOORDHVNVPROC) (const GLhalfNV *fog); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3HNVPROC) (GLhalfNV red, GLhalfNV green, GLhalfNV blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXWEIGHTHNVPROC) (GLhalfNV weight); +typedef void (APIENTRYP PFNGLVERTEXWEIGHTHVNVPROC) (const GLhalfNV *weight); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1HNVPROC) (GLuint index, GLhalfNV x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1HVNVPROC) (GLuint index, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2HNVPROC) (GLuint index, GLhalfNV x, GLhalfNV y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2HVNVPROC) (GLuint index, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3HNVPROC) (GLuint index, GLhalfNV x, GLhalfNV y, GLhalfNV z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3HVNVPROC) (GLuint index, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4HNVPROC) (GLuint index, GLhalfNV x, GLhalfNV y, GLhalfNV z, GLhalfNV w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4HVNVPROC) (GLuint index, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS1HVNVPROC) (GLuint index, GLsizei n, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS2HVNVPROC) (GLuint index, GLsizei n, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS3HVNVPROC) (GLuint index, GLsizei n, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS4HVNVPROC) (GLuint index, GLsizei n, const GLhalfNV *v); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertex2hNV (GLhalfNV x, GLhalfNV y); +GLAPI void APIENTRY glVertex2hvNV (const GLhalfNV *v); +GLAPI void APIENTRY glVertex3hNV (GLhalfNV x, GLhalfNV y, GLhalfNV z); +GLAPI void APIENTRY glVertex3hvNV (const GLhalfNV *v); +GLAPI void APIENTRY glVertex4hNV (GLhalfNV x, GLhalfNV y, GLhalfNV z, GLhalfNV w); +GLAPI void APIENTRY glVertex4hvNV (const GLhalfNV *v); +GLAPI void APIENTRY glNormal3hNV (GLhalfNV nx, GLhalfNV ny, GLhalfNV nz); +GLAPI void APIENTRY glNormal3hvNV (const GLhalfNV *v); +GLAPI void APIENTRY glColor3hNV (GLhalfNV red, GLhalfNV green, GLhalfNV blue); +GLAPI void APIENTRY glColor3hvNV (const GLhalfNV *v); +GLAPI void APIENTRY glColor4hNV (GLhalfNV red, GLhalfNV green, GLhalfNV blue, GLhalfNV alpha); +GLAPI void APIENTRY glColor4hvNV (const GLhalfNV *v); +GLAPI void APIENTRY glTexCoord1hNV (GLhalfNV s); +GLAPI void APIENTRY glTexCoord1hvNV (const GLhalfNV *v); +GLAPI void APIENTRY glTexCoord2hNV (GLhalfNV s, GLhalfNV t); +GLAPI void APIENTRY glTexCoord2hvNV (const GLhalfNV *v); +GLAPI void APIENTRY glTexCoord3hNV (GLhalfNV s, GLhalfNV t, GLhalfNV r); +GLAPI void APIENTRY glTexCoord3hvNV (const GLhalfNV *v); +GLAPI void APIENTRY glTexCoord4hNV (GLhalfNV s, GLhalfNV t, GLhalfNV r, GLhalfNV q); +GLAPI void APIENTRY glTexCoord4hvNV (const GLhalfNV *v); +GLAPI void APIENTRY glMultiTexCoord1hNV (GLenum target, GLhalfNV s); +GLAPI void APIENTRY glMultiTexCoord1hvNV (GLenum target, const GLhalfNV *v); +GLAPI void APIENTRY glMultiTexCoord2hNV (GLenum target, GLhalfNV s, GLhalfNV t); +GLAPI void APIENTRY glMultiTexCoord2hvNV (GLenum target, const GLhalfNV *v); +GLAPI void APIENTRY glMultiTexCoord3hNV (GLenum target, GLhalfNV s, GLhalfNV t, GLhalfNV r); +GLAPI void APIENTRY glMultiTexCoord3hvNV (GLenum target, const GLhalfNV *v); +GLAPI void APIENTRY glMultiTexCoord4hNV (GLenum target, GLhalfNV s, GLhalfNV t, GLhalfNV r, GLhalfNV q); +GLAPI void APIENTRY glMultiTexCoord4hvNV (GLenum target, const GLhalfNV *v); +GLAPI void APIENTRY glFogCoordhNV (GLhalfNV fog); +GLAPI void APIENTRY glFogCoordhvNV (const GLhalfNV *fog); +GLAPI void APIENTRY glSecondaryColor3hNV (GLhalfNV red, GLhalfNV green, GLhalfNV blue); +GLAPI void APIENTRY glSecondaryColor3hvNV (const GLhalfNV *v); +GLAPI void APIENTRY glVertexWeighthNV (GLhalfNV weight); +GLAPI void APIENTRY glVertexWeighthvNV (const GLhalfNV *weight); +GLAPI void APIENTRY glVertexAttrib1hNV (GLuint index, GLhalfNV x); +GLAPI void APIENTRY glVertexAttrib1hvNV (GLuint index, const GLhalfNV *v); +GLAPI void APIENTRY glVertexAttrib2hNV (GLuint index, GLhalfNV x, GLhalfNV y); +GLAPI void APIENTRY glVertexAttrib2hvNV (GLuint index, const GLhalfNV *v); +GLAPI void APIENTRY glVertexAttrib3hNV (GLuint index, GLhalfNV x, GLhalfNV y, GLhalfNV z); +GLAPI void APIENTRY glVertexAttrib3hvNV (GLuint index, const GLhalfNV *v); +GLAPI void APIENTRY glVertexAttrib4hNV (GLuint index, GLhalfNV x, GLhalfNV y, GLhalfNV z, GLhalfNV w); +GLAPI void APIENTRY glVertexAttrib4hvNV (GLuint index, const GLhalfNV *v); +GLAPI void APIENTRY glVertexAttribs1hvNV (GLuint index, GLsizei n, const GLhalfNV *v); +GLAPI void APIENTRY glVertexAttribs2hvNV (GLuint index, GLsizei n, const GLhalfNV *v); +GLAPI void APIENTRY glVertexAttribs3hvNV (GLuint index, GLsizei n, const GLhalfNV *v); +GLAPI void APIENTRY glVertexAttribs4hvNV (GLuint index, GLsizei n, const GLhalfNV *v); +#endif +#endif /* GL_NV_half_float */ + +#ifndef GL_NV_internalformat_sample_query +#define GL_NV_internalformat_sample_query 1 +#define GL_MULTISAMPLES_NV 0x9371 +#define GL_SUPERSAMPLE_SCALE_X_NV 0x9372 +#define GL_SUPERSAMPLE_SCALE_Y_NV 0x9373 +#define GL_CONFORMANT_NV 0x9374 +typedef void (APIENTRYP PFNGLGETINTERNALFORMATSAMPLEIVNVPROC) (GLenum target, GLenum internalformat, GLsizei samples, GLenum pname, GLsizei count, GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGetInternalformatSampleivNV (GLenum target, GLenum internalformat, GLsizei samples, GLenum pname, GLsizei count, GLint *params); +#endif +#endif /* GL_NV_internalformat_sample_query */ + +#ifndef GL_NV_light_max_exponent +#define GL_NV_light_max_exponent 1 +#define GL_MAX_SHININESS_NV 0x8504 +#define GL_MAX_SPOT_EXPONENT_NV 0x8505 +#endif /* GL_NV_light_max_exponent */ + +#ifndef GL_NV_memory_attachment +#define GL_NV_memory_attachment 1 +#define GL_ATTACHED_MEMORY_OBJECT_NV 0x95A4 +#define GL_ATTACHED_MEMORY_OFFSET_NV 0x95A5 +#define GL_MEMORY_ATTACHABLE_ALIGNMENT_NV 0x95A6 +#define GL_MEMORY_ATTACHABLE_SIZE_NV 0x95A7 +#define GL_MEMORY_ATTACHABLE_NV 0x95A8 +#define GL_DETACHED_MEMORY_INCARNATION_NV 0x95A9 +#define GL_DETACHED_TEXTURES_NV 0x95AA +#define GL_DETACHED_BUFFERS_NV 0x95AB +#define GL_MAX_DETACHED_TEXTURES_NV 0x95AC +#define GL_MAX_DETACHED_BUFFERS_NV 0x95AD +typedef void (APIENTRYP PFNGLGETMEMORYOBJECTDETACHEDRESOURCESUIVNVPROC) (GLuint memory, GLenum pname, GLint first, GLsizei count, GLuint *params); +typedef void (APIENTRYP PFNGLRESETMEMORYOBJECTPARAMETERNVPROC) (GLuint memory, GLenum pname); +typedef void (APIENTRYP PFNGLTEXATTACHMEMORYNVPROC) (GLenum target, GLuint memory, GLuint64 offset); +typedef void (APIENTRYP PFNGLBUFFERATTACHMEMORYNVPROC) (GLenum target, GLuint memory, GLuint64 offset); +typedef void (APIENTRYP PFNGLTEXTUREATTACHMEMORYNVPROC) (GLuint texture, GLuint memory, GLuint64 offset); +typedef void (APIENTRYP PFNGLNAMEDBUFFERATTACHMEMORYNVPROC) (GLuint buffer, GLuint memory, GLuint64 offset); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGetMemoryObjectDetachedResourcesuivNV (GLuint memory, GLenum pname, GLint first, GLsizei count, GLuint *params); +GLAPI void APIENTRY glResetMemoryObjectParameterNV (GLuint memory, GLenum pname); +GLAPI void APIENTRY glTexAttachMemoryNV (GLenum target, GLuint memory, GLuint64 offset); +GLAPI void APIENTRY glBufferAttachMemoryNV (GLenum target, GLuint memory, GLuint64 offset); +GLAPI void APIENTRY glTextureAttachMemoryNV (GLuint texture, GLuint memory, GLuint64 offset); +GLAPI void APIENTRY glNamedBufferAttachMemoryNV (GLuint buffer, GLuint memory, GLuint64 offset); +#endif +#endif /* GL_NV_memory_attachment */ + +#ifndef GL_NV_memory_object_sparse +#define GL_NV_memory_object_sparse 1 +typedef void (APIENTRYP PFNGLBUFFERPAGECOMMITMENTMEMNVPROC) (GLenum target, GLintptr offset, GLsizeiptr size, GLuint memory, GLuint64 memOffset, GLboolean commit); +typedef void (APIENTRYP PFNGLTEXPAGECOMMITMENTMEMNVPROC) (GLenum target, GLint layer, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset, GLboolean commit); +typedef void (APIENTRYP PFNGLNAMEDBUFFERPAGECOMMITMENTMEMNVPROC) (GLuint buffer, GLintptr offset, GLsizeiptr size, GLuint memory, GLuint64 memOffset, GLboolean commit); +typedef void (APIENTRYP PFNGLTEXTUREPAGECOMMITMENTMEMNVPROC) (GLuint texture, GLint layer, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset, GLboolean commit); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBufferPageCommitmentMemNV (GLenum target, GLintptr offset, GLsizeiptr size, GLuint memory, GLuint64 memOffset, GLboolean commit); +GLAPI void APIENTRY glTexPageCommitmentMemNV (GLenum target, GLint layer, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset, GLboolean commit); +GLAPI void APIENTRY glNamedBufferPageCommitmentMemNV (GLuint buffer, GLintptr offset, GLsizeiptr size, GLuint memory, GLuint64 memOffset, GLboolean commit); +GLAPI void APIENTRY glTexturePageCommitmentMemNV (GLuint texture, GLint layer, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset, GLboolean commit); +#endif +#endif /* GL_NV_memory_object_sparse */ + +#ifndef GL_NV_mesh_shader +#define GL_NV_mesh_shader 1 +#define GL_MESH_SHADER_NV 0x9559 +#define GL_TASK_SHADER_NV 0x955A +#define GL_MAX_MESH_UNIFORM_BLOCKS_NV 0x8E60 +#define GL_MAX_MESH_TEXTURE_IMAGE_UNITS_NV 0x8E61 +#define GL_MAX_MESH_IMAGE_UNIFORMS_NV 0x8E62 +#define GL_MAX_MESH_UNIFORM_COMPONENTS_NV 0x8E63 +#define GL_MAX_MESH_ATOMIC_COUNTER_BUFFERS_NV 0x8E64 +#define GL_MAX_MESH_ATOMIC_COUNTERS_NV 0x8E65 +#define GL_MAX_MESH_SHADER_STORAGE_BLOCKS_NV 0x8E66 +#define GL_MAX_COMBINED_MESH_UNIFORM_COMPONENTS_NV 0x8E67 +#define GL_MAX_TASK_UNIFORM_BLOCKS_NV 0x8E68 +#define GL_MAX_TASK_TEXTURE_IMAGE_UNITS_NV 0x8E69 +#define GL_MAX_TASK_IMAGE_UNIFORMS_NV 0x8E6A +#define GL_MAX_TASK_UNIFORM_COMPONENTS_NV 0x8E6B +#define GL_MAX_TASK_ATOMIC_COUNTER_BUFFERS_NV 0x8E6C +#define GL_MAX_TASK_ATOMIC_COUNTERS_NV 0x8E6D +#define GL_MAX_TASK_SHADER_STORAGE_BLOCKS_NV 0x8E6E +#define GL_MAX_COMBINED_TASK_UNIFORM_COMPONENTS_NV 0x8E6F +#define GL_MAX_MESH_WORK_GROUP_INVOCATIONS_NV 0x95A2 +#define GL_MAX_TASK_WORK_GROUP_INVOCATIONS_NV 0x95A3 +#define GL_MAX_MESH_TOTAL_MEMORY_SIZE_NV 0x9536 +#define GL_MAX_TASK_TOTAL_MEMORY_SIZE_NV 0x9537 +#define GL_MAX_MESH_OUTPUT_VERTICES_NV 0x9538 +#define GL_MAX_MESH_OUTPUT_PRIMITIVES_NV 0x9539 +#define GL_MAX_TASK_OUTPUT_COUNT_NV 0x953A +#define GL_MAX_DRAW_MESH_TASKS_COUNT_NV 0x953D +#define GL_MAX_MESH_VIEWS_NV 0x9557 +#define GL_MESH_OUTPUT_PER_VERTEX_GRANULARITY_NV 0x92DF +#define GL_MESH_OUTPUT_PER_PRIMITIVE_GRANULARITY_NV 0x9543 +#define GL_MAX_MESH_WORK_GROUP_SIZE_NV 0x953B +#define GL_MAX_TASK_WORK_GROUP_SIZE_NV 0x953C +#define GL_MESH_WORK_GROUP_SIZE_NV 0x953E +#define GL_TASK_WORK_GROUP_SIZE_NV 0x953F +#define GL_MESH_VERTICES_OUT_NV 0x9579 +#define GL_MESH_PRIMITIVES_OUT_NV 0x957A +#define GL_MESH_OUTPUT_TYPE_NV 0x957B +#define GL_UNIFORM_BLOCK_REFERENCED_BY_MESH_SHADER_NV 0x959C +#define GL_UNIFORM_BLOCK_REFERENCED_BY_TASK_SHADER_NV 0x959D +#define GL_REFERENCED_BY_MESH_SHADER_NV 0x95A0 +#define GL_REFERENCED_BY_TASK_SHADER_NV 0x95A1 +#define GL_MESH_SHADER_BIT_NV 0x00000040 +#define GL_TASK_SHADER_BIT_NV 0x00000080 +#define GL_MESH_SUBROUTINE_NV 0x957C +#define GL_TASK_SUBROUTINE_NV 0x957D +#define GL_MESH_SUBROUTINE_UNIFORM_NV 0x957E +#define GL_TASK_SUBROUTINE_UNIFORM_NV 0x957F +#define GL_ATOMIC_COUNTER_BUFFER_REFERENCED_BY_MESH_SHADER_NV 0x959E +#define GL_ATOMIC_COUNTER_BUFFER_REFERENCED_BY_TASK_SHADER_NV 0x959F +typedef void (APIENTRYP PFNGLDRAWMESHTASKSNVPROC) (GLuint first, GLuint count); +typedef void (APIENTRYP PFNGLDRAWMESHTASKSINDIRECTNVPROC) (GLintptr indirect); +typedef void (APIENTRYP PFNGLMULTIDRAWMESHTASKSINDIRECTNVPROC) (GLintptr indirect, GLsizei drawcount, GLsizei stride); +typedef void (APIENTRYP PFNGLMULTIDRAWMESHTASKSINDIRECTCOUNTNVPROC) (GLintptr indirect, GLintptr drawcount, GLsizei maxdrawcount, GLsizei stride); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawMeshTasksNV (GLuint first, GLuint count); +GLAPI void APIENTRY glDrawMeshTasksIndirectNV (GLintptr indirect); +GLAPI void APIENTRY glMultiDrawMeshTasksIndirectNV (GLintptr indirect, GLsizei drawcount, GLsizei stride); +GLAPI void APIENTRY glMultiDrawMeshTasksIndirectCountNV (GLintptr indirect, GLintptr drawcount, GLsizei maxdrawcount, GLsizei stride); +#endif +#endif /* GL_NV_mesh_shader */ + +#ifndef GL_NV_multisample_coverage +#define GL_NV_multisample_coverage 1 +#endif /* GL_NV_multisample_coverage */ + +#ifndef GL_NV_multisample_filter_hint +#define GL_NV_multisample_filter_hint 1 +#define GL_MULTISAMPLE_FILTER_HINT_NV 0x8534 +#endif /* GL_NV_multisample_filter_hint */ + +#ifndef GL_NV_occlusion_query +#define GL_NV_occlusion_query 1 +#define GL_PIXEL_COUNTER_BITS_NV 0x8864 +#define GL_CURRENT_OCCLUSION_QUERY_ID_NV 0x8865 +#define GL_PIXEL_COUNT_NV 0x8866 +#define GL_PIXEL_COUNT_AVAILABLE_NV 0x8867 +typedef void (APIENTRYP PFNGLGENOCCLUSIONQUERIESNVPROC) (GLsizei n, GLuint *ids); +typedef void (APIENTRYP PFNGLDELETEOCCLUSIONQUERIESNVPROC) (GLsizei n, const GLuint *ids); +typedef GLboolean (APIENTRYP PFNGLISOCCLUSIONQUERYNVPROC) (GLuint id); +typedef void (APIENTRYP PFNGLBEGINOCCLUSIONQUERYNVPROC) (GLuint id); +typedef void (APIENTRYP PFNGLENDOCCLUSIONQUERYNVPROC) (void); +typedef void (APIENTRYP PFNGLGETOCCLUSIONQUERYIVNVPROC) (GLuint id, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETOCCLUSIONQUERYUIVNVPROC) (GLuint id, GLenum pname, GLuint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGenOcclusionQueriesNV (GLsizei n, GLuint *ids); +GLAPI void APIENTRY glDeleteOcclusionQueriesNV (GLsizei n, const GLuint *ids); +GLAPI GLboolean APIENTRY glIsOcclusionQueryNV (GLuint id); +GLAPI void APIENTRY glBeginOcclusionQueryNV (GLuint id); +GLAPI void APIENTRY glEndOcclusionQueryNV (void); +GLAPI void APIENTRY glGetOcclusionQueryivNV (GLuint id, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetOcclusionQueryuivNV (GLuint id, GLenum pname, GLuint *params); +#endif +#endif /* GL_NV_occlusion_query */ + +#ifndef GL_NV_packed_depth_stencil +#define GL_NV_packed_depth_stencil 1 +#define GL_DEPTH_STENCIL_NV 0x84F9 +#define GL_UNSIGNED_INT_24_8_NV 0x84FA +#endif /* GL_NV_packed_depth_stencil */ + +#ifndef GL_NV_parameter_buffer_object +#define GL_NV_parameter_buffer_object 1 +#define GL_MAX_PROGRAM_PARAMETER_BUFFER_BINDINGS_NV 0x8DA0 +#define GL_MAX_PROGRAM_PARAMETER_BUFFER_SIZE_NV 0x8DA1 +#define GL_VERTEX_PROGRAM_PARAMETER_BUFFER_NV 0x8DA2 +#define GL_GEOMETRY_PROGRAM_PARAMETER_BUFFER_NV 0x8DA3 +#define GL_FRAGMENT_PROGRAM_PARAMETER_BUFFER_NV 0x8DA4 +typedef void (APIENTRYP PFNGLPROGRAMBUFFERPARAMETERSFVNVPROC) (GLenum target, GLuint bindingIndex, GLuint wordIndex, GLsizei count, const GLfloat *params); +typedef void (APIENTRYP PFNGLPROGRAMBUFFERPARAMETERSIIVNVPROC) (GLenum target, GLuint bindingIndex, GLuint wordIndex, GLsizei count, const GLint *params); +typedef void (APIENTRYP PFNGLPROGRAMBUFFERPARAMETERSIUIVNVPROC) (GLenum target, GLuint bindingIndex, GLuint wordIndex, GLsizei count, const GLuint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glProgramBufferParametersfvNV (GLenum target, GLuint bindingIndex, GLuint wordIndex, GLsizei count, const GLfloat *params); +GLAPI void APIENTRY glProgramBufferParametersIivNV (GLenum target, GLuint bindingIndex, GLuint wordIndex, GLsizei count, const GLint *params); +GLAPI void APIENTRY glProgramBufferParametersIuivNV (GLenum target, GLuint bindingIndex, GLuint wordIndex, GLsizei count, const GLuint *params); +#endif +#endif /* GL_NV_parameter_buffer_object */ + +#ifndef GL_NV_parameter_buffer_object2 +#define GL_NV_parameter_buffer_object2 1 +#endif /* GL_NV_parameter_buffer_object2 */ + +#ifndef GL_NV_path_rendering +#define GL_NV_path_rendering 1 +#define GL_PATH_FORMAT_SVG_NV 0x9070 +#define GL_PATH_FORMAT_PS_NV 0x9071 +#define GL_STANDARD_FONT_NAME_NV 0x9072 +#define GL_SYSTEM_FONT_NAME_NV 0x9073 +#define GL_FILE_NAME_NV 0x9074 +#define GL_PATH_STROKE_WIDTH_NV 0x9075 +#define GL_PATH_END_CAPS_NV 0x9076 +#define GL_PATH_INITIAL_END_CAP_NV 0x9077 +#define GL_PATH_TERMINAL_END_CAP_NV 0x9078 +#define GL_PATH_JOIN_STYLE_NV 0x9079 +#define GL_PATH_MITER_LIMIT_NV 0x907A +#define GL_PATH_DASH_CAPS_NV 0x907B +#define GL_PATH_INITIAL_DASH_CAP_NV 0x907C +#define GL_PATH_TERMINAL_DASH_CAP_NV 0x907D +#define GL_PATH_DASH_OFFSET_NV 0x907E +#define GL_PATH_CLIENT_LENGTH_NV 0x907F +#define GL_PATH_FILL_MODE_NV 0x9080 +#define GL_PATH_FILL_MASK_NV 0x9081 +#define GL_PATH_FILL_COVER_MODE_NV 0x9082 +#define GL_PATH_STROKE_COVER_MODE_NV 0x9083 +#define GL_PATH_STROKE_MASK_NV 0x9084 +#define GL_COUNT_UP_NV 0x9088 +#define GL_COUNT_DOWN_NV 0x9089 +#define GL_PATH_OBJECT_BOUNDING_BOX_NV 0x908A +#define GL_CONVEX_HULL_NV 0x908B +#define GL_BOUNDING_BOX_NV 0x908D +#define GL_TRANSLATE_X_NV 0x908E +#define GL_TRANSLATE_Y_NV 0x908F +#define GL_TRANSLATE_2D_NV 0x9090 +#define GL_TRANSLATE_3D_NV 0x9091 +#define GL_AFFINE_2D_NV 0x9092 +#define GL_AFFINE_3D_NV 0x9094 +#define GL_TRANSPOSE_AFFINE_2D_NV 0x9096 +#define GL_TRANSPOSE_AFFINE_3D_NV 0x9098 +#define GL_UTF8_NV 0x909A +#define GL_UTF16_NV 0x909B +#define GL_BOUNDING_BOX_OF_BOUNDING_BOXES_NV 0x909C +#define GL_PATH_COMMAND_COUNT_NV 0x909D +#define GL_PATH_COORD_COUNT_NV 0x909E +#define GL_PATH_DASH_ARRAY_COUNT_NV 0x909F +#define GL_PATH_COMPUTED_LENGTH_NV 0x90A0 +#define GL_PATH_FILL_BOUNDING_BOX_NV 0x90A1 +#define GL_PATH_STROKE_BOUNDING_BOX_NV 0x90A2 +#define GL_SQUARE_NV 0x90A3 +#define GL_ROUND_NV 0x90A4 +#define GL_TRIANGULAR_NV 0x90A5 +#define GL_BEVEL_NV 0x90A6 +#define GL_MITER_REVERT_NV 0x90A7 +#define GL_MITER_TRUNCATE_NV 0x90A8 +#define GL_SKIP_MISSING_GLYPH_NV 0x90A9 +#define GL_USE_MISSING_GLYPH_NV 0x90AA +#define GL_PATH_ERROR_POSITION_NV 0x90AB +#define GL_ACCUM_ADJACENT_PAIRS_NV 0x90AD +#define GL_ADJACENT_PAIRS_NV 0x90AE +#define GL_FIRST_TO_REST_NV 0x90AF +#define GL_PATH_GEN_MODE_NV 0x90B0 +#define GL_PATH_GEN_COEFF_NV 0x90B1 +#define GL_PATH_GEN_COMPONENTS_NV 0x90B3 +#define GL_PATH_STENCIL_FUNC_NV 0x90B7 +#define GL_PATH_STENCIL_REF_NV 0x90B8 +#define GL_PATH_STENCIL_VALUE_MASK_NV 0x90B9 +#define GL_PATH_STENCIL_DEPTH_OFFSET_FACTOR_NV 0x90BD +#define GL_PATH_STENCIL_DEPTH_OFFSET_UNITS_NV 0x90BE +#define GL_PATH_COVER_DEPTH_FUNC_NV 0x90BF +#define GL_PATH_DASH_OFFSET_RESET_NV 0x90B4 +#define GL_MOVE_TO_RESETS_NV 0x90B5 +#define GL_MOVE_TO_CONTINUES_NV 0x90B6 +#define GL_CLOSE_PATH_NV 0x00 +#define GL_MOVE_TO_NV 0x02 +#define GL_RELATIVE_MOVE_TO_NV 0x03 +#define GL_LINE_TO_NV 0x04 +#define GL_RELATIVE_LINE_TO_NV 0x05 +#define GL_HORIZONTAL_LINE_TO_NV 0x06 +#define GL_RELATIVE_HORIZONTAL_LINE_TO_NV 0x07 +#define GL_VERTICAL_LINE_TO_NV 0x08 +#define GL_RELATIVE_VERTICAL_LINE_TO_NV 0x09 +#define GL_QUADRATIC_CURVE_TO_NV 0x0A +#define GL_RELATIVE_QUADRATIC_CURVE_TO_NV 0x0B +#define GL_CUBIC_CURVE_TO_NV 0x0C +#define GL_RELATIVE_CUBIC_CURVE_TO_NV 0x0D +#define GL_SMOOTH_QUADRATIC_CURVE_TO_NV 0x0E +#define GL_RELATIVE_SMOOTH_QUADRATIC_CURVE_TO_NV 0x0F +#define GL_SMOOTH_CUBIC_CURVE_TO_NV 0x10 +#define GL_RELATIVE_SMOOTH_CUBIC_CURVE_TO_NV 0x11 +#define GL_SMALL_CCW_ARC_TO_NV 0x12 +#define GL_RELATIVE_SMALL_CCW_ARC_TO_NV 0x13 +#define GL_SMALL_CW_ARC_TO_NV 0x14 +#define GL_RELATIVE_SMALL_CW_ARC_TO_NV 0x15 +#define GL_LARGE_CCW_ARC_TO_NV 0x16 +#define GL_RELATIVE_LARGE_CCW_ARC_TO_NV 0x17 +#define GL_LARGE_CW_ARC_TO_NV 0x18 +#define GL_RELATIVE_LARGE_CW_ARC_TO_NV 0x19 +#define GL_RESTART_PATH_NV 0xF0 +#define GL_DUP_FIRST_CUBIC_CURVE_TO_NV 0xF2 +#define GL_DUP_LAST_CUBIC_CURVE_TO_NV 0xF4 +#define GL_RECT_NV 0xF6 +#define GL_CIRCULAR_CCW_ARC_TO_NV 0xF8 +#define GL_CIRCULAR_CW_ARC_TO_NV 0xFA +#define GL_CIRCULAR_TANGENT_ARC_TO_NV 0xFC +#define GL_ARC_TO_NV 0xFE +#define GL_RELATIVE_ARC_TO_NV 0xFF +#define GL_BOLD_BIT_NV 0x01 +#define GL_ITALIC_BIT_NV 0x02 +#define GL_GLYPH_WIDTH_BIT_NV 0x01 +#define GL_GLYPH_HEIGHT_BIT_NV 0x02 +#define GL_GLYPH_HORIZONTAL_BEARING_X_BIT_NV 0x04 +#define GL_GLYPH_HORIZONTAL_BEARING_Y_BIT_NV 0x08 +#define GL_GLYPH_HORIZONTAL_BEARING_ADVANCE_BIT_NV 0x10 +#define GL_GLYPH_VERTICAL_BEARING_X_BIT_NV 0x20 +#define GL_GLYPH_VERTICAL_BEARING_Y_BIT_NV 0x40 +#define GL_GLYPH_VERTICAL_BEARING_ADVANCE_BIT_NV 0x80 +#define GL_GLYPH_HAS_KERNING_BIT_NV 0x100 +#define GL_FONT_X_MIN_BOUNDS_BIT_NV 0x00010000 +#define GL_FONT_Y_MIN_BOUNDS_BIT_NV 0x00020000 +#define GL_FONT_X_MAX_BOUNDS_BIT_NV 0x00040000 +#define GL_FONT_Y_MAX_BOUNDS_BIT_NV 0x00080000 +#define GL_FONT_UNITS_PER_EM_BIT_NV 0x00100000 +#define GL_FONT_ASCENDER_BIT_NV 0x00200000 +#define GL_FONT_DESCENDER_BIT_NV 0x00400000 +#define GL_FONT_HEIGHT_BIT_NV 0x00800000 +#define GL_FONT_MAX_ADVANCE_WIDTH_BIT_NV 0x01000000 +#define GL_FONT_MAX_ADVANCE_HEIGHT_BIT_NV 0x02000000 +#define GL_FONT_UNDERLINE_POSITION_BIT_NV 0x04000000 +#define GL_FONT_UNDERLINE_THICKNESS_BIT_NV 0x08000000 +#define GL_FONT_HAS_KERNING_BIT_NV 0x10000000 +#define GL_ROUNDED_RECT_NV 0xE8 +#define GL_RELATIVE_ROUNDED_RECT_NV 0xE9 +#define GL_ROUNDED_RECT2_NV 0xEA +#define GL_RELATIVE_ROUNDED_RECT2_NV 0xEB +#define GL_ROUNDED_RECT4_NV 0xEC +#define GL_RELATIVE_ROUNDED_RECT4_NV 0xED +#define GL_ROUNDED_RECT8_NV 0xEE +#define GL_RELATIVE_ROUNDED_RECT8_NV 0xEF +#define GL_RELATIVE_RECT_NV 0xF7 +#define GL_FONT_GLYPHS_AVAILABLE_NV 0x9368 +#define GL_FONT_TARGET_UNAVAILABLE_NV 0x9369 +#define GL_FONT_UNAVAILABLE_NV 0x936A +#define GL_FONT_UNINTELLIGIBLE_NV 0x936B +#define GL_CONIC_CURVE_TO_NV 0x1A +#define GL_RELATIVE_CONIC_CURVE_TO_NV 0x1B +#define GL_FONT_NUM_GLYPH_INDICES_BIT_NV 0x20000000 +#define GL_STANDARD_FONT_FORMAT_NV 0x936C +#define GL_2_BYTES_NV 0x1407 +#define GL_3_BYTES_NV 0x1408 +#define GL_4_BYTES_NV 0x1409 +#define GL_EYE_LINEAR_NV 0x2400 +#define GL_OBJECT_LINEAR_NV 0x2401 +#define GL_CONSTANT_NV 0x8576 +#define GL_PATH_FOG_GEN_MODE_NV 0x90AC +#define GL_PRIMARY_COLOR_NV 0x852C +#define GL_SECONDARY_COLOR_NV 0x852D +#define GL_PATH_GEN_COLOR_FORMAT_NV 0x90B2 +#define GL_PATH_PROJECTION_NV 0x1701 +#define GL_PATH_MODELVIEW_NV 0x1700 +#define GL_PATH_MODELVIEW_STACK_DEPTH_NV 0x0BA3 +#define GL_PATH_MODELVIEW_MATRIX_NV 0x0BA6 +#define GL_PATH_MAX_MODELVIEW_STACK_DEPTH_NV 0x0D36 +#define GL_PATH_TRANSPOSE_MODELVIEW_MATRIX_NV 0x84E3 +#define GL_PATH_PROJECTION_STACK_DEPTH_NV 0x0BA4 +#define GL_PATH_PROJECTION_MATRIX_NV 0x0BA7 +#define GL_PATH_MAX_PROJECTION_STACK_DEPTH_NV 0x0D38 +#define GL_PATH_TRANSPOSE_PROJECTION_MATRIX_NV 0x84E4 +#define GL_FRAGMENT_INPUT_NV 0x936D +typedef GLuint (APIENTRYP PFNGLGENPATHSNVPROC) (GLsizei range); +typedef void (APIENTRYP PFNGLDELETEPATHSNVPROC) (GLuint path, GLsizei range); +typedef GLboolean (APIENTRYP PFNGLISPATHNVPROC) (GLuint path); +typedef void (APIENTRYP PFNGLPATHCOMMANDSNVPROC) (GLuint path, GLsizei numCommands, const GLubyte *commands, GLsizei numCoords, GLenum coordType, const void *coords); +typedef void (APIENTRYP PFNGLPATHCOORDSNVPROC) (GLuint path, GLsizei numCoords, GLenum coordType, const void *coords); +typedef void (APIENTRYP PFNGLPATHSUBCOMMANDSNVPROC) (GLuint path, GLsizei commandStart, GLsizei commandsToDelete, GLsizei numCommands, const GLubyte *commands, GLsizei numCoords, GLenum coordType, const void *coords); +typedef void (APIENTRYP PFNGLPATHSUBCOORDSNVPROC) (GLuint path, GLsizei coordStart, GLsizei numCoords, GLenum coordType, const void *coords); +typedef void (APIENTRYP PFNGLPATHSTRINGNVPROC) (GLuint path, GLenum format, GLsizei length, const void *pathString); +typedef void (APIENTRYP PFNGLPATHGLYPHSNVPROC) (GLuint firstPathName, GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLsizei numGlyphs, GLenum type, const void *charcodes, GLenum handleMissingGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +typedef void (APIENTRYP PFNGLPATHGLYPHRANGENVPROC) (GLuint firstPathName, GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLuint firstGlyph, GLsizei numGlyphs, GLenum handleMissingGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +typedef void (APIENTRYP PFNGLWEIGHTPATHSNVPROC) (GLuint resultPath, GLsizei numPaths, const GLuint *paths, const GLfloat *weights); +typedef void (APIENTRYP PFNGLCOPYPATHNVPROC) (GLuint resultPath, GLuint srcPath); +typedef void (APIENTRYP PFNGLINTERPOLATEPATHSNVPROC) (GLuint resultPath, GLuint pathA, GLuint pathB, GLfloat weight); +typedef void (APIENTRYP PFNGLTRANSFORMPATHNVPROC) (GLuint resultPath, GLuint srcPath, GLenum transformType, const GLfloat *transformValues); +typedef void (APIENTRYP PFNGLPATHPARAMETERIVNVPROC) (GLuint path, GLenum pname, const GLint *value); +typedef void (APIENTRYP PFNGLPATHPARAMETERINVPROC) (GLuint path, GLenum pname, GLint value); +typedef void (APIENTRYP PFNGLPATHPARAMETERFVNVPROC) (GLuint path, GLenum pname, const GLfloat *value); +typedef void (APIENTRYP PFNGLPATHPARAMETERFNVPROC) (GLuint path, GLenum pname, GLfloat value); +typedef void (APIENTRYP PFNGLPATHDASHARRAYNVPROC) (GLuint path, GLsizei dashCount, const GLfloat *dashArray); +typedef void (APIENTRYP PFNGLPATHSTENCILFUNCNVPROC) (GLenum func, GLint ref, GLuint mask); +typedef void (APIENTRYP PFNGLPATHSTENCILDEPTHOFFSETNVPROC) (GLfloat factor, GLfloat units); +typedef void (APIENTRYP PFNGLSTENCILFILLPATHNVPROC) (GLuint path, GLenum fillMode, GLuint mask); +typedef void (APIENTRYP PFNGLSTENCILSTROKEPATHNVPROC) (GLuint path, GLint reference, GLuint mask); +typedef void (APIENTRYP PFNGLSTENCILFILLPATHINSTANCEDNVPROC) (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum fillMode, GLuint mask, GLenum transformType, const GLfloat *transformValues); +typedef void (APIENTRYP PFNGLSTENCILSTROKEPATHINSTANCEDNVPROC) (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLint reference, GLuint mask, GLenum transformType, const GLfloat *transformValues); +typedef void (APIENTRYP PFNGLPATHCOVERDEPTHFUNCNVPROC) (GLenum func); +typedef void (APIENTRYP PFNGLCOVERFILLPATHNVPROC) (GLuint path, GLenum coverMode); +typedef void (APIENTRYP PFNGLCOVERSTROKEPATHNVPROC) (GLuint path, GLenum coverMode); +typedef void (APIENTRYP PFNGLCOVERFILLPATHINSTANCEDNVPROC) (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +typedef void (APIENTRYP PFNGLCOVERSTROKEPATHINSTANCEDNVPROC) (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +typedef void (APIENTRYP PFNGLGETPATHPARAMETERIVNVPROC) (GLuint path, GLenum pname, GLint *value); +typedef void (APIENTRYP PFNGLGETPATHPARAMETERFVNVPROC) (GLuint path, GLenum pname, GLfloat *value); +typedef void (APIENTRYP PFNGLGETPATHCOMMANDSNVPROC) (GLuint path, GLubyte *commands); +typedef void (APIENTRYP PFNGLGETPATHCOORDSNVPROC) (GLuint path, GLfloat *coords); +typedef void (APIENTRYP PFNGLGETPATHDASHARRAYNVPROC) (GLuint path, GLfloat *dashArray); +typedef void (APIENTRYP PFNGLGETPATHMETRICSNVPROC) (GLbitfield metricQueryMask, GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLsizei stride, GLfloat *metrics); +typedef void (APIENTRYP PFNGLGETPATHMETRICRANGENVPROC) (GLbitfield metricQueryMask, GLuint firstPathName, GLsizei numPaths, GLsizei stride, GLfloat *metrics); +typedef void (APIENTRYP PFNGLGETPATHSPACINGNVPROC) (GLenum pathListMode, GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLfloat advanceScale, GLfloat kerningScale, GLenum transformType, GLfloat *returnedSpacing); +typedef GLboolean (APIENTRYP PFNGLISPOINTINFILLPATHNVPROC) (GLuint path, GLuint mask, GLfloat x, GLfloat y); +typedef GLboolean (APIENTRYP PFNGLISPOINTINSTROKEPATHNVPROC) (GLuint path, GLfloat x, GLfloat y); +typedef GLfloat (APIENTRYP PFNGLGETPATHLENGTHNVPROC) (GLuint path, GLsizei startSegment, GLsizei numSegments); +typedef GLboolean (APIENTRYP PFNGLPOINTALONGPATHNVPROC) (GLuint path, GLsizei startSegment, GLsizei numSegments, GLfloat distance, GLfloat *x, GLfloat *y, GLfloat *tangentX, GLfloat *tangentY); +typedef void (APIENTRYP PFNGLMATRIXLOAD3X2FNVPROC) (GLenum matrixMode, const GLfloat *m); +typedef void (APIENTRYP PFNGLMATRIXLOAD3X3FNVPROC) (GLenum matrixMode, const GLfloat *m); +typedef void (APIENTRYP PFNGLMATRIXLOADTRANSPOSE3X3FNVPROC) (GLenum matrixMode, const GLfloat *m); +typedef void (APIENTRYP PFNGLMATRIXMULT3X2FNVPROC) (GLenum matrixMode, const GLfloat *m); +typedef void (APIENTRYP PFNGLMATRIXMULT3X3FNVPROC) (GLenum matrixMode, const GLfloat *m); +typedef void (APIENTRYP PFNGLMATRIXMULTTRANSPOSE3X3FNVPROC) (GLenum matrixMode, const GLfloat *m); +typedef void (APIENTRYP PFNGLSTENCILTHENCOVERFILLPATHNVPROC) (GLuint path, GLenum fillMode, GLuint mask, GLenum coverMode); +typedef void (APIENTRYP PFNGLSTENCILTHENCOVERSTROKEPATHNVPROC) (GLuint path, GLint reference, GLuint mask, GLenum coverMode); +typedef void (APIENTRYP PFNGLSTENCILTHENCOVERFILLPATHINSTANCEDNVPROC) (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum fillMode, GLuint mask, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +typedef void (APIENTRYP PFNGLSTENCILTHENCOVERSTROKEPATHINSTANCEDNVPROC) (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLint reference, GLuint mask, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +typedef GLenum (APIENTRYP PFNGLPATHGLYPHINDEXRANGENVPROC) (GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLuint pathParameterTemplate, GLfloat emScale, GLuint *baseAndCount); +typedef GLenum (APIENTRYP PFNGLPATHGLYPHINDEXARRAYNVPROC) (GLuint firstPathName, GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLuint firstGlyphIndex, GLsizei numGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +typedef GLenum (APIENTRYP PFNGLPATHMEMORYGLYPHINDEXARRAYNVPROC) (GLuint firstPathName, GLenum fontTarget, GLsizeiptr fontSize, const void *fontData, GLsizei faceIndex, GLuint firstGlyphIndex, GLsizei numGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +typedef void (APIENTRYP PFNGLPROGRAMPATHFRAGMENTINPUTGENNVPROC) (GLuint program, GLint location, GLenum genMode, GLint components, const GLfloat *coeffs); +typedef void (APIENTRYP PFNGLGETPROGRAMRESOURCEFVNVPROC) (GLuint program, GLenum programInterface, GLuint index, GLsizei propCount, const GLenum *props, GLsizei count, GLsizei *length, GLfloat *params); +typedef void (APIENTRYP PFNGLPATHCOLORGENNVPROC) (GLenum color, GLenum genMode, GLenum colorFormat, const GLfloat *coeffs); +typedef void (APIENTRYP PFNGLPATHTEXGENNVPROC) (GLenum texCoordSet, GLenum genMode, GLint components, const GLfloat *coeffs); +typedef void (APIENTRYP PFNGLPATHFOGGENNVPROC) (GLenum genMode); +typedef void (APIENTRYP PFNGLGETPATHCOLORGENIVNVPROC) (GLenum color, GLenum pname, GLint *value); +typedef void (APIENTRYP PFNGLGETPATHCOLORGENFVNVPROC) (GLenum color, GLenum pname, GLfloat *value); +typedef void (APIENTRYP PFNGLGETPATHTEXGENIVNVPROC) (GLenum texCoordSet, GLenum pname, GLint *value); +typedef void (APIENTRYP PFNGLGETPATHTEXGENFVNVPROC) (GLenum texCoordSet, GLenum pname, GLfloat *value); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLuint APIENTRY glGenPathsNV (GLsizei range); +GLAPI void APIENTRY glDeletePathsNV (GLuint path, GLsizei range); +GLAPI GLboolean APIENTRY glIsPathNV (GLuint path); +GLAPI void APIENTRY glPathCommandsNV (GLuint path, GLsizei numCommands, const GLubyte *commands, GLsizei numCoords, GLenum coordType, const void *coords); +GLAPI void APIENTRY glPathCoordsNV (GLuint path, GLsizei numCoords, GLenum coordType, const void *coords); +GLAPI void APIENTRY glPathSubCommandsNV (GLuint path, GLsizei commandStart, GLsizei commandsToDelete, GLsizei numCommands, const GLubyte *commands, GLsizei numCoords, GLenum coordType, const void *coords); +GLAPI void APIENTRY glPathSubCoordsNV (GLuint path, GLsizei coordStart, GLsizei numCoords, GLenum coordType, const void *coords); +GLAPI void APIENTRY glPathStringNV (GLuint path, GLenum format, GLsizei length, const void *pathString); +GLAPI void APIENTRY glPathGlyphsNV (GLuint firstPathName, GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLsizei numGlyphs, GLenum type, const void *charcodes, GLenum handleMissingGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +GLAPI void APIENTRY glPathGlyphRangeNV (GLuint firstPathName, GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLuint firstGlyph, GLsizei numGlyphs, GLenum handleMissingGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +GLAPI void APIENTRY glWeightPathsNV (GLuint resultPath, GLsizei numPaths, const GLuint *paths, const GLfloat *weights); +GLAPI void APIENTRY glCopyPathNV (GLuint resultPath, GLuint srcPath); +GLAPI void APIENTRY glInterpolatePathsNV (GLuint resultPath, GLuint pathA, GLuint pathB, GLfloat weight); +GLAPI void APIENTRY glTransformPathNV (GLuint resultPath, GLuint srcPath, GLenum transformType, const GLfloat *transformValues); +GLAPI void APIENTRY glPathParameterivNV (GLuint path, GLenum pname, const GLint *value); +GLAPI void APIENTRY glPathParameteriNV (GLuint path, GLenum pname, GLint value); +GLAPI void APIENTRY glPathParameterfvNV (GLuint path, GLenum pname, const GLfloat *value); +GLAPI void APIENTRY glPathParameterfNV (GLuint path, GLenum pname, GLfloat value); +GLAPI void APIENTRY glPathDashArrayNV (GLuint path, GLsizei dashCount, const GLfloat *dashArray); +GLAPI void APIENTRY glPathStencilFuncNV (GLenum func, GLint ref, GLuint mask); +GLAPI void APIENTRY glPathStencilDepthOffsetNV (GLfloat factor, GLfloat units); +GLAPI void APIENTRY glStencilFillPathNV (GLuint path, GLenum fillMode, GLuint mask); +GLAPI void APIENTRY glStencilStrokePathNV (GLuint path, GLint reference, GLuint mask); +GLAPI void APIENTRY glStencilFillPathInstancedNV (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum fillMode, GLuint mask, GLenum transformType, const GLfloat *transformValues); +GLAPI void APIENTRY glStencilStrokePathInstancedNV (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLint reference, GLuint mask, GLenum transformType, const GLfloat *transformValues); +GLAPI void APIENTRY glPathCoverDepthFuncNV (GLenum func); +GLAPI void APIENTRY glCoverFillPathNV (GLuint path, GLenum coverMode); +GLAPI void APIENTRY glCoverStrokePathNV (GLuint path, GLenum coverMode); +GLAPI void APIENTRY glCoverFillPathInstancedNV (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +GLAPI void APIENTRY glCoverStrokePathInstancedNV (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +GLAPI void APIENTRY glGetPathParameterivNV (GLuint path, GLenum pname, GLint *value); +GLAPI void APIENTRY glGetPathParameterfvNV (GLuint path, GLenum pname, GLfloat *value); +GLAPI void APIENTRY glGetPathCommandsNV (GLuint path, GLubyte *commands); +GLAPI void APIENTRY glGetPathCoordsNV (GLuint path, GLfloat *coords); +GLAPI void APIENTRY glGetPathDashArrayNV (GLuint path, GLfloat *dashArray); +GLAPI void APIENTRY glGetPathMetricsNV (GLbitfield metricQueryMask, GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLsizei stride, GLfloat *metrics); +GLAPI void APIENTRY glGetPathMetricRangeNV (GLbitfield metricQueryMask, GLuint firstPathName, GLsizei numPaths, GLsizei stride, GLfloat *metrics); +GLAPI void APIENTRY glGetPathSpacingNV (GLenum pathListMode, GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLfloat advanceScale, GLfloat kerningScale, GLenum transformType, GLfloat *returnedSpacing); +GLAPI GLboolean APIENTRY glIsPointInFillPathNV (GLuint path, GLuint mask, GLfloat x, GLfloat y); +GLAPI GLboolean APIENTRY glIsPointInStrokePathNV (GLuint path, GLfloat x, GLfloat y); +GLAPI GLfloat APIENTRY glGetPathLengthNV (GLuint path, GLsizei startSegment, GLsizei numSegments); +GLAPI GLboolean APIENTRY glPointAlongPathNV (GLuint path, GLsizei startSegment, GLsizei numSegments, GLfloat distance, GLfloat *x, GLfloat *y, GLfloat *tangentX, GLfloat *tangentY); +GLAPI void APIENTRY glMatrixLoad3x2fNV (GLenum matrixMode, const GLfloat *m); +GLAPI void APIENTRY glMatrixLoad3x3fNV (GLenum matrixMode, const GLfloat *m); +GLAPI void APIENTRY glMatrixLoadTranspose3x3fNV (GLenum matrixMode, const GLfloat *m); +GLAPI void APIENTRY glMatrixMult3x2fNV (GLenum matrixMode, const GLfloat *m); +GLAPI void APIENTRY glMatrixMult3x3fNV (GLenum matrixMode, const GLfloat *m); +GLAPI void APIENTRY glMatrixMultTranspose3x3fNV (GLenum matrixMode, const GLfloat *m); +GLAPI void APIENTRY glStencilThenCoverFillPathNV (GLuint path, GLenum fillMode, GLuint mask, GLenum coverMode); +GLAPI void APIENTRY glStencilThenCoverStrokePathNV (GLuint path, GLint reference, GLuint mask, GLenum coverMode); +GLAPI void APIENTRY glStencilThenCoverFillPathInstancedNV (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum fillMode, GLuint mask, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +GLAPI void APIENTRY glStencilThenCoverStrokePathInstancedNV (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLint reference, GLuint mask, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +GLAPI GLenum APIENTRY glPathGlyphIndexRangeNV (GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLuint pathParameterTemplate, GLfloat emScale, GLuint *baseAndCount); +GLAPI GLenum APIENTRY glPathGlyphIndexArrayNV (GLuint firstPathName, GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLuint firstGlyphIndex, GLsizei numGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +GLAPI GLenum APIENTRY glPathMemoryGlyphIndexArrayNV (GLuint firstPathName, GLenum fontTarget, GLsizeiptr fontSize, const void *fontData, GLsizei faceIndex, GLuint firstGlyphIndex, GLsizei numGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +GLAPI void APIENTRY glProgramPathFragmentInputGenNV (GLuint program, GLint location, GLenum genMode, GLint components, const GLfloat *coeffs); +GLAPI void APIENTRY glGetProgramResourcefvNV (GLuint program, GLenum programInterface, GLuint index, GLsizei propCount, const GLenum *props, GLsizei count, GLsizei *length, GLfloat *params); +GLAPI void APIENTRY glPathColorGenNV (GLenum color, GLenum genMode, GLenum colorFormat, const GLfloat *coeffs); +GLAPI void APIENTRY glPathTexGenNV (GLenum texCoordSet, GLenum genMode, GLint components, const GLfloat *coeffs); +GLAPI void APIENTRY glPathFogGenNV (GLenum genMode); +GLAPI void APIENTRY glGetPathColorGenivNV (GLenum color, GLenum pname, GLint *value); +GLAPI void APIENTRY glGetPathColorGenfvNV (GLenum color, GLenum pname, GLfloat *value); +GLAPI void APIENTRY glGetPathTexGenivNV (GLenum texCoordSet, GLenum pname, GLint *value); +GLAPI void APIENTRY glGetPathTexGenfvNV (GLenum texCoordSet, GLenum pname, GLfloat *value); +#endif +#endif /* GL_NV_path_rendering */ + +#ifndef GL_NV_path_rendering_shared_edge +#define GL_NV_path_rendering_shared_edge 1 +#define GL_SHARED_EDGE_NV 0xC0 +#endif /* GL_NV_path_rendering_shared_edge */ + +#ifndef GL_NV_pixel_data_range +#define GL_NV_pixel_data_range 1 +#define GL_WRITE_PIXEL_DATA_RANGE_NV 0x8878 +#define GL_READ_PIXEL_DATA_RANGE_NV 0x8879 +#define GL_WRITE_PIXEL_DATA_RANGE_LENGTH_NV 0x887A +#define GL_READ_PIXEL_DATA_RANGE_LENGTH_NV 0x887B +#define GL_WRITE_PIXEL_DATA_RANGE_POINTER_NV 0x887C +#define GL_READ_PIXEL_DATA_RANGE_POINTER_NV 0x887D +typedef void (APIENTRYP PFNGLPIXELDATARANGENVPROC) (GLenum target, GLsizei length, const void *pointer); +typedef void (APIENTRYP PFNGLFLUSHPIXELDATARANGENVPROC) (GLenum target); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPixelDataRangeNV (GLenum target, GLsizei length, const void *pointer); +GLAPI void APIENTRY glFlushPixelDataRangeNV (GLenum target); +#endif +#endif /* GL_NV_pixel_data_range */ + +#ifndef GL_NV_point_sprite +#define GL_NV_point_sprite 1 +#define GL_POINT_SPRITE_NV 0x8861 +#define GL_COORD_REPLACE_NV 0x8862 +#define GL_POINT_SPRITE_R_MODE_NV 0x8863 +typedef void (APIENTRYP PFNGLPOINTPARAMETERINVPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLPOINTPARAMETERIVNVPROC) (GLenum pname, const GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPointParameteriNV (GLenum pname, GLint param); +GLAPI void APIENTRY glPointParameterivNV (GLenum pname, const GLint *params); +#endif +#endif /* GL_NV_point_sprite */ + +#ifndef GL_NV_present_video +#define GL_NV_present_video 1 +#define GL_FRAME_NV 0x8E26 +#define GL_FIELDS_NV 0x8E27 +#define GL_CURRENT_TIME_NV 0x8E28 +#define GL_NUM_FILL_STREAMS_NV 0x8E29 +#define GL_PRESENT_TIME_NV 0x8E2A +#define GL_PRESENT_DURATION_NV 0x8E2B +typedef void (APIENTRYP PFNGLPRESENTFRAMEKEYEDNVPROC) (GLuint video_slot, GLuint64EXT minPresentTime, GLuint beginPresentTimeId, GLuint presentDurationId, GLenum type, GLenum target0, GLuint fill0, GLuint key0, GLenum target1, GLuint fill1, GLuint key1); +typedef void (APIENTRYP PFNGLPRESENTFRAMEDUALFILLNVPROC) (GLuint video_slot, GLuint64EXT minPresentTime, GLuint beginPresentTimeId, GLuint presentDurationId, GLenum type, GLenum target0, GLuint fill0, GLenum target1, GLuint fill1, GLenum target2, GLuint fill2, GLenum target3, GLuint fill3); +typedef void (APIENTRYP PFNGLGETVIDEOIVNVPROC) (GLuint video_slot, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETVIDEOUIVNVPROC) (GLuint video_slot, GLenum pname, GLuint *params); +typedef void (APIENTRYP PFNGLGETVIDEOI64VNVPROC) (GLuint video_slot, GLenum pname, GLint64EXT *params); +typedef void (APIENTRYP PFNGLGETVIDEOUI64VNVPROC) (GLuint video_slot, GLenum pname, GLuint64EXT *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPresentFrameKeyedNV (GLuint video_slot, GLuint64EXT minPresentTime, GLuint beginPresentTimeId, GLuint presentDurationId, GLenum type, GLenum target0, GLuint fill0, GLuint key0, GLenum target1, GLuint fill1, GLuint key1); +GLAPI void APIENTRY glPresentFrameDualFillNV (GLuint video_slot, GLuint64EXT minPresentTime, GLuint beginPresentTimeId, GLuint presentDurationId, GLenum type, GLenum target0, GLuint fill0, GLenum target1, GLuint fill1, GLenum target2, GLuint fill2, GLenum target3, GLuint fill3); +GLAPI void APIENTRY glGetVideoivNV (GLuint video_slot, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetVideouivNV (GLuint video_slot, GLenum pname, GLuint *params); +GLAPI void APIENTRY glGetVideoi64vNV (GLuint video_slot, GLenum pname, GLint64EXT *params); +GLAPI void APIENTRY glGetVideoui64vNV (GLuint video_slot, GLenum pname, GLuint64EXT *params); +#endif +#endif /* GL_NV_present_video */ + +#ifndef GL_NV_primitive_restart +#define GL_NV_primitive_restart 1 +#define GL_PRIMITIVE_RESTART_NV 0x8558 +#define GL_PRIMITIVE_RESTART_INDEX_NV 0x8559 +typedef void (APIENTRYP PFNGLPRIMITIVERESTARTNVPROC) (void); +typedef void (APIENTRYP PFNGLPRIMITIVERESTARTINDEXNVPROC) (GLuint index); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPrimitiveRestartNV (void); +GLAPI void APIENTRY glPrimitiveRestartIndexNV (GLuint index); +#endif +#endif /* GL_NV_primitive_restart */ + +#ifndef GL_NV_primitive_shading_rate +#define GL_NV_primitive_shading_rate 1 +#define GL_SHADING_RATE_IMAGE_PER_PRIMITIVE_NV 0x95B1 +#define GL_SHADING_RATE_IMAGE_PALETTE_COUNT_NV 0x95B2 +#endif /* GL_NV_primitive_shading_rate */ + +#ifndef GL_NV_query_resource +#define GL_NV_query_resource 1 +#define GL_QUERY_RESOURCE_TYPE_VIDMEM_ALLOC_NV 0x9540 +#define GL_QUERY_RESOURCE_MEMTYPE_VIDMEM_NV 0x9542 +#define GL_QUERY_RESOURCE_SYS_RESERVED_NV 0x9544 +#define GL_QUERY_RESOURCE_TEXTURE_NV 0x9545 +#define GL_QUERY_RESOURCE_RENDERBUFFER_NV 0x9546 +#define GL_QUERY_RESOURCE_BUFFEROBJECT_NV 0x9547 +typedef GLint (APIENTRYP PFNGLQUERYRESOURCENVPROC) (GLenum queryType, GLint tagId, GLuint count, GLint *buffer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLint APIENTRY glQueryResourceNV (GLenum queryType, GLint tagId, GLuint count, GLint *buffer); +#endif +#endif /* GL_NV_query_resource */ + +#ifndef GL_NV_query_resource_tag +#define GL_NV_query_resource_tag 1 +typedef void (APIENTRYP PFNGLGENQUERYRESOURCETAGNVPROC) (GLsizei n, GLint *tagIds); +typedef void (APIENTRYP PFNGLDELETEQUERYRESOURCETAGNVPROC) (GLsizei n, const GLint *tagIds); +typedef void (APIENTRYP PFNGLQUERYRESOURCETAGNVPROC) (GLint tagId, const GLchar *tagString); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGenQueryResourceTagNV (GLsizei n, GLint *tagIds); +GLAPI void APIENTRY glDeleteQueryResourceTagNV (GLsizei n, const GLint *tagIds); +GLAPI void APIENTRY glQueryResourceTagNV (GLint tagId, const GLchar *tagString); +#endif +#endif /* GL_NV_query_resource_tag */ + +#ifndef GL_NV_register_combiners +#define GL_NV_register_combiners 1 +#define GL_REGISTER_COMBINERS_NV 0x8522 +#define GL_VARIABLE_A_NV 0x8523 +#define GL_VARIABLE_B_NV 0x8524 +#define GL_VARIABLE_C_NV 0x8525 +#define GL_VARIABLE_D_NV 0x8526 +#define GL_VARIABLE_E_NV 0x8527 +#define GL_VARIABLE_F_NV 0x8528 +#define GL_VARIABLE_G_NV 0x8529 +#define GL_CONSTANT_COLOR0_NV 0x852A +#define GL_CONSTANT_COLOR1_NV 0x852B +#define GL_SPARE0_NV 0x852E +#define GL_SPARE1_NV 0x852F +#define GL_DISCARD_NV 0x8530 +#define GL_E_TIMES_F_NV 0x8531 +#define GL_SPARE0_PLUS_SECONDARY_COLOR_NV 0x8532 +#define GL_UNSIGNED_IDENTITY_NV 0x8536 +#define GL_UNSIGNED_INVERT_NV 0x8537 +#define GL_EXPAND_NORMAL_NV 0x8538 +#define GL_EXPAND_NEGATE_NV 0x8539 +#define GL_HALF_BIAS_NORMAL_NV 0x853A +#define GL_HALF_BIAS_NEGATE_NV 0x853B +#define GL_SIGNED_IDENTITY_NV 0x853C +#define GL_SIGNED_NEGATE_NV 0x853D +#define GL_SCALE_BY_TWO_NV 0x853E +#define GL_SCALE_BY_FOUR_NV 0x853F +#define GL_SCALE_BY_ONE_HALF_NV 0x8540 +#define GL_BIAS_BY_NEGATIVE_ONE_HALF_NV 0x8541 +#define GL_COMBINER_INPUT_NV 0x8542 +#define GL_COMBINER_MAPPING_NV 0x8543 +#define GL_COMBINER_COMPONENT_USAGE_NV 0x8544 +#define GL_COMBINER_AB_DOT_PRODUCT_NV 0x8545 +#define GL_COMBINER_CD_DOT_PRODUCT_NV 0x8546 +#define GL_COMBINER_MUX_SUM_NV 0x8547 +#define GL_COMBINER_SCALE_NV 0x8548 +#define GL_COMBINER_BIAS_NV 0x8549 +#define GL_COMBINER_AB_OUTPUT_NV 0x854A +#define GL_COMBINER_CD_OUTPUT_NV 0x854B +#define GL_COMBINER_SUM_OUTPUT_NV 0x854C +#define GL_MAX_GENERAL_COMBINERS_NV 0x854D +#define GL_NUM_GENERAL_COMBINERS_NV 0x854E +#define GL_COLOR_SUM_CLAMP_NV 0x854F +#define GL_COMBINER0_NV 0x8550 +#define GL_COMBINER1_NV 0x8551 +#define GL_COMBINER2_NV 0x8552 +#define GL_COMBINER3_NV 0x8553 +#define GL_COMBINER4_NV 0x8554 +#define GL_COMBINER5_NV 0x8555 +#define GL_COMBINER6_NV 0x8556 +#define GL_COMBINER7_NV 0x8557 +typedef void (APIENTRYP PFNGLCOMBINERPARAMETERFVNVPROC) (GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLCOMBINERPARAMETERFNVPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLCOMBINERPARAMETERIVNVPROC) (GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLCOMBINERPARAMETERINVPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLCOMBINERINPUTNVPROC) (GLenum stage, GLenum portion, GLenum variable, GLenum input, GLenum mapping, GLenum componentUsage); +typedef void (APIENTRYP PFNGLCOMBINEROUTPUTNVPROC) (GLenum stage, GLenum portion, GLenum abOutput, GLenum cdOutput, GLenum sumOutput, GLenum scale, GLenum bias, GLboolean abDotProduct, GLboolean cdDotProduct, GLboolean muxSum); +typedef void (APIENTRYP PFNGLFINALCOMBINERINPUTNVPROC) (GLenum variable, GLenum input, GLenum mapping, GLenum componentUsage); +typedef void (APIENTRYP PFNGLGETCOMBINERINPUTPARAMETERFVNVPROC) (GLenum stage, GLenum portion, GLenum variable, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETCOMBINERINPUTPARAMETERIVNVPROC) (GLenum stage, GLenum portion, GLenum variable, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETCOMBINEROUTPUTPARAMETERFVNVPROC) (GLenum stage, GLenum portion, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETCOMBINEROUTPUTPARAMETERIVNVPROC) (GLenum stage, GLenum portion, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETFINALCOMBINERINPUTPARAMETERFVNVPROC) (GLenum variable, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETFINALCOMBINERINPUTPARAMETERIVNVPROC) (GLenum variable, GLenum pname, GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCombinerParameterfvNV (GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glCombinerParameterfNV (GLenum pname, GLfloat param); +GLAPI void APIENTRY glCombinerParameterivNV (GLenum pname, const GLint *params); +GLAPI void APIENTRY glCombinerParameteriNV (GLenum pname, GLint param); +GLAPI void APIENTRY glCombinerInputNV (GLenum stage, GLenum portion, GLenum variable, GLenum input, GLenum mapping, GLenum componentUsage); +GLAPI void APIENTRY glCombinerOutputNV (GLenum stage, GLenum portion, GLenum abOutput, GLenum cdOutput, GLenum sumOutput, GLenum scale, GLenum bias, GLboolean abDotProduct, GLboolean cdDotProduct, GLboolean muxSum); +GLAPI void APIENTRY glFinalCombinerInputNV (GLenum variable, GLenum input, GLenum mapping, GLenum componentUsage); +GLAPI void APIENTRY glGetCombinerInputParameterfvNV (GLenum stage, GLenum portion, GLenum variable, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetCombinerInputParameterivNV (GLenum stage, GLenum portion, GLenum variable, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetCombinerOutputParameterfvNV (GLenum stage, GLenum portion, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetCombinerOutputParameterivNV (GLenum stage, GLenum portion, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetFinalCombinerInputParameterfvNV (GLenum variable, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetFinalCombinerInputParameterivNV (GLenum variable, GLenum pname, GLint *params); +#endif +#endif /* GL_NV_register_combiners */ + +#ifndef GL_NV_register_combiners2 +#define GL_NV_register_combiners2 1 +#define GL_PER_STAGE_CONSTANTS_NV 0x8535 +typedef void (APIENTRYP PFNGLCOMBINERSTAGEPARAMETERFVNVPROC) (GLenum stage, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLGETCOMBINERSTAGEPARAMETERFVNVPROC) (GLenum stage, GLenum pname, GLfloat *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCombinerStageParameterfvNV (GLenum stage, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glGetCombinerStageParameterfvNV (GLenum stage, GLenum pname, GLfloat *params); +#endif +#endif /* GL_NV_register_combiners2 */ + +#ifndef GL_NV_representative_fragment_test +#define GL_NV_representative_fragment_test 1 +#define GL_REPRESENTATIVE_FRAGMENT_TEST_NV 0x937F +#endif /* GL_NV_representative_fragment_test */ + +#ifndef GL_NV_robustness_video_memory_purge +#define GL_NV_robustness_video_memory_purge 1 +#define GL_PURGED_CONTEXT_RESET_NV 0x92BB +#endif /* GL_NV_robustness_video_memory_purge */ + +#ifndef GL_NV_sample_locations +#define GL_NV_sample_locations 1 +#define GL_SAMPLE_LOCATION_SUBPIXEL_BITS_NV 0x933D +#define GL_SAMPLE_LOCATION_PIXEL_GRID_WIDTH_NV 0x933E +#define GL_SAMPLE_LOCATION_PIXEL_GRID_HEIGHT_NV 0x933F +#define GL_PROGRAMMABLE_SAMPLE_LOCATION_TABLE_SIZE_NV 0x9340 +#define GL_SAMPLE_LOCATION_NV 0x8E50 +#define GL_PROGRAMMABLE_SAMPLE_LOCATION_NV 0x9341 +#define GL_FRAMEBUFFER_PROGRAMMABLE_SAMPLE_LOCATIONS_NV 0x9342 +#define GL_FRAMEBUFFER_SAMPLE_LOCATION_PIXEL_GRID_NV 0x9343 +typedef void (APIENTRYP PFNGLFRAMEBUFFERSAMPLELOCATIONSFVNVPROC) (GLenum target, GLuint start, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLNAMEDFRAMEBUFFERSAMPLELOCATIONSFVNVPROC) (GLuint framebuffer, GLuint start, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLRESOLVEDEPTHVALUESNVPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFramebufferSampleLocationsfvNV (GLenum target, GLuint start, GLsizei count, const GLfloat *v); +GLAPI void APIENTRY glNamedFramebufferSampleLocationsfvNV (GLuint framebuffer, GLuint start, GLsizei count, const GLfloat *v); +GLAPI void APIENTRY glResolveDepthValuesNV (void); +#endif +#endif /* GL_NV_sample_locations */ + +#ifndef GL_NV_sample_mask_override_coverage +#define GL_NV_sample_mask_override_coverage 1 +#endif /* GL_NV_sample_mask_override_coverage */ + +#ifndef GL_NV_scissor_exclusive +#define GL_NV_scissor_exclusive 1 +#define GL_SCISSOR_TEST_EXCLUSIVE_NV 0x9555 +#define GL_SCISSOR_BOX_EXCLUSIVE_NV 0x9556 +typedef void (APIENTRYP PFNGLSCISSOREXCLUSIVENVPROC) (GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLSCISSOREXCLUSIVEARRAYVNVPROC) (GLuint first, GLsizei count, const GLint *v); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glScissorExclusiveNV (GLint x, GLint y, GLsizei width, GLsizei height); +GLAPI void APIENTRY glScissorExclusiveArrayvNV (GLuint first, GLsizei count, const GLint *v); +#endif +#endif /* GL_NV_scissor_exclusive */ + +#ifndef GL_NV_shader_atomic_counters +#define GL_NV_shader_atomic_counters 1 +#endif /* GL_NV_shader_atomic_counters */ + +#ifndef GL_NV_shader_atomic_float +#define GL_NV_shader_atomic_float 1 +#endif /* GL_NV_shader_atomic_float */ + +#ifndef GL_NV_shader_atomic_float64 +#define GL_NV_shader_atomic_float64 1 +#endif /* GL_NV_shader_atomic_float64 */ + +#ifndef GL_NV_shader_atomic_fp16_vector +#define GL_NV_shader_atomic_fp16_vector 1 +#endif /* GL_NV_shader_atomic_fp16_vector */ + +#ifndef GL_NV_shader_atomic_int64 +#define GL_NV_shader_atomic_int64 1 +#endif /* GL_NV_shader_atomic_int64 */ + +#ifndef GL_NV_shader_buffer_load +#define GL_NV_shader_buffer_load 1 +#define GL_BUFFER_GPU_ADDRESS_NV 0x8F1D +#define GL_GPU_ADDRESS_NV 0x8F34 +#define GL_MAX_SHADER_BUFFER_ADDRESS_NV 0x8F35 +typedef void (APIENTRYP PFNGLMAKEBUFFERRESIDENTNVPROC) (GLenum target, GLenum access); +typedef void (APIENTRYP PFNGLMAKEBUFFERNONRESIDENTNVPROC) (GLenum target); +typedef GLboolean (APIENTRYP PFNGLISBUFFERRESIDENTNVPROC) (GLenum target); +typedef void (APIENTRYP PFNGLMAKENAMEDBUFFERRESIDENTNVPROC) (GLuint buffer, GLenum access); +typedef void (APIENTRYP PFNGLMAKENAMEDBUFFERNONRESIDENTNVPROC) (GLuint buffer); +typedef GLboolean (APIENTRYP PFNGLISNAMEDBUFFERRESIDENTNVPROC) (GLuint buffer); +typedef void (APIENTRYP PFNGLGETBUFFERPARAMETERUI64VNVPROC) (GLenum target, GLenum pname, GLuint64EXT *params); +typedef void (APIENTRYP PFNGLGETNAMEDBUFFERPARAMETERUI64VNVPROC) (GLuint buffer, GLenum pname, GLuint64EXT *params); +typedef void (APIENTRYP PFNGLGETINTEGERUI64VNVPROC) (GLenum value, GLuint64EXT *result); +typedef void (APIENTRYP PFNGLUNIFORMUI64NVPROC) (GLint location, GLuint64EXT value); +typedef void (APIENTRYP PFNGLUNIFORMUI64VNVPROC) (GLint location, GLsizei count, const GLuint64EXT *value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMUI64NVPROC) (GLuint program, GLint location, GLuint64EXT value); +typedef void (APIENTRYP PFNGLPROGRAMUNIFORMUI64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMakeBufferResidentNV (GLenum target, GLenum access); +GLAPI void APIENTRY glMakeBufferNonResidentNV (GLenum target); +GLAPI GLboolean APIENTRY glIsBufferResidentNV (GLenum target); +GLAPI void APIENTRY glMakeNamedBufferResidentNV (GLuint buffer, GLenum access); +GLAPI void APIENTRY glMakeNamedBufferNonResidentNV (GLuint buffer); +GLAPI GLboolean APIENTRY glIsNamedBufferResidentNV (GLuint buffer); +GLAPI void APIENTRY glGetBufferParameterui64vNV (GLenum target, GLenum pname, GLuint64EXT *params); +GLAPI void APIENTRY glGetNamedBufferParameterui64vNV (GLuint buffer, GLenum pname, GLuint64EXT *params); +GLAPI void APIENTRY glGetIntegerui64vNV (GLenum value, GLuint64EXT *result); +GLAPI void APIENTRY glUniformui64NV (GLint location, GLuint64EXT value); +GLAPI void APIENTRY glUniformui64vNV (GLint location, GLsizei count, const GLuint64EXT *value); +GLAPI void APIENTRY glProgramUniformui64NV (GLuint program, GLint location, GLuint64EXT value); +GLAPI void APIENTRY glProgramUniformui64vNV (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +#endif +#endif /* GL_NV_shader_buffer_load */ + +#ifndef GL_NV_shader_buffer_store +#define GL_NV_shader_buffer_store 1 +#define GL_SHADER_GLOBAL_ACCESS_BARRIER_BIT_NV 0x00000010 +#endif /* GL_NV_shader_buffer_store */ + +#ifndef GL_NV_shader_storage_buffer_object +#define GL_NV_shader_storage_buffer_object 1 +#endif /* GL_NV_shader_storage_buffer_object */ + +#ifndef GL_NV_shader_subgroup_partitioned +#define GL_NV_shader_subgroup_partitioned 1 +#define GL_SUBGROUP_FEATURE_PARTITIONED_BIT_NV 0x00000100 +#endif /* GL_NV_shader_subgroup_partitioned */ + +#ifndef GL_NV_shader_texture_footprint +#define GL_NV_shader_texture_footprint 1 +#endif /* GL_NV_shader_texture_footprint */ + +#ifndef GL_NV_shader_thread_group +#define GL_NV_shader_thread_group 1 +#define GL_WARP_SIZE_NV 0x9339 +#define GL_WARPS_PER_SM_NV 0x933A +#define GL_SM_COUNT_NV 0x933B +#endif /* GL_NV_shader_thread_group */ + +#ifndef GL_NV_shader_thread_shuffle +#define GL_NV_shader_thread_shuffle 1 +#endif /* GL_NV_shader_thread_shuffle */ + +#ifndef GL_NV_shading_rate_image +#define GL_NV_shading_rate_image 1 +#define GL_SHADING_RATE_IMAGE_NV 0x9563 +#define GL_SHADING_RATE_NO_INVOCATIONS_NV 0x9564 +#define GL_SHADING_RATE_1_INVOCATION_PER_PIXEL_NV 0x9565 +#define GL_SHADING_RATE_1_INVOCATION_PER_1X2_PIXELS_NV 0x9566 +#define GL_SHADING_RATE_1_INVOCATION_PER_2X1_PIXELS_NV 0x9567 +#define GL_SHADING_RATE_1_INVOCATION_PER_2X2_PIXELS_NV 0x9568 +#define GL_SHADING_RATE_1_INVOCATION_PER_2X4_PIXELS_NV 0x9569 +#define GL_SHADING_RATE_1_INVOCATION_PER_4X2_PIXELS_NV 0x956A +#define GL_SHADING_RATE_1_INVOCATION_PER_4X4_PIXELS_NV 0x956B +#define GL_SHADING_RATE_2_INVOCATIONS_PER_PIXEL_NV 0x956C +#define GL_SHADING_RATE_4_INVOCATIONS_PER_PIXEL_NV 0x956D +#define GL_SHADING_RATE_8_INVOCATIONS_PER_PIXEL_NV 0x956E +#define GL_SHADING_RATE_16_INVOCATIONS_PER_PIXEL_NV 0x956F +#define GL_SHADING_RATE_IMAGE_BINDING_NV 0x955B +#define GL_SHADING_RATE_IMAGE_TEXEL_WIDTH_NV 0x955C +#define GL_SHADING_RATE_IMAGE_TEXEL_HEIGHT_NV 0x955D +#define GL_SHADING_RATE_IMAGE_PALETTE_SIZE_NV 0x955E +#define GL_MAX_COARSE_FRAGMENT_SAMPLES_NV 0x955F +#define GL_SHADING_RATE_SAMPLE_ORDER_DEFAULT_NV 0x95AE +#define GL_SHADING_RATE_SAMPLE_ORDER_PIXEL_MAJOR_NV 0x95AF +#define GL_SHADING_RATE_SAMPLE_ORDER_SAMPLE_MAJOR_NV 0x95B0 +typedef void (APIENTRYP PFNGLBINDSHADINGRATEIMAGENVPROC) (GLuint texture); +typedef void (APIENTRYP PFNGLGETSHADINGRATEIMAGEPALETTENVPROC) (GLuint viewport, GLuint entry, GLenum *rate); +typedef void (APIENTRYP PFNGLGETSHADINGRATESAMPLELOCATIONIVNVPROC) (GLenum rate, GLuint samples, GLuint index, GLint *location); +typedef void (APIENTRYP PFNGLSHADINGRATEIMAGEBARRIERNVPROC) (GLboolean synchronize); +typedef void (APIENTRYP PFNGLSHADINGRATEIMAGEPALETTENVPROC) (GLuint viewport, GLuint first, GLsizei count, const GLenum *rates); +typedef void (APIENTRYP PFNGLSHADINGRATESAMPLEORDERNVPROC) (GLenum order); +typedef void (APIENTRYP PFNGLSHADINGRATESAMPLEORDERCUSTOMNVPROC) (GLenum rate, GLuint samples, const GLint *locations); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBindShadingRateImageNV (GLuint texture); +GLAPI void APIENTRY glGetShadingRateImagePaletteNV (GLuint viewport, GLuint entry, GLenum *rate); +GLAPI void APIENTRY glGetShadingRateSampleLocationivNV (GLenum rate, GLuint samples, GLuint index, GLint *location); +GLAPI void APIENTRY glShadingRateImageBarrierNV (GLboolean synchronize); +GLAPI void APIENTRY glShadingRateImagePaletteNV (GLuint viewport, GLuint first, GLsizei count, const GLenum *rates); +GLAPI void APIENTRY glShadingRateSampleOrderNV (GLenum order); +GLAPI void APIENTRY glShadingRateSampleOrderCustomNV (GLenum rate, GLuint samples, const GLint *locations); +#endif +#endif /* GL_NV_shading_rate_image */ + +#ifndef GL_NV_stereo_view_rendering +#define GL_NV_stereo_view_rendering 1 +#endif /* GL_NV_stereo_view_rendering */ + +#ifndef GL_NV_tessellation_program5 +#define GL_NV_tessellation_program5 1 +#define GL_MAX_PROGRAM_PATCH_ATTRIBS_NV 0x86D8 +#define GL_TESS_CONTROL_PROGRAM_NV 0x891E +#define GL_TESS_EVALUATION_PROGRAM_NV 0x891F +#define GL_TESS_CONTROL_PROGRAM_PARAMETER_BUFFER_NV 0x8C74 +#define GL_TESS_EVALUATION_PROGRAM_PARAMETER_BUFFER_NV 0x8C75 +#endif /* GL_NV_tessellation_program5 */ + +#ifndef GL_NV_texgen_emboss +#define GL_NV_texgen_emboss 1 +#define GL_EMBOSS_LIGHT_NV 0x855D +#define GL_EMBOSS_CONSTANT_NV 0x855E +#define GL_EMBOSS_MAP_NV 0x855F +#endif /* GL_NV_texgen_emboss */ + +#ifndef GL_NV_texgen_reflection +#define GL_NV_texgen_reflection 1 +#define GL_NORMAL_MAP_NV 0x8511 +#define GL_REFLECTION_MAP_NV 0x8512 +#endif /* GL_NV_texgen_reflection */ + +#ifndef GL_NV_texture_barrier +#define GL_NV_texture_barrier 1 +typedef void (APIENTRYP PFNGLTEXTUREBARRIERNVPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTextureBarrierNV (void); +#endif +#endif /* GL_NV_texture_barrier */ + +#ifndef GL_NV_texture_compression_vtc +#define GL_NV_texture_compression_vtc 1 +#endif /* GL_NV_texture_compression_vtc */ + +#ifndef GL_NV_texture_env_combine4 +#define GL_NV_texture_env_combine4 1 +#define GL_COMBINE4_NV 0x8503 +#define GL_SOURCE3_RGB_NV 0x8583 +#define GL_SOURCE3_ALPHA_NV 0x858B +#define GL_OPERAND3_RGB_NV 0x8593 +#define GL_OPERAND3_ALPHA_NV 0x859B +#endif /* GL_NV_texture_env_combine4 */ + +#ifndef GL_NV_texture_expand_normal +#define GL_NV_texture_expand_normal 1 +#define GL_TEXTURE_UNSIGNED_REMAP_MODE_NV 0x888F +#endif /* GL_NV_texture_expand_normal */ + +#ifndef GL_NV_texture_multisample +#define GL_NV_texture_multisample 1 +#define GL_TEXTURE_COVERAGE_SAMPLES_NV 0x9045 +#define GL_TEXTURE_COLOR_SAMPLES_NV 0x9046 +typedef void (APIENTRYP PFNGLTEXIMAGE2DMULTISAMPLECOVERAGENVPROC) (GLenum target, GLsizei coverageSamples, GLsizei colorSamples, GLint internalFormat, GLsizei width, GLsizei height, GLboolean fixedSampleLocations); +typedef void (APIENTRYP PFNGLTEXIMAGE3DMULTISAMPLECOVERAGENVPROC) (GLenum target, GLsizei coverageSamples, GLsizei colorSamples, GLint internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedSampleLocations); +typedef void (APIENTRYP PFNGLTEXTUREIMAGE2DMULTISAMPLENVPROC) (GLuint texture, GLenum target, GLsizei samples, GLint internalFormat, GLsizei width, GLsizei height, GLboolean fixedSampleLocations); +typedef void (APIENTRYP PFNGLTEXTUREIMAGE3DMULTISAMPLENVPROC) (GLuint texture, GLenum target, GLsizei samples, GLint internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedSampleLocations); +typedef void (APIENTRYP PFNGLTEXTUREIMAGE2DMULTISAMPLECOVERAGENVPROC) (GLuint texture, GLenum target, GLsizei coverageSamples, GLsizei colorSamples, GLint internalFormat, GLsizei width, GLsizei height, GLboolean fixedSampleLocations); +typedef void (APIENTRYP PFNGLTEXTUREIMAGE3DMULTISAMPLECOVERAGENVPROC) (GLuint texture, GLenum target, GLsizei coverageSamples, GLsizei colorSamples, GLint internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedSampleLocations); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTexImage2DMultisampleCoverageNV (GLenum target, GLsizei coverageSamples, GLsizei colorSamples, GLint internalFormat, GLsizei width, GLsizei height, GLboolean fixedSampleLocations); +GLAPI void APIENTRY glTexImage3DMultisampleCoverageNV (GLenum target, GLsizei coverageSamples, GLsizei colorSamples, GLint internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedSampleLocations); +GLAPI void APIENTRY glTextureImage2DMultisampleNV (GLuint texture, GLenum target, GLsizei samples, GLint internalFormat, GLsizei width, GLsizei height, GLboolean fixedSampleLocations); +GLAPI void APIENTRY glTextureImage3DMultisampleNV (GLuint texture, GLenum target, GLsizei samples, GLint internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedSampleLocations); +GLAPI void APIENTRY glTextureImage2DMultisampleCoverageNV (GLuint texture, GLenum target, GLsizei coverageSamples, GLsizei colorSamples, GLint internalFormat, GLsizei width, GLsizei height, GLboolean fixedSampleLocations); +GLAPI void APIENTRY glTextureImage3DMultisampleCoverageNV (GLuint texture, GLenum target, GLsizei coverageSamples, GLsizei colorSamples, GLint internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedSampleLocations); +#endif +#endif /* GL_NV_texture_multisample */ + +#ifndef GL_NV_texture_rectangle +#define GL_NV_texture_rectangle 1 +#define GL_TEXTURE_RECTANGLE_NV 0x84F5 +#define GL_TEXTURE_BINDING_RECTANGLE_NV 0x84F6 +#define GL_PROXY_TEXTURE_RECTANGLE_NV 0x84F7 +#define GL_MAX_RECTANGLE_TEXTURE_SIZE_NV 0x84F8 +#endif /* GL_NV_texture_rectangle */ + +#ifndef GL_NV_texture_rectangle_compressed +#define GL_NV_texture_rectangle_compressed 1 +#endif /* GL_NV_texture_rectangle_compressed */ + +#ifndef GL_NV_texture_shader +#define GL_NV_texture_shader 1 +#define GL_OFFSET_TEXTURE_RECTANGLE_NV 0x864C +#define GL_OFFSET_TEXTURE_RECTANGLE_SCALE_NV 0x864D +#define GL_DOT_PRODUCT_TEXTURE_RECTANGLE_NV 0x864E +#define GL_RGBA_UNSIGNED_DOT_PRODUCT_MAPPING_NV 0x86D9 +#define GL_UNSIGNED_INT_S8_S8_8_8_NV 0x86DA +#define GL_UNSIGNED_INT_8_8_S8_S8_REV_NV 0x86DB +#define GL_DSDT_MAG_INTENSITY_NV 0x86DC +#define GL_SHADER_CONSISTENT_NV 0x86DD +#define GL_TEXTURE_SHADER_NV 0x86DE +#define GL_SHADER_OPERATION_NV 0x86DF +#define GL_CULL_MODES_NV 0x86E0 +#define GL_OFFSET_TEXTURE_MATRIX_NV 0x86E1 +#define GL_OFFSET_TEXTURE_SCALE_NV 0x86E2 +#define GL_OFFSET_TEXTURE_BIAS_NV 0x86E3 +#define GL_OFFSET_TEXTURE_2D_MATRIX_NV 0x86E1 +#define GL_OFFSET_TEXTURE_2D_SCALE_NV 0x86E2 +#define GL_OFFSET_TEXTURE_2D_BIAS_NV 0x86E3 +#define GL_PREVIOUS_TEXTURE_INPUT_NV 0x86E4 +#define GL_CONST_EYE_NV 0x86E5 +#define GL_PASS_THROUGH_NV 0x86E6 +#define GL_CULL_FRAGMENT_NV 0x86E7 +#define GL_OFFSET_TEXTURE_2D_NV 0x86E8 +#define GL_DEPENDENT_AR_TEXTURE_2D_NV 0x86E9 +#define GL_DEPENDENT_GB_TEXTURE_2D_NV 0x86EA +#define GL_DOT_PRODUCT_NV 0x86EC +#define GL_DOT_PRODUCT_DEPTH_REPLACE_NV 0x86ED +#define GL_DOT_PRODUCT_TEXTURE_2D_NV 0x86EE +#define GL_DOT_PRODUCT_TEXTURE_CUBE_MAP_NV 0x86F0 +#define GL_DOT_PRODUCT_DIFFUSE_CUBE_MAP_NV 0x86F1 +#define GL_DOT_PRODUCT_REFLECT_CUBE_MAP_NV 0x86F2 +#define GL_DOT_PRODUCT_CONST_EYE_REFLECT_CUBE_MAP_NV 0x86F3 +#define GL_HILO_NV 0x86F4 +#define GL_DSDT_NV 0x86F5 +#define GL_DSDT_MAG_NV 0x86F6 +#define GL_DSDT_MAG_VIB_NV 0x86F7 +#define GL_HILO16_NV 0x86F8 +#define GL_SIGNED_HILO_NV 0x86F9 +#define GL_SIGNED_HILO16_NV 0x86FA +#define GL_SIGNED_RGBA_NV 0x86FB +#define GL_SIGNED_RGBA8_NV 0x86FC +#define GL_SIGNED_RGB_NV 0x86FE +#define GL_SIGNED_RGB8_NV 0x86FF +#define GL_SIGNED_LUMINANCE_NV 0x8701 +#define GL_SIGNED_LUMINANCE8_NV 0x8702 +#define GL_SIGNED_LUMINANCE_ALPHA_NV 0x8703 +#define GL_SIGNED_LUMINANCE8_ALPHA8_NV 0x8704 +#define GL_SIGNED_ALPHA_NV 0x8705 +#define GL_SIGNED_ALPHA8_NV 0x8706 +#define GL_SIGNED_INTENSITY_NV 0x8707 +#define GL_SIGNED_INTENSITY8_NV 0x8708 +#define GL_DSDT8_NV 0x8709 +#define GL_DSDT8_MAG8_NV 0x870A +#define GL_DSDT8_MAG8_INTENSITY8_NV 0x870B +#define GL_SIGNED_RGB_UNSIGNED_ALPHA_NV 0x870C +#define GL_SIGNED_RGB8_UNSIGNED_ALPHA8_NV 0x870D +#define GL_HI_SCALE_NV 0x870E +#define GL_LO_SCALE_NV 0x870F +#define GL_DS_SCALE_NV 0x8710 +#define GL_DT_SCALE_NV 0x8711 +#define GL_MAGNITUDE_SCALE_NV 0x8712 +#define GL_VIBRANCE_SCALE_NV 0x8713 +#define GL_HI_BIAS_NV 0x8714 +#define GL_LO_BIAS_NV 0x8715 +#define GL_DS_BIAS_NV 0x8716 +#define GL_DT_BIAS_NV 0x8717 +#define GL_MAGNITUDE_BIAS_NV 0x8718 +#define GL_VIBRANCE_BIAS_NV 0x8719 +#define GL_TEXTURE_BORDER_VALUES_NV 0x871A +#define GL_TEXTURE_HI_SIZE_NV 0x871B +#define GL_TEXTURE_LO_SIZE_NV 0x871C +#define GL_TEXTURE_DS_SIZE_NV 0x871D +#define GL_TEXTURE_DT_SIZE_NV 0x871E +#define GL_TEXTURE_MAG_SIZE_NV 0x871F +#endif /* GL_NV_texture_shader */ + +#ifndef GL_NV_texture_shader2 +#define GL_NV_texture_shader2 1 +#define GL_DOT_PRODUCT_TEXTURE_3D_NV 0x86EF +#endif /* GL_NV_texture_shader2 */ + +#ifndef GL_NV_texture_shader3 +#define GL_NV_texture_shader3 1 +#define GL_OFFSET_PROJECTIVE_TEXTURE_2D_NV 0x8850 +#define GL_OFFSET_PROJECTIVE_TEXTURE_2D_SCALE_NV 0x8851 +#define GL_OFFSET_PROJECTIVE_TEXTURE_RECTANGLE_NV 0x8852 +#define GL_OFFSET_PROJECTIVE_TEXTURE_RECTANGLE_SCALE_NV 0x8853 +#define GL_OFFSET_HILO_TEXTURE_2D_NV 0x8854 +#define GL_OFFSET_HILO_TEXTURE_RECTANGLE_NV 0x8855 +#define GL_OFFSET_HILO_PROJECTIVE_TEXTURE_2D_NV 0x8856 +#define GL_OFFSET_HILO_PROJECTIVE_TEXTURE_RECTANGLE_NV 0x8857 +#define GL_DEPENDENT_HILO_TEXTURE_2D_NV 0x8858 +#define GL_DEPENDENT_RGB_TEXTURE_3D_NV 0x8859 +#define GL_DEPENDENT_RGB_TEXTURE_CUBE_MAP_NV 0x885A +#define GL_DOT_PRODUCT_PASS_THROUGH_NV 0x885B +#define GL_DOT_PRODUCT_TEXTURE_1D_NV 0x885C +#define GL_DOT_PRODUCT_AFFINE_DEPTH_REPLACE_NV 0x885D +#define GL_HILO8_NV 0x885E +#define GL_SIGNED_HILO8_NV 0x885F +#define GL_FORCE_BLUE_TO_ONE_NV 0x8860 +#endif /* GL_NV_texture_shader3 */ + +#ifndef GL_NV_timeline_semaphore +#define GL_NV_timeline_semaphore 1 +#define GL_TIMELINE_SEMAPHORE_VALUE_NV 0x9595 +#define GL_SEMAPHORE_TYPE_NV 0x95B3 +#define GL_SEMAPHORE_TYPE_BINARY_NV 0x95B4 +#define GL_SEMAPHORE_TYPE_TIMELINE_NV 0x95B5 +#define GL_MAX_TIMELINE_SEMAPHORE_VALUE_DIFFERENCE_NV 0x95B6 +typedef void (APIENTRYP PFNGLCREATESEMAPHORESNVPROC) (GLsizei n, GLuint *semaphores); +typedef void (APIENTRYP PFNGLSEMAPHOREPARAMETERIVNVPROC) (GLuint semaphore, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLGETSEMAPHOREPARAMETERIVNVPROC) (GLuint semaphore, GLenum pname, GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCreateSemaphoresNV (GLsizei n, GLuint *semaphores); +GLAPI void APIENTRY glSemaphoreParameterivNV (GLuint semaphore, GLenum pname, const GLint *params); +GLAPI void APIENTRY glGetSemaphoreParameterivNV (GLuint semaphore, GLenum pname, GLint *params); +#endif +#endif /* GL_NV_timeline_semaphore */ + +#ifndef GL_NV_transform_feedback +#define GL_NV_transform_feedback 1 +#define GL_BACK_PRIMARY_COLOR_NV 0x8C77 +#define GL_BACK_SECONDARY_COLOR_NV 0x8C78 +#define GL_TEXTURE_COORD_NV 0x8C79 +#define GL_CLIP_DISTANCE_NV 0x8C7A +#define GL_VERTEX_ID_NV 0x8C7B +#define GL_PRIMITIVE_ID_NV 0x8C7C +#define GL_GENERIC_ATTRIB_NV 0x8C7D +#define GL_TRANSFORM_FEEDBACK_ATTRIBS_NV 0x8C7E +#define GL_TRANSFORM_FEEDBACK_BUFFER_MODE_NV 0x8C7F +#define GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_COMPONENTS_NV 0x8C80 +#define GL_ACTIVE_VARYINGS_NV 0x8C81 +#define GL_ACTIVE_VARYING_MAX_LENGTH_NV 0x8C82 +#define GL_TRANSFORM_FEEDBACK_VARYINGS_NV 0x8C83 +#define GL_TRANSFORM_FEEDBACK_BUFFER_START_NV 0x8C84 +#define GL_TRANSFORM_FEEDBACK_BUFFER_SIZE_NV 0x8C85 +#define GL_TRANSFORM_FEEDBACK_RECORD_NV 0x8C86 +#define GL_PRIMITIVES_GENERATED_NV 0x8C87 +#define GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN_NV 0x8C88 +#define GL_RASTERIZER_DISCARD_NV 0x8C89 +#define GL_MAX_TRANSFORM_FEEDBACK_INTERLEAVED_COMPONENTS_NV 0x8C8A +#define GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_ATTRIBS_NV 0x8C8B +#define GL_INTERLEAVED_ATTRIBS_NV 0x8C8C +#define GL_SEPARATE_ATTRIBS_NV 0x8C8D +#define GL_TRANSFORM_FEEDBACK_BUFFER_NV 0x8C8E +#define GL_TRANSFORM_FEEDBACK_BUFFER_BINDING_NV 0x8C8F +#define GL_LAYER_NV 0x8DAA +#define GL_NEXT_BUFFER_NV -2 +#define GL_SKIP_COMPONENTS4_NV -3 +#define GL_SKIP_COMPONENTS3_NV -4 +#define GL_SKIP_COMPONENTS2_NV -5 +#define GL_SKIP_COMPONENTS1_NV -6 +typedef void (APIENTRYP PFNGLBEGINTRANSFORMFEEDBACKNVPROC) (GLenum primitiveMode); +typedef void (APIENTRYP PFNGLENDTRANSFORMFEEDBACKNVPROC) (void); +typedef void (APIENTRYP PFNGLTRANSFORMFEEDBACKATTRIBSNVPROC) (GLsizei count, const GLint *attribs, GLenum bufferMode); +typedef void (APIENTRYP PFNGLBINDBUFFERRANGENVPROC) (GLenum target, GLuint index, GLuint buffer, GLintptr offset, GLsizeiptr size); +typedef void (APIENTRYP PFNGLBINDBUFFEROFFSETNVPROC) (GLenum target, GLuint index, GLuint buffer, GLintptr offset); +typedef void (APIENTRYP PFNGLBINDBUFFERBASENVPROC) (GLenum target, GLuint index, GLuint buffer); +typedef void (APIENTRYP PFNGLTRANSFORMFEEDBACKVARYINGSNVPROC) (GLuint program, GLsizei count, const GLint *locations, GLenum bufferMode); +typedef void (APIENTRYP PFNGLACTIVEVARYINGNVPROC) (GLuint program, const GLchar *name); +typedef GLint (APIENTRYP PFNGLGETVARYINGLOCATIONNVPROC) (GLuint program, const GLchar *name); +typedef void (APIENTRYP PFNGLGETACTIVEVARYINGNVPROC) (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLsizei *size, GLenum *type, GLchar *name); +typedef void (APIENTRYP PFNGLGETTRANSFORMFEEDBACKVARYINGNVPROC) (GLuint program, GLuint index, GLint *location); +typedef void (APIENTRYP PFNGLTRANSFORMFEEDBACKSTREAMATTRIBSNVPROC) (GLsizei count, const GLint *attribs, GLsizei nbuffers, const GLint *bufstreams, GLenum bufferMode); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBeginTransformFeedbackNV (GLenum primitiveMode); +GLAPI void APIENTRY glEndTransformFeedbackNV (void); +GLAPI void APIENTRY glTransformFeedbackAttribsNV (GLsizei count, const GLint *attribs, GLenum bufferMode); +GLAPI void APIENTRY glBindBufferRangeNV (GLenum target, GLuint index, GLuint buffer, GLintptr offset, GLsizeiptr size); +GLAPI void APIENTRY glBindBufferOffsetNV (GLenum target, GLuint index, GLuint buffer, GLintptr offset); +GLAPI void APIENTRY glBindBufferBaseNV (GLenum target, GLuint index, GLuint buffer); +GLAPI void APIENTRY glTransformFeedbackVaryingsNV (GLuint program, GLsizei count, const GLint *locations, GLenum bufferMode); +GLAPI void APIENTRY glActiveVaryingNV (GLuint program, const GLchar *name); +GLAPI GLint APIENTRY glGetVaryingLocationNV (GLuint program, const GLchar *name); +GLAPI void APIENTRY glGetActiveVaryingNV (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLsizei *size, GLenum *type, GLchar *name); +GLAPI void APIENTRY glGetTransformFeedbackVaryingNV (GLuint program, GLuint index, GLint *location); +GLAPI void APIENTRY glTransformFeedbackStreamAttribsNV (GLsizei count, const GLint *attribs, GLsizei nbuffers, const GLint *bufstreams, GLenum bufferMode); +#endif +#endif /* GL_NV_transform_feedback */ + +#ifndef GL_NV_transform_feedback2 +#define GL_NV_transform_feedback2 1 +#define GL_TRANSFORM_FEEDBACK_NV 0x8E22 +#define GL_TRANSFORM_FEEDBACK_BUFFER_PAUSED_NV 0x8E23 +#define GL_TRANSFORM_FEEDBACK_BUFFER_ACTIVE_NV 0x8E24 +#define GL_TRANSFORM_FEEDBACK_BINDING_NV 0x8E25 +typedef void (APIENTRYP PFNGLBINDTRANSFORMFEEDBACKNVPROC) (GLenum target, GLuint id); +typedef void (APIENTRYP PFNGLDELETETRANSFORMFEEDBACKSNVPROC) (GLsizei n, const GLuint *ids); +typedef void (APIENTRYP PFNGLGENTRANSFORMFEEDBACKSNVPROC) (GLsizei n, GLuint *ids); +typedef GLboolean (APIENTRYP PFNGLISTRANSFORMFEEDBACKNVPROC) (GLuint id); +typedef void (APIENTRYP PFNGLPAUSETRANSFORMFEEDBACKNVPROC) (void); +typedef void (APIENTRYP PFNGLRESUMETRANSFORMFEEDBACKNVPROC) (void); +typedef void (APIENTRYP PFNGLDRAWTRANSFORMFEEDBACKNVPROC) (GLenum mode, GLuint id); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBindTransformFeedbackNV (GLenum target, GLuint id); +GLAPI void APIENTRY glDeleteTransformFeedbacksNV (GLsizei n, const GLuint *ids); +GLAPI void APIENTRY glGenTransformFeedbacksNV (GLsizei n, GLuint *ids); +GLAPI GLboolean APIENTRY glIsTransformFeedbackNV (GLuint id); +GLAPI void APIENTRY glPauseTransformFeedbackNV (void); +GLAPI void APIENTRY glResumeTransformFeedbackNV (void); +GLAPI void APIENTRY glDrawTransformFeedbackNV (GLenum mode, GLuint id); +#endif +#endif /* GL_NV_transform_feedback2 */ + +#ifndef GL_NV_uniform_buffer_unified_memory +#define GL_NV_uniform_buffer_unified_memory 1 +#define GL_UNIFORM_BUFFER_UNIFIED_NV 0x936E +#define GL_UNIFORM_BUFFER_ADDRESS_NV 0x936F +#define GL_UNIFORM_BUFFER_LENGTH_NV 0x9370 +#endif /* GL_NV_uniform_buffer_unified_memory */ + +#ifndef GL_NV_vdpau_interop +#define GL_NV_vdpau_interop 1 +typedef GLintptr GLvdpauSurfaceNV; +#define GL_SURFACE_STATE_NV 0x86EB +#define GL_SURFACE_REGISTERED_NV 0x86FD +#define GL_SURFACE_MAPPED_NV 0x8700 +#define GL_WRITE_DISCARD_NV 0x88BE +typedef void (APIENTRYP PFNGLVDPAUINITNVPROC) (const void *vdpDevice, const void *getProcAddress); +typedef void (APIENTRYP PFNGLVDPAUFININVPROC) (void); +typedef GLvdpauSurfaceNV (APIENTRYP PFNGLVDPAUREGISTERVIDEOSURFACENVPROC) (const void *vdpSurface, GLenum target, GLsizei numTextureNames, const GLuint *textureNames); +typedef GLvdpauSurfaceNV (APIENTRYP PFNGLVDPAUREGISTEROUTPUTSURFACENVPROC) (const void *vdpSurface, GLenum target, GLsizei numTextureNames, const GLuint *textureNames); +typedef GLboolean (APIENTRYP PFNGLVDPAUISSURFACENVPROC) (GLvdpauSurfaceNV surface); +typedef void (APIENTRYP PFNGLVDPAUUNREGISTERSURFACENVPROC) (GLvdpauSurfaceNV surface); +typedef void (APIENTRYP PFNGLVDPAUGETSURFACEIVNVPROC) (GLvdpauSurfaceNV surface, GLenum pname, GLsizei count, GLsizei *length, GLint *values); +typedef void (APIENTRYP PFNGLVDPAUSURFACEACCESSNVPROC) (GLvdpauSurfaceNV surface, GLenum access); +typedef void (APIENTRYP PFNGLVDPAUMAPSURFACESNVPROC) (GLsizei numSurfaces, const GLvdpauSurfaceNV *surfaces); +typedef void (APIENTRYP PFNGLVDPAUUNMAPSURFACESNVPROC) (GLsizei numSurface, const GLvdpauSurfaceNV *surfaces); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVDPAUInitNV (const void *vdpDevice, const void *getProcAddress); +GLAPI void APIENTRY glVDPAUFiniNV (void); +GLAPI GLvdpauSurfaceNV APIENTRY glVDPAURegisterVideoSurfaceNV (const void *vdpSurface, GLenum target, GLsizei numTextureNames, const GLuint *textureNames); +GLAPI GLvdpauSurfaceNV APIENTRY glVDPAURegisterOutputSurfaceNV (const void *vdpSurface, GLenum target, GLsizei numTextureNames, const GLuint *textureNames); +GLAPI GLboolean APIENTRY glVDPAUIsSurfaceNV (GLvdpauSurfaceNV surface); +GLAPI void APIENTRY glVDPAUUnregisterSurfaceNV (GLvdpauSurfaceNV surface); +GLAPI void APIENTRY glVDPAUGetSurfaceivNV (GLvdpauSurfaceNV surface, GLenum pname, GLsizei count, GLsizei *length, GLint *values); +GLAPI void APIENTRY glVDPAUSurfaceAccessNV (GLvdpauSurfaceNV surface, GLenum access); +GLAPI void APIENTRY glVDPAUMapSurfacesNV (GLsizei numSurfaces, const GLvdpauSurfaceNV *surfaces); +GLAPI void APIENTRY glVDPAUUnmapSurfacesNV (GLsizei numSurface, const GLvdpauSurfaceNV *surfaces); +#endif +#endif /* GL_NV_vdpau_interop */ + +#ifndef GL_NV_vdpau_interop2 +#define GL_NV_vdpau_interop2 1 +typedef GLvdpauSurfaceNV (APIENTRYP PFNGLVDPAUREGISTERVIDEOSURFACEWITHPICTURESTRUCTURENVPROC) (const void *vdpSurface, GLenum target, GLsizei numTextureNames, const GLuint *textureNames, GLboolean isFrameStructure); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLvdpauSurfaceNV APIENTRY glVDPAURegisterVideoSurfaceWithPictureStructureNV (const void *vdpSurface, GLenum target, GLsizei numTextureNames, const GLuint *textureNames, GLboolean isFrameStructure); +#endif +#endif /* GL_NV_vdpau_interop2 */ + +#ifndef GL_NV_vertex_array_range +#define GL_NV_vertex_array_range 1 +#define GL_VERTEX_ARRAY_RANGE_NV 0x851D +#define GL_VERTEX_ARRAY_RANGE_LENGTH_NV 0x851E +#define GL_VERTEX_ARRAY_RANGE_VALID_NV 0x851F +#define GL_MAX_VERTEX_ARRAY_RANGE_ELEMENT_NV 0x8520 +#define GL_VERTEX_ARRAY_RANGE_POINTER_NV 0x8521 +typedef void (APIENTRYP PFNGLFLUSHVERTEXARRAYRANGENVPROC) (void); +typedef void (APIENTRYP PFNGLVERTEXARRAYRANGENVPROC) (GLsizei length, const void *pointer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFlushVertexArrayRangeNV (void); +GLAPI void APIENTRY glVertexArrayRangeNV (GLsizei length, const void *pointer); +#endif +#endif /* GL_NV_vertex_array_range */ + +#ifndef GL_NV_vertex_array_range2 +#define GL_NV_vertex_array_range2 1 +#define GL_VERTEX_ARRAY_RANGE_WITHOUT_FLUSH_NV 0x8533 +#endif /* GL_NV_vertex_array_range2 */ + +#ifndef GL_NV_vertex_attrib_integer_64bit +#define GL_NV_vertex_attrib_integer_64bit 1 +typedef void (APIENTRYP PFNGLVERTEXATTRIBL1I64NVPROC) (GLuint index, GLint64EXT x); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL2I64NVPROC) (GLuint index, GLint64EXT x, GLint64EXT y); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL3I64NVPROC) (GLuint index, GLint64EXT x, GLint64EXT y, GLint64EXT z); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL4I64NVPROC) (GLuint index, GLint64EXT x, GLint64EXT y, GLint64EXT z, GLint64EXT w); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL1I64VNVPROC) (GLuint index, const GLint64EXT *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL2I64VNVPROC) (GLuint index, const GLint64EXT *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL3I64VNVPROC) (GLuint index, const GLint64EXT *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL4I64VNVPROC) (GLuint index, const GLint64EXT *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL1UI64NVPROC) (GLuint index, GLuint64EXT x); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL2UI64NVPROC) (GLuint index, GLuint64EXT x, GLuint64EXT y); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL3UI64NVPROC) (GLuint index, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL4UI64NVPROC) (GLuint index, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z, GLuint64EXT w); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL1UI64VNVPROC) (GLuint index, const GLuint64EXT *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL2UI64VNVPROC) (GLuint index, const GLuint64EXT *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL3UI64VNVPROC) (GLuint index, const GLuint64EXT *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBL4UI64VNVPROC) (GLuint index, const GLuint64EXT *v); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBLI64VNVPROC) (GLuint index, GLenum pname, GLint64EXT *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBLUI64VNVPROC) (GLuint index, GLenum pname, GLuint64EXT *params); +typedef void (APIENTRYP PFNGLVERTEXATTRIBLFORMATNVPROC) (GLuint index, GLint size, GLenum type, GLsizei stride); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexAttribL1i64NV (GLuint index, GLint64EXT x); +GLAPI void APIENTRY glVertexAttribL2i64NV (GLuint index, GLint64EXT x, GLint64EXT y); +GLAPI void APIENTRY glVertexAttribL3i64NV (GLuint index, GLint64EXT x, GLint64EXT y, GLint64EXT z); +GLAPI void APIENTRY glVertexAttribL4i64NV (GLuint index, GLint64EXT x, GLint64EXT y, GLint64EXT z, GLint64EXT w); +GLAPI void APIENTRY glVertexAttribL1i64vNV (GLuint index, const GLint64EXT *v); +GLAPI void APIENTRY glVertexAttribL2i64vNV (GLuint index, const GLint64EXT *v); +GLAPI void APIENTRY glVertexAttribL3i64vNV (GLuint index, const GLint64EXT *v); +GLAPI void APIENTRY glVertexAttribL4i64vNV (GLuint index, const GLint64EXT *v); +GLAPI void APIENTRY glVertexAttribL1ui64NV (GLuint index, GLuint64EXT x); +GLAPI void APIENTRY glVertexAttribL2ui64NV (GLuint index, GLuint64EXT x, GLuint64EXT y); +GLAPI void APIENTRY glVertexAttribL3ui64NV (GLuint index, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z); +GLAPI void APIENTRY glVertexAttribL4ui64NV (GLuint index, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z, GLuint64EXT w); +GLAPI void APIENTRY glVertexAttribL1ui64vNV (GLuint index, const GLuint64EXT *v); +GLAPI void APIENTRY glVertexAttribL2ui64vNV (GLuint index, const GLuint64EXT *v); +GLAPI void APIENTRY glVertexAttribL3ui64vNV (GLuint index, const GLuint64EXT *v); +GLAPI void APIENTRY glVertexAttribL4ui64vNV (GLuint index, const GLuint64EXT *v); +GLAPI void APIENTRY glGetVertexAttribLi64vNV (GLuint index, GLenum pname, GLint64EXT *params); +GLAPI void APIENTRY glGetVertexAttribLui64vNV (GLuint index, GLenum pname, GLuint64EXT *params); +GLAPI void APIENTRY glVertexAttribLFormatNV (GLuint index, GLint size, GLenum type, GLsizei stride); +#endif +#endif /* GL_NV_vertex_attrib_integer_64bit */ + +#ifndef GL_NV_vertex_buffer_unified_memory +#define GL_NV_vertex_buffer_unified_memory 1 +#define GL_VERTEX_ATTRIB_ARRAY_UNIFIED_NV 0x8F1E +#define GL_ELEMENT_ARRAY_UNIFIED_NV 0x8F1F +#define GL_VERTEX_ATTRIB_ARRAY_ADDRESS_NV 0x8F20 +#define GL_VERTEX_ARRAY_ADDRESS_NV 0x8F21 +#define GL_NORMAL_ARRAY_ADDRESS_NV 0x8F22 +#define GL_COLOR_ARRAY_ADDRESS_NV 0x8F23 +#define GL_INDEX_ARRAY_ADDRESS_NV 0x8F24 +#define GL_TEXTURE_COORD_ARRAY_ADDRESS_NV 0x8F25 +#define GL_EDGE_FLAG_ARRAY_ADDRESS_NV 0x8F26 +#define GL_SECONDARY_COLOR_ARRAY_ADDRESS_NV 0x8F27 +#define GL_FOG_COORD_ARRAY_ADDRESS_NV 0x8F28 +#define GL_ELEMENT_ARRAY_ADDRESS_NV 0x8F29 +#define GL_VERTEX_ATTRIB_ARRAY_LENGTH_NV 0x8F2A +#define GL_VERTEX_ARRAY_LENGTH_NV 0x8F2B +#define GL_NORMAL_ARRAY_LENGTH_NV 0x8F2C +#define GL_COLOR_ARRAY_LENGTH_NV 0x8F2D +#define GL_INDEX_ARRAY_LENGTH_NV 0x8F2E +#define GL_TEXTURE_COORD_ARRAY_LENGTH_NV 0x8F2F +#define GL_EDGE_FLAG_ARRAY_LENGTH_NV 0x8F30 +#define GL_SECONDARY_COLOR_ARRAY_LENGTH_NV 0x8F31 +#define GL_FOG_COORD_ARRAY_LENGTH_NV 0x8F32 +#define GL_ELEMENT_ARRAY_LENGTH_NV 0x8F33 +#define GL_DRAW_INDIRECT_UNIFIED_NV 0x8F40 +#define GL_DRAW_INDIRECT_ADDRESS_NV 0x8F41 +#define GL_DRAW_INDIRECT_LENGTH_NV 0x8F42 +typedef void (APIENTRYP PFNGLBUFFERADDRESSRANGENVPROC) (GLenum pname, GLuint index, GLuint64EXT address, GLsizeiptr length); +typedef void (APIENTRYP PFNGLVERTEXFORMATNVPROC) (GLint size, GLenum type, GLsizei stride); +typedef void (APIENTRYP PFNGLNORMALFORMATNVPROC) (GLenum type, GLsizei stride); +typedef void (APIENTRYP PFNGLCOLORFORMATNVPROC) (GLint size, GLenum type, GLsizei stride); +typedef void (APIENTRYP PFNGLINDEXFORMATNVPROC) (GLenum type, GLsizei stride); +typedef void (APIENTRYP PFNGLTEXCOORDFORMATNVPROC) (GLint size, GLenum type, GLsizei stride); +typedef void (APIENTRYP PFNGLEDGEFLAGFORMATNVPROC) (GLsizei stride); +typedef void (APIENTRYP PFNGLSECONDARYCOLORFORMATNVPROC) (GLint size, GLenum type, GLsizei stride); +typedef void (APIENTRYP PFNGLFOGCOORDFORMATNVPROC) (GLenum type, GLsizei stride); +typedef void (APIENTRYP PFNGLVERTEXATTRIBFORMATNVPROC) (GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride); +typedef void (APIENTRYP PFNGLVERTEXATTRIBIFORMATNVPROC) (GLuint index, GLint size, GLenum type, GLsizei stride); +typedef void (APIENTRYP PFNGLGETINTEGERUI64I_VNVPROC) (GLenum value, GLuint index, GLuint64EXT *result); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBufferAddressRangeNV (GLenum pname, GLuint index, GLuint64EXT address, GLsizeiptr length); +GLAPI void APIENTRY glVertexFormatNV (GLint size, GLenum type, GLsizei stride); +GLAPI void APIENTRY glNormalFormatNV (GLenum type, GLsizei stride); +GLAPI void APIENTRY glColorFormatNV (GLint size, GLenum type, GLsizei stride); +GLAPI void APIENTRY glIndexFormatNV (GLenum type, GLsizei stride); +GLAPI void APIENTRY glTexCoordFormatNV (GLint size, GLenum type, GLsizei stride); +GLAPI void APIENTRY glEdgeFlagFormatNV (GLsizei stride); +GLAPI void APIENTRY glSecondaryColorFormatNV (GLint size, GLenum type, GLsizei stride); +GLAPI void APIENTRY glFogCoordFormatNV (GLenum type, GLsizei stride); +GLAPI void APIENTRY glVertexAttribFormatNV (GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride); +GLAPI void APIENTRY glVertexAttribIFormatNV (GLuint index, GLint size, GLenum type, GLsizei stride); +GLAPI void APIENTRY glGetIntegerui64i_vNV (GLenum value, GLuint index, GLuint64EXT *result); +#endif +#endif /* GL_NV_vertex_buffer_unified_memory */ + +#ifndef GL_NV_vertex_program +#define GL_NV_vertex_program 1 +#define GL_VERTEX_PROGRAM_NV 0x8620 +#define GL_VERTEX_STATE_PROGRAM_NV 0x8621 +#define GL_ATTRIB_ARRAY_SIZE_NV 0x8623 +#define GL_ATTRIB_ARRAY_STRIDE_NV 0x8624 +#define GL_ATTRIB_ARRAY_TYPE_NV 0x8625 +#define GL_CURRENT_ATTRIB_NV 0x8626 +#define GL_PROGRAM_LENGTH_NV 0x8627 +#define GL_PROGRAM_STRING_NV 0x8628 +#define GL_MODELVIEW_PROJECTION_NV 0x8629 +#define GL_IDENTITY_NV 0x862A +#define GL_INVERSE_NV 0x862B +#define GL_TRANSPOSE_NV 0x862C +#define GL_INVERSE_TRANSPOSE_NV 0x862D +#define GL_MAX_TRACK_MATRIX_STACK_DEPTH_NV 0x862E +#define GL_MAX_TRACK_MATRICES_NV 0x862F +#define GL_MATRIX0_NV 0x8630 +#define GL_MATRIX1_NV 0x8631 +#define GL_MATRIX2_NV 0x8632 +#define GL_MATRIX3_NV 0x8633 +#define GL_MATRIX4_NV 0x8634 +#define GL_MATRIX5_NV 0x8635 +#define GL_MATRIX6_NV 0x8636 +#define GL_MATRIX7_NV 0x8637 +#define GL_CURRENT_MATRIX_STACK_DEPTH_NV 0x8640 +#define GL_CURRENT_MATRIX_NV 0x8641 +#define GL_VERTEX_PROGRAM_POINT_SIZE_NV 0x8642 +#define GL_VERTEX_PROGRAM_TWO_SIDE_NV 0x8643 +#define GL_PROGRAM_PARAMETER_NV 0x8644 +#define GL_ATTRIB_ARRAY_POINTER_NV 0x8645 +#define GL_PROGRAM_TARGET_NV 0x8646 +#define GL_PROGRAM_RESIDENT_NV 0x8647 +#define GL_TRACK_MATRIX_NV 0x8648 +#define GL_TRACK_MATRIX_TRANSFORM_NV 0x8649 +#define GL_VERTEX_PROGRAM_BINDING_NV 0x864A +#define GL_PROGRAM_ERROR_POSITION_NV 0x864B +#define GL_VERTEX_ATTRIB_ARRAY0_NV 0x8650 +#define GL_VERTEX_ATTRIB_ARRAY1_NV 0x8651 +#define GL_VERTEX_ATTRIB_ARRAY2_NV 0x8652 +#define GL_VERTEX_ATTRIB_ARRAY3_NV 0x8653 +#define GL_VERTEX_ATTRIB_ARRAY4_NV 0x8654 +#define GL_VERTEX_ATTRIB_ARRAY5_NV 0x8655 +#define GL_VERTEX_ATTRIB_ARRAY6_NV 0x8656 +#define GL_VERTEX_ATTRIB_ARRAY7_NV 0x8657 +#define GL_VERTEX_ATTRIB_ARRAY8_NV 0x8658 +#define GL_VERTEX_ATTRIB_ARRAY9_NV 0x8659 +#define GL_VERTEX_ATTRIB_ARRAY10_NV 0x865A +#define GL_VERTEX_ATTRIB_ARRAY11_NV 0x865B +#define GL_VERTEX_ATTRIB_ARRAY12_NV 0x865C +#define GL_VERTEX_ATTRIB_ARRAY13_NV 0x865D +#define GL_VERTEX_ATTRIB_ARRAY14_NV 0x865E +#define GL_VERTEX_ATTRIB_ARRAY15_NV 0x865F +#define GL_MAP1_VERTEX_ATTRIB0_4_NV 0x8660 +#define GL_MAP1_VERTEX_ATTRIB1_4_NV 0x8661 +#define GL_MAP1_VERTEX_ATTRIB2_4_NV 0x8662 +#define GL_MAP1_VERTEX_ATTRIB3_4_NV 0x8663 +#define GL_MAP1_VERTEX_ATTRIB4_4_NV 0x8664 +#define GL_MAP1_VERTEX_ATTRIB5_4_NV 0x8665 +#define GL_MAP1_VERTEX_ATTRIB6_4_NV 0x8666 +#define GL_MAP1_VERTEX_ATTRIB7_4_NV 0x8667 +#define GL_MAP1_VERTEX_ATTRIB8_4_NV 0x8668 +#define GL_MAP1_VERTEX_ATTRIB9_4_NV 0x8669 +#define GL_MAP1_VERTEX_ATTRIB10_4_NV 0x866A +#define GL_MAP1_VERTEX_ATTRIB11_4_NV 0x866B +#define GL_MAP1_VERTEX_ATTRIB12_4_NV 0x866C +#define GL_MAP1_VERTEX_ATTRIB13_4_NV 0x866D +#define GL_MAP1_VERTEX_ATTRIB14_4_NV 0x866E +#define GL_MAP1_VERTEX_ATTRIB15_4_NV 0x866F +#define GL_MAP2_VERTEX_ATTRIB0_4_NV 0x8670 +#define GL_MAP2_VERTEX_ATTRIB1_4_NV 0x8671 +#define GL_MAP2_VERTEX_ATTRIB2_4_NV 0x8672 +#define GL_MAP2_VERTEX_ATTRIB3_4_NV 0x8673 +#define GL_MAP2_VERTEX_ATTRIB4_4_NV 0x8674 +#define GL_MAP2_VERTEX_ATTRIB5_4_NV 0x8675 +#define GL_MAP2_VERTEX_ATTRIB6_4_NV 0x8676 +#define GL_MAP2_VERTEX_ATTRIB7_4_NV 0x8677 +#define GL_MAP2_VERTEX_ATTRIB8_4_NV 0x8678 +#define GL_MAP2_VERTEX_ATTRIB9_4_NV 0x8679 +#define GL_MAP2_VERTEX_ATTRIB10_4_NV 0x867A +#define GL_MAP2_VERTEX_ATTRIB11_4_NV 0x867B +#define GL_MAP2_VERTEX_ATTRIB12_4_NV 0x867C +#define GL_MAP2_VERTEX_ATTRIB13_4_NV 0x867D +#define GL_MAP2_VERTEX_ATTRIB14_4_NV 0x867E +#define GL_MAP2_VERTEX_ATTRIB15_4_NV 0x867F +typedef GLboolean (APIENTRYP PFNGLAREPROGRAMSRESIDENTNVPROC) (GLsizei n, const GLuint *programs, GLboolean *residences); +typedef void (APIENTRYP PFNGLBINDPROGRAMNVPROC) (GLenum target, GLuint id); +typedef void (APIENTRYP PFNGLDELETEPROGRAMSNVPROC) (GLsizei n, const GLuint *programs); +typedef void (APIENTRYP PFNGLEXECUTEPROGRAMNVPROC) (GLenum target, GLuint id, const GLfloat *params); +typedef void (APIENTRYP PFNGLGENPROGRAMSNVPROC) (GLsizei n, GLuint *programs); +typedef void (APIENTRYP PFNGLGETPROGRAMPARAMETERDVNVPROC) (GLenum target, GLuint index, GLenum pname, GLdouble *params); +typedef void (APIENTRYP PFNGLGETPROGRAMPARAMETERFVNVPROC) (GLenum target, GLuint index, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETPROGRAMIVNVPROC) (GLuint id, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETPROGRAMSTRINGNVPROC) (GLuint id, GLenum pname, GLubyte *program); +typedef void (APIENTRYP PFNGLGETTRACKMATRIXIVNVPROC) (GLenum target, GLuint address, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBDVNVPROC) (GLuint index, GLenum pname, GLdouble *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBFVNVPROC) (GLuint index, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBIVNVPROC) (GLuint index, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBPOINTERVNVPROC) (GLuint index, GLenum pname, void **pointer); +typedef GLboolean (APIENTRYP PFNGLISPROGRAMNVPROC) (GLuint id); +typedef void (APIENTRYP PFNGLLOADPROGRAMNVPROC) (GLenum target, GLuint id, GLsizei len, const GLubyte *program); +typedef void (APIENTRYP PFNGLPROGRAMPARAMETER4DNVPROC) (GLenum target, GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLPROGRAMPARAMETER4DVNVPROC) (GLenum target, GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLPROGRAMPARAMETER4FNVPROC) (GLenum target, GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLPROGRAMPARAMETER4FVNVPROC) (GLenum target, GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLPROGRAMPARAMETERS4DVNVPROC) (GLenum target, GLuint index, GLsizei count, const GLdouble *v); +typedef void (APIENTRYP PFNGLPROGRAMPARAMETERS4FVNVPROC) (GLenum target, GLuint index, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLREQUESTRESIDENTPROGRAMSNVPROC) (GLsizei n, const GLuint *programs); +typedef void (APIENTRYP PFNGLTRACKMATRIXNVPROC) (GLenum target, GLuint address, GLenum matrix, GLenum transform); +typedef void (APIENTRYP PFNGLVERTEXATTRIBPOINTERNVPROC) (GLuint index, GLint fsize, GLenum type, GLsizei stride, const void *pointer); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1DNVPROC) (GLuint index, GLdouble x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1DVNVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1FNVPROC) (GLuint index, GLfloat x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1FVNVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1SNVPROC) (GLuint index, GLshort x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1SVNVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2DNVPROC) (GLuint index, GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2DVNVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2FNVPROC) (GLuint index, GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2FVNVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2SNVPROC) (GLuint index, GLshort x, GLshort y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2SVNVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3DNVPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3DVNVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3FNVPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3FVNVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3SNVPROC) (GLuint index, GLshort x, GLshort y, GLshort z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3SVNVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4DNVPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4DVNVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4FNVPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4FVNVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4SNVPROC) (GLuint index, GLshort x, GLshort y, GLshort z, GLshort w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4SVNVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4UBNVPROC) (GLuint index, GLubyte x, GLubyte y, GLubyte z, GLubyte w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4UBVNVPROC) (GLuint index, const GLubyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS1DVNVPROC) (GLuint index, GLsizei count, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS1FVNVPROC) (GLuint index, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS1SVNVPROC) (GLuint index, GLsizei count, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS2DVNVPROC) (GLuint index, GLsizei count, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS2FVNVPROC) (GLuint index, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS2SVNVPROC) (GLuint index, GLsizei count, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS3DVNVPROC) (GLuint index, GLsizei count, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS3FVNVPROC) (GLuint index, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS3SVNVPROC) (GLuint index, GLsizei count, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS4DVNVPROC) (GLuint index, GLsizei count, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS4FVNVPROC) (GLuint index, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS4SVNVPROC) (GLuint index, GLsizei count, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS4UBVNVPROC) (GLuint index, GLsizei count, const GLubyte *v); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLboolean APIENTRY glAreProgramsResidentNV (GLsizei n, const GLuint *programs, GLboolean *residences); +GLAPI void APIENTRY glBindProgramNV (GLenum target, GLuint id); +GLAPI void APIENTRY glDeleteProgramsNV (GLsizei n, const GLuint *programs); +GLAPI void APIENTRY glExecuteProgramNV (GLenum target, GLuint id, const GLfloat *params); +GLAPI void APIENTRY glGenProgramsNV (GLsizei n, GLuint *programs); +GLAPI void APIENTRY glGetProgramParameterdvNV (GLenum target, GLuint index, GLenum pname, GLdouble *params); +GLAPI void APIENTRY glGetProgramParameterfvNV (GLenum target, GLuint index, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetProgramivNV (GLuint id, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetProgramStringNV (GLuint id, GLenum pname, GLubyte *program); +GLAPI void APIENTRY glGetTrackMatrixivNV (GLenum target, GLuint address, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetVertexAttribdvNV (GLuint index, GLenum pname, GLdouble *params); +GLAPI void APIENTRY glGetVertexAttribfvNV (GLuint index, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetVertexAttribivNV (GLuint index, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetVertexAttribPointervNV (GLuint index, GLenum pname, void **pointer); +GLAPI GLboolean APIENTRY glIsProgramNV (GLuint id); +GLAPI void APIENTRY glLoadProgramNV (GLenum target, GLuint id, GLsizei len, const GLubyte *program); +GLAPI void APIENTRY glProgramParameter4dNV (GLenum target, GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +GLAPI void APIENTRY glProgramParameter4dvNV (GLenum target, GLuint index, const GLdouble *v); +GLAPI void APIENTRY glProgramParameter4fNV (GLenum target, GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +GLAPI void APIENTRY glProgramParameter4fvNV (GLenum target, GLuint index, const GLfloat *v); +GLAPI void APIENTRY glProgramParameters4dvNV (GLenum target, GLuint index, GLsizei count, const GLdouble *v); +GLAPI void APIENTRY glProgramParameters4fvNV (GLenum target, GLuint index, GLsizei count, const GLfloat *v); +GLAPI void APIENTRY glRequestResidentProgramsNV (GLsizei n, const GLuint *programs); +GLAPI void APIENTRY glTrackMatrixNV (GLenum target, GLuint address, GLenum matrix, GLenum transform); +GLAPI void APIENTRY glVertexAttribPointerNV (GLuint index, GLint fsize, GLenum type, GLsizei stride, const void *pointer); +GLAPI void APIENTRY glVertexAttrib1dNV (GLuint index, GLdouble x); +GLAPI void APIENTRY glVertexAttrib1dvNV (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttrib1fNV (GLuint index, GLfloat x); +GLAPI void APIENTRY glVertexAttrib1fvNV (GLuint index, const GLfloat *v); +GLAPI void APIENTRY glVertexAttrib1sNV (GLuint index, GLshort x); +GLAPI void APIENTRY glVertexAttrib1svNV (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttrib2dNV (GLuint index, GLdouble x, GLdouble y); +GLAPI void APIENTRY glVertexAttrib2dvNV (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttrib2fNV (GLuint index, GLfloat x, GLfloat y); +GLAPI void APIENTRY glVertexAttrib2fvNV (GLuint index, const GLfloat *v); +GLAPI void APIENTRY glVertexAttrib2sNV (GLuint index, GLshort x, GLshort y); +GLAPI void APIENTRY glVertexAttrib2svNV (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttrib3dNV (GLuint index, GLdouble x, GLdouble y, GLdouble z); +GLAPI void APIENTRY glVertexAttrib3dvNV (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttrib3fNV (GLuint index, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glVertexAttrib3fvNV (GLuint index, const GLfloat *v); +GLAPI void APIENTRY glVertexAttrib3sNV (GLuint index, GLshort x, GLshort y, GLshort z); +GLAPI void APIENTRY glVertexAttrib3svNV (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttrib4dNV (GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +GLAPI void APIENTRY glVertexAttrib4dvNV (GLuint index, const GLdouble *v); +GLAPI void APIENTRY glVertexAttrib4fNV (GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +GLAPI void APIENTRY glVertexAttrib4fvNV (GLuint index, const GLfloat *v); +GLAPI void APIENTRY glVertexAttrib4sNV (GLuint index, GLshort x, GLshort y, GLshort z, GLshort w); +GLAPI void APIENTRY glVertexAttrib4svNV (GLuint index, const GLshort *v); +GLAPI void APIENTRY glVertexAttrib4ubNV (GLuint index, GLubyte x, GLubyte y, GLubyte z, GLubyte w); +GLAPI void APIENTRY glVertexAttrib4ubvNV (GLuint index, const GLubyte *v); +GLAPI void APIENTRY glVertexAttribs1dvNV (GLuint index, GLsizei count, const GLdouble *v); +GLAPI void APIENTRY glVertexAttribs1fvNV (GLuint index, GLsizei count, const GLfloat *v); +GLAPI void APIENTRY glVertexAttribs1svNV (GLuint index, GLsizei count, const GLshort *v); +GLAPI void APIENTRY glVertexAttribs2dvNV (GLuint index, GLsizei count, const GLdouble *v); +GLAPI void APIENTRY glVertexAttribs2fvNV (GLuint index, GLsizei count, const GLfloat *v); +GLAPI void APIENTRY glVertexAttribs2svNV (GLuint index, GLsizei count, const GLshort *v); +GLAPI void APIENTRY glVertexAttribs3dvNV (GLuint index, GLsizei count, const GLdouble *v); +GLAPI void APIENTRY glVertexAttribs3fvNV (GLuint index, GLsizei count, const GLfloat *v); +GLAPI void APIENTRY glVertexAttribs3svNV (GLuint index, GLsizei count, const GLshort *v); +GLAPI void APIENTRY glVertexAttribs4dvNV (GLuint index, GLsizei count, const GLdouble *v); +GLAPI void APIENTRY glVertexAttribs4fvNV (GLuint index, GLsizei count, const GLfloat *v); +GLAPI void APIENTRY glVertexAttribs4svNV (GLuint index, GLsizei count, const GLshort *v); +GLAPI void APIENTRY glVertexAttribs4ubvNV (GLuint index, GLsizei count, const GLubyte *v); +#endif +#endif /* GL_NV_vertex_program */ + +#ifndef GL_NV_vertex_program1_1 +#define GL_NV_vertex_program1_1 1 +#endif /* GL_NV_vertex_program1_1 */ + +#ifndef GL_NV_vertex_program2 +#define GL_NV_vertex_program2 1 +#endif /* GL_NV_vertex_program2 */ + +#ifndef GL_NV_vertex_program2_option +#define GL_NV_vertex_program2_option 1 +#endif /* GL_NV_vertex_program2_option */ + +#ifndef GL_NV_vertex_program3 +#define GL_NV_vertex_program3 1 +#endif /* GL_NV_vertex_program3 */ + +#ifndef GL_NV_vertex_program4 +#define GL_NV_vertex_program4 1 +#define GL_VERTEX_ATTRIB_ARRAY_INTEGER_NV 0x88FD +#endif /* GL_NV_vertex_program4 */ + +#ifndef GL_NV_video_capture +#define GL_NV_video_capture 1 +#define GL_VIDEO_BUFFER_NV 0x9020 +#define GL_VIDEO_BUFFER_BINDING_NV 0x9021 +#define GL_FIELD_UPPER_NV 0x9022 +#define GL_FIELD_LOWER_NV 0x9023 +#define GL_NUM_VIDEO_CAPTURE_STREAMS_NV 0x9024 +#define GL_NEXT_VIDEO_CAPTURE_BUFFER_STATUS_NV 0x9025 +#define GL_VIDEO_CAPTURE_TO_422_SUPPORTED_NV 0x9026 +#define GL_LAST_VIDEO_CAPTURE_STATUS_NV 0x9027 +#define GL_VIDEO_BUFFER_PITCH_NV 0x9028 +#define GL_VIDEO_COLOR_CONVERSION_MATRIX_NV 0x9029 +#define GL_VIDEO_COLOR_CONVERSION_MAX_NV 0x902A +#define GL_VIDEO_COLOR_CONVERSION_MIN_NV 0x902B +#define GL_VIDEO_COLOR_CONVERSION_OFFSET_NV 0x902C +#define GL_VIDEO_BUFFER_INTERNAL_FORMAT_NV 0x902D +#define GL_PARTIAL_SUCCESS_NV 0x902E +#define GL_SUCCESS_NV 0x902F +#define GL_FAILURE_NV 0x9030 +#define GL_YCBYCR8_422_NV 0x9031 +#define GL_YCBAYCR8A_4224_NV 0x9032 +#define GL_Z6Y10Z6CB10Z6Y10Z6CR10_422_NV 0x9033 +#define GL_Z6Y10Z6CB10Z6A10Z6Y10Z6CR10Z6A10_4224_NV 0x9034 +#define GL_Z4Y12Z4CB12Z4Y12Z4CR12_422_NV 0x9035 +#define GL_Z4Y12Z4CB12Z4A12Z4Y12Z4CR12Z4A12_4224_NV 0x9036 +#define GL_Z4Y12Z4CB12Z4CR12_444_NV 0x9037 +#define GL_VIDEO_CAPTURE_FRAME_WIDTH_NV 0x9038 +#define GL_VIDEO_CAPTURE_FRAME_HEIGHT_NV 0x9039 +#define GL_VIDEO_CAPTURE_FIELD_UPPER_HEIGHT_NV 0x903A +#define GL_VIDEO_CAPTURE_FIELD_LOWER_HEIGHT_NV 0x903B +#define GL_VIDEO_CAPTURE_SURFACE_ORIGIN_NV 0x903C +typedef void (APIENTRYP PFNGLBEGINVIDEOCAPTURENVPROC) (GLuint video_capture_slot); +typedef void (APIENTRYP PFNGLBINDVIDEOCAPTURESTREAMBUFFERNVPROC) (GLuint video_capture_slot, GLuint stream, GLenum frame_region, GLintptrARB offset); +typedef void (APIENTRYP PFNGLBINDVIDEOCAPTURESTREAMTEXTURENVPROC) (GLuint video_capture_slot, GLuint stream, GLenum frame_region, GLenum target, GLuint texture); +typedef void (APIENTRYP PFNGLENDVIDEOCAPTURENVPROC) (GLuint video_capture_slot); +typedef void (APIENTRYP PFNGLGETVIDEOCAPTUREIVNVPROC) (GLuint video_capture_slot, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETVIDEOCAPTURESTREAMIVNVPROC) (GLuint video_capture_slot, GLuint stream, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETVIDEOCAPTURESTREAMFVNVPROC) (GLuint video_capture_slot, GLuint stream, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETVIDEOCAPTURESTREAMDVNVPROC) (GLuint video_capture_slot, GLuint stream, GLenum pname, GLdouble *params); +typedef GLenum (APIENTRYP PFNGLVIDEOCAPTURENVPROC) (GLuint video_capture_slot, GLuint *sequence_num, GLuint64EXT *capture_time); +typedef void (APIENTRYP PFNGLVIDEOCAPTURESTREAMPARAMETERIVNVPROC) (GLuint video_capture_slot, GLuint stream, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLVIDEOCAPTURESTREAMPARAMETERFVNVPROC) (GLuint video_capture_slot, GLuint stream, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLVIDEOCAPTURESTREAMPARAMETERDVNVPROC) (GLuint video_capture_slot, GLuint stream, GLenum pname, const GLdouble *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBeginVideoCaptureNV (GLuint video_capture_slot); +GLAPI void APIENTRY glBindVideoCaptureStreamBufferNV (GLuint video_capture_slot, GLuint stream, GLenum frame_region, GLintptrARB offset); +GLAPI void APIENTRY glBindVideoCaptureStreamTextureNV (GLuint video_capture_slot, GLuint stream, GLenum frame_region, GLenum target, GLuint texture); +GLAPI void APIENTRY glEndVideoCaptureNV (GLuint video_capture_slot); +GLAPI void APIENTRY glGetVideoCaptureivNV (GLuint video_capture_slot, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetVideoCaptureStreamivNV (GLuint video_capture_slot, GLuint stream, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetVideoCaptureStreamfvNV (GLuint video_capture_slot, GLuint stream, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetVideoCaptureStreamdvNV (GLuint video_capture_slot, GLuint stream, GLenum pname, GLdouble *params); +GLAPI GLenum APIENTRY glVideoCaptureNV (GLuint video_capture_slot, GLuint *sequence_num, GLuint64EXT *capture_time); +GLAPI void APIENTRY glVideoCaptureStreamParameterivNV (GLuint video_capture_slot, GLuint stream, GLenum pname, const GLint *params); +GLAPI void APIENTRY glVideoCaptureStreamParameterfvNV (GLuint video_capture_slot, GLuint stream, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glVideoCaptureStreamParameterdvNV (GLuint video_capture_slot, GLuint stream, GLenum pname, const GLdouble *params); +#endif +#endif /* GL_NV_video_capture */ + +#ifndef GL_NV_viewport_array2 +#define GL_NV_viewport_array2 1 +#endif /* GL_NV_viewport_array2 */ + +#ifndef GL_NV_viewport_swizzle +#define GL_NV_viewport_swizzle 1 +#define GL_VIEWPORT_SWIZZLE_POSITIVE_X_NV 0x9350 +#define GL_VIEWPORT_SWIZZLE_NEGATIVE_X_NV 0x9351 +#define GL_VIEWPORT_SWIZZLE_POSITIVE_Y_NV 0x9352 +#define GL_VIEWPORT_SWIZZLE_NEGATIVE_Y_NV 0x9353 +#define GL_VIEWPORT_SWIZZLE_POSITIVE_Z_NV 0x9354 +#define GL_VIEWPORT_SWIZZLE_NEGATIVE_Z_NV 0x9355 +#define GL_VIEWPORT_SWIZZLE_POSITIVE_W_NV 0x9356 +#define GL_VIEWPORT_SWIZZLE_NEGATIVE_W_NV 0x9357 +#define GL_VIEWPORT_SWIZZLE_X_NV 0x9358 +#define GL_VIEWPORT_SWIZZLE_Y_NV 0x9359 +#define GL_VIEWPORT_SWIZZLE_Z_NV 0x935A +#define GL_VIEWPORT_SWIZZLE_W_NV 0x935B +typedef void (APIENTRYP PFNGLVIEWPORTSWIZZLENVPROC) (GLuint index, GLenum swizzlex, GLenum swizzley, GLenum swizzlez, GLenum swizzlew); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glViewportSwizzleNV (GLuint index, GLenum swizzlex, GLenum swizzley, GLenum swizzlez, GLenum swizzlew); +#endif +#endif /* GL_NV_viewport_swizzle */ + +#ifndef GL_OML_interlace +#define GL_OML_interlace 1 +#define GL_INTERLACE_OML 0x8980 +#define GL_INTERLACE_READ_OML 0x8981 +#endif /* GL_OML_interlace */ + +#ifndef GL_OML_resample +#define GL_OML_resample 1 +#define GL_PACK_RESAMPLE_OML 0x8984 +#define GL_UNPACK_RESAMPLE_OML 0x8985 +#define GL_RESAMPLE_REPLICATE_OML 0x8986 +#define GL_RESAMPLE_ZERO_FILL_OML 0x8987 +#define GL_RESAMPLE_AVERAGE_OML 0x8988 +#define GL_RESAMPLE_DECIMATE_OML 0x8989 +#endif /* GL_OML_resample */ + +#ifndef GL_OML_subsample +#define GL_OML_subsample 1 +#define GL_FORMAT_SUBSAMPLE_24_24_OML 0x8982 +#define GL_FORMAT_SUBSAMPLE_244_244_OML 0x8983 +#endif /* GL_OML_subsample */ + +#ifndef GL_OVR_multiview +#define GL_OVR_multiview 1 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_NUM_VIEWS_OVR 0x9630 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_BASE_VIEW_INDEX_OVR 0x9632 +#define GL_MAX_VIEWS_OVR 0x9631 +#define GL_FRAMEBUFFER_INCOMPLETE_VIEW_TARGETS_OVR 0x9633 +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTUREMULTIVIEWOVRPROC) (GLenum target, GLenum attachment, GLuint texture, GLint level, GLint baseViewIndex, GLsizei numViews); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFramebufferTextureMultiviewOVR (GLenum target, GLenum attachment, GLuint texture, GLint level, GLint baseViewIndex, GLsizei numViews); +#endif +#endif /* GL_OVR_multiview */ + +#ifndef GL_OVR_multiview2 +#define GL_OVR_multiview2 1 +#endif /* GL_OVR_multiview2 */ + +#ifndef GL_PGI_misc_hints +#define GL_PGI_misc_hints 1 +#define GL_PREFER_DOUBLEBUFFER_HINT_PGI 0x1A1F8 +#define GL_CONSERVE_MEMORY_HINT_PGI 0x1A1FD +#define GL_RECLAIM_MEMORY_HINT_PGI 0x1A1FE +#define GL_NATIVE_GRAPHICS_HANDLE_PGI 0x1A202 +#define GL_NATIVE_GRAPHICS_BEGIN_HINT_PGI 0x1A203 +#define GL_NATIVE_GRAPHICS_END_HINT_PGI 0x1A204 +#define GL_ALWAYS_FAST_HINT_PGI 0x1A20C +#define GL_ALWAYS_SOFT_HINT_PGI 0x1A20D +#define GL_ALLOW_DRAW_OBJ_HINT_PGI 0x1A20E +#define GL_ALLOW_DRAW_WIN_HINT_PGI 0x1A20F +#define GL_ALLOW_DRAW_FRG_HINT_PGI 0x1A210 +#define GL_ALLOW_DRAW_MEM_HINT_PGI 0x1A211 +#define GL_STRICT_DEPTHFUNC_HINT_PGI 0x1A216 +#define GL_STRICT_LIGHTING_HINT_PGI 0x1A217 +#define GL_STRICT_SCISSOR_HINT_PGI 0x1A218 +#define GL_FULL_STIPPLE_HINT_PGI 0x1A219 +#define GL_CLIP_NEAR_HINT_PGI 0x1A220 +#define GL_CLIP_FAR_HINT_PGI 0x1A221 +#define GL_WIDE_LINE_HINT_PGI 0x1A222 +#define GL_BACK_NORMALS_HINT_PGI 0x1A223 +typedef void (APIENTRYP PFNGLHINTPGIPROC) (GLenum target, GLint mode); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glHintPGI (GLenum target, GLint mode); +#endif +#endif /* GL_PGI_misc_hints */ + +#ifndef GL_PGI_vertex_hints +#define GL_PGI_vertex_hints 1 +#define GL_VERTEX_DATA_HINT_PGI 0x1A22A +#define GL_VERTEX_CONSISTENT_HINT_PGI 0x1A22B +#define GL_MATERIAL_SIDE_HINT_PGI 0x1A22C +#define GL_MAX_VERTEX_HINT_PGI 0x1A22D +#define GL_COLOR3_BIT_PGI 0x00010000 +#define GL_COLOR4_BIT_PGI 0x00020000 +#define GL_EDGEFLAG_BIT_PGI 0x00040000 +#define GL_INDEX_BIT_PGI 0x00080000 +#define GL_MAT_AMBIENT_BIT_PGI 0x00100000 +#define GL_MAT_AMBIENT_AND_DIFFUSE_BIT_PGI 0x00200000 +#define GL_MAT_DIFFUSE_BIT_PGI 0x00400000 +#define GL_MAT_EMISSION_BIT_PGI 0x00800000 +#define GL_MAT_COLOR_INDEXES_BIT_PGI 0x01000000 +#define GL_MAT_SHININESS_BIT_PGI 0x02000000 +#define GL_MAT_SPECULAR_BIT_PGI 0x04000000 +#define GL_NORMAL_BIT_PGI 0x08000000 +#define GL_TEXCOORD1_BIT_PGI 0x10000000 +#define GL_TEXCOORD2_BIT_PGI 0x20000000 +#define GL_TEXCOORD3_BIT_PGI 0x40000000 +#define GL_TEXCOORD4_BIT_PGI 0x80000000 +#define GL_VERTEX23_BIT_PGI 0x00000004 +#define GL_VERTEX4_BIT_PGI 0x00000008 +#endif /* GL_PGI_vertex_hints */ + +#ifndef GL_REND_screen_coordinates +#define GL_REND_screen_coordinates 1 +#define GL_SCREEN_COORDINATES_REND 0x8490 +#define GL_INVERTED_SCREEN_W_REND 0x8491 +#endif /* GL_REND_screen_coordinates */ + +#ifndef GL_S3_s3tc +#define GL_S3_s3tc 1 +#define GL_RGB_S3TC 0x83A0 +#define GL_RGB4_S3TC 0x83A1 +#define GL_RGBA_S3TC 0x83A2 +#define GL_RGBA4_S3TC 0x83A3 +#define GL_RGBA_DXT5_S3TC 0x83A4 +#define GL_RGBA4_DXT5_S3TC 0x83A5 +#endif /* GL_S3_s3tc */ + +#ifndef GL_SGIS_detail_texture +#define GL_SGIS_detail_texture 1 +#define GL_DETAIL_TEXTURE_2D_SGIS 0x8095 +#define GL_DETAIL_TEXTURE_2D_BINDING_SGIS 0x8096 +#define GL_LINEAR_DETAIL_SGIS 0x8097 +#define GL_LINEAR_DETAIL_ALPHA_SGIS 0x8098 +#define GL_LINEAR_DETAIL_COLOR_SGIS 0x8099 +#define GL_DETAIL_TEXTURE_LEVEL_SGIS 0x809A +#define GL_DETAIL_TEXTURE_MODE_SGIS 0x809B +#define GL_DETAIL_TEXTURE_FUNC_POINTS_SGIS 0x809C +typedef void (APIENTRYP PFNGLDETAILTEXFUNCSGISPROC) (GLenum target, GLsizei n, const GLfloat *points); +typedef void (APIENTRYP PFNGLGETDETAILTEXFUNCSGISPROC) (GLenum target, GLfloat *points); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDetailTexFuncSGIS (GLenum target, GLsizei n, const GLfloat *points); +GLAPI void APIENTRY glGetDetailTexFuncSGIS (GLenum target, GLfloat *points); +#endif +#endif /* GL_SGIS_detail_texture */ + +#ifndef GL_SGIS_fog_function +#define GL_SGIS_fog_function 1 +#define GL_FOG_FUNC_SGIS 0x812A +#define GL_FOG_FUNC_POINTS_SGIS 0x812B +#define GL_MAX_FOG_FUNC_POINTS_SGIS 0x812C +typedef void (APIENTRYP PFNGLFOGFUNCSGISPROC) (GLsizei n, const GLfloat *points); +typedef void (APIENTRYP PFNGLGETFOGFUNCSGISPROC) (GLfloat *points); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFogFuncSGIS (GLsizei n, const GLfloat *points); +GLAPI void APIENTRY glGetFogFuncSGIS (GLfloat *points); +#endif +#endif /* GL_SGIS_fog_function */ + +#ifndef GL_SGIS_generate_mipmap +#define GL_SGIS_generate_mipmap 1 +#define GL_GENERATE_MIPMAP_SGIS 0x8191 +#define GL_GENERATE_MIPMAP_HINT_SGIS 0x8192 +#endif /* GL_SGIS_generate_mipmap */ + +#ifndef GL_SGIS_multisample +#define GL_SGIS_multisample 1 +#define GL_MULTISAMPLE_SGIS 0x809D +#define GL_SAMPLE_ALPHA_TO_MASK_SGIS 0x809E +#define GL_SAMPLE_ALPHA_TO_ONE_SGIS 0x809F +#define GL_SAMPLE_MASK_SGIS 0x80A0 +#define GL_1PASS_SGIS 0x80A1 +#define GL_2PASS_0_SGIS 0x80A2 +#define GL_2PASS_1_SGIS 0x80A3 +#define GL_4PASS_0_SGIS 0x80A4 +#define GL_4PASS_1_SGIS 0x80A5 +#define GL_4PASS_2_SGIS 0x80A6 +#define GL_4PASS_3_SGIS 0x80A7 +#define GL_SAMPLE_BUFFERS_SGIS 0x80A8 +#define GL_SAMPLES_SGIS 0x80A9 +#define GL_SAMPLE_MASK_VALUE_SGIS 0x80AA +#define GL_SAMPLE_MASK_INVERT_SGIS 0x80AB +#define GL_SAMPLE_PATTERN_SGIS 0x80AC +typedef void (APIENTRYP PFNGLSAMPLEMASKSGISPROC) (GLclampf value, GLboolean invert); +typedef void (APIENTRYP PFNGLSAMPLEPATTERNSGISPROC) (GLenum pattern); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSampleMaskSGIS (GLclampf value, GLboolean invert); +GLAPI void APIENTRY glSamplePatternSGIS (GLenum pattern); +#endif +#endif /* GL_SGIS_multisample */ + +#ifndef GL_SGIS_pixel_texture +#define GL_SGIS_pixel_texture 1 +#define GL_PIXEL_TEXTURE_SGIS 0x8353 +#define GL_PIXEL_FRAGMENT_RGB_SOURCE_SGIS 0x8354 +#define GL_PIXEL_FRAGMENT_ALPHA_SOURCE_SGIS 0x8355 +#define GL_PIXEL_GROUP_COLOR_SGIS 0x8356 +typedef void (APIENTRYP PFNGLPIXELTEXGENPARAMETERISGISPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLPIXELTEXGENPARAMETERIVSGISPROC) (GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLPIXELTEXGENPARAMETERFSGISPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLPIXELTEXGENPARAMETERFVSGISPROC) (GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLGETPIXELTEXGENPARAMETERIVSGISPROC) (GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETPIXELTEXGENPARAMETERFVSGISPROC) (GLenum pname, GLfloat *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPixelTexGenParameteriSGIS (GLenum pname, GLint param); +GLAPI void APIENTRY glPixelTexGenParameterivSGIS (GLenum pname, const GLint *params); +GLAPI void APIENTRY glPixelTexGenParameterfSGIS (GLenum pname, GLfloat param); +GLAPI void APIENTRY glPixelTexGenParameterfvSGIS (GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glGetPixelTexGenParameterivSGIS (GLenum pname, GLint *params); +GLAPI void APIENTRY glGetPixelTexGenParameterfvSGIS (GLenum pname, GLfloat *params); +#endif +#endif /* GL_SGIS_pixel_texture */ + +#ifndef GL_SGIS_point_line_texgen +#define GL_SGIS_point_line_texgen 1 +#define GL_EYE_DISTANCE_TO_POINT_SGIS 0x81F0 +#define GL_OBJECT_DISTANCE_TO_POINT_SGIS 0x81F1 +#define GL_EYE_DISTANCE_TO_LINE_SGIS 0x81F2 +#define GL_OBJECT_DISTANCE_TO_LINE_SGIS 0x81F3 +#define GL_EYE_POINT_SGIS 0x81F4 +#define GL_OBJECT_POINT_SGIS 0x81F5 +#define GL_EYE_LINE_SGIS 0x81F6 +#define GL_OBJECT_LINE_SGIS 0x81F7 +#endif /* GL_SGIS_point_line_texgen */ + +#ifndef GL_SGIS_point_parameters +#define GL_SGIS_point_parameters 1 +#define GL_POINT_SIZE_MIN_SGIS 0x8126 +#define GL_POINT_SIZE_MAX_SGIS 0x8127 +#define GL_POINT_FADE_THRESHOLD_SIZE_SGIS 0x8128 +#define GL_DISTANCE_ATTENUATION_SGIS 0x8129 +typedef void (APIENTRYP PFNGLPOINTPARAMETERFSGISPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLPOINTPARAMETERFVSGISPROC) (GLenum pname, const GLfloat *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPointParameterfSGIS (GLenum pname, GLfloat param); +GLAPI void APIENTRY glPointParameterfvSGIS (GLenum pname, const GLfloat *params); +#endif +#endif /* GL_SGIS_point_parameters */ + +#ifndef GL_SGIS_sharpen_texture +#define GL_SGIS_sharpen_texture 1 +#define GL_LINEAR_SHARPEN_SGIS 0x80AD +#define GL_LINEAR_SHARPEN_ALPHA_SGIS 0x80AE +#define GL_LINEAR_SHARPEN_COLOR_SGIS 0x80AF +#define GL_SHARPEN_TEXTURE_FUNC_POINTS_SGIS 0x80B0 +typedef void (APIENTRYP PFNGLSHARPENTEXFUNCSGISPROC) (GLenum target, GLsizei n, const GLfloat *points); +typedef void (APIENTRYP PFNGLGETSHARPENTEXFUNCSGISPROC) (GLenum target, GLfloat *points); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSharpenTexFuncSGIS (GLenum target, GLsizei n, const GLfloat *points); +GLAPI void APIENTRY glGetSharpenTexFuncSGIS (GLenum target, GLfloat *points); +#endif +#endif /* GL_SGIS_sharpen_texture */ + +#ifndef GL_SGIS_texture4D +#define GL_SGIS_texture4D 1 +#define GL_PACK_SKIP_VOLUMES_SGIS 0x8130 +#define GL_PACK_IMAGE_DEPTH_SGIS 0x8131 +#define GL_UNPACK_SKIP_VOLUMES_SGIS 0x8132 +#define GL_UNPACK_IMAGE_DEPTH_SGIS 0x8133 +#define GL_TEXTURE_4D_SGIS 0x8134 +#define GL_PROXY_TEXTURE_4D_SGIS 0x8135 +#define GL_TEXTURE_4DSIZE_SGIS 0x8136 +#define GL_TEXTURE_WRAP_Q_SGIS 0x8137 +#define GL_MAX_4D_TEXTURE_SIZE_SGIS 0x8138 +#define GL_TEXTURE_4D_BINDING_SGIS 0x814F +typedef void (APIENTRYP PFNGLTEXIMAGE4DSGISPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLsizei size4d, GLint border, GLenum format, GLenum type, const void *pixels); +typedef void (APIENTRYP PFNGLTEXSUBIMAGE4DSGISPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint woffset, GLsizei width, GLsizei height, GLsizei depth, GLsizei size4d, GLenum format, GLenum type, const void *pixels); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTexImage4DSGIS (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLsizei size4d, GLint border, GLenum format, GLenum type, const void *pixels); +GLAPI void APIENTRY glTexSubImage4DSGIS (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint woffset, GLsizei width, GLsizei height, GLsizei depth, GLsizei size4d, GLenum format, GLenum type, const void *pixels); +#endif +#endif /* GL_SGIS_texture4D */ + +#ifndef GL_SGIS_texture_border_clamp +#define GL_SGIS_texture_border_clamp 1 +#define GL_CLAMP_TO_BORDER_SGIS 0x812D +#endif /* GL_SGIS_texture_border_clamp */ + +#ifndef GL_SGIS_texture_color_mask +#define GL_SGIS_texture_color_mask 1 +#define GL_TEXTURE_COLOR_WRITEMASK_SGIS 0x81EF +typedef void (APIENTRYP PFNGLTEXTURECOLORMASKSGISPROC) (GLboolean red, GLboolean green, GLboolean blue, GLboolean alpha); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTextureColorMaskSGIS (GLboolean red, GLboolean green, GLboolean blue, GLboolean alpha); +#endif +#endif /* GL_SGIS_texture_color_mask */ + +#ifndef GL_SGIS_texture_edge_clamp +#define GL_SGIS_texture_edge_clamp 1 +#define GL_CLAMP_TO_EDGE_SGIS 0x812F +#endif /* GL_SGIS_texture_edge_clamp */ + +#ifndef GL_SGIS_texture_filter4 +#define GL_SGIS_texture_filter4 1 +#define GL_FILTER4_SGIS 0x8146 +#define GL_TEXTURE_FILTER4_SIZE_SGIS 0x8147 +typedef void (APIENTRYP PFNGLGETTEXFILTERFUNCSGISPROC) (GLenum target, GLenum filter, GLfloat *weights); +typedef void (APIENTRYP PFNGLTEXFILTERFUNCSGISPROC) (GLenum target, GLenum filter, GLsizei n, const GLfloat *weights); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGetTexFilterFuncSGIS (GLenum target, GLenum filter, GLfloat *weights); +GLAPI void APIENTRY glTexFilterFuncSGIS (GLenum target, GLenum filter, GLsizei n, const GLfloat *weights); +#endif +#endif /* GL_SGIS_texture_filter4 */ + +#ifndef GL_SGIS_texture_lod +#define GL_SGIS_texture_lod 1 +#define GL_TEXTURE_MIN_LOD_SGIS 0x813A +#define GL_TEXTURE_MAX_LOD_SGIS 0x813B +#define GL_TEXTURE_BASE_LEVEL_SGIS 0x813C +#define GL_TEXTURE_MAX_LEVEL_SGIS 0x813D +#endif /* GL_SGIS_texture_lod */ + +#ifndef GL_SGIS_texture_select +#define GL_SGIS_texture_select 1 +#define GL_DUAL_ALPHA4_SGIS 0x8110 +#define GL_DUAL_ALPHA8_SGIS 0x8111 +#define GL_DUAL_ALPHA12_SGIS 0x8112 +#define GL_DUAL_ALPHA16_SGIS 0x8113 +#define GL_DUAL_LUMINANCE4_SGIS 0x8114 +#define GL_DUAL_LUMINANCE8_SGIS 0x8115 +#define GL_DUAL_LUMINANCE12_SGIS 0x8116 +#define GL_DUAL_LUMINANCE16_SGIS 0x8117 +#define GL_DUAL_INTENSITY4_SGIS 0x8118 +#define GL_DUAL_INTENSITY8_SGIS 0x8119 +#define GL_DUAL_INTENSITY12_SGIS 0x811A +#define GL_DUAL_INTENSITY16_SGIS 0x811B +#define GL_DUAL_LUMINANCE_ALPHA4_SGIS 0x811C +#define GL_DUAL_LUMINANCE_ALPHA8_SGIS 0x811D +#define GL_QUAD_ALPHA4_SGIS 0x811E +#define GL_QUAD_ALPHA8_SGIS 0x811F +#define GL_QUAD_LUMINANCE4_SGIS 0x8120 +#define GL_QUAD_LUMINANCE8_SGIS 0x8121 +#define GL_QUAD_INTENSITY4_SGIS 0x8122 +#define GL_QUAD_INTENSITY8_SGIS 0x8123 +#define GL_DUAL_TEXTURE_SELECT_SGIS 0x8124 +#define GL_QUAD_TEXTURE_SELECT_SGIS 0x8125 +#endif /* GL_SGIS_texture_select */ + +#ifndef GL_SGIX_async +#define GL_SGIX_async 1 +#define GL_ASYNC_MARKER_SGIX 0x8329 +typedef void (APIENTRYP PFNGLASYNCMARKERSGIXPROC) (GLuint marker); +typedef GLint (APIENTRYP PFNGLFINISHASYNCSGIXPROC) (GLuint *markerp); +typedef GLint (APIENTRYP PFNGLPOLLASYNCSGIXPROC) (GLuint *markerp); +typedef GLuint (APIENTRYP PFNGLGENASYNCMARKERSSGIXPROC) (GLsizei range); +typedef void (APIENTRYP PFNGLDELETEASYNCMARKERSSGIXPROC) (GLuint marker, GLsizei range); +typedef GLboolean (APIENTRYP PFNGLISASYNCMARKERSGIXPROC) (GLuint marker); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glAsyncMarkerSGIX (GLuint marker); +GLAPI GLint APIENTRY glFinishAsyncSGIX (GLuint *markerp); +GLAPI GLint APIENTRY glPollAsyncSGIX (GLuint *markerp); +GLAPI GLuint APIENTRY glGenAsyncMarkersSGIX (GLsizei range); +GLAPI void APIENTRY glDeleteAsyncMarkersSGIX (GLuint marker, GLsizei range); +GLAPI GLboolean APIENTRY glIsAsyncMarkerSGIX (GLuint marker); +#endif +#endif /* GL_SGIX_async */ + +#ifndef GL_SGIX_async_histogram +#define GL_SGIX_async_histogram 1 +#define GL_ASYNC_HISTOGRAM_SGIX 0x832C +#define GL_MAX_ASYNC_HISTOGRAM_SGIX 0x832D +#endif /* GL_SGIX_async_histogram */ + +#ifndef GL_SGIX_async_pixel +#define GL_SGIX_async_pixel 1 +#define GL_ASYNC_TEX_IMAGE_SGIX 0x835C +#define GL_ASYNC_DRAW_PIXELS_SGIX 0x835D +#define GL_ASYNC_READ_PIXELS_SGIX 0x835E +#define GL_MAX_ASYNC_TEX_IMAGE_SGIX 0x835F +#define GL_MAX_ASYNC_DRAW_PIXELS_SGIX 0x8360 +#define GL_MAX_ASYNC_READ_PIXELS_SGIX 0x8361 +#endif /* GL_SGIX_async_pixel */ + +#ifndef GL_SGIX_blend_alpha_minmax +#define GL_SGIX_blend_alpha_minmax 1 +#define GL_ALPHA_MIN_SGIX 0x8320 +#define GL_ALPHA_MAX_SGIX 0x8321 +#endif /* GL_SGIX_blend_alpha_minmax */ + +#ifndef GL_SGIX_calligraphic_fragment +#define GL_SGIX_calligraphic_fragment 1 +#define GL_CALLIGRAPHIC_FRAGMENT_SGIX 0x8183 +#endif /* GL_SGIX_calligraphic_fragment */ + +#ifndef GL_SGIX_clipmap +#define GL_SGIX_clipmap 1 +#define GL_LINEAR_CLIPMAP_LINEAR_SGIX 0x8170 +#define GL_TEXTURE_CLIPMAP_CENTER_SGIX 0x8171 +#define GL_TEXTURE_CLIPMAP_FRAME_SGIX 0x8172 +#define GL_TEXTURE_CLIPMAP_OFFSET_SGIX 0x8173 +#define GL_TEXTURE_CLIPMAP_VIRTUAL_DEPTH_SGIX 0x8174 +#define GL_TEXTURE_CLIPMAP_LOD_OFFSET_SGIX 0x8175 +#define GL_TEXTURE_CLIPMAP_DEPTH_SGIX 0x8176 +#define GL_MAX_CLIPMAP_DEPTH_SGIX 0x8177 +#define GL_MAX_CLIPMAP_VIRTUAL_DEPTH_SGIX 0x8178 +#define GL_NEAREST_CLIPMAP_NEAREST_SGIX 0x844D +#define GL_NEAREST_CLIPMAP_LINEAR_SGIX 0x844E +#define GL_LINEAR_CLIPMAP_NEAREST_SGIX 0x844F +#endif /* GL_SGIX_clipmap */ + +#ifndef GL_SGIX_convolution_accuracy +#define GL_SGIX_convolution_accuracy 1 +#define GL_CONVOLUTION_HINT_SGIX 0x8316 +#endif /* GL_SGIX_convolution_accuracy */ + +#ifndef GL_SGIX_depth_pass_instrument +#define GL_SGIX_depth_pass_instrument 1 +#endif /* GL_SGIX_depth_pass_instrument */ + +#ifndef GL_SGIX_depth_texture +#define GL_SGIX_depth_texture 1 +#define GL_DEPTH_COMPONENT16_SGIX 0x81A5 +#define GL_DEPTH_COMPONENT24_SGIX 0x81A6 +#define GL_DEPTH_COMPONENT32_SGIX 0x81A7 +#endif /* GL_SGIX_depth_texture */ + +#ifndef GL_SGIX_flush_raster +#define GL_SGIX_flush_raster 1 +typedef void (APIENTRYP PFNGLFLUSHRASTERSGIXPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFlushRasterSGIX (void); +#endif +#endif /* GL_SGIX_flush_raster */ + +#ifndef GL_SGIX_fog_offset +#define GL_SGIX_fog_offset 1 +#define GL_FOG_OFFSET_SGIX 0x8198 +#define GL_FOG_OFFSET_VALUE_SGIX 0x8199 +#endif /* GL_SGIX_fog_offset */ + +#ifndef GL_SGIX_fragment_lighting +#define GL_SGIX_fragment_lighting 1 +#define GL_FRAGMENT_LIGHTING_SGIX 0x8400 +#define GL_FRAGMENT_COLOR_MATERIAL_SGIX 0x8401 +#define GL_FRAGMENT_COLOR_MATERIAL_FACE_SGIX 0x8402 +#define GL_FRAGMENT_COLOR_MATERIAL_PARAMETER_SGIX 0x8403 +#define GL_MAX_FRAGMENT_LIGHTS_SGIX 0x8404 +#define GL_MAX_ACTIVE_LIGHTS_SGIX 0x8405 +#define GL_CURRENT_RASTER_NORMAL_SGIX 0x8406 +#define GL_LIGHT_ENV_MODE_SGIX 0x8407 +#define GL_FRAGMENT_LIGHT_MODEL_LOCAL_VIEWER_SGIX 0x8408 +#define GL_FRAGMENT_LIGHT_MODEL_TWO_SIDE_SGIX 0x8409 +#define GL_FRAGMENT_LIGHT_MODEL_AMBIENT_SGIX 0x840A +#define GL_FRAGMENT_LIGHT_MODEL_NORMAL_INTERPOLATION_SGIX 0x840B +#define GL_FRAGMENT_LIGHT0_SGIX 0x840C +#define GL_FRAGMENT_LIGHT1_SGIX 0x840D +#define GL_FRAGMENT_LIGHT2_SGIX 0x840E +#define GL_FRAGMENT_LIGHT3_SGIX 0x840F +#define GL_FRAGMENT_LIGHT4_SGIX 0x8410 +#define GL_FRAGMENT_LIGHT5_SGIX 0x8411 +#define GL_FRAGMENT_LIGHT6_SGIX 0x8412 +#define GL_FRAGMENT_LIGHT7_SGIX 0x8413 +typedef void (APIENTRYP PFNGLFRAGMENTCOLORMATERIALSGIXPROC) (GLenum face, GLenum mode); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTFSGIXPROC) (GLenum light, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTFVSGIXPROC) (GLenum light, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTISGIXPROC) (GLenum light, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTIVSGIXPROC) (GLenum light, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTMODELFSGIXPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTMODELFVSGIXPROC) (GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTMODELISGIXPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTMODELIVSGIXPROC) (GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLFRAGMENTMATERIALFSGIXPROC) (GLenum face, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLFRAGMENTMATERIALFVSGIXPROC) (GLenum face, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLFRAGMENTMATERIALISGIXPROC) (GLenum face, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLFRAGMENTMATERIALIVSGIXPROC) (GLenum face, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLGETFRAGMENTLIGHTFVSGIXPROC) (GLenum light, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETFRAGMENTLIGHTIVSGIXPROC) (GLenum light, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETFRAGMENTMATERIALFVSGIXPROC) (GLenum face, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETFRAGMENTMATERIALIVSGIXPROC) (GLenum face, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLLIGHTENVISGIXPROC) (GLenum pname, GLint param); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFragmentColorMaterialSGIX (GLenum face, GLenum mode); +GLAPI void APIENTRY glFragmentLightfSGIX (GLenum light, GLenum pname, GLfloat param); +GLAPI void APIENTRY glFragmentLightfvSGIX (GLenum light, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glFragmentLightiSGIX (GLenum light, GLenum pname, GLint param); +GLAPI void APIENTRY glFragmentLightivSGIX (GLenum light, GLenum pname, const GLint *params); +GLAPI void APIENTRY glFragmentLightModelfSGIX (GLenum pname, GLfloat param); +GLAPI void APIENTRY glFragmentLightModelfvSGIX (GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glFragmentLightModeliSGIX (GLenum pname, GLint param); +GLAPI void APIENTRY glFragmentLightModelivSGIX (GLenum pname, const GLint *params); +GLAPI void APIENTRY glFragmentMaterialfSGIX (GLenum face, GLenum pname, GLfloat param); +GLAPI void APIENTRY glFragmentMaterialfvSGIX (GLenum face, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glFragmentMaterialiSGIX (GLenum face, GLenum pname, GLint param); +GLAPI void APIENTRY glFragmentMaterialivSGIX (GLenum face, GLenum pname, const GLint *params); +GLAPI void APIENTRY glGetFragmentLightfvSGIX (GLenum light, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetFragmentLightivSGIX (GLenum light, GLenum pname, GLint *params); +GLAPI void APIENTRY glGetFragmentMaterialfvSGIX (GLenum face, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetFragmentMaterialivSGIX (GLenum face, GLenum pname, GLint *params); +GLAPI void APIENTRY glLightEnviSGIX (GLenum pname, GLint param); +#endif +#endif /* GL_SGIX_fragment_lighting */ + +#ifndef GL_SGIX_framezoom +#define GL_SGIX_framezoom 1 +#define GL_FRAMEZOOM_SGIX 0x818B +#define GL_FRAMEZOOM_FACTOR_SGIX 0x818C +#define GL_MAX_FRAMEZOOM_FACTOR_SGIX 0x818D +typedef void (APIENTRYP PFNGLFRAMEZOOMSGIXPROC) (GLint factor); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFrameZoomSGIX (GLint factor); +#endif +#endif /* GL_SGIX_framezoom */ + +#ifndef GL_SGIX_igloo_interface +#define GL_SGIX_igloo_interface 1 +typedef void (APIENTRYP PFNGLIGLOOINTERFACESGIXPROC) (GLenum pname, const void *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glIglooInterfaceSGIX (GLenum pname, const void *params); +#endif +#endif /* GL_SGIX_igloo_interface */ + +#ifndef GL_SGIX_instruments +#define GL_SGIX_instruments 1 +#define GL_INSTRUMENT_BUFFER_POINTER_SGIX 0x8180 +#define GL_INSTRUMENT_MEASUREMENTS_SGIX 0x8181 +typedef GLint (APIENTRYP PFNGLGETINSTRUMENTSSGIXPROC) (void); +typedef void (APIENTRYP PFNGLINSTRUMENTSBUFFERSGIXPROC) (GLsizei size, GLint *buffer); +typedef GLint (APIENTRYP PFNGLPOLLINSTRUMENTSSGIXPROC) (GLint *marker_p); +typedef void (APIENTRYP PFNGLREADINSTRUMENTSSGIXPROC) (GLint marker); +typedef void (APIENTRYP PFNGLSTARTINSTRUMENTSSGIXPROC) (void); +typedef void (APIENTRYP PFNGLSTOPINSTRUMENTSSGIXPROC) (GLint marker); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLint APIENTRY glGetInstrumentsSGIX (void); +GLAPI void APIENTRY glInstrumentsBufferSGIX (GLsizei size, GLint *buffer); +GLAPI GLint APIENTRY glPollInstrumentsSGIX (GLint *marker_p); +GLAPI void APIENTRY glReadInstrumentsSGIX (GLint marker); +GLAPI void APIENTRY glStartInstrumentsSGIX (void); +GLAPI void APIENTRY glStopInstrumentsSGIX (GLint marker); +#endif +#endif /* GL_SGIX_instruments */ + +#ifndef GL_SGIX_interlace +#define GL_SGIX_interlace 1 +#define GL_INTERLACE_SGIX 0x8094 +#endif /* GL_SGIX_interlace */ + +#ifndef GL_SGIX_ir_instrument1 +#define GL_SGIX_ir_instrument1 1 +#define GL_IR_INSTRUMENT1_SGIX 0x817F +#endif /* GL_SGIX_ir_instrument1 */ + +#ifndef GL_SGIX_list_priority +#define GL_SGIX_list_priority 1 +#define GL_LIST_PRIORITY_SGIX 0x8182 +typedef void (APIENTRYP PFNGLGETLISTPARAMETERFVSGIXPROC) (GLuint list, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETLISTPARAMETERIVSGIXPROC) (GLuint list, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLLISTPARAMETERFSGIXPROC) (GLuint list, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLLISTPARAMETERFVSGIXPROC) (GLuint list, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLLISTPARAMETERISGIXPROC) (GLuint list, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLLISTPARAMETERIVSGIXPROC) (GLuint list, GLenum pname, const GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGetListParameterfvSGIX (GLuint list, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetListParameterivSGIX (GLuint list, GLenum pname, GLint *params); +GLAPI void APIENTRY glListParameterfSGIX (GLuint list, GLenum pname, GLfloat param); +GLAPI void APIENTRY glListParameterfvSGIX (GLuint list, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glListParameteriSGIX (GLuint list, GLenum pname, GLint param); +GLAPI void APIENTRY glListParameterivSGIX (GLuint list, GLenum pname, const GLint *params); +#endif +#endif /* GL_SGIX_list_priority */ + +#ifndef GL_SGIX_pixel_texture +#define GL_SGIX_pixel_texture 1 +#define GL_PIXEL_TEX_GEN_SGIX 0x8139 +#define GL_PIXEL_TEX_GEN_MODE_SGIX 0x832B +typedef void (APIENTRYP PFNGLPIXELTEXGENSGIXPROC) (GLenum mode); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPixelTexGenSGIX (GLenum mode); +#endif +#endif /* GL_SGIX_pixel_texture */ + +#ifndef GL_SGIX_pixel_tiles +#define GL_SGIX_pixel_tiles 1 +#define GL_PIXEL_TILE_BEST_ALIGNMENT_SGIX 0x813E +#define GL_PIXEL_TILE_CACHE_INCREMENT_SGIX 0x813F +#define GL_PIXEL_TILE_WIDTH_SGIX 0x8140 +#define GL_PIXEL_TILE_HEIGHT_SGIX 0x8141 +#define GL_PIXEL_TILE_GRID_WIDTH_SGIX 0x8142 +#define GL_PIXEL_TILE_GRID_HEIGHT_SGIX 0x8143 +#define GL_PIXEL_TILE_GRID_DEPTH_SGIX 0x8144 +#define GL_PIXEL_TILE_CACHE_SIZE_SGIX 0x8145 +#endif /* GL_SGIX_pixel_tiles */ + +#ifndef GL_SGIX_polynomial_ffd +#define GL_SGIX_polynomial_ffd 1 +#define GL_TEXTURE_DEFORMATION_BIT_SGIX 0x00000001 +#define GL_GEOMETRY_DEFORMATION_BIT_SGIX 0x00000002 +#define GL_GEOMETRY_DEFORMATION_SGIX 0x8194 +#define GL_TEXTURE_DEFORMATION_SGIX 0x8195 +#define GL_DEFORMATIONS_MASK_SGIX 0x8196 +#define GL_MAX_DEFORMATION_ORDER_SGIX 0x8197 +typedef void (APIENTRYP PFNGLDEFORMATIONMAP3DSGIXPROC) (GLenum target, GLdouble u1, GLdouble u2, GLint ustride, GLint uorder, GLdouble v1, GLdouble v2, GLint vstride, GLint vorder, GLdouble w1, GLdouble w2, GLint wstride, GLint worder, const GLdouble *points); +typedef void (APIENTRYP PFNGLDEFORMATIONMAP3FSGIXPROC) (GLenum target, GLfloat u1, GLfloat u2, GLint ustride, GLint uorder, GLfloat v1, GLfloat v2, GLint vstride, GLint vorder, GLfloat w1, GLfloat w2, GLint wstride, GLint worder, const GLfloat *points); +typedef void (APIENTRYP PFNGLDEFORMSGIXPROC) (GLbitfield mask); +typedef void (APIENTRYP PFNGLLOADIDENTITYDEFORMATIONMAPSGIXPROC) (GLbitfield mask); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDeformationMap3dSGIX (GLenum target, GLdouble u1, GLdouble u2, GLint ustride, GLint uorder, GLdouble v1, GLdouble v2, GLint vstride, GLint vorder, GLdouble w1, GLdouble w2, GLint wstride, GLint worder, const GLdouble *points); +GLAPI void APIENTRY glDeformationMap3fSGIX (GLenum target, GLfloat u1, GLfloat u2, GLint ustride, GLint uorder, GLfloat v1, GLfloat v2, GLint vstride, GLint vorder, GLfloat w1, GLfloat w2, GLint wstride, GLint worder, const GLfloat *points); +GLAPI void APIENTRY glDeformSGIX (GLbitfield mask); +GLAPI void APIENTRY glLoadIdentityDeformationMapSGIX (GLbitfield mask); +#endif +#endif /* GL_SGIX_polynomial_ffd */ + +#ifndef GL_SGIX_reference_plane +#define GL_SGIX_reference_plane 1 +#define GL_REFERENCE_PLANE_SGIX 0x817D +#define GL_REFERENCE_PLANE_EQUATION_SGIX 0x817E +typedef void (APIENTRYP PFNGLREFERENCEPLANESGIXPROC) (const GLdouble *equation); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glReferencePlaneSGIX (const GLdouble *equation); +#endif +#endif /* GL_SGIX_reference_plane */ + +#ifndef GL_SGIX_resample +#define GL_SGIX_resample 1 +#define GL_PACK_RESAMPLE_SGIX 0x842E +#define GL_UNPACK_RESAMPLE_SGIX 0x842F +#define GL_RESAMPLE_REPLICATE_SGIX 0x8433 +#define GL_RESAMPLE_ZERO_FILL_SGIX 0x8434 +#define GL_RESAMPLE_DECIMATE_SGIX 0x8430 +#endif /* GL_SGIX_resample */ + +#ifndef GL_SGIX_scalebias_hint +#define GL_SGIX_scalebias_hint 1 +#define GL_SCALEBIAS_HINT_SGIX 0x8322 +#endif /* GL_SGIX_scalebias_hint */ + +#ifndef GL_SGIX_shadow +#define GL_SGIX_shadow 1 +#define GL_TEXTURE_COMPARE_SGIX 0x819A +#define GL_TEXTURE_COMPARE_OPERATOR_SGIX 0x819B +#define GL_TEXTURE_LEQUAL_R_SGIX 0x819C +#define GL_TEXTURE_GEQUAL_R_SGIX 0x819D +#endif /* GL_SGIX_shadow */ + +#ifndef GL_SGIX_shadow_ambient +#define GL_SGIX_shadow_ambient 1 +#define GL_SHADOW_AMBIENT_SGIX 0x80BF +#endif /* GL_SGIX_shadow_ambient */ + +#ifndef GL_SGIX_sprite +#define GL_SGIX_sprite 1 +#define GL_SPRITE_SGIX 0x8148 +#define GL_SPRITE_MODE_SGIX 0x8149 +#define GL_SPRITE_AXIS_SGIX 0x814A +#define GL_SPRITE_TRANSLATION_SGIX 0x814B +#define GL_SPRITE_AXIAL_SGIX 0x814C +#define GL_SPRITE_OBJECT_ALIGNED_SGIX 0x814D +#define GL_SPRITE_EYE_ALIGNED_SGIX 0x814E +typedef void (APIENTRYP PFNGLSPRITEPARAMETERFSGIXPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLSPRITEPARAMETERFVSGIXPROC) (GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLSPRITEPARAMETERISGIXPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLSPRITEPARAMETERIVSGIXPROC) (GLenum pname, const GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSpriteParameterfSGIX (GLenum pname, GLfloat param); +GLAPI void APIENTRY glSpriteParameterfvSGIX (GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glSpriteParameteriSGIX (GLenum pname, GLint param); +GLAPI void APIENTRY glSpriteParameterivSGIX (GLenum pname, const GLint *params); +#endif +#endif /* GL_SGIX_sprite */ + +#ifndef GL_SGIX_subsample +#define GL_SGIX_subsample 1 +#define GL_PACK_SUBSAMPLE_RATE_SGIX 0x85A0 +#define GL_UNPACK_SUBSAMPLE_RATE_SGIX 0x85A1 +#define GL_PIXEL_SUBSAMPLE_4444_SGIX 0x85A2 +#define GL_PIXEL_SUBSAMPLE_2424_SGIX 0x85A3 +#define GL_PIXEL_SUBSAMPLE_4242_SGIX 0x85A4 +#endif /* GL_SGIX_subsample */ + +#ifndef GL_SGIX_tag_sample_buffer +#define GL_SGIX_tag_sample_buffer 1 +typedef void (APIENTRYP PFNGLTAGSAMPLEBUFFERSGIXPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTagSampleBufferSGIX (void); +#endif +#endif /* GL_SGIX_tag_sample_buffer */ + +#ifndef GL_SGIX_texture_add_env +#define GL_SGIX_texture_add_env 1 +#define GL_TEXTURE_ENV_BIAS_SGIX 0x80BE +#endif /* GL_SGIX_texture_add_env */ + +#ifndef GL_SGIX_texture_coordinate_clamp +#define GL_SGIX_texture_coordinate_clamp 1 +#define GL_TEXTURE_MAX_CLAMP_S_SGIX 0x8369 +#define GL_TEXTURE_MAX_CLAMP_T_SGIX 0x836A +#define GL_TEXTURE_MAX_CLAMP_R_SGIX 0x836B +#endif /* GL_SGIX_texture_coordinate_clamp */ + +#ifndef GL_SGIX_texture_lod_bias +#define GL_SGIX_texture_lod_bias 1 +#define GL_TEXTURE_LOD_BIAS_S_SGIX 0x818E +#define GL_TEXTURE_LOD_BIAS_T_SGIX 0x818F +#define GL_TEXTURE_LOD_BIAS_R_SGIX 0x8190 +#endif /* GL_SGIX_texture_lod_bias */ + +#ifndef GL_SGIX_texture_multi_buffer +#define GL_SGIX_texture_multi_buffer 1 +#define GL_TEXTURE_MULTI_BUFFER_HINT_SGIX 0x812E +#endif /* GL_SGIX_texture_multi_buffer */ + +#ifndef GL_SGIX_texture_scale_bias +#define GL_SGIX_texture_scale_bias 1 +#define GL_POST_TEXTURE_FILTER_BIAS_SGIX 0x8179 +#define GL_POST_TEXTURE_FILTER_SCALE_SGIX 0x817A +#define GL_POST_TEXTURE_FILTER_BIAS_RANGE_SGIX 0x817B +#define GL_POST_TEXTURE_FILTER_SCALE_RANGE_SGIX 0x817C +#endif /* GL_SGIX_texture_scale_bias */ + +#ifndef GL_SGIX_vertex_preclip +#define GL_SGIX_vertex_preclip 1 +#define GL_VERTEX_PRECLIP_SGIX 0x83EE +#define GL_VERTEX_PRECLIP_HINT_SGIX 0x83EF +#endif /* GL_SGIX_vertex_preclip */ + +#ifndef GL_SGIX_ycrcb +#define GL_SGIX_ycrcb 1 +#define GL_YCRCB_422_SGIX 0x81BB +#define GL_YCRCB_444_SGIX 0x81BC +#endif /* GL_SGIX_ycrcb */ + +#ifndef GL_SGIX_ycrcb_subsample +#define GL_SGIX_ycrcb_subsample 1 +#endif /* GL_SGIX_ycrcb_subsample */ + +#ifndef GL_SGIX_ycrcba +#define GL_SGIX_ycrcba 1 +#define GL_YCRCB_SGIX 0x8318 +#define GL_YCRCBA_SGIX 0x8319 +#endif /* GL_SGIX_ycrcba */ + +#ifndef GL_SGI_color_matrix +#define GL_SGI_color_matrix 1 +#define GL_COLOR_MATRIX_SGI 0x80B1 +#define GL_COLOR_MATRIX_STACK_DEPTH_SGI 0x80B2 +#define GL_MAX_COLOR_MATRIX_STACK_DEPTH_SGI 0x80B3 +#define GL_POST_COLOR_MATRIX_RED_SCALE_SGI 0x80B4 +#define GL_POST_COLOR_MATRIX_GREEN_SCALE_SGI 0x80B5 +#define GL_POST_COLOR_MATRIX_BLUE_SCALE_SGI 0x80B6 +#define GL_POST_COLOR_MATRIX_ALPHA_SCALE_SGI 0x80B7 +#define GL_POST_COLOR_MATRIX_RED_BIAS_SGI 0x80B8 +#define GL_POST_COLOR_MATRIX_GREEN_BIAS_SGI 0x80B9 +#define GL_POST_COLOR_MATRIX_BLUE_BIAS_SGI 0x80BA +#define GL_POST_COLOR_MATRIX_ALPHA_BIAS_SGI 0x80BB +#endif /* GL_SGI_color_matrix */ + +#ifndef GL_SGI_color_table +#define GL_SGI_color_table 1 +#define GL_COLOR_TABLE_SGI 0x80D0 +#define GL_POST_CONVOLUTION_COLOR_TABLE_SGI 0x80D1 +#define GL_POST_COLOR_MATRIX_COLOR_TABLE_SGI 0x80D2 +#define GL_PROXY_COLOR_TABLE_SGI 0x80D3 +#define GL_PROXY_POST_CONVOLUTION_COLOR_TABLE_SGI 0x80D4 +#define GL_PROXY_POST_COLOR_MATRIX_COLOR_TABLE_SGI 0x80D5 +#define GL_COLOR_TABLE_SCALE_SGI 0x80D6 +#define GL_COLOR_TABLE_BIAS_SGI 0x80D7 +#define GL_COLOR_TABLE_FORMAT_SGI 0x80D8 +#define GL_COLOR_TABLE_WIDTH_SGI 0x80D9 +#define GL_COLOR_TABLE_RED_SIZE_SGI 0x80DA +#define GL_COLOR_TABLE_GREEN_SIZE_SGI 0x80DB +#define GL_COLOR_TABLE_BLUE_SIZE_SGI 0x80DC +#define GL_COLOR_TABLE_ALPHA_SIZE_SGI 0x80DD +#define GL_COLOR_TABLE_LUMINANCE_SIZE_SGI 0x80DE +#define GL_COLOR_TABLE_INTENSITY_SIZE_SGI 0x80DF +typedef void (APIENTRYP PFNGLCOLORTABLESGIPROC) (GLenum target, GLenum internalformat, GLsizei width, GLenum format, GLenum type, const void *table); +typedef void (APIENTRYP PFNGLCOLORTABLEPARAMETERFVSGIPROC) (GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLCOLORTABLEPARAMETERIVSGIPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLCOPYCOLORTABLESGIPROC) (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width); +typedef void (APIENTRYP PFNGLGETCOLORTABLESGIPROC) (GLenum target, GLenum format, GLenum type, void *table); +typedef void (APIENTRYP PFNGLGETCOLORTABLEPARAMETERFVSGIPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETCOLORTABLEPARAMETERIVSGIPROC) (GLenum target, GLenum pname, GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glColorTableSGI (GLenum target, GLenum internalformat, GLsizei width, GLenum format, GLenum type, const void *table); +GLAPI void APIENTRY glColorTableParameterfvSGI (GLenum target, GLenum pname, const GLfloat *params); +GLAPI void APIENTRY glColorTableParameterivSGI (GLenum target, GLenum pname, const GLint *params); +GLAPI void APIENTRY glCopyColorTableSGI (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width); +GLAPI void APIENTRY glGetColorTableSGI (GLenum target, GLenum format, GLenum type, void *table); +GLAPI void APIENTRY glGetColorTableParameterfvSGI (GLenum target, GLenum pname, GLfloat *params); +GLAPI void APIENTRY glGetColorTableParameterivSGI (GLenum target, GLenum pname, GLint *params); +#endif +#endif /* GL_SGI_color_table */ + +#ifndef GL_SGI_texture_color_table +#define GL_SGI_texture_color_table 1 +#define GL_TEXTURE_COLOR_TABLE_SGI 0x80BC +#define GL_PROXY_TEXTURE_COLOR_TABLE_SGI 0x80BD +#endif /* GL_SGI_texture_color_table */ + +#ifndef GL_SUNX_constant_data +#define GL_SUNX_constant_data 1 +#define GL_UNPACK_CONSTANT_DATA_SUNX 0x81D5 +#define GL_TEXTURE_CONSTANT_DATA_SUNX 0x81D6 +typedef void (APIENTRYP PFNGLFINISHTEXTURESUNXPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFinishTextureSUNX (void); +#endif +#endif /* GL_SUNX_constant_data */ + +#ifndef GL_SUN_convolution_border_modes +#define GL_SUN_convolution_border_modes 1 +#define GL_WRAP_BORDER_SUN 0x81D4 +#endif /* GL_SUN_convolution_border_modes */ + +#ifndef GL_SUN_global_alpha +#define GL_SUN_global_alpha 1 +#define GL_GLOBAL_ALPHA_SUN 0x81D9 +#define GL_GLOBAL_ALPHA_FACTOR_SUN 0x81DA +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORBSUNPROC) (GLbyte factor); +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORSSUNPROC) (GLshort factor); +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORISUNPROC) (GLint factor); +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORFSUNPROC) (GLfloat factor); +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORDSUNPROC) (GLdouble factor); +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORUBSUNPROC) (GLubyte factor); +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORUSSUNPROC) (GLushort factor); +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORUISUNPROC) (GLuint factor); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGlobalAlphaFactorbSUN (GLbyte factor); +GLAPI void APIENTRY glGlobalAlphaFactorsSUN (GLshort factor); +GLAPI void APIENTRY glGlobalAlphaFactoriSUN (GLint factor); +GLAPI void APIENTRY glGlobalAlphaFactorfSUN (GLfloat factor); +GLAPI void APIENTRY glGlobalAlphaFactordSUN (GLdouble factor); +GLAPI void APIENTRY glGlobalAlphaFactorubSUN (GLubyte factor); +GLAPI void APIENTRY glGlobalAlphaFactorusSUN (GLushort factor); +GLAPI void APIENTRY glGlobalAlphaFactoruiSUN (GLuint factor); +#endif +#endif /* GL_SUN_global_alpha */ + +#ifndef GL_SUN_mesh_array +#define GL_SUN_mesh_array 1 +#define GL_QUAD_MESH_SUN 0x8614 +#define GL_TRIANGLE_MESH_SUN 0x8615 +typedef void (APIENTRYP PFNGLDRAWMESHARRAYSSUNPROC) (GLenum mode, GLint first, GLsizei count, GLsizei width); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawMeshArraysSUN (GLenum mode, GLint first, GLsizei count, GLsizei width); +#endif +#endif /* GL_SUN_mesh_array */ + +#ifndef GL_SUN_slice_accum +#define GL_SUN_slice_accum 1 +#define GL_SLICE_ACCUM_SUN 0x85CC +#endif /* GL_SUN_slice_accum */ + +#ifndef GL_SUN_triangle_list +#define GL_SUN_triangle_list 1 +#define GL_RESTART_SUN 0x0001 +#define GL_REPLACE_MIDDLE_SUN 0x0002 +#define GL_REPLACE_OLDEST_SUN 0x0003 +#define GL_TRIANGLE_LIST_SUN 0x81D7 +#define GL_REPLACEMENT_CODE_SUN 0x81D8 +#define GL_REPLACEMENT_CODE_ARRAY_SUN 0x85C0 +#define GL_REPLACEMENT_CODE_ARRAY_TYPE_SUN 0x85C1 +#define GL_REPLACEMENT_CODE_ARRAY_STRIDE_SUN 0x85C2 +#define GL_REPLACEMENT_CODE_ARRAY_POINTER_SUN 0x85C3 +#define GL_R1UI_V3F_SUN 0x85C4 +#define GL_R1UI_C4UB_V3F_SUN 0x85C5 +#define GL_R1UI_C3F_V3F_SUN 0x85C6 +#define GL_R1UI_N3F_V3F_SUN 0x85C7 +#define GL_R1UI_C4F_N3F_V3F_SUN 0x85C8 +#define GL_R1UI_T2F_V3F_SUN 0x85C9 +#define GL_R1UI_T2F_N3F_V3F_SUN 0x85CA +#define GL_R1UI_T2F_C4F_N3F_V3F_SUN 0x85CB +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUISUNPROC) (GLuint code); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUSSUNPROC) (GLushort code); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUBSUNPROC) (GLubyte code); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUIVSUNPROC) (const GLuint *code); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUSVSUNPROC) (const GLushort *code); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUBVSUNPROC) (const GLubyte *code); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEPOINTERSUNPROC) (GLenum type, GLsizei stride, const void **pointer); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glReplacementCodeuiSUN (GLuint code); +GLAPI void APIENTRY glReplacementCodeusSUN (GLushort code); +GLAPI void APIENTRY glReplacementCodeubSUN (GLubyte code); +GLAPI void APIENTRY glReplacementCodeuivSUN (const GLuint *code); +GLAPI void APIENTRY glReplacementCodeusvSUN (const GLushort *code); +GLAPI void APIENTRY glReplacementCodeubvSUN (const GLubyte *code); +GLAPI void APIENTRY glReplacementCodePointerSUN (GLenum type, GLsizei stride, const void **pointer); +#endif +#endif /* GL_SUN_triangle_list */ + +#ifndef GL_SUN_vertex +#define GL_SUN_vertex 1 +typedef void (APIENTRYP PFNGLCOLOR4UBVERTEX2FSUNPROC) (GLubyte r, GLubyte g, GLubyte b, GLubyte a, GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLCOLOR4UBVERTEX2FVSUNPROC) (const GLubyte *c, const GLfloat *v); +typedef void (APIENTRYP PFNGLCOLOR4UBVERTEX3FSUNPROC) (GLubyte r, GLubyte g, GLubyte b, GLubyte a, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLCOLOR4UBVERTEX3FVSUNPROC) (const GLubyte *c, const GLfloat *v); +typedef void (APIENTRYP PFNGLCOLOR3FVERTEX3FSUNPROC) (GLfloat r, GLfloat g, GLfloat b, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLCOLOR3FVERTEX3FVSUNPROC) (const GLfloat *c, const GLfloat *v); +typedef void (APIENTRYP PFNGLNORMAL3FVERTEX3FSUNPROC) (GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLNORMAL3FVERTEX3FVSUNPROC) (const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLCOLOR4FNORMAL3FVERTEX3FSUNPROC) (GLfloat r, GLfloat g, GLfloat b, GLfloat a, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLCOLOR4FNORMAL3FVERTEX3FVSUNPROC) (const GLfloat *c, const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLTEXCOORD2FVERTEX3FSUNPROC) (GLfloat s, GLfloat t, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLTEXCOORD2FVERTEX3FVSUNPROC) (const GLfloat *tc, const GLfloat *v); +typedef void (APIENTRYP PFNGLTEXCOORD4FVERTEX4FSUNPROC) (GLfloat s, GLfloat t, GLfloat p, GLfloat q, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLTEXCOORD4FVERTEX4FVSUNPROC) (const GLfloat *tc, const GLfloat *v); +typedef void (APIENTRYP PFNGLTEXCOORD2FCOLOR4UBVERTEX3FSUNPROC) (GLfloat s, GLfloat t, GLubyte r, GLubyte g, GLubyte b, GLubyte a, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLTEXCOORD2FCOLOR4UBVERTEX3FVSUNPROC) (const GLfloat *tc, const GLubyte *c, const GLfloat *v); +typedef void (APIENTRYP PFNGLTEXCOORD2FCOLOR3FVERTEX3FSUNPROC) (GLfloat s, GLfloat t, GLfloat r, GLfloat g, GLfloat b, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLTEXCOORD2FCOLOR3FVERTEX3FVSUNPROC) (const GLfloat *tc, const GLfloat *c, const GLfloat *v); +typedef void (APIENTRYP PFNGLTEXCOORD2FNORMAL3FVERTEX3FSUNPROC) (GLfloat s, GLfloat t, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLTEXCOORD2FNORMAL3FVERTEX3FVSUNPROC) (const GLfloat *tc, const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLTEXCOORD2FCOLOR4FNORMAL3FVERTEX3FSUNPROC) (GLfloat s, GLfloat t, GLfloat r, GLfloat g, GLfloat b, GLfloat a, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLTEXCOORD2FCOLOR4FNORMAL3FVERTEX3FVSUNPROC) (const GLfloat *tc, const GLfloat *c, const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLTEXCOORD4FCOLOR4FNORMAL3FVERTEX4FSUNPROC) (GLfloat s, GLfloat t, GLfloat p, GLfloat q, GLfloat r, GLfloat g, GLfloat b, GLfloat a, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLTEXCOORD4FCOLOR4FNORMAL3FVERTEX4FVSUNPROC) (const GLfloat *tc, const GLfloat *c, const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUIVERTEX3FSUNPROC) (GLuint rc, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUIVERTEX3FVSUNPROC) (const GLuint *rc, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUICOLOR4UBVERTEX3FSUNPROC) (GLuint rc, GLubyte r, GLubyte g, GLubyte b, GLubyte a, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUICOLOR4UBVERTEX3FVSUNPROC) (const GLuint *rc, const GLubyte *c, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUICOLOR3FVERTEX3FSUNPROC) (GLuint rc, GLfloat r, GLfloat g, GLfloat b, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUICOLOR3FVERTEX3FVSUNPROC) (const GLuint *rc, const GLfloat *c, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUINORMAL3FVERTEX3FSUNPROC) (GLuint rc, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUINORMAL3FVERTEX3FVSUNPROC) (const GLuint *rc, const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUICOLOR4FNORMAL3FVERTEX3FSUNPROC) (GLuint rc, GLfloat r, GLfloat g, GLfloat b, GLfloat a, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUICOLOR4FNORMAL3FVERTEX3FVSUNPROC) (const GLuint *rc, const GLfloat *c, const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUITEXCOORD2FVERTEX3FSUNPROC) (GLuint rc, GLfloat s, GLfloat t, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUITEXCOORD2FVERTEX3FVSUNPROC) (const GLuint *rc, const GLfloat *tc, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUITEXCOORD2FNORMAL3FVERTEX3FSUNPROC) (GLuint rc, GLfloat s, GLfloat t, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUITEXCOORD2FNORMAL3FVERTEX3FVSUNPROC) (const GLuint *rc, const GLfloat *tc, const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUITEXCOORD2FCOLOR4FNORMAL3FVERTEX3FSUNPROC) (GLuint rc, GLfloat s, GLfloat t, GLfloat r, GLfloat g, GLfloat b, GLfloat a, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUITEXCOORD2FCOLOR4FNORMAL3FVERTEX3FVSUNPROC) (const GLuint *rc, const GLfloat *tc, const GLfloat *c, const GLfloat *n, const GLfloat *v); +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glColor4ubVertex2fSUN (GLubyte r, GLubyte g, GLubyte b, GLubyte a, GLfloat x, GLfloat y); +GLAPI void APIENTRY glColor4ubVertex2fvSUN (const GLubyte *c, const GLfloat *v); +GLAPI void APIENTRY glColor4ubVertex3fSUN (GLubyte r, GLubyte g, GLubyte b, GLubyte a, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glColor4ubVertex3fvSUN (const GLubyte *c, const GLfloat *v); +GLAPI void APIENTRY glColor3fVertex3fSUN (GLfloat r, GLfloat g, GLfloat b, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glColor3fVertex3fvSUN (const GLfloat *c, const GLfloat *v); +GLAPI void APIENTRY glNormal3fVertex3fSUN (GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glNormal3fVertex3fvSUN (const GLfloat *n, const GLfloat *v); +GLAPI void APIENTRY glColor4fNormal3fVertex3fSUN (GLfloat r, GLfloat g, GLfloat b, GLfloat a, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glColor4fNormal3fVertex3fvSUN (const GLfloat *c, const GLfloat *n, const GLfloat *v); +GLAPI void APIENTRY glTexCoord2fVertex3fSUN (GLfloat s, GLfloat t, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glTexCoord2fVertex3fvSUN (const GLfloat *tc, const GLfloat *v); +GLAPI void APIENTRY glTexCoord4fVertex4fSUN (GLfloat s, GLfloat t, GLfloat p, GLfloat q, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +GLAPI void APIENTRY glTexCoord4fVertex4fvSUN (const GLfloat *tc, const GLfloat *v); +GLAPI void APIENTRY glTexCoord2fColor4ubVertex3fSUN (GLfloat s, GLfloat t, GLubyte r, GLubyte g, GLubyte b, GLubyte a, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glTexCoord2fColor4ubVertex3fvSUN (const GLfloat *tc, const GLubyte *c, const GLfloat *v); +GLAPI void APIENTRY glTexCoord2fColor3fVertex3fSUN (GLfloat s, GLfloat t, GLfloat r, GLfloat g, GLfloat b, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glTexCoord2fColor3fVertex3fvSUN (const GLfloat *tc, const GLfloat *c, const GLfloat *v); +GLAPI void APIENTRY glTexCoord2fNormal3fVertex3fSUN (GLfloat s, GLfloat t, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glTexCoord2fNormal3fVertex3fvSUN (const GLfloat *tc, const GLfloat *n, const GLfloat *v); +GLAPI void APIENTRY glTexCoord2fColor4fNormal3fVertex3fSUN (GLfloat s, GLfloat t, GLfloat r, GLfloat g, GLfloat b, GLfloat a, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glTexCoord2fColor4fNormal3fVertex3fvSUN (const GLfloat *tc, const GLfloat *c, const GLfloat *n, const GLfloat *v); +GLAPI void APIENTRY glTexCoord4fColor4fNormal3fVertex4fSUN (GLfloat s, GLfloat t, GLfloat p, GLfloat q, GLfloat r, GLfloat g, GLfloat b, GLfloat a, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +GLAPI void APIENTRY glTexCoord4fColor4fNormal3fVertex4fvSUN (const GLfloat *tc, const GLfloat *c, const GLfloat *n, const GLfloat *v); +GLAPI void APIENTRY glReplacementCodeuiVertex3fSUN (GLuint rc, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glReplacementCodeuiVertex3fvSUN (const GLuint *rc, const GLfloat *v); +GLAPI void APIENTRY glReplacementCodeuiColor4ubVertex3fSUN (GLuint rc, GLubyte r, GLubyte g, GLubyte b, GLubyte a, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glReplacementCodeuiColor4ubVertex3fvSUN (const GLuint *rc, const GLubyte *c, const GLfloat *v); +GLAPI void APIENTRY glReplacementCodeuiColor3fVertex3fSUN (GLuint rc, GLfloat r, GLfloat g, GLfloat b, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glReplacementCodeuiColor3fVertex3fvSUN (const GLuint *rc, const GLfloat *c, const GLfloat *v); +GLAPI void APIENTRY glReplacementCodeuiNormal3fVertex3fSUN (GLuint rc, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glReplacementCodeuiNormal3fVertex3fvSUN (const GLuint *rc, const GLfloat *n, const GLfloat *v); +GLAPI void APIENTRY glReplacementCodeuiColor4fNormal3fVertex3fSUN (GLuint rc, GLfloat r, GLfloat g, GLfloat b, GLfloat a, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glReplacementCodeuiColor4fNormal3fVertex3fvSUN (const GLuint *rc, const GLfloat *c, const GLfloat *n, const GLfloat *v); +GLAPI void APIENTRY glReplacementCodeuiTexCoord2fVertex3fSUN (GLuint rc, GLfloat s, GLfloat t, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glReplacementCodeuiTexCoord2fVertex3fvSUN (const GLuint *rc, const GLfloat *tc, const GLfloat *v); +GLAPI void APIENTRY glReplacementCodeuiTexCoord2fNormal3fVertex3fSUN (GLuint rc, GLfloat s, GLfloat t, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glReplacementCodeuiTexCoord2fNormal3fVertex3fvSUN (const GLuint *rc, const GLfloat *tc, const GLfloat *n, const GLfloat *v); +GLAPI void APIENTRY glReplacementCodeuiTexCoord2fColor4fNormal3fVertex3fSUN (GLuint rc, GLfloat s, GLfloat t, GLfloat r, GLfloat g, GLfloat b, GLfloat a, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +GLAPI void APIENTRY glReplacementCodeuiTexCoord2fColor4fNormal3fVertex3fvSUN (const GLuint *rc, const GLfloat *tc, const GLfloat *c, const GLfloat *n, const GLfloat *v); +#endif +#endif /* GL_SUN_vertex */ + +#ifndef GL_WIN_phong_shading +#define GL_WIN_phong_shading 1 +#define GL_PHONG_WIN 0x80EA +#define GL_PHONG_HINT_WIN 0x80EB +#endif /* GL_WIN_phong_shading */ + +#ifndef GL_WIN_specular_fog +#define GL_WIN_specular_fog 1 +#define GL_FOG_SPECULAR_TEXTURE_WIN 0x80EC +#endif /* GL_WIN_specular_fog */ + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles.h new file mode 100644 index 00000000..f4465eaa --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles.h @@ -0,0 +1,39 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_opengles.h + * + * This is a simple file to encapsulate the OpenGL ES 1.X API headers. + */ +#include "SDL_config.h" + +#ifdef __IPHONEOS__ +#include +#include +#else +#include +#include +#endif + +#ifndef APIENTRY +#define APIENTRY +#endif diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2.h new file mode 100644 index 00000000..5e3b717d --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2.h @@ -0,0 +1,52 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_opengles2.h + * + * This is a simple file to encapsulate the OpenGL ES 2.0 API headers. + */ +#include "SDL_config.h" + +#if !defined(_MSC_VER) && !defined(SDL_USE_BUILTIN_OPENGL_DEFINITIONS) + +#ifdef __IPHONEOS__ +#include +#include +#else +#include +#include +#include +#endif + +#else /* _MSC_VER */ + +/* OpenGL ES2 headers for Visual Studio */ +#include "SDL_opengles2_khrplatform.h" +#include "SDL_opengles2_gl2platform.h" +#include "SDL_opengles2_gl2.h" +#include "SDL_opengles2_gl2ext.h" + +#endif /* _MSC_VER */ + +#ifndef APIENTRY +#define APIENTRY GL_APIENTRY +#endif diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_gl2.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_gl2.h new file mode 100644 index 00000000..d13622aa --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_gl2.h @@ -0,0 +1,656 @@ +#ifndef __gles2_gl2_h_ +#define __gles2_gl2_h_ 1 + +#ifdef __cplusplus +extern "C" { +#endif + +/* +** Copyright 2013-2020 The Khronos Group Inc. +** SPDX-License-Identifier: MIT +** +** This header is generated from the Khronos OpenGL / OpenGL ES XML +** API Registry. The current version of the Registry, generator scripts +** used to make the header, and the header can be found at +** https://github.com/KhronosGroup/OpenGL-Registry +*/ + +/*#include */ + +#ifndef GL_APIENTRYP +#define GL_APIENTRYP GL_APIENTRY* +#endif + +#ifndef GL_GLES_PROTOTYPES +#define GL_GLES_PROTOTYPES 1 +#endif + +/* Generated on date 20220530 */ + +/* Generated C header for: + * API: gles2 + * Profile: common + * Versions considered: 2\.[0-9] + * Versions emitted: .* + * Default extensions included: None + * Additional extensions included: _nomatch_^ + * Extensions removed: _nomatch_^ + */ + +#ifndef GL_ES_VERSION_2_0 +#define GL_ES_VERSION_2_0 1 +/*#include */ +typedef khronos_int8_t GLbyte; +typedef khronos_float_t GLclampf; +typedef khronos_int32_t GLfixed; +typedef khronos_int16_t GLshort; +typedef khronos_uint16_t GLushort; +typedef void GLvoid; +typedef struct __GLsync *GLsync; +typedef khronos_int64_t GLint64; +typedef khronos_uint64_t GLuint64; +typedef unsigned int GLenum; +typedef unsigned int GLuint; +typedef char GLchar; +typedef khronos_float_t GLfloat; +typedef khronos_ssize_t GLsizeiptr; +typedef khronos_intptr_t GLintptr; +typedef unsigned int GLbitfield; +typedef int GLint; +typedef unsigned char GLboolean; +typedef int GLsizei; +typedef khronos_uint8_t GLubyte; +#define GL_DEPTH_BUFFER_BIT 0x00000100 +#define GL_STENCIL_BUFFER_BIT 0x00000400 +#define GL_COLOR_BUFFER_BIT 0x00004000 +#define GL_FALSE 0 +#define GL_TRUE 1 +#define GL_POINTS 0x0000 +#define GL_LINES 0x0001 +#define GL_LINE_LOOP 0x0002 +#define GL_LINE_STRIP 0x0003 +#define GL_TRIANGLES 0x0004 +#define GL_TRIANGLE_STRIP 0x0005 +#define GL_TRIANGLE_FAN 0x0006 +#define GL_ZERO 0 +#define GL_ONE 1 +#define GL_SRC_COLOR 0x0300 +#define GL_ONE_MINUS_SRC_COLOR 0x0301 +#define GL_SRC_ALPHA 0x0302 +#define GL_ONE_MINUS_SRC_ALPHA 0x0303 +#define GL_DST_ALPHA 0x0304 +#define GL_ONE_MINUS_DST_ALPHA 0x0305 +#define GL_DST_COLOR 0x0306 +#define GL_ONE_MINUS_DST_COLOR 0x0307 +#define GL_SRC_ALPHA_SATURATE 0x0308 +#define GL_FUNC_ADD 0x8006 +#define GL_BLEND_EQUATION 0x8009 +#define GL_BLEND_EQUATION_RGB 0x8009 +#define GL_BLEND_EQUATION_ALPHA 0x883D +#define GL_FUNC_SUBTRACT 0x800A +#define GL_FUNC_REVERSE_SUBTRACT 0x800B +#define GL_BLEND_DST_RGB 0x80C8 +#define GL_BLEND_SRC_RGB 0x80C9 +#define GL_BLEND_DST_ALPHA 0x80CA +#define GL_BLEND_SRC_ALPHA 0x80CB +#define GL_CONSTANT_COLOR 0x8001 +#define GL_ONE_MINUS_CONSTANT_COLOR 0x8002 +#define GL_CONSTANT_ALPHA 0x8003 +#define GL_ONE_MINUS_CONSTANT_ALPHA 0x8004 +#define GL_BLEND_COLOR 0x8005 +#define GL_ARRAY_BUFFER 0x8892 +#define GL_ELEMENT_ARRAY_BUFFER 0x8893 +#define GL_ARRAY_BUFFER_BINDING 0x8894 +#define GL_ELEMENT_ARRAY_BUFFER_BINDING 0x8895 +#define GL_STREAM_DRAW 0x88E0 +#define GL_STATIC_DRAW 0x88E4 +#define GL_DYNAMIC_DRAW 0x88E8 +#define GL_BUFFER_SIZE 0x8764 +#define GL_BUFFER_USAGE 0x8765 +#define GL_CURRENT_VERTEX_ATTRIB 0x8626 +#define GL_FRONT 0x0404 +#define GL_BACK 0x0405 +#define GL_FRONT_AND_BACK 0x0408 +#define GL_TEXTURE_2D 0x0DE1 +#define GL_CULL_FACE 0x0B44 +#define GL_BLEND 0x0BE2 +#define GL_DITHER 0x0BD0 +#define GL_STENCIL_TEST 0x0B90 +#define GL_DEPTH_TEST 0x0B71 +#define GL_SCISSOR_TEST 0x0C11 +#define GL_POLYGON_OFFSET_FILL 0x8037 +#define GL_SAMPLE_ALPHA_TO_COVERAGE 0x809E +#define GL_SAMPLE_COVERAGE 0x80A0 +#define GL_NO_ERROR 0 +#define GL_INVALID_ENUM 0x0500 +#define GL_INVALID_VALUE 0x0501 +#define GL_INVALID_OPERATION 0x0502 +#define GL_OUT_OF_MEMORY 0x0505 +#define GL_CW 0x0900 +#define GL_CCW 0x0901 +#define GL_LINE_WIDTH 0x0B21 +#define GL_ALIASED_POINT_SIZE_RANGE 0x846D +#define GL_ALIASED_LINE_WIDTH_RANGE 0x846E +#define GL_CULL_FACE_MODE 0x0B45 +#define GL_FRONT_FACE 0x0B46 +#define GL_DEPTH_RANGE 0x0B70 +#define GL_DEPTH_WRITEMASK 0x0B72 +#define GL_DEPTH_CLEAR_VALUE 0x0B73 +#define GL_DEPTH_FUNC 0x0B74 +#define GL_STENCIL_CLEAR_VALUE 0x0B91 +#define GL_STENCIL_FUNC 0x0B92 +#define GL_STENCIL_FAIL 0x0B94 +#define GL_STENCIL_PASS_DEPTH_FAIL 0x0B95 +#define GL_STENCIL_PASS_DEPTH_PASS 0x0B96 +#define GL_STENCIL_REF 0x0B97 +#define GL_STENCIL_VALUE_MASK 0x0B93 +#define GL_STENCIL_WRITEMASK 0x0B98 +#define GL_STENCIL_BACK_FUNC 0x8800 +#define GL_STENCIL_BACK_FAIL 0x8801 +#define GL_STENCIL_BACK_PASS_DEPTH_FAIL 0x8802 +#define GL_STENCIL_BACK_PASS_DEPTH_PASS 0x8803 +#define GL_STENCIL_BACK_REF 0x8CA3 +#define GL_STENCIL_BACK_VALUE_MASK 0x8CA4 +#define GL_STENCIL_BACK_WRITEMASK 0x8CA5 +#define GL_VIEWPORT 0x0BA2 +#define GL_SCISSOR_BOX 0x0C10 +#define GL_COLOR_CLEAR_VALUE 0x0C22 +#define GL_COLOR_WRITEMASK 0x0C23 +#define GL_UNPACK_ALIGNMENT 0x0CF5 +#define GL_PACK_ALIGNMENT 0x0D05 +#define GL_MAX_TEXTURE_SIZE 0x0D33 +#define GL_MAX_VIEWPORT_DIMS 0x0D3A +#define GL_SUBPIXEL_BITS 0x0D50 +#define GL_RED_BITS 0x0D52 +#define GL_GREEN_BITS 0x0D53 +#define GL_BLUE_BITS 0x0D54 +#define GL_ALPHA_BITS 0x0D55 +#define GL_DEPTH_BITS 0x0D56 +#define GL_STENCIL_BITS 0x0D57 +#define GL_POLYGON_OFFSET_UNITS 0x2A00 +#define GL_POLYGON_OFFSET_FACTOR 0x8038 +#define GL_TEXTURE_BINDING_2D 0x8069 +#define GL_SAMPLE_BUFFERS 0x80A8 +#define GL_SAMPLES 0x80A9 +#define GL_SAMPLE_COVERAGE_VALUE 0x80AA +#define GL_SAMPLE_COVERAGE_INVERT 0x80AB +#define GL_NUM_COMPRESSED_TEXTURE_FORMATS 0x86A2 +#define GL_COMPRESSED_TEXTURE_FORMATS 0x86A3 +#define GL_DONT_CARE 0x1100 +#define GL_FASTEST 0x1101 +#define GL_NICEST 0x1102 +#define GL_GENERATE_MIPMAP_HINT 0x8192 +#define GL_BYTE 0x1400 +#define GL_UNSIGNED_BYTE 0x1401 +#define GL_SHORT 0x1402 +#define GL_UNSIGNED_SHORT 0x1403 +#define GL_INT 0x1404 +#define GL_UNSIGNED_INT 0x1405 +#define GL_FLOAT 0x1406 +#define GL_FIXED 0x140C +#define GL_DEPTH_COMPONENT 0x1902 +#define GL_ALPHA 0x1906 +#define GL_RGB 0x1907 +#define GL_RGBA 0x1908 +#define GL_LUMINANCE 0x1909 +#define GL_LUMINANCE_ALPHA 0x190A +#define GL_UNSIGNED_SHORT_4_4_4_4 0x8033 +#define GL_UNSIGNED_SHORT_5_5_5_1 0x8034 +#define GL_UNSIGNED_SHORT_5_6_5 0x8363 +#define GL_FRAGMENT_SHADER 0x8B30 +#define GL_VERTEX_SHADER 0x8B31 +#define GL_MAX_VERTEX_ATTRIBS 0x8869 +#define GL_MAX_VERTEX_UNIFORM_VECTORS 0x8DFB +#define GL_MAX_VARYING_VECTORS 0x8DFC +#define GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS 0x8B4D +#define GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS 0x8B4C +#define GL_MAX_TEXTURE_IMAGE_UNITS 0x8872 +#define GL_MAX_FRAGMENT_UNIFORM_VECTORS 0x8DFD +#define GL_SHADER_TYPE 0x8B4F +#define GL_DELETE_STATUS 0x8B80 +#define GL_LINK_STATUS 0x8B82 +#define GL_VALIDATE_STATUS 0x8B83 +#define GL_ATTACHED_SHADERS 0x8B85 +#define GL_ACTIVE_UNIFORMS 0x8B86 +#define GL_ACTIVE_UNIFORM_MAX_LENGTH 0x8B87 +#define GL_ACTIVE_ATTRIBUTES 0x8B89 +#define GL_ACTIVE_ATTRIBUTE_MAX_LENGTH 0x8B8A +#define GL_SHADING_LANGUAGE_VERSION 0x8B8C +#define GL_CURRENT_PROGRAM 0x8B8D +#define GL_NEVER 0x0200 +#define GL_LESS 0x0201 +#define GL_EQUAL 0x0202 +#define GL_LEQUAL 0x0203 +#define GL_GREATER 0x0204 +#define GL_NOTEQUAL 0x0205 +#define GL_GEQUAL 0x0206 +#define GL_ALWAYS 0x0207 +#define GL_KEEP 0x1E00 +#define GL_REPLACE 0x1E01 +#define GL_INCR 0x1E02 +#define GL_DECR 0x1E03 +#define GL_INVERT 0x150A +#define GL_INCR_WRAP 0x8507 +#define GL_DECR_WRAP 0x8508 +#define GL_VENDOR 0x1F00 +#define GL_RENDERER 0x1F01 +#define GL_VERSION 0x1F02 +#define GL_EXTENSIONS 0x1F03 +#define GL_NEAREST 0x2600 +#define GL_LINEAR 0x2601 +#define GL_NEAREST_MIPMAP_NEAREST 0x2700 +#define GL_LINEAR_MIPMAP_NEAREST 0x2701 +#define GL_NEAREST_MIPMAP_LINEAR 0x2702 +#define GL_LINEAR_MIPMAP_LINEAR 0x2703 +#define GL_TEXTURE_MAG_FILTER 0x2800 +#define GL_TEXTURE_MIN_FILTER 0x2801 +#define GL_TEXTURE_WRAP_S 0x2802 +#define GL_TEXTURE_WRAP_T 0x2803 +#define GL_TEXTURE 0x1702 +#define GL_TEXTURE_CUBE_MAP 0x8513 +#define GL_TEXTURE_BINDING_CUBE_MAP 0x8514 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_X 0x8515 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_X 0x8516 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Y 0x8517 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Y 0x8518 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Z 0x8519 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Z 0x851A +#define GL_MAX_CUBE_MAP_TEXTURE_SIZE 0x851C +#define GL_TEXTURE0 0x84C0 +#define GL_TEXTURE1 0x84C1 +#define GL_TEXTURE2 0x84C2 +#define GL_TEXTURE3 0x84C3 +#define GL_TEXTURE4 0x84C4 +#define GL_TEXTURE5 0x84C5 +#define GL_TEXTURE6 0x84C6 +#define GL_TEXTURE7 0x84C7 +#define GL_TEXTURE8 0x84C8 +#define GL_TEXTURE9 0x84C9 +#define GL_TEXTURE10 0x84CA +#define GL_TEXTURE11 0x84CB +#define GL_TEXTURE12 0x84CC +#define GL_TEXTURE13 0x84CD +#define GL_TEXTURE14 0x84CE +#define GL_TEXTURE15 0x84CF +#define GL_TEXTURE16 0x84D0 +#define GL_TEXTURE17 0x84D1 +#define GL_TEXTURE18 0x84D2 +#define GL_TEXTURE19 0x84D3 +#define GL_TEXTURE20 0x84D4 +#define GL_TEXTURE21 0x84D5 +#define GL_TEXTURE22 0x84D6 +#define GL_TEXTURE23 0x84D7 +#define GL_TEXTURE24 0x84D8 +#define GL_TEXTURE25 0x84D9 +#define GL_TEXTURE26 0x84DA +#define GL_TEXTURE27 0x84DB +#define GL_TEXTURE28 0x84DC +#define GL_TEXTURE29 0x84DD +#define GL_TEXTURE30 0x84DE +#define GL_TEXTURE31 0x84DF +#define GL_ACTIVE_TEXTURE 0x84E0 +#define GL_REPEAT 0x2901 +#define GL_CLAMP_TO_EDGE 0x812F +#define GL_MIRRORED_REPEAT 0x8370 +#define GL_FLOAT_VEC2 0x8B50 +#define GL_FLOAT_VEC3 0x8B51 +#define GL_FLOAT_VEC4 0x8B52 +#define GL_INT_VEC2 0x8B53 +#define GL_INT_VEC3 0x8B54 +#define GL_INT_VEC4 0x8B55 +#define GL_BOOL 0x8B56 +#define GL_BOOL_VEC2 0x8B57 +#define GL_BOOL_VEC3 0x8B58 +#define GL_BOOL_VEC4 0x8B59 +#define GL_FLOAT_MAT2 0x8B5A +#define GL_FLOAT_MAT3 0x8B5B +#define GL_FLOAT_MAT4 0x8B5C +#define GL_SAMPLER_2D 0x8B5E +#define GL_SAMPLER_CUBE 0x8B60 +#define GL_VERTEX_ATTRIB_ARRAY_ENABLED 0x8622 +#define GL_VERTEX_ATTRIB_ARRAY_SIZE 0x8623 +#define GL_VERTEX_ATTRIB_ARRAY_STRIDE 0x8624 +#define GL_VERTEX_ATTRIB_ARRAY_TYPE 0x8625 +#define GL_VERTEX_ATTRIB_ARRAY_NORMALIZED 0x886A +#define GL_VERTEX_ATTRIB_ARRAY_POINTER 0x8645 +#define GL_VERTEX_ATTRIB_ARRAY_BUFFER_BINDING 0x889F +#define GL_IMPLEMENTATION_COLOR_READ_TYPE 0x8B9A +#define GL_IMPLEMENTATION_COLOR_READ_FORMAT 0x8B9B +#define GL_COMPILE_STATUS 0x8B81 +#define GL_INFO_LOG_LENGTH 0x8B84 +#define GL_SHADER_SOURCE_LENGTH 0x8B88 +#define GL_SHADER_COMPILER 0x8DFA +#define GL_SHADER_BINARY_FORMATS 0x8DF8 +#define GL_NUM_SHADER_BINARY_FORMATS 0x8DF9 +#define GL_LOW_FLOAT 0x8DF0 +#define GL_MEDIUM_FLOAT 0x8DF1 +#define GL_HIGH_FLOAT 0x8DF2 +#define GL_LOW_INT 0x8DF3 +#define GL_MEDIUM_INT 0x8DF4 +#define GL_HIGH_INT 0x8DF5 +#define GL_FRAMEBUFFER 0x8D40 +#define GL_RENDERBUFFER 0x8D41 +#define GL_RGBA4 0x8056 +#define GL_RGB5_A1 0x8057 +#define GL_RGB565 0x8D62 +#define GL_DEPTH_COMPONENT16 0x81A5 +#define GL_STENCIL_INDEX8 0x8D48 +#define GL_RENDERBUFFER_WIDTH 0x8D42 +#define GL_RENDERBUFFER_HEIGHT 0x8D43 +#define GL_RENDERBUFFER_INTERNAL_FORMAT 0x8D44 +#define GL_RENDERBUFFER_RED_SIZE 0x8D50 +#define GL_RENDERBUFFER_GREEN_SIZE 0x8D51 +#define GL_RENDERBUFFER_BLUE_SIZE 0x8D52 +#define GL_RENDERBUFFER_ALPHA_SIZE 0x8D53 +#define GL_RENDERBUFFER_DEPTH_SIZE 0x8D54 +#define GL_RENDERBUFFER_STENCIL_SIZE 0x8D55 +#define GL_FRAMEBUFFER_ATTACHMENT_OBJECT_TYPE 0x8CD0 +#define GL_FRAMEBUFFER_ATTACHMENT_OBJECT_NAME 0x8CD1 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_LEVEL 0x8CD2 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_CUBE_MAP_FACE 0x8CD3 +#define GL_COLOR_ATTACHMENT0 0x8CE0 +#define GL_DEPTH_ATTACHMENT 0x8D00 +#define GL_STENCIL_ATTACHMENT 0x8D20 +#define GL_NONE 0 +#define GL_FRAMEBUFFER_COMPLETE 0x8CD5 +#define GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT 0x8CD6 +#define GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT 0x8CD7 +#define GL_FRAMEBUFFER_INCOMPLETE_DIMENSIONS 0x8CD9 +#define GL_FRAMEBUFFER_UNSUPPORTED 0x8CDD +#define GL_FRAMEBUFFER_BINDING 0x8CA6 +#define GL_RENDERBUFFER_BINDING 0x8CA7 +#define GL_MAX_RENDERBUFFER_SIZE 0x84E8 +#define GL_INVALID_FRAMEBUFFER_OPERATION 0x0506 +typedef void (GL_APIENTRYP PFNGLACTIVETEXTUREPROC) (GLenum texture); +typedef void (GL_APIENTRYP PFNGLATTACHSHADERPROC) (GLuint program, GLuint shader); +typedef void (GL_APIENTRYP PFNGLBINDATTRIBLOCATIONPROC) (GLuint program, GLuint index, const GLchar *name); +typedef void (GL_APIENTRYP PFNGLBINDBUFFERPROC) (GLenum target, GLuint buffer); +typedef void (GL_APIENTRYP PFNGLBINDFRAMEBUFFERPROC) (GLenum target, GLuint framebuffer); +typedef void (GL_APIENTRYP PFNGLBINDRENDERBUFFERPROC) (GLenum target, GLuint renderbuffer); +typedef void (GL_APIENTRYP PFNGLBINDTEXTUREPROC) (GLenum target, GLuint texture); +typedef void (GL_APIENTRYP PFNGLBLENDCOLORPROC) (GLfloat red, GLfloat green, GLfloat blue, GLfloat alpha); +typedef void (GL_APIENTRYP PFNGLBLENDEQUATIONPROC) (GLenum mode); +typedef void (GL_APIENTRYP PFNGLBLENDEQUATIONSEPARATEPROC) (GLenum modeRGB, GLenum modeAlpha); +typedef void (GL_APIENTRYP PFNGLBLENDFUNCPROC) (GLenum sfactor, GLenum dfactor); +typedef void (GL_APIENTRYP PFNGLBLENDFUNCSEPARATEPROC) (GLenum sfactorRGB, GLenum dfactorRGB, GLenum sfactorAlpha, GLenum dfactorAlpha); +typedef void (GL_APIENTRYP PFNGLBUFFERDATAPROC) (GLenum target, GLsizeiptr size, const void *data, GLenum usage); +typedef void (GL_APIENTRYP PFNGLBUFFERSUBDATAPROC) (GLenum target, GLintptr offset, GLsizeiptr size, const void *data); +typedef GLenum (GL_APIENTRYP PFNGLCHECKFRAMEBUFFERSTATUSPROC) (GLenum target); +typedef void (GL_APIENTRYP PFNGLCLEARPROC) (GLbitfield mask); +typedef void (GL_APIENTRYP PFNGLCLEARCOLORPROC) (GLfloat red, GLfloat green, GLfloat blue, GLfloat alpha); +typedef void (GL_APIENTRYP PFNGLCLEARDEPTHFPROC) (GLfloat d); +typedef void (GL_APIENTRYP PFNGLCLEARSTENCILPROC) (GLint s); +typedef void (GL_APIENTRYP PFNGLCOLORMASKPROC) (GLboolean red, GLboolean green, GLboolean blue, GLboolean alpha); +typedef void (GL_APIENTRYP PFNGLCOMPILESHADERPROC) (GLuint shader); +typedef void (GL_APIENTRYP PFNGLCOMPRESSEDTEXIMAGE2DPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLsizei imageSize, const void *data); +typedef void (GL_APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE2DPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const void *data); +typedef void (GL_APIENTRYP PFNGLCOPYTEXIMAGE2DPROC) (GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height, GLint border); +typedef void (GL_APIENTRYP PFNGLCOPYTEXSUBIMAGE2DPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint x, GLint y, GLsizei width, GLsizei height); +typedef GLuint (GL_APIENTRYP PFNGLCREATEPROGRAMPROC) (void); +typedef GLuint (GL_APIENTRYP PFNGLCREATESHADERPROC) (GLenum type); +typedef void (GL_APIENTRYP PFNGLCULLFACEPROC) (GLenum mode); +typedef void (GL_APIENTRYP PFNGLDELETEBUFFERSPROC) (GLsizei n, const GLuint *buffers); +typedef void (GL_APIENTRYP PFNGLDELETEFRAMEBUFFERSPROC) (GLsizei n, const GLuint *framebuffers); +typedef void (GL_APIENTRYP PFNGLDELETEPROGRAMPROC) (GLuint program); +typedef void (GL_APIENTRYP PFNGLDELETERENDERBUFFERSPROC) (GLsizei n, const GLuint *renderbuffers); +typedef void (GL_APIENTRYP PFNGLDELETESHADERPROC) (GLuint shader); +typedef void (GL_APIENTRYP PFNGLDELETETEXTURESPROC) (GLsizei n, const GLuint *textures); +typedef void (GL_APIENTRYP PFNGLDEPTHFUNCPROC) (GLenum func); +typedef void (GL_APIENTRYP PFNGLDEPTHMASKPROC) (GLboolean flag); +typedef void (GL_APIENTRYP PFNGLDEPTHRANGEFPROC) (GLfloat n, GLfloat f); +typedef void (GL_APIENTRYP PFNGLDETACHSHADERPROC) (GLuint program, GLuint shader); +typedef void (GL_APIENTRYP PFNGLDISABLEPROC) (GLenum cap); +typedef void (GL_APIENTRYP PFNGLDISABLEVERTEXATTRIBARRAYPROC) (GLuint index); +typedef void (GL_APIENTRYP PFNGLDRAWARRAYSPROC) (GLenum mode, GLint first, GLsizei count); +typedef void (GL_APIENTRYP PFNGLDRAWELEMENTSPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices); +typedef void (GL_APIENTRYP PFNGLENABLEPROC) (GLenum cap); +typedef void (GL_APIENTRYP PFNGLENABLEVERTEXATTRIBARRAYPROC) (GLuint index); +typedef void (GL_APIENTRYP PFNGLFINISHPROC) (void); +typedef void (GL_APIENTRYP PFNGLFLUSHPROC) (void); +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERRENDERBUFFERPROC) (GLenum target, GLenum attachment, GLenum renderbuffertarget, GLuint renderbuffer); +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERTEXTURE2DPROC) (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +typedef void (GL_APIENTRYP PFNGLFRONTFACEPROC) (GLenum mode); +typedef void (GL_APIENTRYP PFNGLGENBUFFERSPROC) (GLsizei n, GLuint *buffers); +typedef void (GL_APIENTRYP PFNGLGENERATEMIPMAPPROC) (GLenum target); +typedef void (GL_APIENTRYP PFNGLGENFRAMEBUFFERSPROC) (GLsizei n, GLuint *framebuffers); +typedef void (GL_APIENTRYP PFNGLGENRENDERBUFFERSPROC) (GLsizei n, GLuint *renderbuffers); +typedef void (GL_APIENTRYP PFNGLGENTEXTURESPROC) (GLsizei n, GLuint *textures); +typedef void (GL_APIENTRYP PFNGLGETACTIVEATTRIBPROC) (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLint *size, GLenum *type, GLchar *name); +typedef void (GL_APIENTRYP PFNGLGETACTIVEUNIFORMPROC) (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLint *size, GLenum *type, GLchar *name); +typedef void (GL_APIENTRYP PFNGLGETATTACHEDSHADERSPROC) (GLuint program, GLsizei maxCount, GLsizei *count, GLuint *shaders); +typedef GLint (GL_APIENTRYP PFNGLGETATTRIBLOCATIONPROC) (GLuint program, const GLchar *name); +typedef void (GL_APIENTRYP PFNGLGETBOOLEANVPROC) (GLenum pname, GLboolean *data); +typedef void (GL_APIENTRYP PFNGLGETBUFFERPARAMETERIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef GLenum (GL_APIENTRYP PFNGLGETERRORPROC) (void); +typedef void (GL_APIENTRYP PFNGLGETFLOATVPROC) (GLenum pname, GLfloat *data); +typedef void (GL_APIENTRYP PFNGLGETFRAMEBUFFERATTACHMENTPARAMETERIVPROC) (GLenum target, GLenum attachment, GLenum pname, GLint *params); +typedef void (GL_APIENTRYP PFNGLGETINTEGERVPROC) (GLenum pname, GLint *data); +typedef void (GL_APIENTRYP PFNGLGETPROGRAMIVPROC) (GLuint program, GLenum pname, GLint *params); +typedef void (GL_APIENTRYP PFNGLGETPROGRAMINFOLOGPROC) (GLuint program, GLsizei bufSize, GLsizei *length, GLchar *infoLog); +typedef void (GL_APIENTRYP PFNGLGETRENDERBUFFERPARAMETERIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (GL_APIENTRYP PFNGLGETSHADERIVPROC) (GLuint shader, GLenum pname, GLint *params); +typedef void (GL_APIENTRYP PFNGLGETSHADERINFOLOGPROC) (GLuint shader, GLsizei bufSize, GLsizei *length, GLchar *infoLog); +typedef void (GL_APIENTRYP PFNGLGETSHADERPRECISIONFORMATPROC) (GLenum shadertype, GLenum precisiontype, GLint *range, GLint *precision); +typedef void (GL_APIENTRYP PFNGLGETSHADERSOURCEPROC) (GLuint shader, GLsizei bufSize, GLsizei *length, GLchar *source); +typedef const GLubyte *(GL_APIENTRYP PFNGLGETSTRINGPROC) (GLenum name); +typedef void (GL_APIENTRYP PFNGLGETTEXPARAMETERFVPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (GL_APIENTRYP PFNGLGETTEXPARAMETERIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (GL_APIENTRYP PFNGLGETUNIFORMFVPROC) (GLuint program, GLint location, GLfloat *params); +typedef void (GL_APIENTRYP PFNGLGETUNIFORMIVPROC) (GLuint program, GLint location, GLint *params); +typedef GLint (GL_APIENTRYP PFNGLGETUNIFORMLOCATIONPROC) (GLuint program, const GLchar *name); +typedef void (GL_APIENTRYP PFNGLGETVERTEXATTRIBFVPROC) (GLuint index, GLenum pname, GLfloat *params); +typedef void (GL_APIENTRYP PFNGLGETVERTEXATTRIBIVPROC) (GLuint index, GLenum pname, GLint *params); +typedef void (GL_APIENTRYP PFNGLGETVERTEXATTRIBPOINTERVPROC) (GLuint index, GLenum pname, void **pointer); +typedef void (GL_APIENTRYP PFNGLHINTPROC) (GLenum target, GLenum mode); +typedef GLboolean (GL_APIENTRYP PFNGLISBUFFERPROC) (GLuint buffer); +typedef GLboolean (GL_APIENTRYP PFNGLISENABLEDPROC) (GLenum cap); +typedef GLboolean (GL_APIENTRYP PFNGLISFRAMEBUFFERPROC) (GLuint framebuffer); +typedef GLboolean (GL_APIENTRYP PFNGLISPROGRAMPROC) (GLuint program); +typedef GLboolean (GL_APIENTRYP PFNGLISRENDERBUFFERPROC) (GLuint renderbuffer); +typedef GLboolean (GL_APIENTRYP PFNGLISSHADERPROC) (GLuint shader); +typedef GLboolean (GL_APIENTRYP PFNGLISTEXTUREPROC) (GLuint texture); +typedef void (GL_APIENTRYP PFNGLLINEWIDTHPROC) (GLfloat width); +typedef void (GL_APIENTRYP PFNGLLINKPROGRAMPROC) (GLuint program); +typedef void (GL_APIENTRYP PFNGLPIXELSTOREIPROC) (GLenum pname, GLint param); +typedef void (GL_APIENTRYP PFNGLPOLYGONOFFSETPROC) (GLfloat factor, GLfloat units); +typedef void (GL_APIENTRYP PFNGLREADPIXELSPROC) (GLint x, GLint y, GLsizei width, GLsizei height, GLenum format, GLenum type, void *pixels); +typedef void (GL_APIENTRYP PFNGLRELEASESHADERCOMPILERPROC) (void); +typedef void (GL_APIENTRYP PFNGLRENDERBUFFERSTORAGEPROC) (GLenum target, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (GL_APIENTRYP PFNGLSAMPLECOVERAGEPROC) (GLfloat value, GLboolean invert); +typedef void (GL_APIENTRYP PFNGLSCISSORPROC) (GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (GL_APIENTRYP PFNGLSHADERBINARYPROC) (GLsizei count, const GLuint *shaders, GLenum binaryFormat, const void *binary, GLsizei length); +typedef void (GL_APIENTRYP PFNGLSHADERSOURCEPROC) (GLuint shader, GLsizei count, const GLchar *const*string, const GLint *length); +typedef void (GL_APIENTRYP PFNGLSTENCILFUNCPROC) (GLenum func, GLint ref, GLuint mask); +typedef void (GL_APIENTRYP PFNGLSTENCILFUNCSEPARATEPROC) (GLenum face, GLenum func, GLint ref, GLuint mask); +typedef void (GL_APIENTRYP PFNGLSTENCILMASKPROC) (GLuint mask); +typedef void (GL_APIENTRYP PFNGLSTENCILMASKSEPARATEPROC) (GLenum face, GLuint mask); +typedef void (GL_APIENTRYP PFNGLSTENCILOPPROC) (GLenum fail, GLenum zfail, GLenum zpass); +typedef void (GL_APIENTRYP PFNGLSTENCILOPSEPARATEPROC) (GLenum face, GLenum sfail, GLenum dpfail, GLenum dppass); +typedef void (GL_APIENTRYP PFNGLTEXIMAGE2DPROC) (GLenum target, GLint level, GLint internalformat, GLsizei width, GLsizei height, GLint border, GLenum format, GLenum type, const void *pixels); +typedef void (GL_APIENTRYP PFNGLTEXPARAMETERFPROC) (GLenum target, GLenum pname, GLfloat param); +typedef void (GL_APIENTRYP PFNGLTEXPARAMETERFVPROC) (GLenum target, GLenum pname, const GLfloat *params); +typedef void (GL_APIENTRYP PFNGLTEXPARAMETERIPROC) (GLenum target, GLenum pname, GLint param); +typedef void (GL_APIENTRYP PFNGLTEXPARAMETERIVPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (GL_APIENTRYP PFNGLTEXSUBIMAGE2DPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *pixels); +typedef void (GL_APIENTRYP PFNGLUNIFORM1FPROC) (GLint location, GLfloat v0); +typedef void (GL_APIENTRYP PFNGLUNIFORM1FVPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLUNIFORM1IPROC) (GLint location, GLint v0); +typedef void (GL_APIENTRYP PFNGLUNIFORM1IVPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (GL_APIENTRYP PFNGLUNIFORM2FPROC) (GLint location, GLfloat v0, GLfloat v1); +typedef void (GL_APIENTRYP PFNGLUNIFORM2FVPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLUNIFORM2IPROC) (GLint location, GLint v0, GLint v1); +typedef void (GL_APIENTRYP PFNGLUNIFORM2IVPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (GL_APIENTRYP PFNGLUNIFORM3FPROC) (GLint location, GLfloat v0, GLfloat v1, GLfloat v2); +typedef void (GL_APIENTRYP PFNGLUNIFORM3FVPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLUNIFORM3IPROC) (GLint location, GLint v0, GLint v1, GLint v2); +typedef void (GL_APIENTRYP PFNGLUNIFORM3IVPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (GL_APIENTRYP PFNGLUNIFORM4FPROC) (GLint location, GLfloat v0, GLfloat v1, GLfloat v2, GLfloat v3); +typedef void (GL_APIENTRYP PFNGLUNIFORM4FVPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLUNIFORM4IPROC) (GLint location, GLint v0, GLint v1, GLint v2, GLint v3); +typedef void (GL_APIENTRYP PFNGLUNIFORM4IVPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (GL_APIENTRYP PFNGLUNIFORMMATRIX2FVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLUNIFORMMATRIX3FVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLUNIFORMMATRIX4FVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLUSEPROGRAMPROC) (GLuint program); +typedef void (GL_APIENTRYP PFNGLVALIDATEPROGRAMPROC) (GLuint program); +typedef void (GL_APIENTRYP PFNGLVERTEXATTRIB1FPROC) (GLuint index, GLfloat x); +typedef void (GL_APIENTRYP PFNGLVERTEXATTRIB1FVPROC) (GLuint index, const GLfloat *v); +typedef void (GL_APIENTRYP PFNGLVERTEXATTRIB2FPROC) (GLuint index, GLfloat x, GLfloat y); +typedef void (GL_APIENTRYP PFNGLVERTEXATTRIB2FVPROC) (GLuint index, const GLfloat *v); +typedef void (GL_APIENTRYP PFNGLVERTEXATTRIB3FPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat z); +typedef void (GL_APIENTRYP PFNGLVERTEXATTRIB3FVPROC) (GLuint index, const GLfloat *v); +typedef void (GL_APIENTRYP PFNGLVERTEXATTRIB4FPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (GL_APIENTRYP PFNGLVERTEXATTRIB4FVPROC) (GLuint index, const GLfloat *v); +typedef void (GL_APIENTRYP PFNGLVERTEXATTRIBPOINTERPROC) (GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride, const void *pointer); +typedef void (GL_APIENTRYP PFNGLVIEWPORTPROC) (GLint x, GLint y, GLsizei width, GLsizei height); +#if GL_GLES_PROTOTYPES +GL_APICALL void GL_APIENTRY glActiveTexture (GLenum texture); +GL_APICALL void GL_APIENTRY glAttachShader (GLuint program, GLuint shader); +GL_APICALL void GL_APIENTRY glBindAttribLocation (GLuint program, GLuint index, const GLchar *name); +GL_APICALL void GL_APIENTRY glBindBuffer (GLenum target, GLuint buffer); +GL_APICALL void GL_APIENTRY glBindFramebuffer (GLenum target, GLuint framebuffer); +GL_APICALL void GL_APIENTRY glBindRenderbuffer (GLenum target, GLuint renderbuffer); +GL_APICALL void GL_APIENTRY glBindTexture (GLenum target, GLuint texture); +GL_APICALL void GL_APIENTRY glBlendColor (GLfloat red, GLfloat green, GLfloat blue, GLfloat alpha); +GL_APICALL void GL_APIENTRY glBlendEquation (GLenum mode); +GL_APICALL void GL_APIENTRY glBlendEquationSeparate (GLenum modeRGB, GLenum modeAlpha); +GL_APICALL void GL_APIENTRY glBlendFunc (GLenum sfactor, GLenum dfactor); +GL_APICALL void GL_APIENTRY glBlendFuncSeparate (GLenum sfactorRGB, GLenum dfactorRGB, GLenum sfactorAlpha, GLenum dfactorAlpha); +GL_APICALL void GL_APIENTRY glBufferData (GLenum target, GLsizeiptr size, const void *data, GLenum usage); +GL_APICALL void GL_APIENTRY glBufferSubData (GLenum target, GLintptr offset, GLsizeiptr size, const void *data); +GL_APICALL GLenum GL_APIENTRY glCheckFramebufferStatus (GLenum target); +GL_APICALL void GL_APIENTRY glClear (GLbitfield mask); +GL_APICALL void GL_APIENTRY glClearColor (GLfloat red, GLfloat green, GLfloat blue, GLfloat alpha); +GL_APICALL void GL_APIENTRY glClearDepthf (GLfloat d); +GL_APICALL void GL_APIENTRY glClearStencil (GLint s); +GL_APICALL void GL_APIENTRY glColorMask (GLboolean red, GLboolean green, GLboolean blue, GLboolean alpha); +GL_APICALL void GL_APIENTRY glCompileShader (GLuint shader); +GL_APICALL void GL_APIENTRY glCompressedTexImage2D (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLsizei imageSize, const void *data); +GL_APICALL void GL_APIENTRY glCompressedTexSubImage2D (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const void *data); +GL_APICALL void GL_APIENTRY glCopyTexImage2D (GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height, GLint border); +GL_APICALL void GL_APIENTRY glCopyTexSubImage2D (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint x, GLint y, GLsizei width, GLsizei height); +GL_APICALL GLuint GL_APIENTRY glCreateProgram (void); +GL_APICALL GLuint GL_APIENTRY glCreateShader (GLenum type); +GL_APICALL void GL_APIENTRY glCullFace (GLenum mode); +GL_APICALL void GL_APIENTRY glDeleteBuffers (GLsizei n, const GLuint *buffers); +GL_APICALL void GL_APIENTRY glDeleteFramebuffers (GLsizei n, const GLuint *framebuffers); +GL_APICALL void GL_APIENTRY glDeleteProgram (GLuint program); +GL_APICALL void GL_APIENTRY glDeleteRenderbuffers (GLsizei n, const GLuint *renderbuffers); +GL_APICALL void GL_APIENTRY glDeleteShader (GLuint shader); +GL_APICALL void GL_APIENTRY glDeleteTextures (GLsizei n, const GLuint *textures); +GL_APICALL void GL_APIENTRY glDepthFunc (GLenum func); +GL_APICALL void GL_APIENTRY glDepthMask (GLboolean flag); +GL_APICALL void GL_APIENTRY glDepthRangef (GLfloat n, GLfloat f); +GL_APICALL void GL_APIENTRY glDetachShader (GLuint program, GLuint shader); +GL_APICALL void GL_APIENTRY glDisable (GLenum cap); +GL_APICALL void GL_APIENTRY glDisableVertexAttribArray (GLuint index); +GL_APICALL void GL_APIENTRY glDrawArrays (GLenum mode, GLint first, GLsizei count); +GL_APICALL void GL_APIENTRY glDrawElements (GLenum mode, GLsizei count, GLenum type, const void *indices); +GL_APICALL void GL_APIENTRY glEnable (GLenum cap); +GL_APICALL void GL_APIENTRY glEnableVertexAttribArray (GLuint index); +GL_APICALL void GL_APIENTRY glFinish (void); +GL_APICALL void GL_APIENTRY glFlush (void); +GL_APICALL void GL_APIENTRY glFramebufferRenderbuffer (GLenum target, GLenum attachment, GLenum renderbuffertarget, GLuint renderbuffer); +GL_APICALL void GL_APIENTRY glFramebufferTexture2D (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +GL_APICALL void GL_APIENTRY glFrontFace (GLenum mode); +GL_APICALL void GL_APIENTRY glGenBuffers (GLsizei n, GLuint *buffers); +GL_APICALL void GL_APIENTRY glGenerateMipmap (GLenum target); +GL_APICALL void GL_APIENTRY glGenFramebuffers (GLsizei n, GLuint *framebuffers); +GL_APICALL void GL_APIENTRY glGenRenderbuffers (GLsizei n, GLuint *renderbuffers); +GL_APICALL void GL_APIENTRY glGenTextures (GLsizei n, GLuint *textures); +GL_APICALL void GL_APIENTRY glGetActiveAttrib (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLint *size, GLenum *type, GLchar *name); +GL_APICALL void GL_APIENTRY glGetActiveUniform (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLint *size, GLenum *type, GLchar *name); +GL_APICALL void GL_APIENTRY glGetAttachedShaders (GLuint program, GLsizei maxCount, GLsizei *count, GLuint *shaders); +GL_APICALL GLint GL_APIENTRY glGetAttribLocation (GLuint program, const GLchar *name); +GL_APICALL void GL_APIENTRY glGetBooleanv (GLenum pname, GLboolean *data); +GL_APICALL void GL_APIENTRY glGetBufferParameteriv (GLenum target, GLenum pname, GLint *params); +GL_APICALL GLenum GL_APIENTRY glGetError (void); +GL_APICALL void GL_APIENTRY glGetFloatv (GLenum pname, GLfloat *data); +GL_APICALL void GL_APIENTRY glGetFramebufferAttachmentParameteriv (GLenum target, GLenum attachment, GLenum pname, GLint *params); +GL_APICALL void GL_APIENTRY glGetIntegerv (GLenum pname, GLint *data); +GL_APICALL void GL_APIENTRY glGetProgramiv (GLuint program, GLenum pname, GLint *params); +GL_APICALL void GL_APIENTRY glGetProgramInfoLog (GLuint program, GLsizei bufSize, GLsizei *length, GLchar *infoLog); +GL_APICALL void GL_APIENTRY glGetRenderbufferParameteriv (GLenum target, GLenum pname, GLint *params); +GL_APICALL void GL_APIENTRY glGetShaderiv (GLuint shader, GLenum pname, GLint *params); +GL_APICALL void GL_APIENTRY glGetShaderInfoLog (GLuint shader, GLsizei bufSize, GLsizei *length, GLchar *infoLog); +GL_APICALL void GL_APIENTRY glGetShaderPrecisionFormat (GLenum shadertype, GLenum precisiontype, GLint *range, GLint *precision); +GL_APICALL void GL_APIENTRY glGetShaderSource (GLuint shader, GLsizei bufSize, GLsizei *length, GLchar *source); +GL_APICALL const GLubyte *GL_APIENTRY glGetString (GLenum name); +GL_APICALL void GL_APIENTRY glGetTexParameterfv (GLenum target, GLenum pname, GLfloat *params); +GL_APICALL void GL_APIENTRY glGetTexParameteriv (GLenum target, GLenum pname, GLint *params); +GL_APICALL void GL_APIENTRY glGetUniformfv (GLuint program, GLint location, GLfloat *params); +GL_APICALL void GL_APIENTRY glGetUniformiv (GLuint program, GLint location, GLint *params); +GL_APICALL GLint GL_APIENTRY glGetUniformLocation (GLuint program, const GLchar *name); +GL_APICALL void GL_APIENTRY glGetVertexAttribfv (GLuint index, GLenum pname, GLfloat *params); +GL_APICALL void GL_APIENTRY glGetVertexAttribiv (GLuint index, GLenum pname, GLint *params); +GL_APICALL void GL_APIENTRY glGetVertexAttribPointerv (GLuint index, GLenum pname, void **pointer); +GL_APICALL void GL_APIENTRY glHint (GLenum target, GLenum mode); +GL_APICALL GLboolean GL_APIENTRY glIsBuffer (GLuint buffer); +GL_APICALL GLboolean GL_APIENTRY glIsEnabled (GLenum cap); +GL_APICALL GLboolean GL_APIENTRY glIsFramebuffer (GLuint framebuffer); +GL_APICALL GLboolean GL_APIENTRY glIsProgram (GLuint program); +GL_APICALL GLboolean GL_APIENTRY glIsRenderbuffer (GLuint renderbuffer); +GL_APICALL GLboolean GL_APIENTRY glIsShader (GLuint shader); +GL_APICALL GLboolean GL_APIENTRY glIsTexture (GLuint texture); +GL_APICALL void GL_APIENTRY glLineWidth (GLfloat width); +GL_APICALL void GL_APIENTRY glLinkProgram (GLuint program); +GL_APICALL void GL_APIENTRY glPixelStorei (GLenum pname, GLint param); +GL_APICALL void GL_APIENTRY glPolygonOffset (GLfloat factor, GLfloat units); +GL_APICALL void GL_APIENTRY glReadPixels (GLint x, GLint y, GLsizei width, GLsizei height, GLenum format, GLenum type, void *pixels); +GL_APICALL void GL_APIENTRY glReleaseShaderCompiler (void); +GL_APICALL void GL_APIENTRY glRenderbufferStorage (GLenum target, GLenum internalformat, GLsizei width, GLsizei height); +GL_APICALL void GL_APIENTRY glSampleCoverage (GLfloat value, GLboolean invert); +GL_APICALL void GL_APIENTRY glScissor (GLint x, GLint y, GLsizei width, GLsizei height); +GL_APICALL void GL_APIENTRY glShaderBinary (GLsizei count, const GLuint *shaders, GLenum binaryFormat, const void *binary, GLsizei length); +GL_APICALL void GL_APIENTRY glShaderSource (GLuint shader, GLsizei count, const GLchar *const*string, const GLint *length); +GL_APICALL void GL_APIENTRY glStencilFunc (GLenum func, GLint ref, GLuint mask); +GL_APICALL void GL_APIENTRY glStencilFuncSeparate (GLenum face, GLenum func, GLint ref, GLuint mask); +GL_APICALL void GL_APIENTRY glStencilMask (GLuint mask); +GL_APICALL void GL_APIENTRY glStencilMaskSeparate (GLenum face, GLuint mask); +GL_APICALL void GL_APIENTRY glStencilOp (GLenum fail, GLenum zfail, GLenum zpass); +GL_APICALL void GL_APIENTRY glStencilOpSeparate (GLenum face, GLenum sfail, GLenum dpfail, GLenum dppass); +GL_APICALL void GL_APIENTRY glTexImage2D (GLenum target, GLint level, GLint internalformat, GLsizei width, GLsizei height, GLint border, GLenum format, GLenum type, const void *pixels); +GL_APICALL void GL_APIENTRY glTexParameterf (GLenum target, GLenum pname, GLfloat param); +GL_APICALL void GL_APIENTRY glTexParameterfv (GLenum target, GLenum pname, const GLfloat *params); +GL_APICALL void GL_APIENTRY glTexParameteri (GLenum target, GLenum pname, GLint param); +GL_APICALL void GL_APIENTRY glTexParameteriv (GLenum target, GLenum pname, const GLint *params); +GL_APICALL void GL_APIENTRY glTexSubImage2D (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLenum type, const void *pixels); +GL_APICALL void GL_APIENTRY glUniform1f (GLint location, GLfloat v0); +GL_APICALL void GL_APIENTRY glUniform1fv (GLint location, GLsizei count, const GLfloat *value); +GL_APICALL void GL_APIENTRY glUniform1i (GLint location, GLint v0); +GL_APICALL void GL_APIENTRY glUniform1iv (GLint location, GLsizei count, const GLint *value); +GL_APICALL void GL_APIENTRY glUniform2f (GLint location, GLfloat v0, GLfloat v1); +GL_APICALL void GL_APIENTRY glUniform2fv (GLint location, GLsizei count, const GLfloat *value); +GL_APICALL void GL_APIENTRY glUniform2i (GLint location, GLint v0, GLint v1); +GL_APICALL void GL_APIENTRY glUniform2iv (GLint location, GLsizei count, const GLint *value); +GL_APICALL void GL_APIENTRY glUniform3f (GLint location, GLfloat v0, GLfloat v1, GLfloat v2); +GL_APICALL void GL_APIENTRY glUniform3fv (GLint location, GLsizei count, const GLfloat *value); +GL_APICALL void GL_APIENTRY glUniform3i (GLint location, GLint v0, GLint v1, GLint v2); +GL_APICALL void GL_APIENTRY glUniform3iv (GLint location, GLsizei count, const GLint *value); +GL_APICALL void GL_APIENTRY glUniform4f (GLint location, GLfloat v0, GLfloat v1, GLfloat v2, GLfloat v3); +GL_APICALL void GL_APIENTRY glUniform4fv (GLint location, GLsizei count, const GLfloat *value); +GL_APICALL void GL_APIENTRY glUniform4i (GLint location, GLint v0, GLint v1, GLint v2, GLint v3); +GL_APICALL void GL_APIENTRY glUniform4iv (GLint location, GLsizei count, const GLint *value); +GL_APICALL void GL_APIENTRY glUniformMatrix2fv (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glUniformMatrix3fv (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glUniformMatrix4fv (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glUseProgram (GLuint program); +GL_APICALL void GL_APIENTRY glValidateProgram (GLuint program); +GL_APICALL void GL_APIENTRY glVertexAttrib1f (GLuint index, GLfloat x); +GL_APICALL void GL_APIENTRY glVertexAttrib1fv (GLuint index, const GLfloat *v); +GL_APICALL void GL_APIENTRY glVertexAttrib2f (GLuint index, GLfloat x, GLfloat y); +GL_APICALL void GL_APIENTRY glVertexAttrib2fv (GLuint index, const GLfloat *v); +GL_APICALL void GL_APIENTRY glVertexAttrib3f (GLuint index, GLfloat x, GLfloat y, GLfloat z); +GL_APICALL void GL_APIENTRY glVertexAttrib3fv (GLuint index, const GLfloat *v); +GL_APICALL void GL_APIENTRY glVertexAttrib4f (GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +GL_APICALL void GL_APIENTRY glVertexAttrib4fv (GLuint index, const GLfloat *v); +GL_APICALL void GL_APIENTRY glVertexAttribPointer (GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride, const void *pointer); +GL_APICALL void GL_APIENTRY glViewport (GLint x, GLint y, GLsizei width, GLsizei height); +#endif +#endif /* GL_ES_VERSION_2_0 */ + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_gl2ext.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_gl2ext.h new file mode 100644 index 00000000..9448ce09 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_gl2ext.h @@ -0,0 +1,4033 @@ +#ifndef __gles2_gl2ext_h_ +#define __gles2_gl2ext_h_ 1 + +#ifdef __cplusplus +extern "C" { +#endif + +/* +** Copyright 2013-2020 The Khronos Group Inc. +** SPDX-License-Identifier: MIT +** +** This header is generated from the Khronos OpenGL / OpenGL ES XML +** API Registry. The current version of the Registry, generator scripts +** used to make the header, and the header can be found at +** https://github.com/KhronosGroup/OpenGL-Registry +*/ + +#ifndef GL_APIENTRYP +#define GL_APIENTRYP GL_APIENTRY* +#endif + +/* Generated on date 20220530 */ + +/* Generated C header for: + * API: gles2 + * Profile: common + * Versions considered: 2\.[0-9] + * Versions emitted: _nomatch_^ + * Default extensions included: gles2 + * Additional extensions included: _nomatch_^ + * Extensions removed: _nomatch_^ + */ + +#ifndef GL_KHR_blend_equation_advanced +#define GL_KHR_blend_equation_advanced 1 +#define GL_MULTIPLY_KHR 0x9294 +#define GL_SCREEN_KHR 0x9295 +#define GL_OVERLAY_KHR 0x9296 +#define GL_DARKEN_KHR 0x9297 +#define GL_LIGHTEN_KHR 0x9298 +#define GL_COLORDODGE_KHR 0x9299 +#define GL_COLORBURN_KHR 0x929A +#define GL_HARDLIGHT_KHR 0x929B +#define GL_SOFTLIGHT_KHR 0x929C +#define GL_DIFFERENCE_KHR 0x929E +#define GL_EXCLUSION_KHR 0x92A0 +#define GL_HSL_HUE_KHR 0x92AD +#define GL_HSL_SATURATION_KHR 0x92AE +#define GL_HSL_COLOR_KHR 0x92AF +#define GL_HSL_LUMINOSITY_KHR 0x92B0 +typedef void (GL_APIENTRYP PFNGLBLENDBARRIERKHRPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glBlendBarrierKHR (void); +#endif +#endif /* GL_KHR_blend_equation_advanced */ + +#ifndef GL_KHR_blend_equation_advanced_coherent +#define GL_KHR_blend_equation_advanced_coherent 1 +#define GL_BLEND_ADVANCED_COHERENT_KHR 0x9285 +#endif /* GL_KHR_blend_equation_advanced_coherent */ + +#ifndef GL_KHR_context_flush_control +#define GL_KHR_context_flush_control 1 +#define GL_CONTEXT_RELEASE_BEHAVIOR_KHR 0x82FB +#define GL_CONTEXT_RELEASE_BEHAVIOR_FLUSH_KHR 0x82FC +#endif /* GL_KHR_context_flush_control */ + +#ifndef GL_KHR_debug +#define GL_KHR_debug 1 +typedef void (GL_APIENTRY *GLDEBUGPROCKHR)(GLenum source,GLenum type,GLuint id,GLenum severity,GLsizei length,const GLchar *message,const void *userParam); +#define GL_SAMPLER 0x82E6 +#define GL_DEBUG_OUTPUT_SYNCHRONOUS_KHR 0x8242 +#define GL_DEBUG_NEXT_LOGGED_MESSAGE_LENGTH_KHR 0x8243 +#define GL_DEBUG_CALLBACK_FUNCTION_KHR 0x8244 +#define GL_DEBUG_CALLBACK_USER_PARAM_KHR 0x8245 +#define GL_DEBUG_SOURCE_API_KHR 0x8246 +#define GL_DEBUG_SOURCE_WINDOW_SYSTEM_KHR 0x8247 +#define GL_DEBUG_SOURCE_SHADER_COMPILER_KHR 0x8248 +#define GL_DEBUG_SOURCE_THIRD_PARTY_KHR 0x8249 +#define GL_DEBUG_SOURCE_APPLICATION_KHR 0x824A +#define GL_DEBUG_SOURCE_OTHER_KHR 0x824B +#define GL_DEBUG_TYPE_ERROR_KHR 0x824C +#define GL_DEBUG_TYPE_DEPRECATED_BEHAVIOR_KHR 0x824D +#define GL_DEBUG_TYPE_UNDEFINED_BEHAVIOR_KHR 0x824E +#define GL_DEBUG_TYPE_PORTABILITY_KHR 0x824F +#define GL_DEBUG_TYPE_PERFORMANCE_KHR 0x8250 +#define GL_DEBUG_TYPE_OTHER_KHR 0x8251 +#define GL_DEBUG_TYPE_MARKER_KHR 0x8268 +#define GL_DEBUG_TYPE_PUSH_GROUP_KHR 0x8269 +#define GL_DEBUG_TYPE_POP_GROUP_KHR 0x826A +#define GL_DEBUG_SEVERITY_NOTIFICATION_KHR 0x826B +#define GL_MAX_DEBUG_GROUP_STACK_DEPTH_KHR 0x826C +#define GL_DEBUG_GROUP_STACK_DEPTH_KHR 0x826D +#define GL_BUFFER_KHR 0x82E0 +#define GL_SHADER_KHR 0x82E1 +#define GL_PROGRAM_KHR 0x82E2 +#define GL_VERTEX_ARRAY_KHR 0x8074 +#define GL_QUERY_KHR 0x82E3 +#define GL_PROGRAM_PIPELINE_KHR 0x82E4 +#define GL_SAMPLER_KHR 0x82E6 +#define GL_MAX_LABEL_LENGTH_KHR 0x82E8 +#define GL_MAX_DEBUG_MESSAGE_LENGTH_KHR 0x9143 +#define GL_MAX_DEBUG_LOGGED_MESSAGES_KHR 0x9144 +#define GL_DEBUG_LOGGED_MESSAGES_KHR 0x9145 +#define GL_DEBUG_SEVERITY_HIGH_KHR 0x9146 +#define GL_DEBUG_SEVERITY_MEDIUM_KHR 0x9147 +#define GL_DEBUG_SEVERITY_LOW_KHR 0x9148 +#define GL_DEBUG_OUTPUT_KHR 0x92E0 +#define GL_CONTEXT_FLAG_DEBUG_BIT_KHR 0x00000002 +#define GL_STACK_OVERFLOW_KHR 0x0503 +#define GL_STACK_UNDERFLOW_KHR 0x0504 +typedef void (GL_APIENTRYP PFNGLDEBUGMESSAGECONTROLKHRPROC) (GLenum source, GLenum type, GLenum severity, GLsizei count, const GLuint *ids, GLboolean enabled); +typedef void (GL_APIENTRYP PFNGLDEBUGMESSAGEINSERTKHRPROC) (GLenum source, GLenum type, GLuint id, GLenum severity, GLsizei length, const GLchar *buf); +typedef void (GL_APIENTRYP PFNGLDEBUGMESSAGECALLBACKKHRPROC) (GLDEBUGPROCKHR callback, const void *userParam); +typedef GLuint (GL_APIENTRYP PFNGLGETDEBUGMESSAGELOGKHRPROC) (GLuint count, GLsizei bufSize, GLenum *sources, GLenum *types, GLuint *ids, GLenum *severities, GLsizei *lengths, GLchar *messageLog); +typedef void (GL_APIENTRYP PFNGLPUSHDEBUGGROUPKHRPROC) (GLenum source, GLuint id, GLsizei length, const GLchar *message); +typedef void (GL_APIENTRYP PFNGLPOPDEBUGGROUPKHRPROC) (void); +typedef void (GL_APIENTRYP PFNGLOBJECTLABELKHRPROC) (GLenum identifier, GLuint name, GLsizei length, const GLchar *label); +typedef void (GL_APIENTRYP PFNGLGETOBJECTLABELKHRPROC) (GLenum identifier, GLuint name, GLsizei bufSize, GLsizei *length, GLchar *label); +typedef void (GL_APIENTRYP PFNGLOBJECTPTRLABELKHRPROC) (const void *ptr, GLsizei length, const GLchar *label); +typedef void (GL_APIENTRYP PFNGLGETOBJECTPTRLABELKHRPROC) (const void *ptr, GLsizei bufSize, GLsizei *length, GLchar *label); +typedef void (GL_APIENTRYP PFNGLGETPOINTERVKHRPROC) (GLenum pname, void **params); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glDebugMessageControlKHR (GLenum source, GLenum type, GLenum severity, GLsizei count, const GLuint *ids, GLboolean enabled); +GL_APICALL void GL_APIENTRY glDebugMessageInsertKHR (GLenum source, GLenum type, GLuint id, GLenum severity, GLsizei length, const GLchar *buf); +GL_APICALL void GL_APIENTRY glDebugMessageCallbackKHR (GLDEBUGPROCKHR callback, const void *userParam); +GL_APICALL GLuint GL_APIENTRY glGetDebugMessageLogKHR (GLuint count, GLsizei bufSize, GLenum *sources, GLenum *types, GLuint *ids, GLenum *severities, GLsizei *lengths, GLchar *messageLog); +GL_APICALL void GL_APIENTRY glPushDebugGroupKHR (GLenum source, GLuint id, GLsizei length, const GLchar *message); +GL_APICALL void GL_APIENTRY glPopDebugGroupKHR (void); +GL_APICALL void GL_APIENTRY glObjectLabelKHR (GLenum identifier, GLuint name, GLsizei length, const GLchar *label); +GL_APICALL void GL_APIENTRY glGetObjectLabelKHR (GLenum identifier, GLuint name, GLsizei bufSize, GLsizei *length, GLchar *label); +GL_APICALL void GL_APIENTRY glObjectPtrLabelKHR (const void *ptr, GLsizei length, const GLchar *label); +GL_APICALL void GL_APIENTRY glGetObjectPtrLabelKHR (const void *ptr, GLsizei bufSize, GLsizei *length, GLchar *label); +GL_APICALL void GL_APIENTRY glGetPointervKHR (GLenum pname, void **params); +#endif +#endif /* GL_KHR_debug */ + +#ifndef GL_KHR_no_error +#define GL_KHR_no_error 1 +#define GL_CONTEXT_FLAG_NO_ERROR_BIT_KHR 0x00000008 +#endif /* GL_KHR_no_error */ + +#ifndef GL_KHR_parallel_shader_compile +#define GL_KHR_parallel_shader_compile 1 +#define GL_MAX_SHADER_COMPILER_THREADS_KHR 0x91B0 +#define GL_COMPLETION_STATUS_KHR 0x91B1 +typedef void (GL_APIENTRYP PFNGLMAXSHADERCOMPILERTHREADSKHRPROC) (GLuint count); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glMaxShaderCompilerThreadsKHR (GLuint count); +#endif +#endif /* GL_KHR_parallel_shader_compile */ + +#ifndef GL_KHR_robust_buffer_access_behavior +#define GL_KHR_robust_buffer_access_behavior 1 +#endif /* GL_KHR_robust_buffer_access_behavior */ + +#ifndef GL_KHR_robustness +#define GL_KHR_robustness 1 +#define GL_CONTEXT_ROBUST_ACCESS_KHR 0x90F3 +#define GL_LOSE_CONTEXT_ON_RESET_KHR 0x8252 +#define GL_GUILTY_CONTEXT_RESET_KHR 0x8253 +#define GL_INNOCENT_CONTEXT_RESET_KHR 0x8254 +#define GL_UNKNOWN_CONTEXT_RESET_KHR 0x8255 +#define GL_RESET_NOTIFICATION_STRATEGY_KHR 0x8256 +#define GL_NO_RESET_NOTIFICATION_KHR 0x8261 +#define GL_CONTEXT_LOST_KHR 0x0507 +typedef GLenum (GL_APIENTRYP PFNGLGETGRAPHICSRESETSTATUSKHRPROC) (void); +typedef void (GL_APIENTRYP PFNGLREADNPIXELSKHRPROC) (GLint x, GLint y, GLsizei width, GLsizei height, GLenum format, GLenum type, GLsizei bufSize, void *data); +typedef void (GL_APIENTRYP PFNGLGETNUNIFORMFVKHRPROC) (GLuint program, GLint location, GLsizei bufSize, GLfloat *params); +typedef void (GL_APIENTRYP PFNGLGETNUNIFORMIVKHRPROC) (GLuint program, GLint location, GLsizei bufSize, GLint *params); +typedef void (GL_APIENTRYP PFNGLGETNUNIFORMUIVKHRPROC) (GLuint program, GLint location, GLsizei bufSize, GLuint *params); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL GLenum GL_APIENTRY glGetGraphicsResetStatusKHR (void); +GL_APICALL void GL_APIENTRY glReadnPixelsKHR (GLint x, GLint y, GLsizei width, GLsizei height, GLenum format, GLenum type, GLsizei bufSize, void *data); +GL_APICALL void GL_APIENTRY glGetnUniformfvKHR (GLuint program, GLint location, GLsizei bufSize, GLfloat *params); +GL_APICALL void GL_APIENTRY glGetnUniformivKHR (GLuint program, GLint location, GLsizei bufSize, GLint *params); +GL_APICALL void GL_APIENTRY glGetnUniformuivKHR (GLuint program, GLint location, GLsizei bufSize, GLuint *params); +#endif +#endif /* GL_KHR_robustness */ + +#ifndef GL_KHR_shader_subgroup +#define GL_KHR_shader_subgroup 1 +#define GL_SUBGROUP_SIZE_KHR 0x9532 +#define GL_SUBGROUP_SUPPORTED_STAGES_KHR 0x9533 +#define GL_SUBGROUP_SUPPORTED_FEATURES_KHR 0x9534 +#define GL_SUBGROUP_QUAD_ALL_STAGES_KHR 0x9535 +#define GL_SUBGROUP_FEATURE_BASIC_BIT_KHR 0x00000001 +#define GL_SUBGROUP_FEATURE_VOTE_BIT_KHR 0x00000002 +#define GL_SUBGROUP_FEATURE_ARITHMETIC_BIT_KHR 0x00000004 +#define GL_SUBGROUP_FEATURE_BALLOT_BIT_KHR 0x00000008 +#define GL_SUBGROUP_FEATURE_SHUFFLE_BIT_KHR 0x00000010 +#define GL_SUBGROUP_FEATURE_SHUFFLE_RELATIVE_BIT_KHR 0x00000020 +#define GL_SUBGROUP_FEATURE_CLUSTERED_BIT_KHR 0x00000040 +#define GL_SUBGROUP_FEATURE_QUAD_BIT_KHR 0x00000080 +#endif /* GL_KHR_shader_subgroup */ + +#ifndef GL_KHR_texture_compression_astc_hdr +#define GL_KHR_texture_compression_astc_hdr 1 +#define GL_COMPRESSED_RGBA_ASTC_4x4_KHR 0x93B0 +#define GL_COMPRESSED_RGBA_ASTC_5x4_KHR 0x93B1 +#define GL_COMPRESSED_RGBA_ASTC_5x5_KHR 0x93B2 +#define GL_COMPRESSED_RGBA_ASTC_6x5_KHR 0x93B3 +#define GL_COMPRESSED_RGBA_ASTC_6x6_KHR 0x93B4 +#define GL_COMPRESSED_RGBA_ASTC_8x5_KHR 0x93B5 +#define GL_COMPRESSED_RGBA_ASTC_8x6_KHR 0x93B6 +#define GL_COMPRESSED_RGBA_ASTC_8x8_KHR 0x93B7 +#define GL_COMPRESSED_RGBA_ASTC_10x5_KHR 0x93B8 +#define GL_COMPRESSED_RGBA_ASTC_10x6_KHR 0x93B9 +#define GL_COMPRESSED_RGBA_ASTC_10x8_KHR 0x93BA +#define GL_COMPRESSED_RGBA_ASTC_10x10_KHR 0x93BB +#define GL_COMPRESSED_RGBA_ASTC_12x10_KHR 0x93BC +#define GL_COMPRESSED_RGBA_ASTC_12x12_KHR 0x93BD +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR 0x93D0 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_5x4_KHR 0x93D1 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_5x5_KHR 0x93D2 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_6x5_KHR 0x93D3 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_6x6_KHR 0x93D4 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_8x5_KHR 0x93D5 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_8x6_KHR 0x93D6 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_8x8_KHR 0x93D7 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x5_KHR 0x93D8 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x6_KHR 0x93D9 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x8_KHR 0x93DA +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x10_KHR 0x93DB +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_12x10_KHR 0x93DC +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_12x12_KHR 0x93DD +#endif /* GL_KHR_texture_compression_astc_hdr */ + +#ifndef GL_KHR_texture_compression_astc_ldr +#define GL_KHR_texture_compression_astc_ldr 1 +#endif /* GL_KHR_texture_compression_astc_ldr */ + +#ifndef GL_KHR_texture_compression_astc_sliced_3d +#define GL_KHR_texture_compression_astc_sliced_3d 1 +#endif /* GL_KHR_texture_compression_astc_sliced_3d */ + +#ifndef GL_OES_EGL_image +#define GL_OES_EGL_image 1 +typedef void *GLeglImageOES; +typedef void (GL_APIENTRYP PFNGLEGLIMAGETARGETTEXTURE2DOESPROC) (GLenum target, GLeglImageOES image); +typedef void (GL_APIENTRYP PFNGLEGLIMAGETARGETRENDERBUFFERSTORAGEOESPROC) (GLenum target, GLeglImageOES image); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glEGLImageTargetTexture2DOES (GLenum target, GLeglImageOES image); +GL_APICALL void GL_APIENTRY glEGLImageTargetRenderbufferStorageOES (GLenum target, GLeglImageOES image); +#endif +#endif /* GL_OES_EGL_image */ + +#ifndef GL_OES_EGL_image_external +#define GL_OES_EGL_image_external 1 +#define GL_TEXTURE_EXTERNAL_OES 0x8D65 +#define GL_TEXTURE_BINDING_EXTERNAL_OES 0x8D67 +#define GL_REQUIRED_TEXTURE_IMAGE_UNITS_OES 0x8D68 +#define GL_SAMPLER_EXTERNAL_OES 0x8D66 +#endif /* GL_OES_EGL_image_external */ + +#ifndef GL_OES_EGL_image_external_essl3 +#define GL_OES_EGL_image_external_essl3 1 +#endif /* GL_OES_EGL_image_external_essl3 */ + +#ifndef GL_OES_compressed_ETC1_RGB8_sub_texture +#define GL_OES_compressed_ETC1_RGB8_sub_texture 1 +#endif /* GL_OES_compressed_ETC1_RGB8_sub_texture */ + +#ifndef GL_OES_compressed_ETC1_RGB8_texture +#define GL_OES_compressed_ETC1_RGB8_texture 1 +#define GL_ETC1_RGB8_OES 0x8D64 +#endif /* GL_OES_compressed_ETC1_RGB8_texture */ + +#ifndef GL_OES_compressed_paletted_texture +#define GL_OES_compressed_paletted_texture 1 +#define GL_PALETTE4_RGB8_OES 0x8B90 +#define GL_PALETTE4_RGBA8_OES 0x8B91 +#define GL_PALETTE4_R5_G6_B5_OES 0x8B92 +#define GL_PALETTE4_RGBA4_OES 0x8B93 +#define GL_PALETTE4_RGB5_A1_OES 0x8B94 +#define GL_PALETTE8_RGB8_OES 0x8B95 +#define GL_PALETTE8_RGBA8_OES 0x8B96 +#define GL_PALETTE8_R5_G6_B5_OES 0x8B97 +#define GL_PALETTE8_RGBA4_OES 0x8B98 +#define GL_PALETTE8_RGB5_A1_OES 0x8B99 +#endif /* GL_OES_compressed_paletted_texture */ + +#ifndef GL_OES_copy_image +#define GL_OES_copy_image 1 +typedef void (GL_APIENTRYP PFNGLCOPYIMAGESUBDATAOESPROC) (GLuint srcName, GLenum srcTarget, GLint srcLevel, GLint srcX, GLint srcY, GLint srcZ, GLuint dstName, GLenum dstTarget, GLint dstLevel, GLint dstX, GLint dstY, GLint dstZ, GLsizei srcWidth, GLsizei srcHeight, GLsizei srcDepth); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glCopyImageSubDataOES (GLuint srcName, GLenum srcTarget, GLint srcLevel, GLint srcX, GLint srcY, GLint srcZ, GLuint dstName, GLenum dstTarget, GLint dstLevel, GLint dstX, GLint dstY, GLint dstZ, GLsizei srcWidth, GLsizei srcHeight, GLsizei srcDepth); +#endif +#endif /* GL_OES_copy_image */ + +#ifndef GL_OES_depth24 +#define GL_OES_depth24 1 +#define GL_DEPTH_COMPONENT24_OES 0x81A6 +#endif /* GL_OES_depth24 */ + +#ifndef GL_OES_depth32 +#define GL_OES_depth32 1 +#define GL_DEPTH_COMPONENT32_OES 0x81A7 +#endif /* GL_OES_depth32 */ + +#ifndef GL_OES_depth_texture +#define GL_OES_depth_texture 1 +#endif /* GL_OES_depth_texture */ + +#ifndef GL_OES_draw_buffers_indexed +#define GL_OES_draw_buffers_indexed 1 +#define GL_MIN 0x8007 +#define GL_MAX 0x8008 +typedef void (GL_APIENTRYP PFNGLENABLEIOESPROC) (GLenum target, GLuint index); +typedef void (GL_APIENTRYP PFNGLDISABLEIOESPROC) (GLenum target, GLuint index); +typedef void (GL_APIENTRYP PFNGLBLENDEQUATIONIOESPROC) (GLuint buf, GLenum mode); +typedef void (GL_APIENTRYP PFNGLBLENDEQUATIONSEPARATEIOESPROC) (GLuint buf, GLenum modeRGB, GLenum modeAlpha); +typedef void (GL_APIENTRYP PFNGLBLENDFUNCIOESPROC) (GLuint buf, GLenum src, GLenum dst); +typedef void (GL_APIENTRYP PFNGLBLENDFUNCSEPARATEIOESPROC) (GLuint buf, GLenum srcRGB, GLenum dstRGB, GLenum srcAlpha, GLenum dstAlpha); +typedef void (GL_APIENTRYP PFNGLCOLORMASKIOESPROC) (GLuint index, GLboolean r, GLboolean g, GLboolean b, GLboolean a); +typedef GLboolean (GL_APIENTRYP PFNGLISENABLEDIOESPROC) (GLenum target, GLuint index); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glEnableiOES (GLenum target, GLuint index); +GL_APICALL void GL_APIENTRY glDisableiOES (GLenum target, GLuint index); +GL_APICALL void GL_APIENTRY glBlendEquationiOES (GLuint buf, GLenum mode); +GL_APICALL void GL_APIENTRY glBlendEquationSeparateiOES (GLuint buf, GLenum modeRGB, GLenum modeAlpha); +GL_APICALL void GL_APIENTRY glBlendFunciOES (GLuint buf, GLenum src, GLenum dst); +GL_APICALL void GL_APIENTRY glBlendFuncSeparateiOES (GLuint buf, GLenum srcRGB, GLenum dstRGB, GLenum srcAlpha, GLenum dstAlpha); +GL_APICALL void GL_APIENTRY glColorMaskiOES (GLuint index, GLboolean r, GLboolean g, GLboolean b, GLboolean a); +GL_APICALL GLboolean GL_APIENTRY glIsEnablediOES (GLenum target, GLuint index); +#endif +#endif /* GL_OES_draw_buffers_indexed */ + +#ifndef GL_OES_draw_elements_base_vertex +#define GL_OES_draw_elements_base_vertex 1 +typedef void (GL_APIENTRYP PFNGLDRAWELEMENTSBASEVERTEXOESPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLint basevertex); +typedef void (GL_APIENTRYP PFNGLDRAWRANGEELEMENTSBASEVERTEXOESPROC) (GLenum mode, GLuint start, GLuint end, GLsizei count, GLenum type, const void *indices, GLint basevertex); +typedef void (GL_APIENTRYP PFNGLDRAWELEMENTSINSTANCEDBASEVERTEXOESPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount, GLint basevertex); +typedef void (GL_APIENTRYP PFNGLMULTIDRAWELEMENTSBASEVERTEXEXTPROC) (GLenum mode, const GLsizei *count, GLenum type, const void *const*indices, GLsizei drawcount, const GLint *basevertex); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glDrawElementsBaseVertexOES (GLenum mode, GLsizei count, GLenum type, const void *indices, GLint basevertex); +GL_APICALL void GL_APIENTRY glDrawRangeElementsBaseVertexOES (GLenum mode, GLuint start, GLuint end, GLsizei count, GLenum type, const void *indices, GLint basevertex); +GL_APICALL void GL_APIENTRY glDrawElementsInstancedBaseVertexOES (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount, GLint basevertex); +GL_APICALL void GL_APIENTRY glMultiDrawElementsBaseVertexEXT (GLenum mode, const GLsizei *count, GLenum type, const void *const*indices, GLsizei drawcount, const GLint *basevertex); +#endif +#endif /* GL_OES_draw_elements_base_vertex */ + +#ifndef GL_OES_element_index_uint +#define GL_OES_element_index_uint 1 +#endif /* GL_OES_element_index_uint */ + +#ifndef GL_OES_fbo_render_mipmap +#define GL_OES_fbo_render_mipmap 1 +#endif /* GL_OES_fbo_render_mipmap */ + +#ifndef GL_OES_fragment_precision_high +#define GL_OES_fragment_precision_high 1 +#endif /* GL_OES_fragment_precision_high */ + +#ifndef GL_OES_geometry_point_size +#define GL_OES_geometry_point_size 1 +#endif /* GL_OES_geometry_point_size */ + +#ifndef GL_OES_geometry_shader +#define GL_OES_geometry_shader 1 +#define GL_GEOMETRY_SHADER_OES 0x8DD9 +#define GL_GEOMETRY_SHADER_BIT_OES 0x00000004 +#define GL_GEOMETRY_LINKED_VERTICES_OUT_OES 0x8916 +#define GL_GEOMETRY_LINKED_INPUT_TYPE_OES 0x8917 +#define GL_GEOMETRY_LINKED_OUTPUT_TYPE_OES 0x8918 +#define GL_GEOMETRY_SHADER_INVOCATIONS_OES 0x887F +#define GL_LAYER_PROVOKING_VERTEX_OES 0x825E +#define GL_LINES_ADJACENCY_OES 0x000A +#define GL_LINE_STRIP_ADJACENCY_OES 0x000B +#define GL_TRIANGLES_ADJACENCY_OES 0x000C +#define GL_TRIANGLE_STRIP_ADJACENCY_OES 0x000D +#define GL_MAX_GEOMETRY_UNIFORM_COMPONENTS_OES 0x8DDF +#define GL_MAX_GEOMETRY_UNIFORM_BLOCKS_OES 0x8A2C +#define GL_MAX_COMBINED_GEOMETRY_UNIFORM_COMPONENTS_OES 0x8A32 +#define GL_MAX_GEOMETRY_INPUT_COMPONENTS_OES 0x9123 +#define GL_MAX_GEOMETRY_OUTPUT_COMPONENTS_OES 0x9124 +#define GL_MAX_GEOMETRY_OUTPUT_VERTICES_OES 0x8DE0 +#define GL_MAX_GEOMETRY_TOTAL_OUTPUT_COMPONENTS_OES 0x8DE1 +#define GL_MAX_GEOMETRY_SHADER_INVOCATIONS_OES 0x8E5A +#define GL_MAX_GEOMETRY_TEXTURE_IMAGE_UNITS_OES 0x8C29 +#define GL_MAX_GEOMETRY_ATOMIC_COUNTER_BUFFERS_OES 0x92CF +#define GL_MAX_GEOMETRY_ATOMIC_COUNTERS_OES 0x92D5 +#define GL_MAX_GEOMETRY_IMAGE_UNIFORMS_OES 0x90CD +#define GL_MAX_GEOMETRY_SHADER_STORAGE_BLOCKS_OES 0x90D7 +#define GL_FIRST_VERTEX_CONVENTION_OES 0x8E4D +#define GL_LAST_VERTEX_CONVENTION_OES 0x8E4E +#define GL_UNDEFINED_VERTEX_OES 0x8260 +#define GL_PRIMITIVES_GENERATED_OES 0x8C87 +#define GL_FRAMEBUFFER_DEFAULT_LAYERS_OES 0x9312 +#define GL_MAX_FRAMEBUFFER_LAYERS_OES 0x9317 +#define GL_FRAMEBUFFER_INCOMPLETE_LAYER_TARGETS_OES 0x8DA8 +#define GL_FRAMEBUFFER_ATTACHMENT_LAYERED_OES 0x8DA7 +#define GL_REFERENCED_BY_GEOMETRY_SHADER_OES 0x9309 +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERTEXTUREOESPROC) (GLenum target, GLenum attachment, GLuint texture, GLint level); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glFramebufferTextureOES (GLenum target, GLenum attachment, GLuint texture, GLint level); +#endif +#endif /* GL_OES_geometry_shader */ + +#ifndef GL_OES_get_program_binary +#define GL_OES_get_program_binary 1 +#define GL_PROGRAM_BINARY_LENGTH_OES 0x8741 +#define GL_NUM_PROGRAM_BINARY_FORMATS_OES 0x87FE +#define GL_PROGRAM_BINARY_FORMATS_OES 0x87FF +typedef void (GL_APIENTRYP PFNGLGETPROGRAMBINARYOESPROC) (GLuint program, GLsizei bufSize, GLsizei *length, GLenum *binaryFormat, void *binary); +typedef void (GL_APIENTRYP PFNGLPROGRAMBINARYOESPROC) (GLuint program, GLenum binaryFormat, const void *binary, GLint length); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glGetProgramBinaryOES (GLuint program, GLsizei bufSize, GLsizei *length, GLenum *binaryFormat, void *binary); +GL_APICALL void GL_APIENTRY glProgramBinaryOES (GLuint program, GLenum binaryFormat, const void *binary, GLint length); +#endif +#endif /* GL_OES_get_program_binary */ + +#ifndef GL_OES_gpu_shader5 +#define GL_OES_gpu_shader5 1 +#endif /* GL_OES_gpu_shader5 */ + +#ifndef GL_OES_mapbuffer +#define GL_OES_mapbuffer 1 +#define GL_WRITE_ONLY_OES 0x88B9 +#define GL_BUFFER_ACCESS_OES 0x88BB +#define GL_BUFFER_MAPPED_OES 0x88BC +#define GL_BUFFER_MAP_POINTER_OES 0x88BD +typedef void *(GL_APIENTRYP PFNGLMAPBUFFEROESPROC) (GLenum target, GLenum access); +typedef GLboolean (GL_APIENTRYP PFNGLUNMAPBUFFEROESPROC) (GLenum target); +typedef void (GL_APIENTRYP PFNGLGETBUFFERPOINTERVOESPROC) (GLenum target, GLenum pname, void **params); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void *GL_APIENTRY glMapBufferOES (GLenum target, GLenum access); +GL_APICALL GLboolean GL_APIENTRY glUnmapBufferOES (GLenum target); +GL_APICALL void GL_APIENTRY glGetBufferPointervOES (GLenum target, GLenum pname, void **params); +#endif +#endif /* GL_OES_mapbuffer */ + +#ifndef GL_OES_packed_depth_stencil +#define GL_OES_packed_depth_stencil 1 +#define GL_DEPTH_STENCIL_OES 0x84F9 +#define GL_UNSIGNED_INT_24_8_OES 0x84FA +#define GL_DEPTH24_STENCIL8_OES 0x88F0 +#endif /* GL_OES_packed_depth_stencil */ + +#ifndef GL_OES_primitive_bounding_box +#define GL_OES_primitive_bounding_box 1 +#define GL_PRIMITIVE_BOUNDING_BOX_OES 0x92BE +typedef void (GL_APIENTRYP PFNGLPRIMITIVEBOUNDINGBOXOESPROC) (GLfloat minX, GLfloat minY, GLfloat minZ, GLfloat minW, GLfloat maxX, GLfloat maxY, GLfloat maxZ, GLfloat maxW); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glPrimitiveBoundingBoxOES (GLfloat minX, GLfloat minY, GLfloat minZ, GLfloat minW, GLfloat maxX, GLfloat maxY, GLfloat maxZ, GLfloat maxW); +#endif +#endif /* GL_OES_primitive_bounding_box */ + +#ifndef GL_OES_required_internalformat +#define GL_OES_required_internalformat 1 +#define GL_ALPHA8_OES 0x803C +#define GL_DEPTH_COMPONENT16_OES 0x81A5 +#define GL_LUMINANCE4_ALPHA4_OES 0x8043 +#define GL_LUMINANCE8_ALPHA8_OES 0x8045 +#define GL_LUMINANCE8_OES 0x8040 +#define GL_RGBA4_OES 0x8056 +#define GL_RGB5_A1_OES 0x8057 +#define GL_RGB565_OES 0x8D62 +#define GL_RGB8_OES 0x8051 +#define GL_RGBA8_OES 0x8058 +#define GL_RGB10_EXT 0x8052 +#define GL_RGB10_A2_EXT 0x8059 +#endif /* GL_OES_required_internalformat */ + +#ifndef GL_OES_rgb8_rgba8 +#define GL_OES_rgb8_rgba8 1 +#endif /* GL_OES_rgb8_rgba8 */ + +#ifndef GL_OES_sample_shading +#define GL_OES_sample_shading 1 +#define GL_SAMPLE_SHADING_OES 0x8C36 +#define GL_MIN_SAMPLE_SHADING_VALUE_OES 0x8C37 +typedef void (GL_APIENTRYP PFNGLMINSAMPLESHADINGOESPROC) (GLfloat value); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glMinSampleShadingOES (GLfloat value); +#endif +#endif /* GL_OES_sample_shading */ + +#ifndef GL_OES_sample_variables +#define GL_OES_sample_variables 1 +#endif /* GL_OES_sample_variables */ + +#ifndef GL_OES_shader_image_atomic +#define GL_OES_shader_image_atomic 1 +#endif /* GL_OES_shader_image_atomic */ + +#ifndef GL_OES_shader_io_blocks +#define GL_OES_shader_io_blocks 1 +#endif /* GL_OES_shader_io_blocks */ + +#ifndef GL_OES_shader_multisample_interpolation +#define GL_OES_shader_multisample_interpolation 1 +#define GL_MIN_FRAGMENT_INTERPOLATION_OFFSET_OES 0x8E5B +#define GL_MAX_FRAGMENT_INTERPOLATION_OFFSET_OES 0x8E5C +#define GL_FRAGMENT_INTERPOLATION_OFFSET_BITS_OES 0x8E5D +#endif /* GL_OES_shader_multisample_interpolation */ + +#ifndef GL_OES_standard_derivatives +#define GL_OES_standard_derivatives 1 +#define GL_FRAGMENT_SHADER_DERIVATIVE_HINT_OES 0x8B8B +#endif /* GL_OES_standard_derivatives */ + +#ifndef GL_OES_stencil1 +#define GL_OES_stencil1 1 +#define GL_STENCIL_INDEX1_OES 0x8D46 +#endif /* GL_OES_stencil1 */ + +#ifndef GL_OES_stencil4 +#define GL_OES_stencil4 1 +#define GL_STENCIL_INDEX4_OES 0x8D47 +#endif /* GL_OES_stencil4 */ + +#ifndef GL_OES_surfaceless_context +#define GL_OES_surfaceless_context 1 +#define GL_FRAMEBUFFER_UNDEFINED_OES 0x8219 +#endif /* GL_OES_surfaceless_context */ + +#ifndef GL_OES_tessellation_point_size +#define GL_OES_tessellation_point_size 1 +#endif /* GL_OES_tessellation_point_size */ + +#ifndef GL_OES_tessellation_shader +#define GL_OES_tessellation_shader 1 +#define GL_PATCHES_OES 0x000E +#define GL_PATCH_VERTICES_OES 0x8E72 +#define GL_TESS_CONTROL_OUTPUT_VERTICES_OES 0x8E75 +#define GL_TESS_GEN_MODE_OES 0x8E76 +#define GL_TESS_GEN_SPACING_OES 0x8E77 +#define GL_TESS_GEN_VERTEX_ORDER_OES 0x8E78 +#define GL_TESS_GEN_POINT_MODE_OES 0x8E79 +#define GL_ISOLINES_OES 0x8E7A +#define GL_QUADS_OES 0x0007 +#define GL_FRACTIONAL_ODD_OES 0x8E7B +#define GL_FRACTIONAL_EVEN_OES 0x8E7C +#define GL_MAX_PATCH_VERTICES_OES 0x8E7D +#define GL_MAX_TESS_GEN_LEVEL_OES 0x8E7E +#define GL_MAX_TESS_CONTROL_UNIFORM_COMPONENTS_OES 0x8E7F +#define GL_MAX_TESS_EVALUATION_UNIFORM_COMPONENTS_OES 0x8E80 +#define GL_MAX_TESS_CONTROL_TEXTURE_IMAGE_UNITS_OES 0x8E81 +#define GL_MAX_TESS_EVALUATION_TEXTURE_IMAGE_UNITS_OES 0x8E82 +#define GL_MAX_TESS_CONTROL_OUTPUT_COMPONENTS_OES 0x8E83 +#define GL_MAX_TESS_PATCH_COMPONENTS_OES 0x8E84 +#define GL_MAX_TESS_CONTROL_TOTAL_OUTPUT_COMPONENTS_OES 0x8E85 +#define GL_MAX_TESS_EVALUATION_OUTPUT_COMPONENTS_OES 0x8E86 +#define GL_MAX_TESS_CONTROL_UNIFORM_BLOCKS_OES 0x8E89 +#define GL_MAX_TESS_EVALUATION_UNIFORM_BLOCKS_OES 0x8E8A +#define GL_MAX_TESS_CONTROL_INPUT_COMPONENTS_OES 0x886C +#define GL_MAX_TESS_EVALUATION_INPUT_COMPONENTS_OES 0x886D +#define GL_MAX_COMBINED_TESS_CONTROL_UNIFORM_COMPONENTS_OES 0x8E1E +#define GL_MAX_COMBINED_TESS_EVALUATION_UNIFORM_COMPONENTS_OES 0x8E1F +#define GL_MAX_TESS_CONTROL_ATOMIC_COUNTER_BUFFERS_OES 0x92CD +#define GL_MAX_TESS_EVALUATION_ATOMIC_COUNTER_BUFFERS_OES 0x92CE +#define GL_MAX_TESS_CONTROL_ATOMIC_COUNTERS_OES 0x92D3 +#define GL_MAX_TESS_EVALUATION_ATOMIC_COUNTERS_OES 0x92D4 +#define GL_MAX_TESS_CONTROL_IMAGE_UNIFORMS_OES 0x90CB +#define GL_MAX_TESS_EVALUATION_IMAGE_UNIFORMS_OES 0x90CC +#define GL_MAX_TESS_CONTROL_SHADER_STORAGE_BLOCKS_OES 0x90D8 +#define GL_MAX_TESS_EVALUATION_SHADER_STORAGE_BLOCKS_OES 0x90D9 +#define GL_PRIMITIVE_RESTART_FOR_PATCHES_SUPPORTED_OES 0x8221 +#define GL_IS_PER_PATCH_OES 0x92E7 +#define GL_REFERENCED_BY_TESS_CONTROL_SHADER_OES 0x9307 +#define GL_REFERENCED_BY_TESS_EVALUATION_SHADER_OES 0x9308 +#define GL_TESS_CONTROL_SHADER_OES 0x8E88 +#define GL_TESS_EVALUATION_SHADER_OES 0x8E87 +#define GL_TESS_CONTROL_SHADER_BIT_OES 0x00000008 +#define GL_TESS_EVALUATION_SHADER_BIT_OES 0x00000010 +typedef void (GL_APIENTRYP PFNGLPATCHPARAMETERIOESPROC) (GLenum pname, GLint value); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glPatchParameteriOES (GLenum pname, GLint value); +#endif +#endif /* GL_OES_tessellation_shader */ + +#ifndef GL_OES_texture_3D +#define GL_OES_texture_3D 1 +#define GL_TEXTURE_WRAP_R_OES 0x8072 +#define GL_TEXTURE_3D_OES 0x806F +#define GL_TEXTURE_BINDING_3D_OES 0x806A +#define GL_MAX_3D_TEXTURE_SIZE_OES 0x8073 +#define GL_SAMPLER_3D_OES 0x8B5F +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_3D_ZOFFSET_OES 0x8CD4 +typedef void (GL_APIENTRYP PFNGLTEXIMAGE3DOESPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLenum format, GLenum type, const void *pixels); +typedef void (GL_APIENTRYP PFNGLTEXSUBIMAGE3DOESPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *pixels); +typedef void (GL_APIENTRYP PFNGLCOPYTEXSUBIMAGE3DOESPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (GL_APIENTRYP PFNGLCOMPRESSEDTEXIMAGE3DOESPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLsizei imageSize, const void *data); +typedef void (GL_APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE3DOESPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const void *data); +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERTEXTURE3DOESPROC) (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level, GLint zoffset); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glTexImage3DOES (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLenum format, GLenum type, const void *pixels); +GL_APICALL void GL_APIENTRY glTexSubImage3DOES (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *pixels); +GL_APICALL void GL_APIENTRY glCopyTexSubImage3DOES (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint x, GLint y, GLsizei width, GLsizei height); +GL_APICALL void GL_APIENTRY glCompressedTexImage3DOES (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLsizei imageSize, const void *data); +GL_APICALL void GL_APIENTRY glCompressedTexSubImage3DOES (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const void *data); +GL_APICALL void GL_APIENTRY glFramebufferTexture3DOES (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level, GLint zoffset); +#endif +#endif /* GL_OES_texture_3D */ + +#ifndef GL_OES_texture_border_clamp +#define GL_OES_texture_border_clamp 1 +#define GL_TEXTURE_BORDER_COLOR_OES 0x1004 +#define GL_CLAMP_TO_BORDER_OES 0x812D +typedef void (GL_APIENTRYP PFNGLTEXPARAMETERIIVOESPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (GL_APIENTRYP PFNGLTEXPARAMETERIUIVOESPROC) (GLenum target, GLenum pname, const GLuint *params); +typedef void (GL_APIENTRYP PFNGLGETTEXPARAMETERIIVOESPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (GL_APIENTRYP PFNGLGETTEXPARAMETERIUIVOESPROC) (GLenum target, GLenum pname, GLuint *params); +typedef void (GL_APIENTRYP PFNGLSAMPLERPARAMETERIIVOESPROC) (GLuint sampler, GLenum pname, const GLint *param); +typedef void (GL_APIENTRYP PFNGLSAMPLERPARAMETERIUIVOESPROC) (GLuint sampler, GLenum pname, const GLuint *param); +typedef void (GL_APIENTRYP PFNGLGETSAMPLERPARAMETERIIVOESPROC) (GLuint sampler, GLenum pname, GLint *params); +typedef void (GL_APIENTRYP PFNGLGETSAMPLERPARAMETERIUIVOESPROC) (GLuint sampler, GLenum pname, GLuint *params); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glTexParameterIivOES (GLenum target, GLenum pname, const GLint *params); +GL_APICALL void GL_APIENTRY glTexParameterIuivOES (GLenum target, GLenum pname, const GLuint *params); +GL_APICALL void GL_APIENTRY glGetTexParameterIivOES (GLenum target, GLenum pname, GLint *params); +GL_APICALL void GL_APIENTRY glGetTexParameterIuivOES (GLenum target, GLenum pname, GLuint *params); +GL_APICALL void GL_APIENTRY glSamplerParameterIivOES (GLuint sampler, GLenum pname, const GLint *param); +GL_APICALL void GL_APIENTRY glSamplerParameterIuivOES (GLuint sampler, GLenum pname, const GLuint *param); +GL_APICALL void GL_APIENTRY glGetSamplerParameterIivOES (GLuint sampler, GLenum pname, GLint *params); +GL_APICALL void GL_APIENTRY glGetSamplerParameterIuivOES (GLuint sampler, GLenum pname, GLuint *params); +#endif +#endif /* GL_OES_texture_border_clamp */ + +#ifndef GL_OES_texture_buffer +#define GL_OES_texture_buffer 1 +#define GL_TEXTURE_BUFFER_OES 0x8C2A +#define GL_TEXTURE_BUFFER_BINDING_OES 0x8C2A +#define GL_MAX_TEXTURE_BUFFER_SIZE_OES 0x8C2B +#define GL_TEXTURE_BINDING_BUFFER_OES 0x8C2C +#define GL_TEXTURE_BUFFER_DATA_STORE_BINDING_OES 0x8C2D +#define GL_TEXTURE_BUFFER_OFFSET_ALIGNMENT_OES 0x919F +#define GL_SAMPLER_BUFFER_OES 0x8DC2 +#define GL_INT_SAMPLER_BUFFER_OES 0x8DD0 +#define GL_UNSIGNED_INT_SAMPLER_BUFFER_OES 0x8DD8 +#define GL_IMAGE_BUFFER_OES 0x9051 +#define GL_INT_IMAGE_BUFFER_OES 0x905C +#define GL_UNSIGNED_INT_IMAGE_BUFFER_OES 0x9067 +#define GL_TEXTURE_BUFFER_OFFSET_OES 0x919D +#define GL_TEXTURE_BUFFER_SIZE_OES 0x919E +typedef void (GL_APIENTRYP PFNGLTEXBUFFEROESPROC) (GLenum target, GLenum internalformat, GLuint buffer); +typedef void (GL_APIENTRYP PFNGLTEXBUFFERRANGEOESPROC) (GLenum target, GLenum internalformat, GLuint buffer, GLintptr offset, GLsizeiptr size); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glTexBufferOES (GLenum target, GLenum internalformat, GLuint buffer); +GL_APICALL void GL_APIENTRY glTexBufferRangeOES (GLenum target, GLenum internalformat, GLuint buffer, GLintptr offset, GLsizeiptr size); +#endif +#endif /* GL_OES_texture_buffer */ + +#ifndef GL_OES_texture_compression_astc +#define GL_OES_texture_compression_astc 1 +#define GL_COMPRESSED_RGBA_ASTC_3x3x3_OES 0x93C0 +#define GL_COMPRESSED_RGBA_ASTC_4x3x3_OES 0x93C1 +#define GL_COMPRESSED_RGBA_ASTC_4x4x3_OES 0x93C2 +#define GL_COMPRESSED_RGBA_ASTC_4x4x4_OES 0x93C3 +#define GL_COMPRESSED_RGBA_ASTC_5x4x4_OES 0x93C4 +#define GL_COMPRESSED_RGBA_ASTC_5x5x4_OES 0x93C5 +#define GL_COMPRESSED_RGBA_ASTC_5x5x5_OES 0x93C6 +#define GL_COMPRESSED_RGBA_ASTC_6x5x5_OES 0x93C7 +#define GL_COMPRESSED_RGBA_ASTC_6x6x5_OES 0x93C8 +#define GL_COMPRESSED_RGBA_ASTC_6x6x6_OES 0x93C9 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_3x3x3_OES 0x93E0 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_4x3x3_OES 0x93E1 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_4x4x3_OES 0x93E2 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_4x4x4_OES 0x93E3 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_5x4x4_OES 0x93E4 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_5x5x4_OES 0x93E5 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_5x5x5_OES 0x93E6 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_6x5x5_OES 0x93E7 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_6x6x5_OES 0x93E8 +#define GL_COMPRESSED_SRGB8_ALPHA8_ASTC_6x6x6_OES 0x93E9 +#endif /* GL_OES_texture_compression_astc */ + +#ifndef GL_OES_texture_cube_map_array +#define GL_OES_texture_cube_map_array 1 +#define GL_TEXTURE_CUBE_MAP_ARRAY_OES 0x9009 +#define GL_TEXTURE_BINDING_CUBE_MAP_ARRAY_OES 0x900A +#define GL_SAMPLER_CUBE_MAP_ARRAY_OES 0x900C +#define GL_SAMPLER_CUBE_MAP_ARRAY_SHADOW_OES 0x900D +#define GL_INT_SAMPLER_CUBE_MAP_ARRAY_OES 0x900E +#define GL_UNSIGNED_INT_SAMPLER_CUBE_MAP_ARRAY_OES 0x900F +#define GL_IMAGE_CUBE_MAP_ARRAY_OES 0x9054 +#define GL_INT_IMAGE_CUBE_MAP_ARRAY_OES 0x905F +#define GL_UNSIGNED_INT_IMAGE_CUBE_MAP_ARRAY_OES 0x906A +#endif /* GL_OES_texture_cube_map_array */ + +#ifndef GL_OES_texture_float +#define GL_OES_texture_float 1 +#endif /* GL_OES_texture_float */ + +#ifndef GL_OES_texture_float_linear +#define GL_OES_texture_float_linear 1 +#endif /* GL_OES_texture_float_linear */ + +#ifndef GL_OES_texture_half_float +#define GL_OES_texture_half_float 1 +#define GL_HALF_FLOAT_OES 0x8D61 +#endif /* GL_OES_texture_half_float */ + +#ifndef GL_OES_texture_half_float_linear +#define GL_OES_texture_half_float_linear 1 +#endif /* GL_OES_texture_half_float_linear */ + +#ifndef GL_OES_texture_npot +#define GL_OES_texture_npot 1 +#endif /* GL_OES_texture_npot */ + +#ifndef GL_OES_texture_stencil8 +#define GL_OES_texture_stencil8 1 +#define GL_STENCIL_INDEX_OES 0x1901 +#define GL_STENCIL_INDEX8_OES 0x8D48 +#endif /* GL_OES_texture_stencil8 */ + +#ifndef GL_OES_texture_storage_multisample_2d_array +#define GL_OES_texture_storage_multisample_2d_array 1 +#define GL_TEXTURE_2D_MULTISAMPLE_ARRAY_OES 0x9102 +#define GL_TEXTURE_BINDING_2D_MULTISAMPLE_ARRAY_OES 0x9105 +#define GL_SAMPLER_2D_MULTISAMPLE_ARRAY_OES 0x910B +#define GL_INT_SAMPLER_2D_MULTISAMPLE_ARRAY_OES 0x910C +#define GL_UNSIGNED_INT_SAMPLER_2D_MULTISAMPLE_ARRAY_OES 0x910D +typedef void (GL_APIENTRYP PFNGLTEXSTORAGE3DMULTISAMPLEOESPROC) (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedsamplelocations); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glTexStorage3DMultisampleOES (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedsamplelocations); +#endif +#endif /* GL_OES_texture_storage_multisample_2d_array */ + +#ifndef GL_OES_texture_view +#define GL_OES_texture_view 1 +#define GL_TEXTURE_VIEW_MIN_LEVEL_OES 0x82DB +#define GL_TEXTURE_VIEW_NUM_LEVELS_OES 0x82DC +#define GL_TEXTURE_VIEW_MIN_LAYER_OES 0x82DD +#define GL_TEXTURE_VIEW_NUM_LAYERS_OES 0x82DE +#define GL_TEXTURE_IMMUTABLE_LEVELS 0x82DF +typedef void (GL_APIENTRYP PFNGLTEXTUREVIEWOESPROC) (GLuint texture, GLenum target, GLuint origtexture, GLenum internalformat, GLuint minlevel, GLuint numlevels, GLuint minlayer, GLuint numlayers); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glTextureViewOES (GLuint texture, GLenum target, GLuint origtexture, GLenum internalformat, GLuint minlevel, GLuint numlevels, GLuint minlayer, GLuint numlayers); +#endif +#endif /* GL_OES_texture_view */ + +#ifndef GL_OES_vertex_array_object +#define GL_OES_vertex_array_object 1 +#define GL_VERTEX_ARRAY_BINDING_OES 0x85B5 +typedef void (GL_APIENTRYP PFNGLBINDVERTEXARRAYOESPROC) (GLuint array); +typedef void (GL_APIENTRYP PFNGLDELETEVERTEXARRAYSOESPROC) (GLsizei n, const GLuint *arrays); +typedef void (GL_APIENTRYP PFNGLGENVERTEXARRAYSOESPROC) (GLsizei n, GLuint *arrays); +typedef GLboolean (GL_APIENTRYP PFNGLISVERTEXARRAYOESPROC) (GLuint array); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glBindVertexArrayOES (GLuint array); +GL_APICALL void GL_APIENTRY glDeleteVertexArraysOES (GLsizei n, const GLuint *arrays); +GL_APICALL void GL_APIENTRY glGenVertexArraysOES (GLsizei n, GLuint *arrays); +GL_APICALL GLboolean GL_APIENTRY glIsVertexArrayOES (GLuint array); +#endif +#endif /* GL_OES_vertex_array_object */ + +#ifndef GL_OES_vertex_half_float +#define GL_OES_vertex_half_float 1 +#endif /* GL_OES_vertex_half_float */ + +#ifndef GL_OES_vertex_type_10_10_10_2 +#define GL_OES_vertex_type_10_10_10_2 1 +#define GL_UNSIGNED_INT_10_10_10_2_OES 0x8DF6 +#define GL_INT_10_10_10_2_OES 0x8DF7 +#endif /* GL_OES_vertex_type_10_10_10_2 */ + +#ifndef GL_OES_viewport_array +#define GL_OES_viewport_array 1 +#define GL_MAX_VIEWPORTS_OES 0x825B +#define GL_VIEWPORT_SUBPIXEL_BITS_OES 0x825C +#define GL_VIEWPORT_BOUNDS_RANGE_OES 0x825D +#define GL_VIEWPORT_INDEX_PROVOKING_VERTEX_OES 0x825F +typedef void (GL_APIENTRYP PFNGLVIEWPORTARRAYVOESPROC) (GLuint first, GLsizei count, const GLfloat *v); +typedef void (GL_APIENTRYP PFNGLVIEWPORTINDEXEDFOESPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat w, GLfloat h); +typedef void (GL_APIENTRYP PFNGLVIEWPORTINDEXEDFVOESPROC) (GLuint index, const GLfloat *v); +typedef void (GL_APIENTRYP PFNGLSCISSORARRAYVOESPROC) (GLuint first, GLsizei count, const GLint *v); +typedef void (GL_APIENTRYP PFNGLSCISSORINDEXEDOESPROC) (GLuint index, GLint left, GLint bottom, GLsizei width, GLsizei height); +typedef void (GL_APIENTRYP PFNGLSCISSORINDEXEDVOESPROC) (GLuint index, const GLint *v); +typedef void (GL_APIENTRYP PFNGLDEPTHRANGEARRAYFVOESPROC) (GLuint first, GLsizei count, const GLfloat *v); +typedef void (GL_APIENTRYP PFNGLDEPTHRANGEINDEXEDFOESPROC) (GLuint index, GLfloat n, GLfloat f); +typedef void (GL_APIENTRYP PFNGLGETFLOATI_VOESPROC) (GLenum target, GLuint index, GLfloat *data); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glViewportArrayvOES (GLuint first, GLsizei count, const GLfloat *v); +GL_APICALL void GL_APIENTRY glViewportIndexedfOES (GLuint index, GLfloat x, GLfloat y, GLfloat w, GLfloat h); +GL_APICALL void GL_APIENTRY glViewportIndexedfvOES (GLuint index, const GLfloat *v); +GL_APICALL void GL_APIENTRY glScissorArrayvOES (GLuint first, GLsizei count, const GLint *v); +GL_APICALL void GL_APIENTRY glScissorIndexedOES (GLuint index, GLint left, GLint bottom, GLsizei width, GLsizei height); +GL_APICALL void GL_APIENTRY glScissorIndexedvOES (GLuint index, const GLint *v); +GL_APICALL void GL_APIENTRY glDepthRangeArrayfvOES (GLuint first, GLsizei count, const GLfloat *v); +GL_APICALL void GL_APIENTRY glDepthRangeIndexedfOES (GLuint index, GLfloat n, GLfloat f); +GL_APICALL void GL_APIENTRY glGetFloati_vOES (GLenum target, GLuint index, GLfloat *data); +#endif +#endif /* GL_OES_viewport_array */ + +#ifndef GL_AMD_compressed_3DC_texture +#define GL_AMD_compressed_3DC_texture 1 +#define GL_3DC_X_AMD 0x87F9 +#define GL_3DC_XY_AMD 0x87FA +#endif /* GL_AMD_compressed_3DC_texture */ + +#ifndef GL_AMD_compressed_ATC_texture +#define GL_AMD_compressed_ATC_texture 1 +#define GL_ATC_RGB_AMD 0x8C92 +#define GL_ATC_RGBA_EXPLICIT_ALPHA_AMD 0x8C93 +#define GL_ATC_RGBA_INTERPOLATED_ALPHA_AMD 0x87EE +#endif /* GL_AMD_compressed_ATC_texture */ + +#ifndef GL_AMD_framebuffer_multisample_advanced +#define GL_AMD_framebuffer_multisample_advanced 1 +#define GL_RENDERBUFFER_STORAGE_SAMPLES_AMD 0x91B2 +#define GL_MAX_COLOR_FRAMEBUFFER_SAMPLES_AMD 0x91B3 +#define GL_MAX_COLOR_FRAMEBUFFER_STORAGE_SAMPLES_AMD 0x91B4 +#define GL_MAX_DEPTH_STENCIL_FRAMEBUFFER_SAMPLES_AMD 0x91B5 +#define GL_NUM_SUPPORTED_MULTISAMPLE_MODES_AMD 0x91B6 +#define GL_SUPPORTED_MULTISAMPLE_MODES_AMD 0x91B7 +typedef void (GL_APIENTRYP PFNGLRENDERBUFFERSTORAGEMULTISAMPLEADVANCEDAMDPROC) (GLenum target, GLsizei samples, GLsizei storageSamples, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (GL_APIENTRYP PFNGLNAMEDRENDERBUFFERSTORAGEMULTISAMPLEADVANCEDAMDPROC) (GLuint renderbuffer, GLsizei samples, GLsizei storageSamples, GLenum internalformat, GLsizei width, GLsizei height); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glRenderbufferStorageMultisampleAdvancedAMD (GLenum target, GLsizei samples, GLsizei storageSamples, GLenum internalformat, GLsizei width, GLsizei height); +GL_APICALL void GL_APIENTRY glNamedRenderbufferStorageMultisampleAdvancedAMD (GLuint renderbuffer, GLsizei samples, GLsizei storageSamples, GLenum internalformat, GLsizei width, GLsizei height); +#endif +#endif /* GL_AMD_framebuffer_multisample_advanced */ + +#ifndef GL_AMD_performance_monitor +#define GL_AMD_performance_monitor 1 +#define GL_COUNTER_TYPE_AMD 0x8BC0 +#define GL_COUNTER_RANGE_AMD 0x8BC1 +#define GL_UNSIGNED_INT64_AMD 0x8BC2 +#define GL_PERCENTAGE_AMD 0x8BC3 +#define GL_PERFMON_RESULT_AVAILABLE_AMD 0x8BC4 +#define GL_PERFMON_RESULT_SIZE_AMD 0x8BC5 +#define GL_PERFMON_RESULT_AMD 0x8BC6 +typedef void (GL_APIENTRYP PFNGLGETPERFMONITORGROUPSAMDPROC) (GLint *numGroups, GLsizei groupsSize, GLuint *groups); +typedef void (GL_APIENTRYP PFNGLGETPERFMONITORCOUNTERSAMDPROC) (GLuint group, GLint *numCounters, GLint *maxActiveCounters, GLsizei counterSize, GLuint *counters); +typedef void (GL_APIENTRYP PFNGLGETPERFMONITORGROUPSTRINGAMDPROC) (GLuint group, GLsizei bufSize, GLsizei *length, GLchar *groupString); +typedef void (GL_APIENTRYP PFNGLGETPERFMONITORCOUNTERSTRINGAMDPROC) (GLuint group, GLuint counter, GLsizei bufSize, GLsizei *length, GLchar *counterString); +typedef void (GL_APIENTRYP PFNGLGETPERFMONITORCOUNTERINFOAMDPROC) (GLuint group, GLuint counter, GLenum pname, void *data); +typedef void (GL_APIENTRYP PFNGLGENPERFMONITORSAMDPROC) (GLsizei n, GLuint *monitors); +typedef void (GL_APIENTRYP PFNGLDELETEPERFMONITORSAMDPROC) (GLsizei n, GLuint *monitors); +typedef void (GL_APIENTRYP PFNGLSELECTPERFMONITORCOUNTERSAMDPROC) (GLuint monitor, GLboolean enable, GLuint group, GLint numCounters, GLuint *counterList); +typedef void (GL_APIENTRYP PFNGLBEGINPERFMONITORAMDPROC) (GLuint monitor); +typedef void (GL_APIENTRYP PFNGLENDPERFMONITORAMDPROC) (GLuint monitor); +typedef void (GL_APIENTRYP PFNGLGETPERFMONITORCOUNTERDATAAMDPROC) (GLuint monitor, GLenum pname, GLsizei dataSize, GLuint *data, GLint *bytesWritten); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glGetPerfMonitorGroupsAMD (GLint *numGroups, GLsizei groupsSize, GLuint *groups); +GL_APICALL void GL_APIENTRY glGetPerfMonitorCountersAMD (GLuint group, GLint *numCounters, GLint *maxActiveCounters, GLsizei counterSize, GLuint *counters); +GL_APICALL void GL_APIENTRY glGetPerfMonitorGroupStringAMD (GLuint group, GLsizei bufSize, GLsizei *length, GLchar *groupString); +GL_APICALL void GL_APIENTRY glGetPerfMonitorCounterStringAMD (GLuint group, GLuint counter, GLsizei bufSize, GLsizei *length, GLchar *counterString); +GL_APICALL void GL_APIENTRY glGetPerfMonitorCounterInfoAMD (GLuint group, GLuint counter, GLenum pname, void *data); +GL_APICALL void GL_APIENTRY glGenPerfMonitorsAMD (GLsizei n, GLuint *monitors); +GL_APICALL void GL_APIENTRY glDeletePerfMonitorsAMD (GLsizei n, GLuint *monitors); +GL_APICALL void GL_APIENTRY glSelectPerfMonitorCountersAMD (GLuint monitor, GLboolean enable, GLuint group, GLint numCounters, GLuint *counterList); +GL_APICALL void GL_APIENTRY glBeginPerfMonitorAMD (GLuint monitor); +GL_APICALL void GL_APIENTRY glEndPerfMonitorAMD (GLuint monitor); +GL_APICALL void GL_APIENTRY glGetPerfMonitorCounterDataAMD (GLuint monitor, GLenum pname, GLsizei dataSize, GLuint *data, GLint *bytesWritten); +#endif +#endif /* GL_AMD_performance_monitor */ + +#ifndef GL_AMD_program_binary_Z400 +#define GL_AMD_program_binary_Z400 1 +#define GL_Z400_BINARY_AMD 0x8740 +#endif /* GL_AMD_program_binary_Z400 */ + +#ifndef GL_ANDROID_extension_pack_es31a +#define GL_ANDROID_extension_pack_es31a 1 +#endif /* GL_ANDROID_extension_pack_es31a */ + +#ifndef GL_ANGLE_depth_texture +#define GL_ANGLE_depth_texture 1 +#endif /* GL_ANGLE_depth_texture */ + +#ifndef GL_ANGLE_framebuffer_blit +#define GL_ANGLE_framebuffer_blit 1 +#define GL_READ_FRAMEBUFFER_ANGLE 0x8CA8 +#define GL_DRAW_FRAMEBUFFER_ANGLE 0x8CA9 +#define GL_DRAW_FRAMEBUFFER_BINDING_ANGLE 0x8CA6 +#define GL_READ_FRAMEBUFFER_BINDING_ANGLE 0x8CAA +typedef void (GL_APIENTRYP PFNGLBLITFRAMEBUFFERANGLEPROC) (GLint srcX0, GLint srcY0, GLint srcX1, GLint srcY1, GLint dstX0, GLint dstY0, GLint dstX1, GLint dstY1, GLbitfield mask, GLenum filter); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glBlitFramebufferANGLE (GLint srcX0, GLint srcY0, GLint srcX1, GLint srcY1, GLint dstX0, GLint dstY0, GLint dstX1, GLint dstY1, GLbitfield mask, GLenum filter); +#endif +#endif /* GL_ANGLE_framebuffer_blit */ + +#ifndef GL_ANGLE_framebuffer_multisample +#define GL_ANGLE_framebuffer_multisample 1 +#define GL_RENDERBUFFER_SAMPLES_ANGLE 0x8CAB +#define GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE_ANGLE 0x8D56 +#define GL_MAX_SAMPLES_ANGLE 0x8D57 +typedef void (GL_APIENTRYP PFNGLRENDERBUFFERSTORAGEMULTISAMPLEANGLEPROC) (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glRenderbufferStorageMultisampleANGLE (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +#endif +#endif /* GL_ANGLE_framebuffer_multisample */ + +#ifndef GL_ANGLE_instanced_arrays +#define GL_ANGLE_instanced_arrays 1 +#define GL_VERTEX_ATTRIB_ARRAY_DIVISOR_ANGLE 0x88FE +typedef void (GL_APIENTRYP PFNGLDRAWARRAYSINSTANCEDANGLEPROC) (GLenum mode, GLint first, GLsizei count, GLsizei primcount); +typedef void (GL_APIENTRYP PFNGLDRAWELEMENTSINSTANCEDANGLEPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei primcount); +typedef void (GL_APIENTRYP PFNGLVERTEXATTRIBDIVISORANGLEPROC) (GLuint index, GLuint divisor); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glDrawArraysInstancedANGLE (GLenum mode, GLint first, GLsizei count, GLsizei primcount); +GL_APICALL void GL_APIENTRY glDrawElementsInstancedANGLE (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei primcount); +GL_APICALL void GL_APIENTRY glVertexAttribDivisorANGLE (GLuint index, GLuint divisor); +#endif +#endif /* GL_ANGLE_instanced_arrays */ + +#ifndef GL_ANGLE_pack_reverse_row_order +#define GL_ANGLE_pack_reverse_row_order 1 +#define GL_PACK_REVERSE_ROW_ORDER_ANGLE 0x93A4 +#endif /* GL_ANGLE_pack_reverse_row_order */ + +#ifndef GL_ANGLE_program_binary +#define GL_ANGLE_program_binary 1 +#define GL_PROGRAM_BINARY_ANGLE 0x93A6 +#endif /* GL_ANGLE_program_binary */ + +#ifndef GL_ANGLE_texture_compression_dxt3 +#define GL_ANGLE_texture_compression_dxt3 1 +#define GL_COMPRESSED_RGBA_S3TC_DXT3_ANGLE 0x83F2 +#endif /* GL_ANGLE_texture_compression_dxt3 */ + +#ifndef GL_ANGLE_texture_compression_dxt5 +#define GL_ANGLE_texture_compression_dxt5 1 +#define GL_COMPRESSED_RGBA_S3TC_DXT5_ANGLE 0x83F3 +#endif /* GL_ANGLE_texture_compression_dxt5 */ + +#ifndef GL_ANGLE_texture_usage +#define GL_ANGLE_texture_usage 1 +#define GL_TEXTURE_USAGE_ANGLE 0x93A2 +#define GL_FRAMEBUFFER_ATTACHMENT_ANGLE 0x93A3 +#endif /* GL_ANGLE_texture_usage */ + +#ifndef GL_ANGLE_translated_shader_source +#define GL_ANGLE_translated_shader_source 1 +#define GL_TRANSLATED_SHADER_SOURCE_LENGTH_ANGLE 0x93A0 +typedef void (GL_APIENTRYP PFNGLGETTRANSLATEDSHADERSOURCEANGLEPROC) (GLuint shader, GLsizei bufSize, GLsizei *length, GLchar *source); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glGetTranslatedShaderSourceANGLE (GLuint shader, GLsizei bufSize, GLsizei *length, GLchar *source); +#endif +#endif /* GL_ANGLE_translated_shader_source */ + +#ifndef GL_APPLE_clip_distance +#define GL_APPLE_clip_distance 1 +#define GL_MAX_CLIP_DISTANCES_APPLE 0x0D32 +#define GL_CLIP_DISTANCE0_APPLE 0x3000 +#define GL_CLIP_DISTANCE1_APPLE 0x3001 +#define GL_CLIP_DISTANCE2_APPLE 0x3002 +#define GL_CLIP_DISTANCE3_APPLE 0x3003 +#define GL_CLIP_DISTANCE4_APPLE 0x3004 +#define GL_CLIP_DISTANCE5_APPLE 0x3005 +#define GL_CLIP_DISTANCE6_APPLE 0x3006 +#define GL_CLIP_DISTANCE7_APPLE 0x3007 +#endif /* GL_APPLE_clip_distance */ + +#ifndef GL_APPLE_color_buffer_packed_float +#define GL_APPLE_color_buffer_packed_float 1 +#endif /* GL_APPLE_color_buffer_packed_float */ + +#ifndef GL_APPLE_copy_texture_levels +#define GL_APPLE_copy_texture_levels 1 +typedef void (GL_APIENTRYP PFNGLCOPYTEXTURELEVELSAPPLEPROC) (GLuint destinationTexture, GLuint sourceTexture, GLint sourceBaseLevel, GLsizei sourceLevelCount); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glCopyTextureLevelsAPPLE (GLuint destinationTexture, GLuint sourceTexture, GLint sourceBaseLevel, GLsizei sourceLevelCount); +#endif +#endif /* GL_APPLE_copy_texture_levels */ + +#ifndef GL_APPLE_framebuffer_multisample +#define GL_APPLE_framebuffer_multisample 1 +#define GL_RENDERBUFFER_SAMPLES_APPLE 0x8CAB +#define GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE_APPLE 0x8D56 +#define GL_MAX_SAMPLES_APPLE 0x8D57 +#define GL_READ_FRAMEBUFFER_APPLE 0x8CA8 +#define GL_DRAW_FRAMEBUFFER_APPLE 0x8CA9 +#define GL_DRAW_FRAMEBUFFER_BINDING_APPLE 0x8CA6 +#define GL_READ_FRAMEBUFFER_BINDING_APPLE 0x8CAA +typedef void (GL_APIENTRYP PFNGLRENDERBUFFERSTORAGEMULTISAMPLEAPPLEPROC) (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (GL_APIENTRYP PFNGLRESOLVEMULTISAMPLEFRAMEBUFFERAPPLEPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glRenderbufferStorageMultisampleAPPLE (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +GL_APICALL void GL_APIENTRY glResolveMultisampleFramebufferAPPLE (void); +#endif +#endif /* GL_APPLE_framebuffer_multisample */ + +#ifndef GL_APPLE_rgb_422 +#define GL_APPLE_rgb_422 1 +#define GL_RGB_422_APPLE 0x8A1F +#define GL_UNSIGNED_SHORT_8_8_APPLE 0x85BA +#define GL_UNSIGNED_SHORT_8_8_REV_APPLE 0x85BB +#define GL_RGB_RAW_422_APPLE 0x8A51 +#endif /* GL_APPLE_rgb_422 */ + +#ifndef GL_APPLE_sync +#define GL_APPLE_sync 1 +#define GL_SYNC_OBJECT_APPLE 0x8A53 +#define GL_MAX_SERVER_WAIT_TIMEOUT_APPLE 0x9111 +#define GL_OBJECT_TYPE_APPLE 0x9112 +#define GL_SYNC_CONDITION_APPLE 0x9113 +#define GL_SYNC_STATUS_APPLE 0x9114 +#define GL_SYNC_FLAGS_APPLE 0x9115 +#define GL_SYNC_FENCE_APPLE 0x9116 +#define GL_SYNC_GPU_COMMANDS_COMPLETE_APPLE 0x9117 +#define GL_UNSIGNALED_APPLE 0x9118 +#define GL_SIGNALED_APPLE 0x9119 +#define GL_ALREADY_SIGNALED_APPLE 0x911A +#define GL_TIMEOUT_EXPIRED_APPLE 0x911B +#define GL_CONDITION_SATISFIED_APPLE 0x911C +#define GL_WAIT_FAILED_APPLE 0x911D +#define GL_SYNC_FLUSH_COMMANDS_BIT_APPLE 0x00000001 +#define GL_TIMEOUT_IGNORED_APPLE 0xFFFFFFFFFFFFFFFFull +typedef GLsync (GL_APIENTRYP PFNGLFENCESYNCAPPLEPROC) (GLenum condition, GLbitfield flags); +typedef GLboolean (GL_APIENTRYP PFNGLISSYNCAPPLEPROC) (GLsync sync); +typedef void (GL_APIENTRYP PFNGLDELETESYNCAPPLEPROC) (GLsync sync); +typedef GLenum (GL_APIENTRYP PFNGLCLIENTWAITSYNCAPPLEPROC) (GLsync sync, GLbitfield flags, GLuint64 timeout); +typedef void (GL_APIENTRYP PFNGLWAITSYNCAPPLEPROC) (GLsync sync, GLbitfield flags, GLuint64 timeout); +typedef void (GL_APIENTRYP PFNGLGETINTEGER64VAPPLEPROC) (GLenum pname, GLint64 *params); +typedef void (GL_APIENTRYP PFNGLGETSYNCIVAPPLEPROC) (GLsync sync, GLenum pname, GLsizei count, GLsizei *length, GLint *values); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL GLsync GL_APIENTRY glFenceSyncAPPLE (GLenum condition, GLbitfield flags); +GL_APICALL GLboolean GL_APIENTRY glIsSyncAPPLE (GLsync sync); +GL_APICALL void GL_APIENTRY glDeleteSyncAPPLE (GLsync sync); +GL_APICALL GLenum GL_APIENTRY glClientWaitSyncAPPLE (GLsync sync, GLbitfield flags, GLuint64 timeout); +GL_APICALL void GL_APIENTRY glWaitSyncAPPLE (GLsync sync, GLbitfield flags, GLuint64 timeout); +GL_APICALL void GL_APIENTRY glGetInteger64vAPPLE (GLenum pname, GLint64 *params); +GL_APICALL void GL_APIENTRY glGetSyncivAPPLE (GLsync sync, GLenum pname, GLsizei count, GLsizei *length, GLint *values); +#endif +#endif /* GL_APPLE_sync */ + +#ifndef GL_APPLE_texture_format_BGRA8888 +#define GL_APPLE_texture_format_BGRA8888 1 +#define GL_BGRA_EXT 0x80E1 +#define GL_BGRA8_EXT 0x93A1 +#endif /* GL_APPLE_texture_format_BGRA8888 */ + +#ifndef GL_APPLE_texture_max_level +#define GL_APPLE_texture_max_level 1 +#define GL_TEXTURE_MAX_LEVEL_APPLE 0x813D +#endif /* GL_APPLE_texture_max_level */ + +#ifndef GL_APPLE_texture_packed_float +#define GL_APPLE_texture_packed_float 1 +#define GL_UNSIGNED_INT_10F_11F_11F_REV_APPLE 0x8C3B +#define GL_UNSIGNED_INT_5_9_9_9_REV_APPLE 0x8C3E +#define GL_R11F_G11F_B10F_APPLE 0x8C3A +#define GL_RGB9_E5_APPLE 0x8C3D +#endif /* GL_APPLE_texture_packed_float */ + +#ifndef GL_ARM_mali_program_binary +#define GL_ARM_mali_program_binary 1 +#define GL_MALI_PROGRAM_BINARY_ARM 0x8F61 +#endif /* GL_ARM_mali_program_binary */ + +#ifndef GL_ARM_mali_shader_binary +#define GL_ARM_mali_shader_binary 1 +#define GL_MALI_SHADER_BINARY_ARM 0x8F60 +#endif /* GL_ARM_mali_shader_binary */ + +#ifndef GL_ARM_rgba8 +#define GL_ARM_rgba8 1 +#endif /* GL_ARM_rgba8 */ + +#ifndef GL_ARM_shader_framebuffer_fetch +#define GL_ARM_shader_framebuffer_fetch 1 +#define GL_FETCH_PER_SAMPLE_ARM 0x8F65 +#define GL_FRAGMENT_SHADER_FRAMEBUFFER_FETCH_MRT_ARM 0x8F66 +#endif /* GL_ARM_shader_framebuffer_fetch */ + +#ifndef GL_ARM_shader_framebuffer_fetch_depth_stencil +#define GL_ARM_shader_framebuffer_fetch_depth_stencil 1 +#endif /* GL_ARM_shader_framebuffer_fetch_depth_stencil */ + +#ifndef GL_ARM_texture_unnormalized_coordinates +#define GL_ARM_texture_unnormalized_coordinates 1 +#define GL_TEXTURE_UNNORMALIZED_COORDINATES_ARM 0x8F6A +#endif /* GL_ARM_texture_unnormalized_coordinates */ + +#ifndef GL_DMP_program_binary +#define GL_DMP_program_binary 1 +#define GL_SMAPHS30_PROGRAM_BINARY_DMP 0x9251 +#define GL_SMAPHS_PROGRAM_BINARY_DMP 0x9252 +#define GL_DMP_PROGRAM_BINARY_DMP 0x9253 +#endif /* GL_DMP_program_binary */ + +#ifndef GL_DMP_shader_binary +#define GL_DMP_shader_binary 1 +#define GL_SHADER_BINARY_DMP 0x9250 +#endif /* GL_DMP_shader_binary */ + +#ifndef GL_EXT_EGL_image_array +#define GL_EXT_EGL_image_array 1 +#endif /* GL_EXT_EGL_image_array */ + +#ifndef GL_EXT_EGL_image_storage +#define GL_EXT_EGL_image_storage 1 +typedef void (GL_APIENTRYP PFNGLEGLIMAGETARGETTEXSTORAGEEXTPROC) (GLenum target, GLeglImageOES image, const GLint* attrib_list); +typedef void (GL_APIENTRYP PFNGLEGLIMAGETARGETTEXTURESTORAGEEXTPROC) (GLuint texture, GLeglImageOES image, const GLint* attrib_list); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glEGLImageTargetTexStorageEXT (GLenum target, GLeglImageOES image, const GLint* attrib_list); +GL_APICALL void GL_APIENTRY glEGLImageTargetTextureStorageEXT (GLuint texture, GLeglImageOES image, const GLint* attrib_list); +#endif +#endif /* GL_EXT_EGL_image_storage */ + +#ifndef GL_EXT_EGL_image_storage_compression +#define GL_EXT_EGL_image_storage_compression 1 +#define GL_SURFACE_COMPRESSION_EXT 0x96C0 +#define GL_SURFACE_COMPRESSION_FIXED_RATE_NONE_EXT 0x96C1 +#define GL_SURFACE_COMPRESSION_FIXED_RATE_DEFAULT_EXT 0x96C2 +#endif /* GL_EXT_EGL_image_storage_compression */ + +#ifndef GL_EXT_YUV_target +#define GL_EXT_YUV_target 1 +#define GL_SAMPLER_EXTERNAL_2D_Y2Y_EXT 0x8BE7 +#endif /* GL_EXT_YUV_target */ + +#ifndef GL_EXT_base_instance +#define GL_EXT_base_instance 1 +typedef void (GL_APIENTRYP PFNGLDRAWARRAYSINSTANCEDBASEINSTANCEEXTPROC) (GLenum mode, GLint first, GLsizei count, GLsizei instancecount, GLuint baseinstance); +typedef void (GL_APIENTRYP PFNGLDRAWELEMENTSINSTANCEDBASEINSTANCEEXTPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount, GLuint baseinstance); +typedef void (GL_APIENTRYP PFNGLDRAWELEMENTSINSTANCEDBASEVERTEXBASEINSTANCEEXTPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount, GLint basevertex, GLuint baseinstance); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glDrawArraysInstancedBaseInstanceEXT (GLenum mode, GLint first, GLsizei count, GLsizei instancecount, GLuint baseinstance); +GL_APICALL void GL_APIENTRY glDrawElementsInstancedBaseInstanceEXT (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount, GLuint baseinstance); +GL_APICALL void GL_APIENTRY glDrawElementsInstancedBaseVertexBaseInstanceEXT (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount, GLint basevertex, GLuint baseinstance); +#endif +#endif /* GL_EXT_base_instance */ + +#ifndef GL_EXT_blend_func_extended +#define GL_EXT_blend_func_extended 1 +#define GL_SRC1_COLOR_EXT 0x88F9 +#define GL_SRC1_ALPHA_EXT 0x8589 +#define GL_ONE_MINUS_SRC1_COLOR_EXT 0x88FA +#define GL_ONE_MINUS_SRC1_ALPHA_EXT 0x88FB +#define GL_SRC_ALPHA_SATURATE_EXT 0x0308 +#define GL_LOCATION_INDEX_EXT 0x930F +#define GL_MAX_DUAL_SOURCE_DRAW_BUFFERS_EXT 0x88FC +typedef void (GL_APIENTRYP PFNGLBINDFRAGDATALOCATIONINDEXEDEXTPROC) (GLuint program, GLuint colorNumber, GLuint index, const GLchar *name); +typedef void (GL_APIENTRYP PFNGLBINDFRAGDATALOCATIONEXTPROC) (GLuint program, GLuint color, const GLchar *name); +typedef GLint (GL_APIENTRYP PFNGLGETPROGRAMRESOURCELOCATIONINDEXEXTPROC) (GLuint program, GLenum programInterface, const GLchar *name); +typedef GLint (GL_APIENTRYP PFNGLGETFRAGDATAINDEXEXTPROC) (GLuint program, const GLchar *name); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glBindFragDataLocationIndexedEXT (GLuint program, GLuint colorNumber, GLuint index, const GLchar *name); +GL_APICALL void GL_APIENTRY glBindFragDataLocationEXT (GLuint program, GLuint color, const GLchar *name); +GL_APICALL GLint GL_APIENTRY glGetProgramResourceLocationIndexEXT (GLuint program, GLenum programInterface, const GLchar *name); +GL_APICALL GLint GL_APIENTRY glGetFragDataIndexEXT (GLuint program, const GLchar *name); +#endif +#endif /* GL_EXT_blend_func_extended */ + +#ifndef GL_EXT_blend_minmax +#define GL_EXT_blend_minmax 1 +#define GL_MIN_EXT 0x8007 +#define GL_MAX_EXT 0x8008 +#endif /* GL_EXT_blend_minmax */ + +#ifndef GL_EXT_buffer_storage +#define GL_EXT_buffer_storage 1 +#define GL_MAP_READ_BIT 0x0001 +#define GL_MAP_WRITE_BIT 0x0002 +#define GL_MAP_PERSISTENT_BIT_EXT 0x0040 +#define GL_MAP_COHERENT_BIT_EXT 0x0080 +#define GL_DYNAMIC_STORAGE_BIT_EXT 0x0100 +#define GL_CLIENT_STORAGE_BIT_EXT 0x0200 +#define GL_CLIENT_MAPPED_BUFFER_BARRIER_BIT_EXT 0x00004000 +#define GL_BUFFER_IMMUTABLE_STORAGE_EXT 0x821F +#define GL_BUFFER_STORAGE_FLAGS_EXT 0x8220 +typedef void (GL_APIENTRYP PFNGLBUFFERSTORAGEEXTPROC) (GLenum target, GLsizeiptr size, const void *data, GLbitfield flags); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glBufferStorageEXT (GLenum target, GLsizeiptr size, const void *data, GLbitfield flags); +#endif +#endif /* GL_EXT_buffer_storage */ + +#ifndef GL_EXT_clear_texture +#define GL_EXT_clear_texture 1 +typedef void (GL_APIENTRYP PFNGLCLEARTEXIMAGEEXTPROC) (GLuint texture, GLint level, GLenum format, GLenum type, const void *data); +typedef void (GL_APIENTRYP PFNGLCLEARTEXSUBIMAGEEXTPROC) (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *data); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glClearTexImageEXT (GLuint texture, GLint level, GLenum format, GLenum type, const void *data); +GL_APICALL void GL_APIENTRY glClearTexSubImageEXT (GLuint texture, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const void *data); +#endif +#endif /* GL_EXT_clear_texture */ + +#ifndef GL_EXT_clip_control +#define GL_EXT_clip_control 1 +#define GL_LOWER_LEFT_EXT 0x8CA1 +#define GL_UPPER_LEFT_EXT 0x8CA2 +#define GL_NEGATIVE_ONE_TO_ONE_EXT 0x935E +#define GL_ZERO_TO_ONE_EXT 0x935F +#define GL_CLIP_ORIGIN_EXT 0x935C +#define GL_CLIP_DEPTH_MODE_EXT 0x935D +typedef void (GL_APIENTRYP PFNGLCLIPCONTROLEXTPROC) (GLenum origin, GLenum depth); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glClipControlEXT (GLenum origin, GLenum depth); +#endif +#endif /* GL_EXT_clip_control */ + +#ifndef GL_EXT_clip_cull_distance +#define GL_EXT_clip_cull_distance 1 +#define GL_MAX_CLIP_DISTANCES_EXT 0x0D32 +#define GL_MAX_CULL_DISTANCES_EXT 0x82F9 +#define GL_MAX_COMBINED_CLIP_AND_CULL_DISTANCES_EXT 0x82FA +#define GL_CLIP_DISTANCE0_EXT 0x3000 +#define GL_CLIP_DISTANCE1_EXT 0x3001 +#define GL_CLIP_DISTANCE2_EXT 0x3002 +#define GL_CLIP_DISTANCE3_EXT 0x3003 +#define GL_CLIP_DISTANCE4_EXT 0x3004 +#define GL_CLIP_DISTANCE5_EXT 0x3005 +#define GL_CLIP_DISTANCE6_EXT 0x3006 +#define GL_CLIP_DISTANCE7_EXT 0x3007 +#endif /* GL_EXT_clip_cull_distance */ + +#ifndef GL_EXT_color_buffer_float +#define GL_EXT_color_buffer_float 1 +#endif /* GL_EXT_color_buffer_float */ + +#ifndef GL_EXT_color_buffer_half_float +#define GL_EXT_color_buffer_half_float 1 +#define GL_RGBA16F_EXT 0x881A +#define GL_RGB16F_EXT 0x881B +#define GL_RG16F_EXT 0x822F +#define GL_R16F_EXT 0x822D +#define GL_FRAMEBUFFER_ATTACHMENT_COMPONENT_TYPE_EXT 0x8211 +#define GL_UNSIGNED_NORMALIZED_EXT 0x8C17 +#endif /* GL_EXT_color_buffer_half_float */ + +#ifndef GL_EXT_conservative_depth +#define GL_EXT_conservative_depth 1 +#endif /* GL_EXT_conservative_depth */ + +#ifndef GL_EXT_copy_image +#define GL_EXT_copy_image 1 +typedef void (GL_APIENTRYP PFNGLCOPYIMAGESUBDATAEXTPROC) (GLuint srcName, GLenum srcTarget, GLint srcLevel, GLint srcX, GLint srcY, GLint srcZ, GLuint dstName, GLenum dstTarget, GLint dstLevel, GLint dstX, GLint dstY, GLint dstZ, GLsizei srcWidth, GLsizei srcHeight, GLsizei srcDepth); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glCopyImageSubDataEXT (GLuint srcName, GLenum srcTarget, GLint srcLevel, GLint srcX, GLint srcY, GLint srcZ, GLuint dstName, GLenum dstTarget, GLint dstLevel, GLint dstX, GLint dstY, GLint dstZ, GLsizei srcWidth, GLsizei srcHeight, GLsizei srcDepth); +#endif +#endif /* GL_EXT_copy_image */ + +#ifndef GL_EXT_debug_label +#define GL_EXT_debug_label 1 +#define GL_PROGRAM_PIPELINE_OBJECT_EXT 0x8A4F +#define GL_PROGRAM_OBJECT_EXT 0x8B40 +#define GL_SHADER_OBJECT_EXT 0x8B48 +#define GL_BUFFER_OBJECT_EXT 0x9151 +#define GL_QUERY_OBJECT_EXT 0x9153 +#define GL_VERTEX_ARRAY_OBJECT_EXT 0x9154 +#define GL_TRANSFORM_FEEDBACK 0x8E22 +typedef void (GL_APIENTRYP PFNGLLABELOBJECTEXTPROC) (GLenum type, GLuint object, GLsizei length, const GLchar *label); +typedef void (GL_APIENTRYP PFNGLGETOBJECTLABELEXTPROC) (GLenum type, GLuint object, GLsizei bufSize, GLsizei *length, GLchar *label); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glLabelObjectEXT (GLenum type, GLuint object, GLsizei length, const GLchar *label); +GL_APICALL void GL_APIENTRY glGetObjectLabelEXT (GLenum type, GLuint object, GLsizei bufSize, GLsizei *length, GLchar *label); +#endif +#endif /* GL_EXT_debug_label */ + +#ifndef GL_EXT_debug_marker +#define GL_EXT_debug_marker 1 +typedef void (GL_APIENTRYP PFNGLINSERTEVENTMARKEREXTPROC) (GLsizei length, const GLchar *marker); +typedef void (GL_APIENTRYP PFNGLPUSHGROUPMARKEREXTPROC) (GLsizei length, const GLchar *marker); +typedef void (GL_APIENTRYP PFNGLPOPGROUPMARKEREXTPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glInsertEventMarkerEXT (GLsizei length, const GLchar *marker); +GL_APICALL void GL_APIENTRY glPushGroupMarkerEXT (GLsizei length, const GLchar *marker); +GL_APICALL void GL_APIENTRY glPopGroupMarkerEXT (void); +#endif +#endif /* GL_EXT_debug_marker */ + +#ifndef GL_EXT_depth_clamp +#define GL_EXT_depth_clamp 1 +#define GL_DEPTH_CLAMP_EXT 0x864F +#endif /* GL_EXT_depth_clamp */ + +#ifndef GL_EXT_discard_framebuffer +#define GL_EXT_discard_framebuffer 1 +#define GL_COLOR_EXT 0x1800 +#define GL_DEPTH_EXT 0x1801 +#define GL_STENCIL_EXT 0x1802 +typedef void (GL_APIENTRYP PFNGLDISCARDFRAMEBUFFEREXTPROC) (GLenum target, GLsizei numAttachments, const GLenum *attachments); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glDiscardFramebufferEXT (GLenum target, GLsizei numAttachments, const GLenum *attachments); +#endif +#endif /* GL_EXT_discard_framebuffer */ + +#ifndef GL_EXT_disjoint_timer_query +#define GL_EXT_disjoint_timer_query 1 +#define GL_QUERY_COUNTER_BITS_EXT 0x8864 +#define GL_CURRENT_QUERY_EXT 0x8865 +#define GL_QUERY_RESULT_EXT 0x8866 +#define GL_QUERY_RESULT_AVAILABLE_EXT 0x8867 +#define GL_TIME_ELAPSED_EXT 0x88BF +#define GL_TIMESTAMP_EXT 0x8E28 +#define GL_GPU_DISJOINT_EXT 0x8FBB +typedef void (GL_APIENTRYP PFNGLGENQUERIESEXTPROC) (GLsizei n, GLuint *ids); +typedef void (GL_APIENTRYP PFNGLDELETEQUERIESEXTPROC) (GLsizei n, const GLuint *ids); +typedef GLboolean (GL_APIENTRYP PFNGLISQUERYEXTPROC) (GLuint id); +typedef void (GL_APIENTRYP PFNGLBEGINQUERYEXTPROC) (GLenum target, GLuint id); +typedef void (GL_APIENTRYP PFNGLENDQUERYEXTPROC) (GLenum target); +typedef void (GL_APIENTRYP PFNGLQUERYCOUNTEREXTPROC) (GLuint id, GLenum target); +typedef void (GL_APIENTRYP PFNGLGETQUERYIVEXTPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (GL_APIENTRYP PFNGLGETQUERYOBJECTIVEXTPROC) (GLuint id, GLenum pname, GLint *params); +typedef void (GL_APIENTRYP PFNGLGETQUERYOBJECTUIVEXTPROC) (GLuint id, GLenum pname, GLuint *params); +typedef void (GL_APIENTRYP PFNGLGETQUERYOBJECTI64VEXTPROC) (GLuint id, GLenum pname, GLint64 *params); +typedef void (GL_APIENTRYP PFNGLGETQUERYOBJECTUI64VEXTPROC) (GLuint id, GLenum pname, GLuint64 *params); +typedef void (GL_APIENTRYP PFNGLGETINTEGER64VEXTPROC) (GLenum pname, GLint64 *data); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glGenQueriesEXT (GLsizei n, GLuint *ids); +GL_APICALL void GL_APIENTRY glDeleteQueriesEXT (GLsizei n, const GLuint *ids); +GL_APICALL GLboolean GL_APIENTRY glIsQueryEXT (GLuint id); +GL_APICALL void GL_APIENTRY glBeginQueryEXT (GLenum target, GLuint id); +GL_APICALL void GL_APIENTRY glEndQueryEXT (GLenum target); +GL_APICALL void GL_APIENTRY glQueryCounterEXT (GLuint id, GLenum target); +GL_APICALL void GL_APIENTRY glGetQueryivEXT (GLenum target, GLenum pname, GLint *params); +GL_APICALL void GL_APIENTRY glGetQueryObjectivEXT (GLuint id, GLenum pname, GLint *params); +GL_APICALL void GL_APIENTRY glGetQueryObjectuivEXT (GLuint id, GLenum pname, GLuint *params); +GL_APICALL void GL_APIENTRY glGetQueryObjecti64vEXT (GLuint id, GLenum pname, GLint64 *params); +GL_APICALL void GL_APIENTRY glGetQueryObjectui64vEXT (GLuint id, GLenum pname, GLuint64 *params); +GL_APICALL void GL_APIENTRY glGetInteger64vEXT (GLenum pname, GLint64 *data); +#endif +#endif /* GL_EXT_disjoint_timer_query */ + +#ifndef GL_EXT_draw_buffers +#define GL_EXT_draw_buffers 1 +#define GL_MAX_COLOR_ATTACHMENTS_EXT 0x8CDF +#define GL_MAX_DRAW_BUFFERS_EXT 0x8824 +#define GL_DRAW_BUFFER0_EXT 0x8825 +#define GL_DRAW_BUFFER1_EXT 0x8826 +#define GL_DRAW_BUFFER2_EXT 0x8827 +#define GL_DRAW_BUFFER3_EXT 0x8828 +#define GL_DRAW_BUFFER4_EXT 0x8829 +#define GL_DRAW_BUFFER5_EXT 0x882A +#define GL_DRAW_BUFFER6_EXT 0x882B +#define GL_DRAW_BUFFER7_EXT 0x882C +#define GL_DRAW_BUFFER8_EXT 0x882D +#define GL_DRAW_BUFFER9_EXT 0x882E +#define GL_DRAW_BUFFER10_EXT 0x882F +#define GL_DRAW_BUFFER11_EXT 0x8830 +#define GL_DRAW_BUFFER12_EXT 0x8831 +#define GL_DRAW_BUFFER13_EXT 0x8832 +#define GL_DRAW_BUFFER14_EXT 0x8833 +#define GL_DRAW_BUFFER15_EXT 0x8834 +#define GL_COLOR_ATTACHMENT0_EXT 0x8CE0 +#define GL_COLOR_ATTACHMENT1_EXT 0x8CE1 +#define GL_COLOR_ATTACHMENT2_EXT 0x8CE2 +#define GL_COLOR_ATTACHMENT3_EXT 0x8CE3 +#define GL_COLOR_ATTACHMENT4_EXT 0x8CE4 +#define GL_COLOR_ATTACHMENT5_EXT 0x8CE5 +#define GL_COLOR_ATTACHMENT6_EXT 0x8CE6 +#define GL_COLOR_ATTACHMENT7_EXT 0x8CE7 +#define GL_COLOR_ATTACHMENT8_EXT 0x8CE8 +#define GL_COLOR_ATTACHMENT9_EXT 0x8CE9 +#define GL_COLOR_ATTACHMENT10_EXT 0x8CEA +#define GL_COLOR_ATTACHMENT11_EXT 0x8CEB +#define GL_COLOR_ATTACHMENT12_EXT 0x8CEC +#define GL_COLOR_ATTACHMENT13_EXT 0x8CED +#define GL_COLOR_ATTACHMENT14_EXT 0x8CEE +#define GL_COLOR_ATTACHMENT15_EXT 0x8CEF +typedef void (GL_APIENTRYP PFNGLDRAWBUFFERSEXTPROC) (GLsizei n, const GLenum *bufs); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glDrawBuffersEXT (GLsizei n, const GLenum *bufs); +#endif +#endif /* GL_EXT_draw_buffers */ + +#ifndef GL_EXT_draw_buffers_indexed +#define GL_EXT_draw_buffers_indexed 1 +typedef void (GL_APIENTRYP PFNGLENABLEIEXTPROC) (GLenum target, GLuint index); +typedef void (GL_APIENTRYP PFNGLDISABLEIEXTPROC) (GLenum target, GLuint index); +typedef void (GL_APIENTRYP PFNGLBLENDEQUATIONIEXTPROC) (GLuint buf, GLenum mode); +typedef void (GL_APIENTRYP PFNGLBLENDEQUATIONSEPARATEIEXTPROC) (GLuint buf, GLenum modeRGB, GLenum modeAlpha); +typedef void (GL_APIENTRYP PFNGLBLENDFUNCIEXTPROC) (GLuint buf, GLenum src, GLenum dst); +typedef void (GL_APIENTRYP PFNGLBLENDFUNCSEPARATEIEXTPROC) (GLuint buf, GLenum srcRGB, GLenum dstRGB, GLenum srcAlpha, GLenum dstAlpha); +typedef void (GL_APIENTRYP PFNGLCOLORMASKIEXTPROC) (GLuint index, GLboolean r, GLboolean g, GLboolean b, GLboolean a); +typedef GLboolean (GL_APIENTRYP PFNGLISENABLEDIEXTPROC) (GLenum target, GLuint index); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glEnableiEXT (GLenum target, GLuint index); +GL_APICALL void GL_APIENTRY glDisableiEXT (GLenum target, GLuint index); +GL_APICALL void GL_APIENTRY glBlendEquationiEXT (GLuint buf, GLenum mode); +GL_APICALL void GL_APIENTRY glBlendEquationSeparateiEXT (GLuint buf, GLenum modeRGB, GLenum modeAlpha); +GL_APICALL void GL_APIENTRY glBlendFunciEXT (GLuint buf, GLenum src, GLenum dst); +GL_APICALL void GL_APIENTRY glBlendFuncSeparateiEXT (GLuint buf, GLenum srcRGB, GLenum dstRGB, GLenum srcAlpha, GLenum dstAlpha); +GL_APICALL void GL_APIENTRY glColorMaskiEXT (GLuint index, GLboolean r, GLboolean g, GLboolean b, GLboolean a); +GL_APICALL GLboolean GL_APIENTRY glIsEnablediEXT (GLenum target, GLuint index); +#endif +#endif /* GL_EXT_draw_buffers_indexed */ + +#ifndef GL_EXT_draw_elements_base_vertex +#define GL_EXT_draw_elements_base_vertex 1 +typedef void (GL_APIENTRYP PFNGLDRAWELEMENTSBASEVERTEXEXTPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLint basevertex); +typedef void (GL_APIENTRYP PFNGLDRAWRANGEELEMENTSBASEVERTEXEXTPROC) (GLenum mode, GLuint start, GLuint end, GLsizei count, GLenum type, const void *indices, GLint basevertex); +typedef void (GL_APIENTRYP PFNGLDRAWELEMENTSINSTANCEDBASEVERTEXEXTPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount, GLint basevertex); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glDrawElementsBaseVertexEXT (GLenum mode, GLsizei count, GLenum type, const void *indices, GLint basevertex); +GL_APICALL void GL_APIENTRY glDrawRangeElementsBaseVertexEXT (GLenum mode, GLuint start, GLuint end, GLsizei count, GLenum type, const void *indices, GLint basevertex); +GL_APICALL void GL_APIENTRY glDrawElementsInstancedBaseVertexEXT (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei instancecount, GLint basevertex); +#endif +#endif /* GL_EXT_draw_elements_base_vertex */ + +#ifndef GL_EXT_draw_instanced +#define GL_EXT_draw_instanced 1 +typedef void (GL_APIENTRYP PFNGLDRAWARRAYSINSTANCEDEXTPROC) (GLenum mode, GLint start, GLsizei count, GLsizei primcount); +typedef void (GL_APIENTRYP PFNGLDRAWELEMENTSINSTANCEDEXTPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei primcount); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glDrawArraysInstancedEXT (GLenum mode, GLint start, GLsizei count, GLsizei primcount); +GL_APICALL void GL_APIENTRY glDrawElementsInstancedEXT (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei primcount); +#endif +#endif /* GL_EXT_draw_instanced */ + +#ifndef GL_EXT_draw_transform_feedback +#define GL_EXT_draw_transform_feedback 1 +typedef void (GL_APIENTRYP PFNGLDRAWTRANSFORMFEEDBACKEXTPROC) (GLenum mode, GLuint id); +typedef void (GL_APIENTRYP PFNGLDRAWTRANSFORMFEEDBACKINSTANCEDEXTPROC) (GLenum mode, GLuint id, GLsizei instancecount); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glDrawTransformFeedbackEXT (GLenum mode, GLuint id); +GL_APICALL void GL_APIENTRY glDrawTransformFeedbackInstancedEXT (GLenum mode, GLuint id, GLsizei instancecount); +#endif +#endif /* GL_EXT_draw_transform_feedback */ + +#ifndef GL_EXT_external_buffer +#define GL_EXT_external_buffer 1 +typedef void *GLeglClientBufferEXT; +typedef void (GL_APIENTRYP PFNGLBUFFERSTORAGEEXTERNALEXTPROC) (GLenum target, GLintptr offset, GLsizeiptr size, GLeglClientBufferEXT clientBuffer, GLbitfield flags); +typedef void (GL_APIENTRYP PFNGLNAMEDBUFFERSTORAGEEXTERNALEXTPROC) (GLuint buffer, GLintptr offset, GLsizeiptr size, GLeglClientBufferEXT clientBuffer, GLbitfield flags); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glBufferStorageExternalEXT (GLenum target, GLintptr offset, GLsizeiptr size, GLeglClientBufferEXT clientBuffer, GLbitfield flags); +GL_APICALL void GL_APIENTRY glNamedBufferStorageExternalEXT (GLuint buffer, GLintptr offset, GLsizeiptr size, GLeglClientBufferEXT clientBuffer, GLbitfield flags); +#endif +#endif /* GL_EXT_external_buffer */ + +#ifndef GL_EXT_float_blend +#define GL_EXT_float_blend 1 +#endif /* GL_EXT_float_blend */ + +#ifndef GL_EXT_fragment_shading_rate +#define GL_EXT_fragment_shading_rate 1 +#define GL_SHADING_RATE_1X1_PIXELS_EXT 0x96A6 +#define GL_SHADING_RATE_1X2_PIXELS_EXT 0x96A7 +#define GL_SHADING_RATE_2X1_PIXELS_EXT 0x96A8 +#define GL_SHADING_RATE_2X2_PIXELS_EXT 0x96A9 +#define GL_SHADING_RATE_1X4_PIXELS_EXT 0x96AA +#define GL_SHADING_RATE_4X1_PIXELS_EXT 0x96AB +#define GL_SHADING_RATE_4X2_PIXELS_EXT 0x96AC +#define GL_SHADING_RATE_2X4_PIXELS_EXT 0x96AD +#define GL_SHADING_RATE_4X4_PIXELS_EXT 0x96AE +#define GL_SHADING_RATE_EXT 0x96D0 +#define GL_SHADING_RATE_ATTACHMENT_EXT 0x96D1 +#define GL_FRAGMENT_SHADING_RATE_COMBINER_OP_KEEP_EXT 0x96D2 +#define GL_FRAGMENT_SHADING_RATE_COMBINER_OP_REPLACE_EXT 0x96D3 +#define GL_FRAGMENT_SHADING_RATE_COMBINER_OP_MIN_EXT 0x96D4 +#define GL_FRAGMENT_SHADING_RATE_COMBINER_OP_MAX_EXT 0x96D5 +#define GL_FRAGMENT_SHADING_RATE_COMBINER_OP_MUL_EXT 0x96D6 +#define GL_MIN_FRAGMENT_SHADING_RATE_ATTACHMENT_TEXEL_WIDTH_EXT 0x96D7 +#define GL_MAX_FRAGMENT_SHADING_RATE_ATTACHMENT_TEXEL_WIDTH_EXT 0x96D8 +#define GL_MIN_FRAGMENT_SHADING_RATE_ATTACHMENT_TEXEL_HEIGHT_EXT 0x96D9 +#define GL_MAX_FRAGMENT_SHADING_RATE_ATTACHMENT_TEXEL_HEIGHT_EXT 0x96DA +#define GL_MAX_FRAGMENT_SHADING_RATE_ATTACHMENT_TEXEL_ASPECT_RATIO_EXT 0x96DB +#define GL_MAX_FRAGMENT_SHADING_RATE_ATTACHMENT_LAYERS_EXT 0x96DC +#define GL_FRAGMENT_SHADING_RATE_WITH_SHADER_DEPTH_STENCIL_WRITES_SUPPORTED_EXT 0x96DD +#define GL_FRAGMENT_SHADING_RATE_WITH_SAMPLE_MASK_SUPPORTED_EXT 0x96DE +#define GL_FRAGMENT_SHADING_RATE_ATTACHMENT_WITH_DEFAULT_FRAMEBUFFER_SUPPORTED_EXT 0x96DF +#define GL_FRAGMENT_SHADING_RATE_NON_TRIVIAL_COMBINERS_SUPPORTED_EXT 0x8F6F +typedef void (GL_APIENTRYP PFNGLGETFRAGMENTSHADINGRATESEXTPROC) (GLsizei samples, GLsizei maxCount, GLsizei *count, GLenum *shadingRates); +typedef void (GL_APIENTRYP PFNGLSHADINGRATEEXTPROC) (GLenum rate); +typedef void (GL_APIENTRYP PFNGLSHADINGRATECOMBINEROPSEXTPROC) (GLenum combinerOp0, GLenum combinerOp1); +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERSHADINGRATEEXTPROC) (GLenum target, GLenum attachment, GLuint texture, GLint baseLayer, GLsizei numLayers, GLsizei texelWidth, GLsizei texelHeight); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glGetFragmentShadingRatesEXT (GLsizei samples, GLsizei maxCount, GLsizei *count, GLenum *shadingRates); +GL_APICALL void GL_APIENTRY glShadingRateEXT (GLenum rate); +GL_APICALL void GL_APIENTRY glShadingRateCombinerOpsEXT (GLenum combinerOp0, GLenum combinerOp1); +GL_APICALL void GL_APIENTRY glFramebufferShadingRateEXT (GLenum target, GLenum attachment, GLuint texture, GLint baseLayer, GLsizei numLayers, GLsizei texelWidth, GLsizei texelHeight); +#endif +#endif /* GL_EXT_fragment_shading_rate */ + +#ifndef GL_EXT_geometry_point_size +#define GL_EXT_geometry_point_size 1 +#endif /* GL_EXT_geometry_point_size */ + +#ifndef GL_EXT_geometry_shader +#define GL_EXT_geometry_shader 1 +#define GL_GEOMETRY_SHADER_EXT 0x8DD9 +#define GL_GEOMETRY_SHADER_BIT_EXT 0x00000004 +#define GL_GEOMETRY_LINKED_VERTICES_OUT_EXT 0x8916 +#define GL_GEOMETRY_LINKED_INPUT_TYPE_EXT 0x8917 +#define GL_GEOMETRY_LINKED_OUTPUT_TYPE_EXT 0x8918 +#define GL_GEOMETRY_SHADER_INVOCATIONS_EXT 0x887F +#define GL_LAYER_PROVOKING_VERTEX_EXT 0x825E +#define GL_LINES_ADJACENCY_EXT 0x000A +#define GL_LINE_STRIP_ADJACENCY_EXT 0x000B +#define GL_TRIANGLES_ADJACENCY_EXT 0x000C +#define GL_TRIANGLE_STRIP_ADJACENCY_EXT 0x000D +#define GL_MAX_GEOMETRY_UNIFORM_COMPONENTS_EXT 0x8DDF +#define GL_MAX_GEOMETRY_UNIFORM_BLOCKS_EXT 0x8A2C +#define GL_MAX_COMBINED_GEOMETRY_UNIFORM_COMPONENTS_EXT 0x8A32 +#define GL_MAX_GEOMETRY_INPUT_COMPONENTS_EXT 0x9123 +#define GL_MAX_GEOMETRY_OUTPUT_COMPONENTS_EXT 0x9124 +#define GL_MAX_GEOMETRY_OUTPUT_VERTICES_EXT 0x8DE0 +#define GL_MAX_GEOMETRY_TOTAL_OUTPUT_COMPONENTS_EXT 0x8DE1 +#define GL_MAX_GEOMETRY_SHADER_INVOCATIONS_EXT 0x8E5A +#define GL_MAX_GEOMETRY_TEXTURE_IMAGE_UNITS_EXT 0x8C29 +#define GL_MAX_GEOMETRY_ATOMIC_COUNTER_BUFFERS_EXT 0x92CF +#define GL_MAX_GEOMETRY_ATOMIC_COUNTERS_EXT 0x92D5 +#define GL_MAX_GEOMETRY_IMAGE_UNIFORMS_EXT 0x90CD +#define GL_MAX_GEOMETRY_SHADER_STORAGE_BLOCKS_EXT 0x90D7 +#define GL_FIRST_VERTEX_CONVENTION_EXT 0x8E4D +#define GL_LAST_VERTEX_CONVENTION_EXT 0x8E4E +#define GL_UNDEFINED_VERTEX_EXT 0x8260 +#define GL_PRIMITIVES_GENERATED_EXT 0x8C87 +#define GL_FRAMEBUFFER_DEFAULT_LAYERS_EXT 0x9312 +#define GL_MAX_FRAMEBUFFER_LAYERS_EXT 0x9317 +#define GL_FRAMEBUFFER_INCOMPLETE_LAYER_TARGETS_EXT 0x8DA8 +#define GL_FRAMEBUFFER_ATTACHMENT_LAYERED_EXT 0x8DA7 +#define GL_REFERENCED_BY_GEOMETRY_SHADER_EXT 0x9309 +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERTEXTUREEXTPROC) (GLenum target, GLenum attachment, GLuint texture, GLint level); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glFramebufferTextureEXT (GLenum target, GLenum attachment, GLuint texture, GLint level); +#endif +#endif /* GL_EXT_geometry_shader */ + +#ifndef GL_EXT_gpu_shader5 +#define GL_EXT_gpu_shader5 1 +#endif /* GL_EXT_gpu_shader5 */ + +#ifndef GL_EXT_instanced_arrays +#define GL_EXT_instanced_arrays 1 +#define GL_VERTEX_ATTRIB_ARRAY_DIVISOR_EXT 0x88FE +typedef void (GL_APIENTRYP PFNGLVERTEXATTRIBDIVISOREXTPROC) (GLuint index, GLuint divisor); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glVertexAttribDivisorEXT (GLuint index, GLuint divisor); +#endif +#endif /* GL_EXT_instanced_arrays */ + +#ifndef GL_EXT_map_buffer_range +#define GL_EXT_map_buffer_range 1 +#define GL_MAP_READ_BIT_EXT 0x0001 +#define GL_MAP_WRITE_BIT_EXT 0x0002 +#define GL_MAP_INVALIDATE_RANGE_BIT_EXT 0x0004 +#define GL_MAP_INVALIDATE_BUFFER_BIT_EXT 0x0008 +#define GL_MAP_FLUSH_EXPLICIT_BIT_EXT 0x0010 +#define GL_MAP_UNSYNCHRONIZED_BIT_EXT 0x0020 +typedef void *(GL_APIENTRYP PFNGLMAPBUFFERRANGEEXTPROC) (GLenum target, GLintptr offset, GLsizeiptr length, GLbitfield access); +typedef void (GL_APIENTRYP PFNGLFLUSHMAPPEDBUFFERRANGEEXTPROC) (GLenum target, GLintptr offset, GLsizeiptr length); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void *GL_APIENTRY glMapBufferRangeEXT (GLenum target, GLintptr offset, GLsizeiptr length, GLbitfield access); +GL_APICALL void GL_APIENTRY glFlushMappedBufferRangeEXT (GLenum target, GLintptr offset, GLsizeiptr length); +#endif +#endif /* GL_EXT_map_buffer_range */ + +#ifndef GL_EXT_memory_object +#define GL_EXT_memory_object 1 +#define GL_TEXTURE_TILING_EXT 0x9580 +#define GL_DEDICATED_MEMORY_OBJECT_EXT 0x9581 +#define GL_PROTECTED_MEMORY_OBJECT_EXT 0x959B +#define GL_NUM_TILING_TYPES_EXT 0x9582 +#define GL_TILING_TYPES_EXT 0x9583 +#define GL_OPTIMAL_TILING_EXT 0x9584 +#define GL_LINEAR_TILING_EXT 0x9585 +#define GL_NUM_DEVICE_UUIDS_EXT 0x9596 +#define GL_DEVICE_UUID_EXT 0x9597 +#define GL_DRIVER_UUID_EXT 0x9598 +#define GL_UUID_SIZE_EXT 16 +typedef void (GL_APIENTRYP PFNGLGETUNSIGNEDBYTEVEXTPROC) (GLenum pname, GLubyte *data); +typedef void (GL_APIENTRYP PFNGLGETUNSIGNEDBYTEI_VEXTPROC) (GLenum target, GLuint index, GLubyte *data); +typedef void (GL_APIENTRYP PFNGLDELETEMEMORYOBJECTSEXTPROC) (GLsizei n, const GLuint *memoryObjects); +typedef GLboolean (GL_APIENTRYP PFNGLISMEMORYOBJECTEXTPROC) (GLuint memoryObject); +typedef void (GL_APIENTRYP PFNGLCREATEMEMORYOBJECTSEXTPROC) (GLsizei n, GLuint *memoryObjects); +typedef void (GL_APIENTRYP PFNGLMEMORYOBJECTPARAMETERIVEXTPROC) (GLuint memoryObject, GLenum pname, const GLint *params); +typedef void (GL_APIENTRYP PFNGLGETMEMORYOBJECTPARAMETERIVEXTPROC) (GLuint memoryObject, GLenum pname, GLint *params); +typedef void (GL_APIENTRYP PFNGLTEXSTORAGEMEM2DEXTPROC) (GLenum target, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLuint memory, GLuint64 offset); +typedef void (GL_APIENTRYP PFNGLTEXSTORAGEMEM2DMULTISAMPLEEXTPROC) (GLenum target, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +typedef void (GL_APIENTRYP PFNGLTEXSTORAGEMEM3DEXTPROC) (GLenum target, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset); +typedef void (GL_APIENTRYP PFNGLTEXSTORAGEMEM3DMULTISAMPLEEXTPROC) (GLenum target, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +typedef void (GL_APIENTRYP PFNGLBUFFERSTORAGEMEMEXTPROC) (GLenum target, GLsizeiptr size, GLuint memory, GLuint64 offset); +typedef void (GL_APIENTRYP PFNGLTEXTURESTORAGEMEM2DEXTPROC) (GLuint texture, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLuint memory, GLuint64 offset); +typedef void (GL_APIENTRYP PFNGLTEXTURESTORAGEMEM2DMULTISAMPLEEXTPROC) (GLuint texture, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +typedef void (GL_APIENTRYP PFNGLTEXTURESTORAGEMEM3DEXTPROC) (GLuint texture, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset); +typedef void (GL_APIENTRYP PFNGLTEXTURESTORAGEMEM3DMULTISAMPLEEXTPROC) (GLuint texture, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +typedef void (GL_APIENTRYP PFNGLNAMEDBUFFERSTORAGEMEMEXTPROC) (GLuint buffer, GLsizeiptr size, GLuint memory, GLuint64 offset); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glGetUnsignedBytevEXT (GLenum pname, GLubyte *data); +GL_APICALL void GL_APIENTRY glGetUnsignedBytei_vEXT (GLenum target, GLuint index, GLubyte *data); +GL_APICALL void GL_APIENTRY glDeleteMemoryObjectsEXT (GLsizei n, const GLuint *memoryObjects); +GL_APICALL GLboolean GL_APIENTRY glIsMemoryObjectEXT (GLuint memoryObject); +GL_APICALL void GL_APIENTRY glCreateMemoryObjectsEXT (GLsizei n, GLuint *memoryObjects); +GL_APICALL void GL_APIENTRY glMemoryObjectParameterivEXT (GLuint memoryObject, GLenum pname, const GLint *params); +GL_APICALL void GL_APIENTRY glGetMemoryObjectParameterivEXT (GLuint memoryObject, GLenum pname, GLint *params); +GL_APICALL void GL_APIENTRY glTexStorageMem2DEXT (GLenum target, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLuint memory, GLuint64 offset); +GL_APICALL void GL_APIENTRY glTexStorageMem2DMultisampleEXT (GLenum target, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +GL_APICALL void GL_APIENTRY glTexStorageMem3DEXT (GLenum target, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset); +GL_APICALL void GL_APIENTRY glTexStorageMem3DMultisampleEXT (GLenum target, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +GL_APICALL void GL_APIENTRY glBufferStorageMemEXT (GLenum target, GLsizeiptr size, GLuint memory, GLuint64 offset); +GL_APICALL void GL_APIENTRY glTextureStorageMem2DEXT (GLuint texture, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLuint memory, GLuint64 offset); +GL_APICALL void GL_APIENTRY glTextureStorageMem2DMultisampleEXT (GLuint texture, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +GL_APICALL void GL_APIENTRY glTextureStorageMem3DEXT (GLuint texture, GLsizei levels, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset); +GL_APICALL void GL_APIENTRY glTextureStorageMem3DMultisampleEXT (GLuint texture, GLsizei samples, GLenum internalFormat, GLsizei width, GLsizei height, GLsizei depth, GLboolean fixedSampleLocations, GLuint memory, GLuint64 offset); +GL_APICALL void GL_APIENTRY glNamedBufferStorageMemEXT (GLuint buffer, GLsizeiptr size, GLuint memory, GLuint64 offset); +#endif +#endif /* GL_EXT_memory_object */ + +#ifndef GL_EXT_memory_object_fd +#define GL_EXT_memory_object_fd 1 +#define GL_HANDLE_TYPE_OPAQUE_FD_EXT 0x9586 +typedef void (GL_APIENTRYP PFNGLIMPORTMEMORYFDEXTPROC) (GLuint memory, GLuint64 size, GLenum handleType, GLint fd); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glImportMemoryFdEXT (GLuint memory, GLuint64 size, GLenum handleType, GLint fd); +#endif +#endif /* GL_EXT_memory_object_fd */ + +#ifndef GL_EXT_memory_object_win32 +#define GL_EXT_memory_object_win32 1 +#define GL_HANDLE_TYPE_OPAQUE_WIN32_EXT 0x9587 +#define GL_HANDLE_TYPE_OPAQUE_WIN32_KMT_EXT 0x9588 +#define GL_DEVICE_LUID_EXT 0x9599 +#define GL_DEVICE_NODE_MASK_EXT 0x959A +#define GL_LUID_SIZE_EXT 8 +#define GL_HANDLE_TYPE_D3D12_TILEPOOL_EXT 0x9589 +#define GL_HANDLE_TYPE_D3D12_RESOURCE_EXT 0x958A +#define GL_HANDLE_TYPE_D3D11_IMAGE_EXT 0x958B +#define GL_HANDLE_TYPE_D3D11_IMAGE_KMT_EXT 0x958C +typedef void (GL_APIENTRYP PFNGLIMPORTMEMORYWIN32HANDLEEXTPROC) (GLuint memory, GLuint64 size, GLenum handleType, void *handle); +typedef void (GL_APIENTRYP PFNGLIMPORTMEMORYWIN32NAMEEXTPROC) (GLuint memory, GLuint64 size, GLenum handleType, const void *name); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glImportMemoryWin32HandleEXT (GLuint memory, GLuint64 size, GLenum handleType, void *handle); +GL_APICALL void GL_APIENTRY glImportMemoryWin32NameEXT (GLuint memory, GLuint64 size, GLenum handleType, const void *name); +#endif +#endif /* GL_EXT_memory_object_win32 */ + +#ifndef GL_EXT_multi_draw_arrays +#define GL_EXT_multi_draw_arrays 1 +typedef void (GL_APIENTRYP PFNGLMULTIDRAWARRAYSEXTPROC) (GLenum mode, const GLint *first, const GLsizei *count, GLsizei primcount); +typedef void (GL_APIENTRYP PFNGLMULTIDRAWELEMENTSEXTPROC) (GLenum mode, const GLsizei *count, GLenum type, const void *const*indices, GLsizei primcount); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glMultiDrawArraysEXT (GLenum mode, const GLint *first, const GLsizei *count, GLsizei primcount); +GL_APICALL void GL_APIENTRY glMultiDrawElementsEXT (GLenum mode, const GLsizei *count, GLenum type, const void *const*indices, GLsizei primcount); +#endif +#endif /* GL_EXT_multi_draw_arrays */ + +#ifndef GL_EXT_multi_draw_indirect +#define GL_EXT_multi_draw_indirect 1 +typedef void (GL_APIENTRYP PFNGLMULTIDRAWARRAYSINDIRECTEXTPROC) (GLenum mode, const void *indirect, GLsizei drawcount, GLsizei stride); +typedef void (GL_APIENTRYP PFNGLMULTIDRAWELEMENTSINDIRECTEXTPROC) (GLenum mode, GLenum type, const void *indirect, GLsizei drawcount, GLsizei stride); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glMultiDrawArraysIndirectEXT (GLenum mode, const void *indirect, GLsizei drawcount, GLsizei stride); +GL_APICALL void GL_APIENTRY glMultiDrawElementsIndirectEXT (GLenum mode, GLenum type, const void *indirect, GLsizei drawcount, GLsizei stride); +#endif +#endif /* GL_EXT_multi_draw_indirect */ + +#ifndef GL_EXT_multisampled_compatibility +#define GL_EXT_multisampled_compatibility 1 +#define GL_MULTISAMPLE_EXT 0x809D +#define GL_SAMPLE_ALPHA_TO_ONE_EXT 0x809F +#endif /* GL_EXT_multisampled_compatibility */ + +#ifndef GL_EXT_multisampled_render_to_texture +#define GL_EXT_multisampled_render_to_texture 1 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_SAMPLES_EXT 0x8D6C +#define GL_RENDERBUFFER_SAMPLES_EXT 0x8CAB +#define GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE_EXT 0x8D56 +#define GL_MAX_SAMPLES_EXT 0x8D57 +typedef void (GL_APIENTRYP PFNGLRENDERBUFFERSTORAGEMULTISAMPLEEXTPROC) (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERTEXTURE2DMULTISAMPLEEXTPROC) (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level, GLsizei samples); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glRenderbufferStorageMultisampleEXT (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +GL_APICALL void GL_APIENTRY glFramebufferTexture2DMultisampleEXT (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level, GLsizei samples); +#endif +#endif /* GL_EXT_multisampled_render_to_texture */ + +#ifndef GL_EXT_multisampled_render_to_texture2 +#define GL_EXT_multisampled_render_to_texture2 1 +#endif /* GL_EXT_multisampled_render_to_texture2 */ + +#ifndef GL_EXT_multiview_draw_buffers +#define GL_EXT_multiview_draw_buffers 1 +#define GL_COLOR_ATTACHMENT_EXT 0x90F0 +#define GL_MULTIVIEW_EXT 0x90F1 +#define GL_DRAW_BUFFER_EXT 0x0C01 +#define GL_READ_BUFFER_EXT 0x0C02 +#define GL_MAX_MULTIVIEW_BUFFERS_EXT 0x90F2 +typedef void (GL_APIENTRYP PFNGLREADBUFFERINDEXEDEXTPROC) (GLenum src, GLint index); +typedef void (GL_APIENTRYP PFNGLDRAWBUFFERSINDEXEDEXTPROC) (GLint n, const GLenum *location, const GLint *indices); +typedef void (GL_APIENTRYP PFNGLGETINTEGERI_VEXTPROC) (GLenum target, GLuint index, GLint *data); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glReadBufferIndexedEXT (GLenum src, GLint index); +GL_APICALL void GL_APIENTRY glDrawBuffersIndexedEXT (GLint n, const GLenum *location, const GLint *indices); +GL_APICALL void GL_APIENTRY glGetIntegeri_vEXT (GLenum target, GLuint index, GLint *data); +#endif +#endif /* GL_EXT_multiview_draw_buffers */ + +#ifndef GL_EXT_multiview_tessellation_geometry_shader +#define GL_EXT_multiview_tessellation_geometry_shader 1 +#endif /* GL_EXT_multiview_tessellation_geometry_shader */ + +#ifndef GL_EXT_multiview_texture_multisample +#define GL_EXT_multiview_texture_multisample 1 +#endif /* GL_EXT_multiview_texture_multisample */ + +#ifndef GL_EXT_multiview_timer_query +#define GL_EXT_multiview_timer_query 1 +#endif /* GL_EXT_multiview_timer_query */ + +#ifndef GL_EXT_occlusion_query_boolean +#define GL_EXT_occlusion_query_boolean 1 +#define GL_ANY_SAMPLES_PASSED_EXT 0x8C2F +#define GL_ANY_SAMPLES_PASSED_CONSERVATIVE_EXT 0x8D6A +#endif /* GL_EXT_occlusion_query_boolean */ + +#ifndef GL_EXT_polygon_offset_clamp +#define GL_EXT_polygon_offset_clamp 1 +#define GL_POLYGON_OFFSET_CLAMP_EXT 0x8E1B +typedef void (GL_APIENTRYP PFNGLPOLYGONOFFSETCLAMPEXTPROC) (GLfloat factor, GLfloat units, GLfloat clamp); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glPolygonOffsetClampEXT (GLfloat factor, GLfloat units, GLfloat clamp); +#endif +#endif /* GL_EXT_polygon_offset_clamp */ + +#ifndef GL_EXT_post_depth_coverage +#define GL_EXT_post_depth_coverage 1 +#endif /* GL_EXT_post_depth_coverage */ + +#ifndef GL_EXT_primitive_bounding_box +#define GL_EXT_primitive_bounding_box 1 +#define GL_PRIMITIVE_BOUNDING_BOX_EXT 0x92BE +typedef void (GL_APIENTRYP PFNGLPRIMITIVEBOUNDINGBOXEXTPROC) (GLfloat minX, GLfloat minY, GLfloat minZ, GLfloat minW, GLfloat maxX, GLfloat maxY, GLfloat maxZ, GLfloat maxW); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glPrimitiveBoundingBoxEXT (GLfloat minX, GLfloat minY, GLfloat minZ, GLfloat minW, GLfloat maxX, GLfloat maxY, GLfloat maxZ, GLfloat maxW); +#endif +#endif /* GL_EXT_primitive_bounding_box */ + +#ifndef GL_EXT_protected_textures +#define GL_EXT_protected_textures 1 +#define GL_CONTEXT_FLAG_PROTECTED_CONTENT_BIT_EXT 0x00000010 +#define GL_TEXTURE_PROTECTED_EXT 0x8BFA +#endif /* GL_EXT_protected_textures */ + +#ifndef GL_EXT_pvrtc_sRGB +#define GL_EXT_pvrtc_sRGB 1 +#define GL_COMPRESSED_SRGB_PVRTC_2BPPV1_EXT 0x8A54 +#define GL_COMPRESSED_SRGB_PVRTC_4BPPV1_EXT 0x8A55 +#define GL_COMPRESSED_SRGB_ALPHA_PVRTC_2BPPV1_EXT 0x8A56 +#define GL_COMPRESSED_SRGB_ALPHA_PVRTC_4BPPV1_EXT 0x8A57 +#define GL_COMPRESSED_SRGB_ALPHA_PVRTC_2BPPV2_IMG 0x93F0 +#define GL_COMPRESSED_SRGB_ALPHA_PVRTC_4BPPV2_IMG 0x93F1 +#endif /* GL_EXT_pvrtc_sRGB */ + +#ifndef GL_EXT_raster_multisample +#define GL_EXT_raster_multisample 1 +#define GL_RASTER_MULTISAMPLE_EXT 0x9327 +#define GL_RASTER_SAMPLES_EXT 0x9328 +#define GL_MAX_RASTER_SAMPLES_EXT 0x9329 +#define GL_RASTER_FIXED_SAMPLE_LOCATIONS_EXT 0x932A +#define GL_MULTISAMPLE_RASTERIZATION_ALLOWED_EXT 0x932B +#define GL_EFFECTIVE_RASTER_SAMPLES_EXT 0x932C +typedef void (GL_APIENTRYP PFNGLRASTERSAMPLESEXTPROC) (GLuint samples, GLboolean fixedsamplelocations); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glRasterSamplesEXT (GLuint samples, GLboolean fixedsamplelocations); +#endif +#endif /* GL_EXT_raster_multisample */ + +#ifndef GL_EXT_read_format_bgra +#define GL_EXT_read_format_bgra 1 +#define GL_UNSIGNED_SHORT_4_4_4_4_REV_EXT 0x8365 +#define GL_UNSIGNED_SHORT_1_5_5_5_REV_EXT 0x8366 +#endif /* GL_EXT_read_format_bgra */ + +#ifndef GL_EXT_render_snorm +#define GL_EXT_render_snorm 1 +#define GL_R8_SNORM 0x8F94 +#define GL_RG8_SNORM 0x8F95 +#define GL_RGBA8_SNORM 0x8F97 +#define GL_R16_SNORM_EXT 0x8F98 +#define GL_RG16_SNORM_EXT 0x8F99 +#define GL_RGBA16_SNORM_EXT 0x8F9B +#endif /* GL_EXT_render_snorm */ + +#ifndef GL_EXT_robustness +#define GL_EXT_robustness 1 +#define GL_GUILTY_CONTEXT_RESET_EXT 0x8253 +#define GL_INNOCENT_CONTEXT_RESET_EXT 0x8254 +#define GL_UNKNOWN_CONTEXT_RESET_EXT 0x8255 +#define GL_CONTEXT_ROBUST_ACCESS_EXT 0x90F3 +#define GL_RESET_NOTIFICATION_STRATEGY_EXT 0x8256 +#define GL_LOSE_CONTEXT_ON_RESET_EXT 0x8252 +#define GL_NO_RESET_NOTIFICATION_EXT 0x8261 +typedef GLenum (GL_APIENTRYP PFNGLGETGRAPHICSRESETSTATUSEXTPROC) (void); +typedef void (GL_APIENTRYP PFNGLREADNPIXELSEXTPROC) (GLint x, GLint y, GLsizei width, GLsizei height, GLenum format, GLenum type, GLsizei bufSize, void *data); +typedef void (GL_APIENTRYP PFNGLGETNUNIFORMFVEXTPROC) (GLuint program, GLint location, GLsizei bufSize, GLfloat *params); +typedef void (GL_APIENTRYP PFNGLGETNUNIFORMIVEXTPROC) (GLuint program, GLint location, GLsizei bufSize, GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL GLenum GL_APIENTRY glGetGraphicsResetStatusEXT (void); +GL_APICALL void GL_APIENTRY glReadnPixelsEXT (GLint x, GLint y, GLsizei width, GLsizei height, GLenum format, GLenum type, GLsizei bufSize, void *data); +GL_APICALL void GL_APIENTRY glGetnUniformfvEXT (GLuint program, GLint location, GLsizei bufSize, GLfloat *params); +GL_APICALL void GL_APIENTRY glGetnUniformivEXT (GLuint program, GLint location, GLsizei bufSize, GLint *params); +#endif +#endif /* GL_EXT_robustness */ + +#ifndef GL_EXT_sRGB +#define GL_EXT_sRGB 1 +#define GL_SRGB_EXT 0x8C40 +#define GL_SRGB_ALPHA_EXT 0x8C42 +#define GL_SRGB8_ALPHA8_EXT 0x8C43 +#define GL_FRAMEBUFFER_ATTACHMENT_COLOR_ENCODING_EXT 0x8210 +#endif /* GL_EXT_sRGB */ + +#ifndef GL_EXT_sRGB_write_control +#define GL_EXT_sRGB_write_control 1 +#define GL_FRAMEBUFFER_SRGB_EXT 0x8DB9 +#endif /* GL_EXT_sRGB_write_control */ + +#ifndef GL_EXT_semaphore +#define GL_EXT_semaphore 1 +#define GL_LAYOUT_GENERAL_EXT 0x958D +#define GL_LAYOUT_COLOR_ATTACHMENT_EXT 0x958E +#define GL_LAYOUT_DEPTH_STENCIL_ATTACHMENT_EXT 0x958F +#define GL_LAYOUT_DEPTH_STENCIL_READ_ONLY_EXT 0x9590 +#define GL_LAYOUT_SHADER_READ_ONLY_EXT 0x9591 +#define GL_LAYOUT_TRANSFER_SRC_EXT 0x9592 +#define GL_LAYOUT_TRANSFER_DST_EXT 0x9593 +#define GL_LAYOUT_DEPTH_READ_ONLY_STENCIL_ATTACHMENT_EXT 0x9530 +#define GL_LAYOUT_DEPTH_ATTACHMENT_STENCIL_READ_ONLY_EXT 0x9531 +typedef void (GL_APIENTRYP PFNGLGENSEMAPHORESEXTPROC) (GLsizei n, GLuint *semaphores); +typedef void (GL_APIENTRYP PFNGLDELETESEMAPHORESEXTPROC) (GLsizei n, const GLuint *semaphores); +typedef GLboolean (GL_APIENTRYP PFNGLISSEMAPHOREEXTPROC) (GLuint semaphore); +typedef void (GL_APIENTRYP PFNGLSEMAPHOREPARAMETERUI64VEXTPROC) (GLuint semaphore, GLenum pname, const GLuint64 *params); +typedef void (GL_APIENTRYP PFNGLGETSEMAPHOREPARAMETERUI64VEXTPROC) (GLuint semaphore, GLenum pname, GLuint64 *params); +typedef void (GL_APIENTRYP PFNGLWAITSEMAPHOREEXTPROC) (GLuint semaphore, GLuint numBufferBarriers, const GLuint *buffers, GLuint numTextureBarriers, const GLuint *textures, const GLenum *srcLayouts); +typedef void (GL_APIENTRYP PFNGLSIGNALSEMAPHOREEXTPROC) (GLuint semaphore, GLuint numBufferBarriers, const GLuint *buffers, GLuint numTextureBarriers, const GLuint *textures, const GLenum *dstLayouts); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glGenSemaphoresEXT (GLsizei n, GLuint *semaphores); +GL_APICALL void GL_APIENTRY glDeleteSemaphoresEXT (GLsizei n, const GLuint *semaphores); +GL_APICALL GLboolean GL_APIENTRY glIsSemaphoreEXT (GLuint semaphore); +GL_APICALL void GL_APIENTRY glSemaphoreParameterui64vEXT (GLuint semaphore, GLenum pname, const GLuint64 *params); +GL_APICALL void GL_APIENTRY glGetSemaphoreParameterui64vEXT (GLuint semaphore, GLenum pname, GLuint64 *params); +GL_APICALL void GL_APIENTRY glWaitSemaphoreEXT (GLuint semaphore, GLuint numBufferBarriers, const GLuint *buffers, GLuint numTextureBarriers, const GLuint *textures, const GLenum *srcLayouts); +GL_APICALL void GL_APIENTRY glSignalSemaphoreEXT (GLuint semaphore, GLuint numBufferBarriers, const GLuint *buffers, GLuint numTextureBarriers, const GLuint *textures, const GLenum *dstLayouts); +#endif +#endif /* GL_EXT_semaphore */ + +#ifndef GL_EXT_semaphore_fd +#define GL_EXT_semaphore_fd 1 +typedef void (GL_APIENTRYP PFNGLIMPORTSEMAPHOREFDEXTPROC) (GLuint semaphore, GLenum handleType, GLint fd); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glImportSemaphoreFdEXT (GLuint semaphore, GLenum handleType, GLint fd); +#endif +#endif /* GL_EXT_semaphore_fd */ + +#ifndef GL_EXT_semaphore_win32 +#define GL_EXT_semaphore_win32 1 +#define GL_HANDLE_TYPE_D3D12_FENCE_EXT 0x9594 +#define GL_D3D12_FENCE_VALUE_EXT 0x9595 +typedef void (GL_APIENTRYP PFNGLIMPORTSEMAPHOREWIN32HANDLEEXTPROC) (GLuint semaphore, GLenum handleType, void *handle); +typedef void (GL_APIENTRYP PFNGLIMPORTSEMAPHOREWIN32NAMEEXTPROC) (GLuint semaphore, GLenum handleType, const void *name); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glImportSemaphoreWin32HandleEXT (GLuint semaphore, GLenum handleType, void *handle); +GL_APICALL void GL_APIENTRY glImportSemaphoreWin32NameEXT (GLuint semaphore, GLenum handleType, const void *name); +#endif +#endif /* GL_EXT_semaphore_win32 */ + +#ifndef GL_EXT_separate_depth_stencil +#define GL_EXT_separate_depth_stencil 1 +#endif /* GL_EXT_separate_depth_stencil */ + +#ifndef GL_EXT_separate_shader_objects +#define GL_EXT_separate_shader_objects 1 +#define GL_ACTIVE_PROGRAM_EXT 0x8259 +#define GL_VERTEX_SHADER_BIT_EXT 0x00000001 +#define GL_FRAGMENT_SHADER_BIT_EXT 0x00000002 +#define GL_ALL_SHADER_BITS_EXT 0xFFFFFFFF +#define GL_PROGRAM_SEPARABLE_EXT 0x8258 +#define GL_PROGRAM_PIPELINE_BINDING_EXT 0x825A +typedef void (GL_APIENTRYP PFNGLACTIVESHADERPROGRAMEXTPROC) (GLuint pipeline, GLuint program); +typedef void (GL_APIENTRYP PFNGLBINDPROGRAMPIPELINEEXTPROC) (GLuint pipeline); +typedef GLuint (GL_APIENTRYP PFNGLCREATESHADERPROGRAMVEXTPROC) (GLenum type, GLsizei count, const GLchar **strings); +typedef void (GL_APIENTRYP PFNGLDELETEPROGRAMPIPELINESEXTPROC) (GLsizei n, const GLuint *pipelines); +typedef void (GL_APIENTRYP PFNGLGENPROGRAMPIPELINESEXTPROC) (GLsizei n, GLuint *pipelines); +typedef void (GL_APIENTRYP PFNGLGETPROGRAMPIPELINEINFOLOGEXTPROC) (GLuint pipeline, GLsizei bufSize, GLsizei *length, GLchar *infoLog); +typedef void (GL_APIENTRYP PFNGLGETPROGRAMPIPELINEIVEXTPROC) (GLuint pipeline, GLenum pname, GLint *params); +typedef GLboolean (GL_APIENTRYP PFNGLISPROGRAMPIPELINEEXTPROC) (GLuint pipeline); +typedef void (GL_APIENTRYP PFNGLPROGRAMPARAMETERIEXTPROC) (GLuint program, GLenum pname, GLint value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM1FEXTPROC) (GLuint program, GLint location, GLfloat v0); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM1FVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM1IEXTPROC) (GLuint program, GLint location, GLint v0); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM1IVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLint *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM2FEXTPROC) (GLuint program, GLint location, GLfloat v0, GLfloat v1); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM2FVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM2IEXTPROC) (GLuint program, GLint location, GLint v0, GLint v1); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM2IVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLint *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM3FEXTPROC) (GLuint program, GLint location, GLfloat v0, GLfloat v1, GLfloat v2); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM3FVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM3IEXTPROC) (GLuint program, GLint location, GLint v0, GLint v1, GLint v2); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM3IVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLint *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM4FEXTPROC) (GLuint program, GLint location, GLfloat v0, GLfloat v1, GLfloat v2, GLfloat v3); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM4FVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM4IEXTPROC) (GLuint program, GLint location, GLint v0, GLint v1, GLint v2, GLint v3); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM4IVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLint *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORMMATRIX2FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORMMATRIX3FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORMMATRIX4FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLUSEPROGRAMSTAGESEXTPROC) (GLuint pipeline, GLbitfield stages, GLuint program); +typedef void (GL_APIENTRYP PFNGLVALIDATEPROGRAMPIPELINEEXTPROC) (GLuint pipeline); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM1UIEXTPROC) (GLuint program, GLint location, GLuint v0); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM2UIEXTPROC) (GLuint program, GLint location, GLuint v0, GLuint v1); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM3UIEXTPROC) (GLuint program, GLint location, GLuint v0, GLuint v1, GLuint v2); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM4UIEXTPROC) (GLuint program, GLint location, GLuint v0, GLuint v1, GLuint v2, GLuint v3); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM1UIVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLuint *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM2UIVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLuint *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM3UIVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLuint *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM4UIVEXTPROC) (GLuint program, GLint location, GLsizei count, const GLuint *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORMMATRIX2X3FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORMMATRIX3X2FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORMMATRIX2X4FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORMMATRIX4X2FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORMMATRIX3X4FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORMMATRIX4X3FVEXTPROC) (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glActiveShaderProgramEXT (GLuint pipeline, GLuint program); +GL_APICALL void GL_APIENTRY glBindProgramPipelineEXT (GLuint pipeline); +GL_APICALL GLuint GL_APIENTRY glCreateShaderProgramvEXT (GLenum type, GLsizei count, const GLchar **strings); +GL_APICALL void GL_APIENTRY glDeleteProgramPipelinesEXT (GLsizei n, const GLuint *pipelines); +GL_APICALL void GL_APIENTRY glGenProgramPipelinesEXT (GLsizei n, GLuint *pipelines); +GL_APICALL void GL_APIENTRY glGetProgramPipelineInfoLogEXT (GLuint pipeline, GLsizei bufSize, GLsizei *length, GLchar *infoLog); +GL_APICALL void GL_APIENTRY glGetProgramPipelineivEXT (GLuint pipeline, GLenum pname, GLint *params); +GL_APICALL GLboolean GL_APIENTRY glIsProgramPipelineEXT (GLuint pipeline); +GL_APICALL void GL_APIENTRY glProgramParameteriEXT (GLuint program, GLenum pname, GLint value); +GL_APICALL void GL_APIENTRY glProgramUniform1fEXT (GLuint program, GLint location, GLfloat v0); +GL_APICALL void GL_APIENTRY glProgramUniform1fvEXT (GLuint program, GLint location, GLsizei count, const GLfloat *value); +GL_APICALL void GL_APIENTRY glProgramUniform1iEXT (GLuint program, GLint location, GLint v0); +GL_APICALL void GL_APIENTRY glProgramUniform1ivEXT (GLuint program, GLint location, GLsizei count, const GLint *value); +GL_APICALL void GL_APIENTRY glProgramUniform2fEXT (GLuint program, GLint location, GLfloat v0, GLfloat v1); +GL_APICALL void GL_APIENTRY glProgramUniform2fvEXT (GLuint program, GLint location, GLsizei count, const GLfloat *value); +GL_APICALL void GL_APIENTRY glProgramUniform2iEXT (GLuint program, GLint location, GLint v0, GLint v1); +GL_APICALL void GL_APIENTRY glProgramUniform2ivEXT (GLuint program, GLint location, GLsizei count, const GLint *value); +GL_APICALL void GL_APIENTRY glProgramUniform3fEXT (GLuint program, GLint location, GLfloat v0, GLfloat v1, GLfloat v2); +GL_APICALL void GL_APIENTRY glProgramUniform3fvEXT (GLuint program, GLint location, GLsizei count, const GLfloat *value); +GL_APICALL void GL_APIENTRY glProgramUniform3iEXT (GLuint program, GLint location, GLint v0, GLint v1, GLint v2); +GL_APICALL void GL_APIENTRY glProgramUniform3ivEXT (GLuint program, GLint location, GLsizei count, const GLint *value); +GL_APICALL void GL_APIENTRY glProgramUniform4fEXT (GLuint program, GLint location, GLfloat v0, GLfloat v1, GLfloat v2, GLfloat v3); +GL_APICALL void GL_APIENTRY glProgramUniform4fvEXT (GLuint program, GLint location, GLsizei count, const GLfloat *value); +GL_APICALL void GL_APIENTRY glProgramUniform4iEXT (GLuint program, GLint location, GLint v0, GLint v1, GLint v2, GLint v3); +GL_APICALL void GL_APIENTRY glProgramUniform4ivEXT (GLuint program, GLint location, GLsizei count, const GLint *value); +GL_APICALL void GL_APIENTRY glProgramUniformMatrix2fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glProgramUniformMatrix3fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glProgramUniformMatrix4fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glUseProgramStagesEXT (GLuint pipeline, GLbitfield stages, GLuint program); +GL_APICALL void GL_APIENTRY glValidateProgramPipelineEXT (GLuint pipeline); +GL_APICALL void GL_APIENTRY glProgramUniform1uiEXT (GLuint program, GLint location, GLuint v0); +GL_APICALL void GL_APIENTRY glProgramUniform2uiEXT (GLuint program, GLint location, GLuint v0, GLuint v1); +GL_APICALL void GL_APIENTRY glProgramUniform3uiEXT (GLuint program, GLint location, GLuint v0, GLuint v1, GLuint v2); +GL_APICALL void GL_APIENTRY glProgramUniform4uiEXT (GLuint program, GLint location, GLuint v0, GLuint v1, GLuint v2, GLuint v3); +GL_APICALL void GL_APIENTRY glProgramUniform1uivEXT (GLuint program, GLint location, GLsizei count, const GLuint *value); +GL_APICALL void GL_APIENTRY glProgramUniform2uivEXT (GLuint program, GLint location, GLsizei count, const GLuint *value); +GL_APICALL void GL_APIENTRY glProgramUniform3uivEXT (GLuint program, GLint location, GLsizei count, const GLuint *value); +GL_APICALL void GL_APIENTRY glProgramUniform4uivEXT (GLuint program, GLint location, GLsizei count, const GLuint *value); +GL_APICALL void GL_APIENTRY glProgramUniformMatrix2x3fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glProgramUniformMatrix3x2fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glProgramUniformMatrix2x4fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glProgramUniformMatrix4x2fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glProgramUniformMatrix3x4fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glProgramUniformMatrix4x3fvEXT (GLuint program, GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +#endif +#endif /* GL_EXT_separate_shader_objects */ + +#ifndef GL_EXT_shader_framebuffer_fetch +#define GL_EXT_shader_framebuffer_fetch 1 +#define GL_FRAGMENT_SHADER_DISCARDS_SAMPLES_EXT 0x8A52 +#endif /* GL_EXT_shader_framebuffer_fetch */ + +#ifndef GL_EXT_shader_framebuffer_fetch_non_coherent +#define GL_EXT_shader_framebuffer_fetch_non_coherent 1 +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERFETCHBARRIEREXTPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glFramebufferFetchBarrierEXT (void); +#endif +#endif /* GL_EXT_shader_framebuffer_fetch_non_coherent */ + +#ifndef GL_EXT_shader_group_vote +#define GL_EXT_shader_group_vote 1 +#endif /* GL_EXT_shader_group_vote */ + +#ifndef GL_EXT_shader_implicit_conversions +#define GL_EXT_shader_implicit_conversions 1 +#endif /* GL_EXT_shader_implicit_conversions */ + +#ifndef GL_EXT_shader_integer_mix +#define GL_EXT_shader_integer_mix 1 +#endif /* GL_EXT_shader_integer_mix */ + +#ifndef GL_EXT_shader_io_blocks +#define GL_EXT_shader_io_blocks 1 +#endif /* GL_EXT_shader_io_blocks */ + +#ifndef GL_EXT_shader_non_constant_global_initializers +#define GL_EXT_shader_non_constant_global_initializers 1 +#endif /* GL_EXT_shader_non_constant_global_initializers */ + +#ifndef GL_EXT_shader_pixel_local_storage +#define GL_EXT_shader_pixel_local_storage 1 +#define GL_MAX_SHADER_PIXEL_LOCAL_STORAGE_FAST_SIZE_EXT 0x8F63 +#define GL_MAX_SHADER_PIXEL_LOCAL_STORAGE_SIZE_EXT 0x8F67 +#define GL_SHADER_PIXEL_LOCAL_STORAGE_EXT 0x8F64 +#endif /* GL_EXT_shader_pixel_local_storage */ + +#ifndef GL_EXT_shader_pixel_local_storage2 +#define GL_EXT_shader_pixel_local_storage2 1 +#define GL_MAX_SHADER_COMBINED_LOCAL_STORAGE_FAST_SIZE_EXT 0x9650 +#define GL_MAX_SHADER_COMBINED_LOCAL_STORAGE_SIZE_EXT 0x9651 +#define GL_FRAMEBUFFER_INCOMPLETE_INSUFFICIENT_SHADER_COMBINED_LOCAL_STORAGE_EXT 0x9652 +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERPIXELLOCALSTORAGESIZEEXTPROC) (GLuint target, GLsizei size); +typedef GLsizei (GL_APIENTRYP PFNGLGETFRAMEBUFFERPIXELLOCALSTORAGESIZEEXTPROC) (GLuint target); +typedef void (GL_APIENTRYP PFNGLCLEARPIXELLOCALSTORAGEUIEXTPROC) (GLsizei offset, GLsizei n, const GLuint *values); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glFramebufferPixelLocalStorageSizeEXT (GLuint target, GLsizei size); +GL_APICALL GLsizei GL_APIENTRY glGetFramebufferPixelLocalStorageSizeEXT (GLuint target); +GL_APICALL void GL_APIENTRY glClearPixelLocalStorageuiEXT (GLsizei offset, GLsizei n, const GLuint *values); +#endif +#endif /* GL_EXT_shader_pixel_local_storage2 */ + +#ifndef GL_EXT_shader_samples_identical +#define GL_EXT_shader_samples_identical 1 +#endif /* GL_EXT_shader_samples_identical */ + +#ifndef GL_EXT_shader_texture_lod +#define GL_EXT_shader_texture_lod 1 +#endif /* GL_EXT_shader_texture_lod */ + +#ifndef GL_EXT_shadow_samplers +#define GL_EXT_shadow_samplers 1 +#define GL_TEXTURE_COMPARE_MODE_EXT 0x884C +#define GL_TEXTURE_COMPARE_FUNC_EXT 0x884D +#define GL_COMPARE_REF_TO_TEXTURE_EXT 0x884E +#define GL_SAMPLER_2D_SHADOW_EXT 0x8B62 +#endif /* GL_EXT_shadow_samplers */ + +#ifndef GL_EXT_sparse_texture +#define GL_EXT_sparse_texture 1 +#define GL_TEXTURE_SPARSE_EXT 0x91A6 +#define GL_VIRTUAL_PAGE_SIZE_INDEX_EXT 0x91A7 +#define GL_NUM_SPARSE_LEVELS_EXT 0x91AA +#define GL_NUM_VIRTUAL_PAGE_SIZES_EXT 0x91A8 +#define GL_VIRTUAL_PAGE_SIZE_X_EXT 0x9195 +#define GL_VIRTUAL_PAGE_SIZE_Y_EXT 0x9196 +#define GL_VIRTUAL_PAGE_SIZE_Z_EXT 0x9197 +#define GL_TEXTURE_2D_ARRAY 0x8C1A +#define GL_TEXTURE_3D 0x806F +#define GL_MAX_SPARSE_TEXTURE_SIZE_EXT 0x9198 +#define GL_MAX_SPARSE_3D_TEXTURE_SIZE_EXT 0x9199 +#define GL_MAX_SPARSE_ARRAY_TEXTURE_LAYERS_EXT 0x919A +#define GL_SPARSE_TEXTURE_FULL_ARRAY_CUBE_MIPMAPS_EXT 0x91A9 +typedef void (GL_APIENTRYP PFNGLTEXPAGECOMMITMENTEXTPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLboolean commit); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glTexPageCommitmentEXT (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLboolean commit); +#endif +#endif /* GL_EXT_sparse_texture */ + +#ifndef GL_EXT_sparse_texture2 +#define GL_EXT_sparse_texture2 1 +#endif /* GL_EXT_sparse_texture2 */ + +#ifndef GL_EXT_tessellation_point_size +#define GL_EXT_tessellation_point_size 1 +#endif /* GL_EXT_tessellation_point_size */ + +#ifndef GL_EXT_tessellation_shader +#define GL_EXT_tessellation_shader 1 +#define GL_PATCHES_EXT 0x000E +#define GL_PATCH_VERTICES_EXT 0x8E72 +#define GL_TESS_CONTROL_OUTPUT_VERTICES_EXT 0x8E75 +#define GL_TESS_GEN_MODE_EXT 0x8E76 +#define GL_TESS_GEN_SPACING_EXT 0x8E77 +#define GL_TESS_GEN_VERTEX_ORDER_EXT 0x8E78 +#define GL_TESS_GEN_POINT_MODE_EXT 0x8E79 +#define GL_ISOLINES_EXT 0x8E7A +#define GL_QUADS_EXT 0x0007 +#define GL_FRACTIONAL_ODD_EXT 0x8E7B +#define GL_FRACTIONAL_EVEN_EXT 0x8E7C +#define GL_MAX_PATCH_VERTICES_EXT 0x8E7D +#define GL_MAX_TESS_GEN_LEVEL_EXT 0x8E7E +#define GL_MAX_TESS_CONTROL_UNIFORM_COMPONENTS_EXT 0x8E7F +#define GL_MAX_TESS_EVALUATION_UNIFORM_COMPONENTS_EXT 0x8E80 +#define GL_MAX_TESS_CONTROL_TEXTURE_IMAGE_UNITS_EXT 0x8E81 +#define GL_MAX_TESS_EVALUATION_TEXTURE_IMAGE_UNITS_EXT 0x8E82 +#define GL_MAX_TESS_CONTROL_OUTPUT_COMPONENTS_EXT 0x8E83 +#define GL_MAX_TESS_PATCH_COMPONENTS_EXT 0x8E84 +#define GL_MAX_TESS_CONTROL_TOTAL_OUTPUT_COMPONENTS_EXT 0x8E85 +#define GL_MAX_TESS_EVALUATION_OUTPUT_COMPONENTS_EXT 0x8E86 +#define GL_MAX_TESS_CONTROL_UNIFORM_BLOCKS_EXT 0x8E89 +#define GL_MAX_TESS_EVALUATION_UNIFORM_BLOCKS_EXT 0x8E8A +#define GL_MAX_TESS_CONTROL_INPUT_COMPONENTS_EXT 0x886C +#define GL_MAX_TESS_EVALUATION_INPUT_COMPONENTS_EXT 0x886D +#define GL_MAX_COMBINED_TESS_CONTROL_UNIFORM_COMPONENTS_EXT 0x8E1E +#define GL_MAX_COMBINED_TESS_EVALUATION_UNIFORM_COMPONENTS_EXT 0x8E1F +#define GL_MAX_TESS_CONTROL_ATOMIC_COUNTER_BUFFERS_EXT 0x92CD +#define GL_MAX_TESS_EVALUATION_ATOMIC_COUNTER_BUFFERS_EXT 0x92CE +#define GL_MAX_TESS_CONTROL_ATOMIC_COUNTERS_EXT 0x92D3 +#define GL_MAX_TESS_EVALUATION_ATOMIC_COUNTERS_EXT 0x92D4 +#define GL_MAX_TESS_CONTROL_IMAGE_UNIFORMS_EXT 0x90CB +#define GL_MAX_TESS_EVALUATION_IMAGE_UNIFORMS_EXT 0x90CC +#define GL_MAX_TESS_CONTROL_SHADER_STORAGE_BLOCKS_EXT 0x90D8 +#define GL_MAX_TESS_EVALUATION_SHADER_STORAGE_BLOCKS_EXT 0x90D9 +#define GL_PRIMITIVE_RESTART_FOR_PATCHES_SUPPORTED 0x8221 +#define GL_IS_PER_PATCH_EXT 0x92E7 +#define GL_REFERENCED_BY_TESS_CONTROL_SHADER_EXT 0x9307 +#define GL_REFERENCED_BY_TESS_EVALUATION_SHADER_EXT 0x9308 +#define GL_TESS_CONTROL_SHADER_EXT 0x8E88 +#define GL_TESS_EVALUATION_SHADER_EXT 0x8E87 +#define GL_TESS_CONTROL_SHADER_BIT_EXT 0x00000008 +#define GL_TESS_EVALUATION_SHADER_BIT_EXT 0x00000010 +typedef void (GL_APIENTRYP PFNGLPATCHPARAMETERIEXTPROC) (GLenum pname, GLint value); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glPatchParameteriEXT (GLenum pname, GLint value); +#endif +#endif /* GL_EXT_tessellation_shader */ + +#ifndef GL_EXT_texture_border_clamp +#define GL_EXT_texture_border_clamp 1 +#define GL_TEXTURE_BORDER_COLOR_EXT 0x1004 +#define GL_CLAMP_TO_BORDER_EXT 0x812D +typedef void (GL_APIENTRYP PFNGLTEXPARAMETERIIVEXTPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (GL_APIENTRYP PFNGLTEXPARAMETERIUIVEXTPROC) (GLenum target, GLenum pname, const GLuint *params); +typedef void (GL_APIENTRYP PFNGLGETTEXPARAMETERIIVEXTPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (GL_APIENTRYP PFNGLGETTEXPARAMETERIUIVEXTPROC) (GLenum target, GLenum pname, GLuint *params); +typedef void (GL_APIENTRYP PFNGLSAMPLERPARAMETERIIVEXTPROC) (GLuint sampler, GLenum pname, const GLint *param); +typedef void (GL_APIENTRYP PFNGLSAMPLERPARAMETERIUIVEXTPROC) (GLuint sampler, GLenum pname, const GLuint *param); +typedef void (GL_APIENTRYP PFNGLGETSAMPLERPARAMETERIIVEXTPROC) (GLuint sampler, GLenum pname, GLint *params); +typedef void (GL_APIENTRYP PFNGLGETSAMPLERPARAMETERIUIVEXTPROC) (GLuint sampler, GLenum pname, GLuint *params); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glTexParameterIivEXT (GLenum target, GLenum pname, const GLint *params); +GL_APICALL void GL_APIENTRY glTexParameterIuivEXT (GLenum target, GLenum pname, const GLuint *params); +GL_APICALL void GL_APIENTRY glGetTexParameterIivEXT (GLenum target, GLenum pname, GLint *params); +GL_APICALL void GL_APIENTRY glGetTexParameterIuivEXT (GLenum target, GLenum pname, GLuint *params); +GL_APICALL void GL_APIENTRY glSamplerParameterIivEXT (GLuint sampler, GLenum pname, const GLint *param); +GL_APICALL void GL_APIENTRY glSamplerParameterIuivEXT (GLuint sampler, GLenum pname, const GLuint *param); +GL_APICALL void GL_APIENTRY glGetSamplerParameterIivEXT (GLuint sampler, GLenum pname, GLint *params); +GL_APICALL void GL_APIENTRY glGetSamplerParameterIuivEXT (GLuint sampler, GLenum pname, GLuint *params); +#endif +#endif /* GL_EXT_texture_border_clamp */ + +#ifndef GL_EXT_texture_buffer +#define GL_EXT_texture_buffer 1 +#define GL_TEXTURE_BUFFER_EXT 0x8C2A +#define GL_TEXTURE_BUFFER_BINDING_EXT 0x8C2A +#define GL_MAX_TEXTURE_BUFFER_SIZE_EXT 0x8C2B +#define GL_TEXTURE_BINDING_BUFFER_EXT 0x8C2C +#define GL_TEXTURE_BUFFER_DATA_STORE_BINDING_EXT 0x8C2D +#define GL_TEXTURE_BUFFER_OFFSET_ALIGNMENT_EXT 0x919F +#define GL_SAMPLER_BUFFER_EXT 0x8DC2 +#define GL_INT_SAMPLER_BUFFER_EXT 0x8DD0 +#define GL_UNSIGNED_INT_SAMPLER_BUFFER_EXT 0x8DD8 +#define GL_IMAGE_BUFFER_EXT 0x9051 +#define GL_INT_IMAGE_BUFFER_EXT 0x905C +#define GL_UNSIGNED_INT_IMAGE_BUFFER_EXT 0x9067 +#define GL_TEXTURE_BUFFER_OFFSET_EXT 0x919D +#define GL_TEXTURE_BUFFER_SIZE_EXT 0x919E +typedef void (GL_APIENTRYP PFNGLTEXBUFFEREXTPROC) (GLenum target, GLenum internalformat, GLuint buffer); +typedef void (GL_APIENTRYP PFNGLTEXBUFFERRANGEEXTPROC) (GLenum target, GLenum internalformat, GLuint buffer, GLintptr offset, GLsizeiptr size); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glTexBufferEXT (GLenum target, GLenum internalformat, GLuint buffer); +GL_APICALL void GL_APIENTRY glTexBufferRangeEXT (GLenum target, GLenum internalformat, GLuint buffer, GLintptr offset, GLsizeiptr size); +#endif +#endif /* GL_EXT_texture_buffer */ + +#ifndef GL_EXT_texture_compression_astc_decode_mode +#define GL_EXT_texture_compression_astc_decode_mode 1 +#define GL_TEXTURE_ASTC_DECODE_PRECISION_EXT 0x8F69 +#endif /* GL_EXT_texture_compression_astc_decode_mode */ + +#ifndef GL_EXT_texture_compression_bptc +#define GL_EXT_texture_compression_bptc 1 +#define GL_COMPRESSED_RGBA_BPTC_UNORM_EXT 0x8E8C +#define GL_COMPRESSED_SRGB_ALPHA_BPTC_UNORM_EXT 0x8E8D +#define GL_COMPRESSED_RGB_BPTC_SIGNED_FLOAT_EXT 0x8E8E +#define GL_COMPRESSED_RGB_BPTC_UNSIGNED_FLOAT_EXT 0x8E8F +#endif /* GL_EXT_texture_compression_bptc */ + +#ifndef GL_EXT_texture_compression_dxt1 +#define GL_EXT_texture_compression_dxt1 1 +#define GL_COMPRESSED_RGB_S3TC_DXT1_EXT 0x83F0 +#define GL_COMPRESSED_RGBA_S3TC_DXT1_EXT 0x83F1 +#endif /* GL_EXT_texture_compression_dxt1 */ + +#ifndef GL_EXT_texture_compression_rgtc +#define GL_EXT_texture_compression_rgtc 1 +#define GL_COMPRESSED_RED_RGTC1_EXT 0x8DBB +#define GL_COMPRESSED_SIGNED_RED_RGTC1_EXT 0x8DBC +#define GL_COMPRESSED_RED_GREEN_RGTC2_EXT 0x8DBD +#define GL_COMPRESSED_SIGNED_RED_GREEN_RGTC2_EXT 0x8DBE +#endif /* GL_EXT_texture_compression_rgtc */ + +#ifndef GL_EXT_texture_compression_s3tc +#define GL_EXT_texture_compression_s3tc 1 +#define GL_COMPRESSED_RGBA_S3TC_DXT3_EXT 0x83F2 +#define GL_COMPRESSED_RGBA_S3TC_DXT5_EXT 0x83F3 +#endif /* GL_EXT_texture_compression_s3tc */ + +#ifndef GL_EXT_texture_compression_s3tc_srgb +#define GL_EXT_texture_compression_s3tc_srgb 1 +#define GL_COMPRESSED_SRGB_S3TC_DXT1_EXT 0x8C4C +#define GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT 0x8C4D +#define GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT3_EXT 0x8C4E +#define GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT 0x8C4F +#endif /* GL_EXT_texture_compression_s3tc_srgb */ + +#ifndef GL_EXT_texture_cube_map_array +#define GL_EXT_texture_cube_map_array 1 +#define GL_TEXTURE_CUBE_MAP_ARRAY_EXT 0x9009 +#define GL_TEXTURE_BINDING_CUBE_MAP_ARRAY_EXT 0x900A +#define GL_SAMPLER_CUBE_MAP_ARRAY_EXT 0x900C +#define GL_SAMPLER_CUBE_MAP_ARRAY_SHADOW_EXT 0x900D +#define GL_INT_SAMPLER_CUBE_MAP_ARRAY_EXT 0x900E +#define GL_UNSIGNED_INT_SAMPLER_CUBE_MAP_ARRAY_EXT 0x900F +#define GL_IMAGE_CUBE_MAP_ARRAY_EXT 0x9054 +#define GL_INT_IMAGE_CUBE_MAP_ARRAY_EXT 0x905F +#define GL_UNSIGNED_INT_IMAGE_CUBE_MAP_ARRAY_EXT 0x906A +#endif /* GL_EXT_texture_cube_map_array */ + +#ifndef GL_EXT_texture_filter_anisotropic +#define GL_EXT_texture_filter_anisotropic 1 +#define GL_TEXTURE_MAX_ANISOTROPY_EXT 0x84FE +#define GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT 0x84FF +#endif /* GL_EXT_texture_filter_anisotropic */ + +#ifndef GL_EXT_texture_filter_minmax +#define GL_EXT_texture_filter_minmax 1 +#define GL_TEXTURE_REDUCTION_MODE_EXT 0x9366 +#define GL_WEIGHTED_AVERAGE_EXT 0x9367 +#endif /* GL_EXT_texture_filter_minmax */ + +#ifndef GL_EXT_texture_format_BGRA8888 +#define GL_EXT_texture_format_BGRA8888 1 +#endif /* GL_EXT_texture_format_BGRA8888 */ + +#ifndef GL_EXT_texture_format_sRGB_override +#define GL_EXT_texture_format_sRGB_override 1 +#define GL_TEXTURE_FORMAT_SRGB_OVERRIDE_EXT 0x8FBF +#endif /* GL_EXT_texture_format_sRGB_override */ + +#ifndef GL_EXT_texture_mirror_clamp_to_edge +#define GL_EXT_texture_mirror_clamp_to_edge 1 +#define GL_MIRROR_CLAMP_TO_EDGE_EXT 0x8743 +#endif /* GL_EXT_texture_mirror_clamp_to_edge */ + +#ifndef GL_EXT_texture_norm16 +#define GL_EXT_texture_norm16 1 +#define GL_R16_EXT 0x822A +#define GL_RG16_EXT 0x822C +#define GL_RGBA16_EXT 0x805B +#define GL_RGB16_EXT 0x8054 +#define GL_RGB16_SNORM_EXT 0x8F9A +#endif /* GL_EXT_texture_norm16 */ + +#ifndef GL_EXT_texture_query_lod +#define GL_EXT_texture_query_lod 1 +#endif /* GL_EXT_texture_query_lod */ + +#ifndef GL_EXT_texture_rg +#define GL_EXT_texture_rg 1 +#define GL_RED_EXT 0x1903 +#define GL_RG_EXT 0x8227 +#define GL_R8_EXT 0x8229 +#define GL_RG8_EXT 0x822B +#endif /* GL_EXT_texture_rg */ + +#ifndef GL_EXT_texture_sRGB_R8 +#define GL_EXT_texture_sRGB_R8 1 +#define GL_SR8_EXT 0x8FBD +#endif /* GL_EXT_texture_sRGB_R8 */ + +#ifndef GL_EXT_texture_sRGB_RG8 +#define GL_EXT_texture_sRGB_RG8 1 +#define GL_SRG8_EXT 0x8FBE +#endif /* GL_EXT_texture_sRGB_RG8 */ + +#ifndef GL_EXT_texture_sRGB_decode +#define GL_EXT_texture_sRGB_decode 1 +#define GL_TEXTURE_SRGB_DECODE_EXT 0x8A48 +#define GL_DECODE_EXT 0x8A49 +#define GL_SKIP_DECODE_EXT 0x8A4A +#endif /* GL_EXT_texture_sRGB_decode */ + +#ifndef GL_EXT_texture_shadow_lod +#define GL_EXT_texture_shadow_lod 1 +#endif /* GL_EXT_texture_shadow_lod */ + +#ifndef GL_EXT_texture_storage +#define GL_EXT_texture_storage 1 +#define GL_TEXTURE_IMMUTABLE_FORMAT_EXT 0x912F +#define GL_ALPHA8_EXT 0x803C +#define GL_LUMINANCE8_EXT 0x8040 +#define GL_LUMINANCE8_ALPHA8_EXT 0x8045 +#define GL_RGBA32F_EXT 0x8814 +#define GL_RGB32F_EXT 0x8815 +#define GL_ALPHA32F_EXT 0x8816 +#define GL_LUMINANCE32F_EXT 0x8818 +#define GL_LUMINANCE_ALPHA32F_EXT 0x8819 +#define GL_ALPHA16F_EXT 0x881C +#define GL_LUMINANCE16F_EXT 0x881E +#define GL_LUMINANCE_ALPHA16F_EXT 0x881F +#define GL_R32F_EXT 0x822E +#define GL_RG32F_EXT 0x8230 +typedef void (GL_APIENTRYP PFNGLTEXSTORAGE1DEXTPROC) (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width); +typedef void (GL_APIENTRYP PFNGLTEXSTORAGE2DEXTPROC) (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (GL_APIENTRYP PFNGLTEXSTORAGE3DEXTPROC) (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth); +typedef void (GL_APIENTRYP PFNGLTEXTURESTORAGE1DEXTPROC) (GLuint texture, GLenum target, GLsizei levels, GLenum internalformat, GLsizei width); +typedef void (GL_APIENTRYP PFNGLTEXTURESTORAGE2DEXTPROC) (GLuint texture, GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (GL_APIENTRYP PFNGLTEXTURESTORAGE3DEXTPROC) (GLuint texture, GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glTexStorage1DEXT (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width); +GL_APICALL void GL_APIENTRY glTexStorage2DEXT (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height); +GL_APICALL void GL_APIENTRY glTexStorage3DEXT (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth); +GL_APICALL void GL_APIENTRY glTextureStorage1DEXT (GLuint texture, GLenum target, GLsizei levels, GLenum internalformat, GLsizei width); +GL_APICALL void GL_APIENTRY glTextureStorage2DEXT (GLuint texture, GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height); +GL_APICALL void GL_APIENTRY glTextureStorage3DEXT (GLuint texture, GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth); +#endif +#endif /* GL_EXT_texture_storage */ + +#ifndef GL_EXT_texture_storage_compression +#define GL_EXT_texture_storage_compression 1 +#define GL_NUM_SURFACE_COMPRESSION_FIXED_RATES_EXT 0x8F6E +#define GL_SURFACE_COMPRESSION_FIXED_RATE_1BPC_EXT 0x96C4 +#define GL_SURFACE_COMPRESSION_FIXED_RATE_2BPC_EXT 0x96C5 +#define GL_SURFACE_COMPRESSION_FIXED_RATE_3BPC_EXT 0x96C6 +#define GL_SURFACE_COMPRESSION_FIXED_RATE_4BPC_EXT 0x96C7 +#define GL_SURFACE_COMPRESSION_FIXED_RATE_5BPC_EXT 0x96C8 +#define GL_SURFACE_COMPRESSION_FIXED_RATE_6BPC_EXT 0x96C9 +#define GL_SURFACE_COMPRESSION_FIXED_RATE_7BPC_EXT 0x96CA +#define GL_SURFACE_COMPRESSION_FIXED_RATE_8BPC_EXT 0x96CB +#define GL_SURFACE_COMPRESSION_FIXED_RATE_9BPC_EXT 0x96CC +#define GL_SURFACE_COMPRESSION_FIXED_RATE_10BPC_EXT 0x96CD +#define GL_SURFACE_COMPRESSION_FIXED_RATE_11BPC_EXT 0x96CE +#define GL_SURFACE_COMPRESSION_FIXED_RATE_12BPC_EXT 0x96CF +typedef void (GL_APIENTRYP PFNGLTEXSTORAGEATTRIBS2DEXTPROC) (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, const GLint* attrib_list); +typedef void (GL_APIENTRYP PFNGLTEXSTORAGEATTRIBS3DEXTPROC) (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, const GLint* attrib_list); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glTexStorageAttribs2DEXT (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, const GLint* attrib_list); +GL_APICALL void GL_APIENTRY glTexStorageAttribs3DEXT (GLenum target, GLsizei levels, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, const GLint* attrib_list); +#endif +#endif /* GL_EXT_texture_storage_compression */ + +#ifndef GL_EXT_texture_type_2_10_10_10_REV +#define GL_EXT_texture_type_2_10_10_10_REV 1 +#define GL_UNSIGNED_INT_2_10_10_10_REV_EXT 0x8368 +#endif /* GL_EXT_texture_type_2_10_10_10_REV */ + +#ifndef GL_EXT_texture_view +#define GL_EXT_texture_view 1 +#define GL_TEXTURE_VIEW_MIN_LEVEL_EXT 0x82DB +#define GL_TEXTURE_VIEW_NUM_LEVELS_EXT 0x82DC +#define GL_TEXTURE_VIEW_MIN_LAYER_EXT 0x82DD +#define GL_TEXTURE_VIEW_NUM_LAYERS_EXT 0x82DE +typedef void (GL_APIENTRYP PFNGLTEXTUREVIEWEXTPROC) (GLuint texture, GLenum target, GLuint origtexture, GLenum internalformat, GLuint minlevel, GLuint numlevels, GLuint minlayer, GLuint numlayers); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glTextureViewEXT (GLuint texture, GLenum target, GLuint origtexture, GLenum internalformat, GLuint minlevel, GLuint numlevels, GLuint minlayer, GLuint numlayers); +#endif +#endif /* GL_EXT_texture_view */ + +#ifndef GL_EXT_unpack_subimage +#define GL_EXT_unpack_subimage 1 +#define GL_UNPACK_ROW_LENGTH_EXT 0x0CF2 +#define GL_UNPACK_SKIP_ROWS_EXT 0x0CF3 +#define GL_UNPACK_SKIP_PIXELS_EXT 0x0CF4 +#endif /* GL_EXT_unpack_subimage */ + +#ifndef GL_EXT_win32_keyed_mutex +#define GL_EXT_win32_keyed_mutex 1 +typedef GLboolean (GL_APIENTRYP PFNGLACQUIREKEYEDMUTEXWIN32EXTPROC) (GLuint memory, GLuint64 key, GLuint timeout); +typedef GLboolean (GL_APIENTRYP PFNGLRELEASEKEYEDMUTEXWIN32EXTPROC) (GLuint memory, GLuint64 key); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL GLboolean GL_APIENTRY glAcquireKeyedMutexWin32EXT (GLuint memory, GLuint64 key, GLuint timeout); +GL_APICALL GLboolean GL_APIENTRY glReleaseKeyedMutexWin32EXT (GLuint memory, GLuint64 key); +#endif +#endif /* GL_EXT_win32_keyed_mutex */ + +#ifndef GL_EXT_window_rectangles +#define GL_EXT_window_rectangles 1 +#define GL_INCLUSIVE_EXT 0x8F10 +#define GL_EXCLUSIVE_EXT 0x8F11 +#define GL_WINDOW_RECTANGLE_EXT 0x8F12 +#define GL_WINDOW_RECTANGLE_MODE_EXT 0x8F13 +#define GL_MAX_WINDOW_RECTANGLES_EXT 0x8F14 +#define GL_NUM_WINDOW_RECTANGLES_EXT 0x8F15 +typedef void (GL_APIENTRYP PFNGLWINDOWRECTANGLESEXTPROC) (GLenum mode, GLsizei count, const GLint *box); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glWindowRectanglesEXT (GLenum mode, GLsizei count, const GLint *box); +#endif +#endif /* GL_EXT_window_rectangles */ + +#ifndef GL_FJ_shader_binary_GCCSO +#define GL_FJ_shader_binary_GCCSO 1 +#define GL_GCCSO_SHADER_BINARY_FJ 0x9260 +#endif /* GL_FJ_shader_binary_GCCSO */ + +#ifndef GL_IMG_bindless_texture +#define GL_IMG_bindless_texture 1 +typedef GLuint64 (GL_APIENTRYP PFNGLGETTEXTUREHANDLEIMGPROC) (GLuint texture); +typedef GLuint64 (GL_APIENTRYP PFNGLGETTEXTURESAMPLERHANDLEIMGPROC) (GLuint texture, GLuint sampler); +typedef void (GL_APIENTRYP PFNGLUNIFORMHANDLEUI64IMGPROC) (GLint location, GLuint64 value); +typedef void (GL_APIENTRYP PFNGLUNIFORMHANDLEUI64VIMGPROC) (GLint location, GLsizei count, const GLuint64 *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORMHANDLEUI64IMGPROC) (GLuint program, GLint location, GLuint64 value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORMHANDLEUI64VIMGPROC) (GLuint program, GLint location, GLsizei count, const GLuint64 *values); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL GLuint64 GL_APIENTRY glGetTextureHandleIMG (GLuint texture); +GL_APICALL GLuint64 GL_APIENTRY glGetTextureSamplerHandleIMG (GLuint texture, GLuint sampler); +GL_APICALL void GL_APIENTRY glUniformHandleui64IMG (GLint location, GLuint64 value); +GL_APICALL void GL_APIENTRY glUniformHandleui64vIMG (GLint location, GLsizei count, const GLuint64 *value); +GL_APICALL void GL_APIENTRY glProgramUniformHandleui64IMG (GLuint program, GLint location, GLuint64 value); +GL_APICALL void GL_APIENTRY glProgramUniformHandleui64vIMG (GLuint program, GLint location, GLsizei count, const GLuint64 *values); +#endif +#endif /* GL_IMG_bindless_texture */ + +#ifndef GL_IMG_framebuffer_downsample +#define GL_IMG_framebuffer_downsample 1 +#define GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE_AND_DOWNSAMPLE_IMG 0x913C +#define GL_NUM_DOWNSAMPLE_SCALES_IMG 0x913D +#define GL_DOWNSAMPLE_SCALES_IMG 0x913E +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_SCALE_IMG 0x913F +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERTEXTURE2DDOWNSAMPLEIMGPROC) (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level, GLint xscale, GLint yscale); +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERTEXTURELAYERDOWNSAMPLEIMGPROC) (GLenum target, GLenum attachment, GLuint texture, GLint level, GLint layer, GLint xscale, GLint yscale); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glFramebufferTexture2DDownsampleIMG (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level, GLint xscale, GLint yscale); +GL_APICALL void GL_APIENTRY glFramebufferTextureLayerDownsampleIMG (GLenum target, GLenum attachment, GLuint texture, GLint level, GLint layer, GLint xscale, GLint yscale); +#endif +#endif /* GL_IMG_framebuffer_downsample */ + +#ifndef GL_IMG_multisampled_render_to_texture +#define GL_IMG_multisampled_render_to_texture 1 +#define GL_RENDERBUFFER_SAMPLES_IMG 0x9133 +#define GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE_IMG 0x9134 +#define GL_MAX_SAMPLES_IMG 0x9135 +#define GL_TEXTURE_SAMPLES_IMG 0x9136 +typedef void (GL_APIENTRYP PFNGLRENDERBUFFERSTORAGEMULTISAMPLEIMGPROC) (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERTEXTURE2DMULTISAMPLEIMGPROC) (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level, GLsizei samples); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glRenderbufferStorageMultisampleIMG (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +GL_APICALL void GL_APIENTRY glFramebufferTexture2DMultisampleIMG (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level, GLsizei samples); +#endif +#endif /* GL_IMG_multisampled_render_to_texture */ + +#ifndef GL_IMG_program_binary +#define GL_IMG_program_binary 1 +#define GL_SGX_PROGRAM_BINARY_IMG 0x9130 +#endif /* GL_IMG_program_binary */ + +#ifndef GL_IMG_read_format +#define GL_IMG_read_format 1 +#define GL_BGRA_IMG 0x80E1 +#define GL_UNSIGNED_SHORT_4_4_4_4_REV_IMG 0x8365 +#endif /* GL_IMG_read_format */ + +#ifndef GL_IMG_shader_binary +#define GL_IMG_shader_binary 1 +#define GL_SGX_BINARY_IMG 0x8C0A +#endif /* GL_IMG_shader_binary */ + +#ifndef GL_IMG_texture_compression_pvrtc +#define GL_IMG_texture_compression_pvrtc 1 +#define GL_COMPRESSED_RGB_PVRTC_4BPPV1_IMG 0x8C00 +#define GL_COMPRESSED_RGB_PVRTC_2BPPV1_IMG 0x8C01 +#define GL_COMPRESSED_RGBA_PVRTC_4BPPV1_IMG 0x8C02 +#define GL_COMPRESSED_RGBA_PVRTC_2BPPV1_IMG 0x8C03 +#endif /* GL_IMG_texture_compression_pvrtc */ + +#ifndef GL_IMG_texture_compression_pvrtc2 +#define GL_IMG_texture_compression_pvrtc2 1 +#define GL_COMPRESSED_RGBA_PVRTC_2BPPV2_IMG 0x9137 +#define GL_COMPRESSED_RGBA_PVRTC_4BPPV2_IMG 0x9138 +#endif /* GL_IMG_texture_compression_pvrtc2 */ + +#ifndef GL_IMG_texture_filter_cubic +#define GL_IMG_texture_filter_cubic 1 +#define GL_CUBIC_IMG 0x9139 +#define GL_CUBIC_MIPMAP_NEAREST_IMG 0x913A +#define GL_CUBIC_MIPMAP_LINEAR_IMG 0x913B +#endif /* GL_IMG_texture_filter_cubic */ + +#ifndef GL_INTEL_blackhole_render +#define GL_INTEL_blackhole_render 1 +#define GL_BLACKHOLE_RENDER_INTEL 0x83FC +#endif /* GL_INTEL_blackhole_render */ + +#ifndef GL_INTEL_conservative_rasterization +#define GL_INTEL_conservative_rasterization 1 +#define GL_CONSERVATIVE_RASTERIZATION_INTEL 0x83FE +#endif /* GL_INTEL_conservative_rasterization */ + +#ifndef GL_INTEL_framebuffer_CMAA +#define GL_INTEL_framebuffer_CMAA 1 +typedef void (GL_APIENTRYP PFNGLAPPLYFRAMEBUFFERATTACHMENTCMAAINTELPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glApplyFramebufferAttachmentCMAAINTEL (void); +#endif +#endif /* GL_INTEL_framebuffer_CMAA */ + +#ifndef GL_INTEL_performance_query +#define GL_INTEL_performance_query 1 +#define GL_PERFQUERY_SINGLE_CONTEXT_INTEL 0x00000000 +#define GL_PERFQUERY_GLOBAL_CONTEXT_INTEL 0x00000001 +#define GL_PERFQUERY_WAIT_INTEL 0x83FB +#define GL_PERFQUERY_FLUSH_INTEL 0x83FA +#define GL_PERFQUERY_DONOT_FLUSH_INTEL 0x83F9 +#define GL_PERFQUERY_COUNTER_EVENT_INTEL 0x94F0 +#define GL_PERFQUERY_COUNTER_DURATION_NORM_INTEL 0x94F1 +#define GL_PERFQUERY_COUNTER_DURATION_RAW_INTEL 0x94F2 +#define GL_PERFQUERY_COUNTER_THROUGHPUT_INTEL 0x94F3 +#define GL_PERFQUERY_COUNTER_RAW_INTEL 0x94F4 +#define GL_PERFQUERY_COUNTER_TIMESTAMP_INTEL 0x94F5 +#define GL_PERFQUERY_COUNTER_DATA_UINT32_INTEL 0x94F8 +#define GL_PERFQUERY_COUNTER_DATA_UINT64_INTEL 0x94F9 +#define GL_PERFQUERY_COUNTER_DATA_FLOAT_INTEL 0x94FA +#define GL_PERFQUERY_COUNTER_DATA_DOUBLE_INTEL 0x94FB +#define GL_PERFQUERY_COUNTER_DATA_BOOL32_INTEL 0x94FC +#define GL_PERFQUERY_QUERY_NAME_LENGTH_MAX_INTEL 0x94FD +#define GL_PERFQUERY_COUNTER_NAME_LENGTH_MAX_INTEL 0x94FE +#define GL_PERFQUERY_COUNTER_DESC_LENGTH_MAX_INTEL 0x94FF +#define GL_PERFQUERY_GPA_EXTENDED_COUNTERS_INTEL 0x9500 +typedef void (GL_APIENTRYP PFNGLBEGINPERFQUERYINTELPROC) (GLuint queryHandle); +typedef void (GL_APIENTRYP PFNGLCREATEPERFQUERYINTELPROC) (GLuint queryId, GLuint *queryHandle); +typedef void (GL_APIENTRYP PFNGLDELETEPERFQUERYINTELPROC) (GLuint queryHandle); +typedef void (GL_APIENTRYP PFNGLENDPERFQUERYINTELPROC) (GLuint queryHandle); +typedef void (GL_APIENTRYP PFNGLGETFIRSTPERFQUERYIDINTELPROC) (GLuint *queryId); +typedef void (GL_APIENTRYP PFNGLGETNEXTPERFQUERYIDINTELPROC) (GLuint queryId, GLuint *nextQueryId); +typedef void (GL_APIENTRYP PFNGLGETPERFCOUNTERINFOINTELPROC) (GLuint queryId, GLuint counterId, GLuint counterNameLength, GLchar *counterName, GLuint counterDescLength, GLchar *counterDesc, GLuint *counterOffset, GLuint *counterDataSize, GLuint *counterTypeEnum, GLuint *counterDataTypeEnum, GLuint64 *rawCounterMaxValue); +typedef void (GL_APIENTRYP PFNGLGETPERFQUERYDATAINTELPROC) (GLuint queryHandle, GLuint flags, GLsizei dataSize, void *data, GLuint *bytesWritten); +typedef void (GL_APIENTRYP PFNGLGETPERFQUERYIDBYNAMEINTELPROC) (GLchar *queryName, GLuint *queryId); +typedef void (GL_APIENTRYP PFNGLGETPERFQUERYINFOINTELPROC) (GLuint queryId, GLuint queryNameLength, GLchar *queryName, GLuint *dataSize, GLuint *noCounters, GLuint *noInstances, GLuint *capsMask); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glBeginPerfQueryINTEL (GLuint queryHandle); +GL_APICALL void GL_APIENTRY glCreatePerfQueryINTEL (GLuint queryId, GLuint *queryHandle); +GL_APICALL void GL_APIENTRY glDeletePerfQueryINTEL (GLuint queryHandle); +GL_APICALL void GL_APIENTRY glEndPerfQueryINTEL (GLuint queryHandle); +GL_APICALL void GL_APIENTRY glGetFirstPerfQueryIdINTEL (GLuint *queryId); +GL_APICALL void GL_APIENTRY glGetNextPerfQueryIdINTEL (GLuint queryId, GLuint *nextQueryId); +GL_APICALL void GL_APIENTRY glGetPerfCounterInfoINTEL (GLuint queryId, GLuint counterId, GLuint counterNameLength, GLchar *counterName, GLuint counterDescLength, GLchar *counterDesc, GLuint *counterOffset, GLuint *counterDataSize, GLuint *counterTypeEnum, GLuint *counterDataTypeEnum, GLuint64 *rawCounterMaxValue); +GL_APICALL void GL_APIENTRY glGetPerfQueryDataINTEL (GLuint queryHandle, GLuint flags, GLsizei dataSize, void *data, GLuint *bytesWritten); +GL_APICALL void GL_APIENTRY glGetPerfQueryIdByNameINTEL (GLchar *queryName, GLuint *queryId); +GL_APICALL void GL_APIENTRY glGetPerfQueryInfoINTEL (GLuint queryId, GLuint queryNameLength, GLchar *queryName, GLuint *dataSize, GLuint *noCounters, GLuint *noInstances, GLuint *capsMask); +#endif +#endif /* GL_INTEL_performance_query */ + +#ifndef GL_MESA_bgra +#define GL_MESA_bgra 1 +#define GL_BGR_EXT 0x80E0 +#endif /* GL_MESA_bgra */ + +#ifndef GL_MESA_framebuffer_flip_x +#define GL_MESA_framebuffer_flip_x 1 +#define GL_FRAMEBUFFER_FLIP_X_MESA 0x8BBC +#endif /* GL_MESA_framebuffer_flip_x */ + +#ifndef GL_MESA_framebuffer_flip_y +#define GL_MESA_framebuffer_flip_y 1 +#define GL_FRAMEBUFFER_FLIP_Y_MESA 0x8BBB +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERPARAMETERIMESAPROC) (GLenum target, GLenum pname, GLint param); +typedef void (GL_APIENTRYP PFNGLGETFRAMEBUFFERPARAMETERIVMESAPROC) (GLenum target, GLenum pname, GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glFramebufferParameteriMESA (GLenum target, GLenum pname, GLint param); +GL_APICALL void GL_APIENTRY glGetFramebufferParameterivMESA (GLenum target, GLenum pname, GLint *params); +#endif +#endif /* GL_MESA_framebuffer_flip_y */ + +#ifndef GL_MESA_framebuffer_swap_xy +#define GL_MESA_framebuffer_swap_xy 1 +#define GL_FRAMEBUFFER_SWAP_XY_MESA 0x8BBD +#endif /* GL_MESA_framebuffer_swap_xy */ + +#ifndef GL_MESA_program_binary_formats +#define GL_MESA_program_binary_formats 1 +#define GL_PROGRAM_BINARY_FORMAT_MESA 0x875F +#endif /* GL_MESA_program_binary_formats */ + +#ifndef GL_MESA_shader_integer_functions +#define GL_MESA_shader_integer_functions 1 +#endif /* GL_MESA_shader_integer_functions */ + +#ifndef GL_NVX_blend_equation_advanced_multi_draw_buffers +#define GL_NVX_blend_equation_advanced_multi_draw_buffers 1 +#endif /* GL_NVX_blend_equation_advanced_multi_draw_buffers */ + +#ifndef GL_NV_bindless_texture +#define GL_NV_bindless_texture 1 +typedef GLuint64 (GL_APIENTRYP PFNGLGETTEXTUREHANDLENVPROC) (GLuint texture); +typedef GLuint64 (GL_APIENTRYP PFNGLGETTEXTURESAMPLERHANDLENVPROC) (GLuint texture, GLuint sampler); +typedef void (GL_APIENTRYP PFNGLMAKETEXTUREHANDLERESIDENTNVPROC) (GLuint64 handle); +typedef void (GL_APIENTRYP PFNGLMAKETEXTUREHANDLENONRESIDENTNVPROC) (GLuint64 handle); +typedef GLuint64 (GL_APIENTRYP PFNGLGETIMAGEHANDLENVPROC) (GLuint texture, GLint level, GLboolean layered, GLint layer, GLenum format); +typedef void (GL_APIENTRYP PFNGLMAKEIMAGEHANDLERESIDENTNVPROC) (GLuint64 handle, GLenum access); +typedef void (GL_APIENTRYP PFNGLMAKEIMAGEHANDLENONRESIDENTNVPROC) (GLuint64 handle); +typedef void (GL_APIENTRYP PFNGLUNIFORMHANDLEUI64NVPROC) (GLint location, GLuint64 value); +typedef void (GL_APIENTRYP PFNGLUNIFORMHANDLEUI64VNVPROC) (GLint location, GLsizei count, const GLuint64 *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORMHANDLEUI64NVPROC) (GLuint program, GLint location, GLuint64 value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORMHANDLEUI64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLuint64 *values); +typedef GLboolean (GL_APIENTRYP PFNGLISTEXTUREHANDLERESIDENTNVPROC) (GLuint64 handle); +typedef GLboolean (GL_APIENTRYP PFNGLISIMAGEHANDLERESIDENTNVPROC) (GLuint64 handle); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL GLuint64 GL_APIENTRY glGetTextureHandleNV (GLuint texture); +GL_APICALL GLuint64 GL_APIENTRY glGetTextureSamplerHandleNV (GLuint texture, GLuint sampler); +GL_APICALL void GL_APIENTRY glMakeTextureHandleResidentNV (GLuint64 handle); +GL_APICALL void GL_APIENTRY glMakeTextureHandleNonResidentNV (GLuint64 handle); +GL_APICALL GLuint64 GL_APIENTRY glGetImageHandleNV (GLuint texture, GLint level, GLboolean layered, GLint layer, GLenum format); +GL_APICALL void GL_APIENTRY glMakeImageHandleResidentNV (GLuint64 handle, GLenum access); +GL_APICALL void GL_APIENTRY glMakeImageHandleNonResidentNV (GLuint64 handle); +GL_APICALL void GL_APIENTRY glUniformHandleui64NV (GLint location, GLuint64 value); +GL_APICALL void GL_APIENTRY glUniformHandleui64vNV (GLint location, GLsizei count, const GLuint64 *value); +GL_APICALL void GL_APIENTRY glProgramUniformHandleui64NV (GLuint program, GLint location, GLuint64 value); +GL_APICALL void GL_APIENTRY glProgramUniformHandleui64vNV (GLuint program, GLint location, GLsizei count, const GLuint64 *values); +GL_APICALL GLboolean GL_APIENTRY glIsTextureHandleResidentNV (GLuint64 handle); +GL_APICALL GLboolean GL_APIENTRY glIsImageHandleResidentNV (GLuint64 handle); +#endif +#endif /* GL_NV_bindless_texture */ + +#ifndef GL_NV_blend_equation_advanced +#define GL_NV_blend_equation_advanced 1 +#define GL_BLEND_OVERLAP_NV 0x9281 +#define GL_BLEND_PREMULTIPLIED_SRC_NV 0x9280 +#define GL_BLUE_NV 0x1905 +#define GL_COLORBURN_NV 0x929A +#define GL_COLORDODGE_NV 0x9299 +#define GL_CONJOINT_NV 0x9284 +#define GL_CONTRAST_NV 0x92A1 +#define GL_DARKEN_NV 0x9297 +#define GL_DIFFERENCE_NV 0x929E +#define GL_DISJOINT_NV 0x9283 +#define GL_DST_ATOP_NV 0x928F +#define GL_DST_IN_NV 0x928B +#define GL_DST_NV 0x9287 +#define GL_DST_OUT_NV 0x928D +#define GL_DST_OVER_NV 0x9289 +#define GL_EXCLUSION_NV 0x92A0 +#define GL_GREEN_NV 0x1904 +#define GL_HARDLIGHT_NV 0x929B +#define GL_HARDMIX_NV 0x92A9 +#define GL_HSL_COLOR_NV 0x92AF +#define GL_HSL_HUE_NV 0x92AD +#define GL_HSL_LUMINOSITY_NV 0x92B0 +#define GL_HSL_SATURATION_NV 0x92AE +#define GL_INVERT_OVG_NV 0x92B4 +#define GL_INVERT_RGB_NV 0x92A3 +#define GL_LIGHTEN_NV 0x9298 +#define GL_LINEARBURN_NV 0x92A5 +#define GL_LINEARDODGE_NV 0x92A4 +#define GL_LINEARLIGHT_NV 0x92A7 +#define GL_MINUS_CLAMPED_NV 0x92B3 +#define GL_MINUS_NV 0x929F +#define GL_MULTIPLY_NV 0x9294 +#define GL_OVERLAY_NV 0x9296 +#define GL_PINLIGHT_NV 0x92A8 +#define GL_PLUS_CLAMPED_ALPHA_NV 0x92B2 +#define GL_PLUS_CLAMPED_NV 0x92B1 +#define GL_PLUS_DARKER_NV 0x9292 +#define GL_PLUS_NV 0x9291 +#define GL_RED_NV 0x1903 +#define GL_SCREEN_NV 0x9295 +#define GL_SOFTLIGHT_NV 0x929C +#define GL_SRC_ATOP_NV 0x928E +#define GL_SRC_IN_NV 0x928A +#define GL_SRC_NV 0x9286 +#define GL_SRC_OUT_NV 0x928C +#define GL_SRC_OVER_NV 0x9288 +#define GL_UNCORRELATED_NV 0x9282 +#define GL_VIVIDLIGHT_NV 0x92A6 +#define GL_XOR_NV 0x1506 +typedef void (GL_APIENTRYP PFNGLBLENDPARAMETERINVPROC) (GLenum pname, GLint value); +typedef void (GL_APIENTRYP PFNGLBLENDBARRIERNVPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glBlendParameteriNV (GLenum pname, GLint value); +GL_APICALL void GL_APIENTRY glBlendBarrierNV (void); +#endif +#endif /* GL_NV_blend_equation_advanced */ + +#ifndef GL_NV_blend_equation_advanced_coherent +#define GL_NV_blend_equation_advanced_coherent 1 +#define GL_BLEND_ADVANCED_COHERENT_NV 0x9285 +#endif /* GL_NV_blend_equation_advanced_coherent */ + +#ifndef GL_NV_blend_minmax_factor +#define GL_NV_blend_minmax_factor 1 +#define GL_FACTOR_MIN_AMD 0x901C +#define GL_FACTOR_MAX_AMD 0x901D +#endif /* GL_NV_blend_minmax_factor */ + +#ifndef GL_NV_clip_space_w_scaling +#define GL_NV_clip_space_w_scaling 1 +#define GL_VIEWPORT_POSITION_W_SCALE_NV 0x937C +#define GL_VIEWPORT_POSITION_W_SCALE_X_COEFF_NV 0x937D +#define GL_VIEWPORT_POSITION_W_SCALE_Y_COEFF_NV 0x937E +typedef void (GL_APIENTRYP PFNGLVIEWPORTPOSITIONWSCALENVPROC) (GLuint index, GLfloat xcoeff, GLfloat ycoeff); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glViewportPositionWScaleNV (GLuint index, GLfloat xcoeff, GLfloat ycoeff); +#endif +#endif /* GL_NV_clip_space_w_scaling */ + +#ifndef GL_NV_compute_shader_derivatives +#define GL_NV_compute_shader_derivatives 1 +#endif /* GL_NV_compute_shader_derivatives */ + +#ifndef GL_NV_conditional_render +#define GL_NV_conditional_render 1 +#define GL_QUERY_WAIT_NV 0x8E13 +#define GL_QUERY_NO_WAIT_NV 0x8E14 +#define GL_QUERY_BY_REGION_WAIT_NV 0x8E15 +#define GL_QUERY_BY_REGION_NO_WAIT_NV 0x8E16 +typedef void (GL_APIENTRYP PFNGLBEGINCONDITIONALRENDERNVPROC) (GLuint id, GLenum mode); +typedef void (GL_APIENTRYP PFNGLENDCONDITIONALRENDERNVPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glBeginConditionalRenderNV (GLuint id, GLenum mode); +GL_APICALL void GL_APIENTRY glEndConditionalRenderNV (void); +#endif +#endif /* GL_NV_conditional_render */ + +#ifndef GL_NV_conservative_raster +#define GL_NV_conservative_raster 1 +#define GL_CONSERVATIVE_RASTERIZATION_NV 0x9346 +#define GL_SUBPIXEL_PRECISION_BIAS_X_BITS_NV 0x9347 +#define GL_SUBPIXEL_PRECISION_BIAS_Y_BITS_NV 0x9348 +#define GL_MAX_SUBPIXEL_PRECISION_BIAS_BITS_NV 0x9349 +typedef void (GL_APIENTRYP PFNGLSUBPIXELPRECISIONBIASNVPROC) (GLuint xbits, GLuint ybits); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glSubpixelPrecisionBiasNV (GLuint xbits, GLuint ybits); +#endif +#endif /* GL_NV_conservative_raster */ + +#ifndef GL_NV_conservative_raster_pre_snap +#define GL_NV_conservative_raster_pre_snap 1 +#define GL_CONSERVATIVE_RASTER_MODE_PRE_SNAP_NV 0x9550 +#endif /* GL_NV_conservative_raster_pre_snap */ + +#ifndef GL_NV_conservative_raster_pre_snap_triangles +#define GL_NV_conservative_raster_pre_snap_triangles 1 +#define GL_CONSERVATIVE_RASTER_MODE_NV 0x954D +#define GL_CONSERVATIVE_RASTER_MODE_POST_SNAP_NV 0x954E +#define GL_CONSERVATIVE_RASTER_MODE_PRE_SNAP_TRIANGLES_NV 0x954F +typedef void (GL_APIENTRYP PFNGLCONSERVATIVERASTERPARAMETERINVPROC) (GLenum pname, GLint param); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glConservativeRasterParameteriNV (GLenum pname, GLint param); +#endif +#endif /* GL_NV_conservative_raster_pre_snap_triangles */ + +#ifndef GL_NV_copy_buffer +#define GL_NV_copy_buffer 1 +#define GL_COPY_READ_BUFFER_NV 0x8F36 +#define GL_COPY_WRITE_BUFFER_NV 0x8F37 +typedef void (GL_APIENTRYP PFNGLCOPYBUFFERSUBDATANVPROC) (GLenum readTarget, GLenum writeTarget, GLintptr readOffset, GLintptr writeOffset, GLsizeiptr size); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glCopyBufferSubDataNV (GLenum readTarget, GLenum writeTarget, GLintptr readOffset, GLintptr writeOffset, GLsizeiptr size); +#endif +#endif /* GL_NV_copy_buffer */ + +#ifndef GL_NV_coverage_sample +#define GL_NV_coverage_sample 1 +#define GL_COVERAGE_COMPONENT_NV 0x8ED0 +#define GL_COVERAGE_COMPONENT4_NV 0x8ED1 +#define GL_COVERAGE_ATTACHMENT_NV 0x8ED2 +#define GL_COVERAGE_BUFFERS_NV 0x8ED3 +#define GL_COVERAGE_SAMPLES_NV 0x8ED4 +#define GL_COVERAGE_ALL_FRAGMENTS_NV 0x8ED5 +#define GL_COVERAGE_EDGE_FRAGMENTS_NV 0x8ED6 +#define GL_COVERAGE_AUTOMATIC_NV 0x8ED7 +#define GL_COVERAGE_BUFFER_BIT_NV 0x00008000 +typedef void (GL_APIENTRYP PFNGLCOVERAGEMASKNVPROC) (GLboolean mask); +typedef void (GL_APIENTRYP PFNGLCOVERAGEOPERATIONNVPROC) (GLenum operation); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glCoverageMaskNV (GLboolean mask); +GL_APICALL void GL_APIENTRY glCoverageOperationNV (GLenum operation); +#endif +#endif /* GL_NV_coverage_sample */ + +#ifndef GL_NV_depth_nonlinear +#define GL_NV_depth_nonlinear 1 +#define GL_DEPTH_COMPONENT16_NONLINEAR_NV 0x8E2C +#endif /* GL_NV_depth_nonlinear */ + +#ifndef GL_NV_draw_buffers +#define GL_NV_draw_buffers 1 +#define GL_MAX_DRAW_BUFFERS_NV 0x8824 +#define GL_DRAW_BUFFER0_NV 0x8825 +#define GL_DRAW_BUFFER1_NV 0x8826 +#define GL_DRAW_BUFFER2_NV 0x8827 +#define GL_DRAW_BUFFER3_NV 0x8828 +#define GL_DRAW_BUFFER4_NV 0x8829 +#define GL_DRAW_BUFFER5_NV 0x882A +#define GL_DRAW_BUFFER6_NV 0x882B +#define GL_DRAW_BUFFER7_NV 0x882C +#define GL_DRAW_BUFFER8_NV 0x882D +#define GL_DRAW_BUFFER9_NV 0x882E +#define GL_DRAW_BUFFER10_NV 0x882F +#define GL_DRAW_BUFFER11_NV 0x8830 +#define GL_DRAW_BUFFER12_NV 0x8831 +#define GL_DRAW_BUFFER13_NV 0x8832 +#define GL_DRAW_BUFFER14_NV 0x8833 +#define GL_DRAW_BUFFER15_NV 0x8834 +#define GL_COLOR_ATTACHMENT0_NV 0x8CE0 +#define GL_COLOR_ATTACHMENT1_NV 0x8CE1 +#define GL_COLOR_ATTACHMENT2_NV 0x8CE2 +#define GL_COLOR_ATTACHMENT3_NV 0x8CE3 +#define GL_COLOR_ATTACHMENT4_NV 0x8CE4 +#define GL_COLOR_ATTACHMENT5_NV 0x8CE5 +#define GL_COLOR_ATTACHMENT6_NV 0x8CE6 +#define GL_COLOR_ATTACHMENT7_NV 0x8CE7 +#define GL_COLOR_ATTACHMENT8_NV 0x8CE8 +#define GL_COLOR_ATTACHMENT9_NV 0x8CE9 +#define GL_COLOR_ATTACHMENT10_NV 0x8CEA +#define GL_COLOR_ATTACHMENT11_NV 0x8CEB +#define GL_COLOR_ATTACHMENT12_NV 0x8CEC +#define GL_COLOR_ATTACHMENT13_NV 0x8CED +#define GL_COLOR_ATTACHMENT14_NV 0x8CEE +#define GL_COLOR_ATTACHMENT15_NV 0x8CEF +typedef void (GL_APIENTRYP PFNGLDRAWBUFFERSNVPROC) (GLsizei n, const GLenum *bufs); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glDrawBuffersNV (GLsizei n, const GLenum *bufs); +#endif +#endif /* GL_NV_draw_buffers */ + +#ifndef GL_NV_draw_instanced +#define GL_NV_draw_instanced 1 +typedef void (GL_APIENTRYP PFNGLDRAWARRAYSINSTANCEDNVPROC) (GLenum mode, GLint first, GLsizei count, GLsizei primcount); +typedef void (GL_APIENTRYP PFNGLDRAWELEMENTSINSTANCEDNVPROC) (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei primcount); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glDrawArraysInstancedNV (GLenum mode, GLint first, GLsizei count, GLsizei primcount); +GL_APICALL void GL_APIENTRY glDrawElementsInstancedNV (GLenum mode, GLsizei count, GLenum type, const void *indices, GLsizei primcount); +#endif +#endif /* GL_NV_draw_instanced */ + +#ifndef GL_NV_draw_vulkan_image +#define GL_NV_draw_vulkan_image 1 +typedef void (GL_APIENTRY *GLVULKANPROCNV)(void); +typedef void (GL_APIENTRYP PFNGLDRAWVKIMAGENVPROC) (GLuint64 vkImage, GLuint sampler, GLfloat x0, GLfloat y0, GLfloat x1, GLfloat y1, GLfloat z, GLfloat s0, GLfloat t0, GLfloat s1, GLfloat t1); +typedef GLVULKANPROCNV (GL_APIENTRYP PFNGLGETVKPROCADDRNVPROC) (const GLchar *name); +typedef void (GL_APIENTRYP PFNGLWAITVKSEMAPHORENVPROC) (GLuint64 vkSemaphore); +typedef void (GL_APIENTRYP PFNGLSIGNALVKSEMAPHORENVPROC) (GLuint64 vkSemaphore); +typedef void (GL_APIENTRYP PFNGLSIGNALVKFENCENVPROC) (GLuint64 vkFence); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glDrawVkImageNV (GLuint64 vkImage, GLuint sampler, GLfloat x0, GLfloat y0, GLfloat x1, GLfloat y1, GLfloat z, GLfloat s0, GLfloat t0, GLfloat s1, GLfloat t1); +GL_APICALL GLVULKANPROCNV GL_APIENTRY glGetVkProcAddrNV (const GLchar *name); +GL_APICALL void GL_APIENTRY glWaitVkSemaphoreNV (GLuint64 vkSemaphore); +GL_APICALL void GL_APIENTRY glSignalVkSemaphoreNV (GLuint64 vkSemaphore); +GL_APICALL void GL_APIENTRY glSignalVkFenceNV (GLuint64 vkFence); +#endif +#endif /* GL_NV_draw_vulkan_image */ + +#ifndef GL_NV_explicit_attrib_location +#define GL_NV_explicit_attrib_location 1 +#endif /* GL_NV_explicit_attrib_location */ + +#ifndef GL_NV_fbo_color_attachments +#define GL_NV_fbo_color_attachments 1 +#define GL_MAX_COLOR_ATTACHMENTS_NV 0x8CDF +#endif /* GL_NV_fbo_color_attachments */ + +#ifndef GL_NV_fence +#define GL_NV_fence 1 +#define GL_ALL_COMPLETED_NV 0x84F2 +#define GL_FENCE_STATUS_NV 0x84F3 +#define GL_FENCE_CONDITION_NV 0x84F4 +typedef void (GL_APIENTRYP PFNGLDELETEFENCESNVPROC) (GLsizei n, const GLuint *fences); +typedef void (GL_APIENTRYP PFNGLGENFENCESNVPROC) (GLsizei n, GLuint *fences); +typedef GLboolean (GL_APIENTRYP PFNGLISFENCENVPROC) (GLuint fence); +typedef GLboolean (GL_APIENTRYP PFNGLTESTFENCENVPROC) (GLuint fence); +typedef void (GL_APIENTRYP PFNGLGETFENCEIVNVPROC) (GLuint fence, GLenum pname, GLint *params); +typedef void (GL_APIENTRYP PFNGLFINISHFENCENVPROC) (GLuint fence); +typedef void (GL_APIENTRYP PFNGLSETFENCENVPROC) (GLuint fence, GLenum condition); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glDeleteFencesNV (GLsizei n, const GLuint *fences); +GL_APICALL void GL_APIENTRY glGenFencesNV (GLsizei n, GLuint *fences); +GL_APICALL GLboolean GL_APIENTRY glIsFenceNV (GLuint fence); +GL_APICALL GLboolean GL_APIENTRY glTestFenceNV (GLuint fence); +GL_APICALL void GL_APIENTRY glGetFenceivNV (GLuint fence, GLenum pname, GLint *params); +GL_APICALL void GL_APIENTRY glFinishFenceNV (GLuint fence); +GL_APICALL void GL_APIENTRY glSetFenceNV (GLuint fence, GLenum condition); +#endif +#endif /* GL_NV_fence */ + +#ifndef GL_NV_fill_rectangle +#define GL_NV_fill_rectangle 1 +#define GL_FILL_RECTANGLE_NV 0x933C +#endif /* GL_NV_fill_rectangle */ + +#ifndef GL_NV_fragment_coverage_to_color +#define GL_NV_fragment_coverage_to_color 1 +#define GL_FRAGMENT_COVERAGE_TO_COLOR_NV 0x92DD +#define GL_FRAGMENT_COVERAGE_COLOR_NV 0x92DE +typedef void (GL_APIENTRYP PFNGLFRAGMENTCOVERAGECOLORNVPROC) (GLuint color); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glFragmentCoverageColorNV (GLuint color); +#endif +#endif /* GL_NV_fragment_coverage_to_color */ + +#ifndef GL_NV_fragment_shader_barycentric +#define GL_NV_fragment_shader_barycentric 1 +#endif /* GL_NV_fragment_shader_barycentric */ + +#ifndef GL_NV_fragment_shader_interlock +#define GL_NV_fragment_shader_interlock 1 +#endif /* GL_NV_fragment_shader_interlock */ + +#ifndef GL_NV_framebuffer_blit +#define GL_NV_framebuffer_blit 1 +#define GL_READ_FRAMEBUFFER_NV 0x8CA8 +#define GL_DRAW_FRAMEBUFFER_NV 0x8CA9 +#define GL_DRAW_FRAMEBUFFER_BINDING_NV 0x8CA6 +#define GL_READ_FRAMEBUFFER_BINDING_NV 0x8CAA +typedef void (GL_APIENTRYP PFNGLBLITFRAMEBUFFERNVPROC) (GLint srcX0, GLint srcY0, GLint srcX1, GLint srcY1, GLint dstX0, GLint dstY0, GLint dstX1, GLint dstY1, GLbitfield mask, GLenum filter); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glBlitFramebufferNV (GLint srcX0, GLint srcY0, GLint srcX1, GLint srcY1, GLint dstX0, GLint dstY0, GLint dstX1, GLint dstY1, GLbitfield mask, GLenum filter); +#endif +#endif /* GL_NV_framebuffer_blit */ + +#ifndef GL_NV_framebuffer_mixed_samples +#define GL_NV_framebuffer_mixed_samples 1 +#define GL_COVERAGE_MODULATION_TABLE_NV 0x9331 +#define GL_COLOR_SAMPLES_NV 0x8E20 +#define GL_DEPTH_SAMPLES_NV 0x932D +#define GL_STENCIL_SAMPLES_NV 0x932E +#define GL_MIXED_DEPTH_SAMPLES_SUPPORTED_NV 0x932F +#define GL_MIXED_STENCIL_SAMPLES_SUPPORTED_NV 0x9330 +#define GL_COVERAGE_MODULATION_NV 0x9332 +#define GL_COVERAGE_MODULATION_TABLE_SIZE_NV 0x9333 +typedef void (GL_APIENTRYP PFNGLCOVERAGEMODULATIONTABLENVPROC) (GLsizei n, const GLfloat *v); +typedef void (GL_APIENTRYP PFNGLGETCOVERAGEMODULATIONTABLENVPROC) (GLsizei bufSize, GLfloat *v); +typedef void (GL_APIENTRYP PFNGLCOVERAGEMODULATIONNVPROC) (GLenum components); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glCoverageModulationTableNV (GLsizei n, const GLfloat *v); +GL_APICALL void GL_APIENTRY glGetCoverageModulationTableNV (GLsizei bufSize, GLfloat *v); +GL_APICALL void GL_APIENTRY glCoverageModulationNV (GLenum components); +#endif +#endif /* GL_NV_framebuffer_mixed_samples */ + +#ifndef GL_NV_framebuffer_multisample +#define GL_NV_framebuffer_multisample 1 +#define GL_RENDERBUFFER_SAMPLES_NV 0x8CAB +#define GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE_NV 0x8D56 +#define GL_MAX_SAMPLES_NV 0x8D57 +typedef void (GL_APIENTRYP PFNGLRENDERBUFFERSTORAGEMULTISAMPLENVPROC) (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glRenderbufferStorageMultisampleNV (GLenum target, GLsizei samples, GLenum internalformat, GLsizei width, GLsizei height); +#endif +#endif /* GL_NV_framebuffer_multisample */ + +#ifndef GL_NV_generate_mipmap_sRGB +#define GL_NV_generate_mipmap_sRGB 1 +#endif /* GL_NV_generate_mipmap_sRGB */ + +#ifndef GL_NV_geometry_shader_passthrough +#define GL_NV_geometry_shader_passthrough 1 +#endif /* GL_NV_geometry_shader_passthrough */ + +#ifndef GL_NV_gpu_shader5 +#define GL_NV_gpu_shader5 1 +typedef khronos_int64_t GLint64EXT; +typedef khronos_uint64_t GLuint64EXT; +#define GL_INT64_NV 0x140E +#define GL_UNSIGNED_INT64_NV 0x140F +#define GL_INT8_NV 0x8FE0 +#define GL_INT8_VEC2_NV 0x8FE1 +#define GL_INT8_VEC3_NV 0x8FE2 +#define GL_INT8_VEC4_NV 0x8FE3 +#define GL_INT16_NV 0x8FE4 +#define GL_INT16_VEC2_NV 0x8FE5 +#define GL_INT16_VEC3_NV 0x8FE6 +#define GL_INT16_VEC4_NV 0x8FE7 +#define GL_INT64_VEC2_NV 0x8FE9 +#define GL_INT64_VEC3_NV 0x8FEA +#define GL_INT64_VEC4_NV 0x8FEB +#define GL_UNSIGNED_INT8_NV 0x8FEC +#define GL_UNSIGNED_INT8_VEC2_NV 0x8FED +#define GL_UNSIGNED_INT8_VEC3_NV 0x8FEE +#define GL_UNSIGNED_INT8_VEC4_NV 0x8FEF +#define GL_UNSIGNED_INT16_NV 0x8FF0 +#define GL_UNSIGNED_INT16_VEC2_NV 0x8FF1 +#define GL_UNSIGNED_INT16_VEC3_NV 0x8FF2 +#define GL_UNSIGNED_INT16_VEC4_NV 0x8FF3 +#define GL_UNSIGNED_INT64_VEC2_NV 0x8FF5 +#define GL_UNSIGNED_INT64_VEC3_NV 0x8FF6 +#define GL_UNSIGNED_INT64_VEC4_NV 0x8FF7 +#define GL_FLOAT16_NV 0x8FF8 +#define GL_FLOAT16_VEC2_NV 0x8FF9 +#define GL_FLOAT16_VEC3_NV 0x8FFA +#define GL_FLOAT16_VEC4_NV 0x8FFB +#define GL_PATCHES 0x000E +typedef void (GL_APIENTRYP PFNGLUNIFORM1I64NVPROC) (GLint location, GLint64EXT x); +typedef void (GL_APIENTRYP PFNGLUNIFORM2I64NVPROC) (GLint location, GLint64EXT x, GLint64EXT y); +typedef void (GL_APIENTRYP PFNGLUNIFORM3I64NVPROC) (GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z); +typedef void (GL_APIENTRYP PFNGLUNIFORM4I64NVPROC) (GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z, GLint64EXT w); +typedef void (GL_APIENTRYP PFNGLUNIFORM1I64VNVPROC) (GLint location, GLsizei count, const GLint64EXT *value); +typedef void (GL_APIENTRYP PFNGLUNIFORM2I64VNVPROC) (GLint location, GLsizei count, const GLint64EXT *value); +typedef void (GL_APIENTRYP PFNGLUNIFORM3I64VNVPROC) (GLint location, GLsizei count, const GLint64EXT *value); +typedef void (GL_APIENTRYP PFNGLUNIFORM4I64VNVPROC) (GLint location, GLsizei count, const GLint64EXT *value); +typedef void (GL_APIENTRYP PFNGLUNIFORM1UI64NVPROC) (GLint location, GLuint64EXT x); +typedef void (GL_APIENTRYP PFNGLUNIFORM2UI64NVPROC) (GLint location, GLuint64EXT x, GLuint64EXT y); +typedef void (GL_APIENTRYP PFNGLUNIFORM3UI64NVPROC) (GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z); +typedef void (GL_APIENTRYP PFNGLUNIFORM4UI64NVPROC) (GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z, GLuint64EXT w); +typedef void (GL_APIENTRYP PFNGLUNIFORM1UI64VNVPROC) (GLint location, GLsizei count, const GLuint64EXT *value); +typedef void (GL_APIENTRYP PFNGLUNIFORM2UI64VNVPROC) (GLint location, GLsizei count, const GLuint64EXT *value); +typedef void (GL_APIENTRYP PFNGLUNIFORM3UI64VNVPROC) (GLint location, GLsizei count, const GLuint64EXT *value); +typedef void (GL_APIENTRYP PFNGLUNIFORM4UI64VNVPROC) (GLint location, GLsizei count, const GLuint64EXT *value); +typedef void (GL_APIENTRYP PFNGLGETUNIFORMI64VNVPROC) (GLuint program, GLint location, GLint64EXT *params); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM1I64NVPROC) (GLuint program, GLint location, GLint64EXT x); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM2I64NVPROC) (GLuint program, GLint location, GLint64EXT x, GLint64EXT y); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM3I64NVPROC) (GLuint program, GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM4I64NVPROC) (GLuint program, GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z, GLint64EXT w); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM1I64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM2I64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM3I64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM4I64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM1UI64NVPROC) (GLuint program, GLint location, GLuint64EXT x); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM2UI64NVPROC) (GLuint program, GLint location, GLuint64EXT x, GLuint64EXT y); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM3UI64NVPROC) (GLuint program, GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM4UI64NVPROC) (GLuint program, GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z, GLuint64EXT w); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM1UI64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM2UI64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM3UI64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +typedef void (GL_APIENTRYP PFNGLPROGRAMUNIFORM4UI64VNVPROC) (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glUniform1i64NV (GLint location, GLint64EXT x); +GL_APICALL void GL_APIENTRY glUniform2i64NV (GLint location, GLint64EXT x, GLint64EXT y); +GL_APICALL void GL_APIENTRY glUniform3i64NV (GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z); +GL_APICALL void GL_APIENTRY glUniform4i64NV (GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z, GLint64EXT w); +GL_APICALL void GL_APIENTRY glUniform1i64vNV (GLint location, GLsizei count, const GLint64EXT *value); +GL_APICALL void GL_APIENTRY glUniform2i64vNV (GLint location, GLsizei count, const GLint64EXT *value); +GL_APICALL void GL_APIENTRY glUniform3i64vNV (GLint location, GLsizei count, const GLint64EXT *value); +GL_APICALL void GL_APIENTRY glUniform4i64vNV (GLint location, GLsizei count, const GLint64EXT *value); +GL_APICALL void GL_APIENTRY glUniform1ui64NV (GLint location, GLuint64EXT x); +GL_APICALL void GL_APIENTRY glUniform2ui64NV (GLint location, GLuint64EXT x, GLuint64EXT y); +GL_APICALL void GL_APIENTRY glUniform3ui64NV (GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z); +GL_APICALL void GL_APIENTRY glUniform4ui64NV (GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z, GLuint64EXT w); +GL_APICALL void GL_APIENTRY glUniform1ui64vNV (GLint location, GLsizei count, const GLuint64EXT *value); +GL_APICALL void GL_APIENTRY glUniform2ui64vNV (GLint location, GLsizei count, const GLuint64EXT *value); +GL_APICALL void GL_APIENTRY glUniform3ui64vNV (GLint location, GLsizei count, const GLuint64EXT *value); +GL_APICALL void GL_APIENTRY glUniform4ui64vNV (GLint location, GLsizei count, const GLuint64EXT *value); +GL_APICALL void GL_APIENTRY glGetUniformi64vNV (GLuint program, GLint location, GLint64EXT *params); +GL_APICALL void GL_APIENTRY glProgramUniform1i64NV (GLuint program, GLint location, GLint64EXT x); +GL_APICALL void GL_APIENTRY glProgramUniform2i64NV (GLuint program, GLint location, GLint64EXT x, GLint64EXT y); +GL_APICALL void GL_APIENTRY glProgramUniform3i64NV (GLuint program, GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z); +GL_APICALL void GL_APIENTRY glProgramUniform4i64NV (GLuint program, GLint location, GLint64EXT x, GLint64EXT y, GLint64EXT z, GLint64EXT w); +GL_APICALL void GL_APIENTRY glProgramUniform1i64vNV (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +GL_APICALL void GL_APIENTRY glProgramUniform2i64vNV (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +GL_APICALL void GL_APIENTRY glProgramUniform3i64vNV (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +GL_APICALL void GL_APIENTRY glProgramUniform4i64vNV (GLuint program, GLint location, GLsizei count, const GLint64EXT *value); +GL_APICALL void GL_APIENTRY glProgramUniform1ui64NV (GLuint program, GLint location, GLuint64EXT x); +GL_APICALL void GL_APIENTRY glProgramUniform2ui64NV (GLuint program, GLint location, GLuint64EXT x, GLuint64EXT y); +GL_APICALL void GL_APIENTRY glProgramUniform3ui64NV (GLuint program, GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z); +GL_APICALL void GL_APIENTRY glProgramUniform4ui64NV (GLuint program, GLint location, GLuint64EXT x, GLuint64EXT y, GLuint64EXT z, GLuint64EXT w); +GL_APICALL void GL_APIENTRY glProgramUniform1ui64vNV (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +GL_APICALL void GL_APIENTRY glProgramUniform2ui64vNV (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +GL_APICALL void GL_APIENTRY glProgramUniform3ui64vNV (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +GL_APICALL void GL_APIENTRY glProgramUniform4ui64vNV (GLuint program, GLint location, GLsizei count, const GLuint64EXT *value); +#endif +#endif /* GL_NV_gpu_shader5 */ + +#ifndef GL_NV_image_formats +#define GL_NV_image_formats 1 +#endif /* GL_NV_image_formats */ + +#ifndef GL_NV_instanced_arrays +#define GL_NV_instanced_arrays 1 +#define GL_VERTEX_ATTRIB_ARRAY_DIVISOR_NV 0x88FE +typedef void (GL_APIENTRYP PFNGLVERTEXATTRIBDIVISORNVPROC) (GLuint index, GLuint divisor); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glVertexAttribDivisorNV (GLuint index, GLuint divisor); +#endif +#endif /* GL_NV_instanced_arrays */ + +#ifndef GL_NV_internalformat_sample_query +#define GL_NV_internalformat_sample_query 1 +#define GL_TEXTURE_2D_MULTISAMPLE 0x9100 +#define GL_TEXTURE_2D_MULTISAMPLE_ARRAY 0x9102 +#define GL_MULTISAMPLES_NV 0x9371 +#define GL_SUPERSAMPLE_SCALE_X_NV 0x9372 +#define GL_SUPERSAMPLE_SCALE_Y_NV 0x9373 +#define GL_CONFORMANT_NV 0x9374 +typedef void (GL_APIENTRYP PFNGLGETINTERNALFORMATSAMPLEIVNVPROC) (GLenum target, GLenum internalformat, GLsizei samples, GLenum pname, GLsizei count, GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glGetInternalformatSampleivNV (GLenum target, GLenum internalformat, GLsizei samples, GLenum pname, GLsizei count, GLint *params); +#endif +#endif /* GL_NV_internalformat_sample_query */ + +#ifndef GL_NV_memory_attachment +#define GL_NV_memory_attachment 1 +#define GL_ATTACHED_MEMORY_OBJECT_NV 0x95A4 +#define GL_ATTACHED_MEMORY_OFFSET_NV 0x95A5 +#define GL_MEMORY_ATTACHABLE_ALIGNMENT_NV 0x95A6 +#define GL_MEMORY_ATTACHABLE_SIZE_NV 0x95A7 +#define GL_MEMORY_ATTACHABLE_NV 0x95A8 +#define GL_DETACHED_MEMORY_INCARNATION_NV 0x95A9 +#define GL_DETACHED_TEXTURES_NV 0x95AA +#define GL_DETACHED_BUFFERS_NV 0x95AB +#define GL_MAX_DETACHED_TEXTURES_NV 0x95AC +#define GL_MAX_DETACHED_BUFFERS_NV 0x95AD +typedef void (GL_APIENTRYP PFNGLGETMEMORYOBJECTDETACHEDRESOURCESUIVNVPROC) (GLuint memory, GLenum pname, GLint first, GLsizei count, GLuint *params); +typedef void (GL_APIENTRYP PFNGLRESETMEMORYOBJECTPARAMETERNVPROC) (GLuint memory, GLenum pname); +typedef void (GL_APIENTRYP PFNGLTEXATTACHMEMORYNVPROC) (GLenum target, GLuint memory, GLuint64 offset); +typedef void (GL_APIENTRYP PFNGLBUFFERATTACHMEMORYNVPROC) (GLenum target, GLuint memory, GLuint64 offset); +typedef void (GL_APIENTRYP PFNGLTEXTUREATTACHMEMORYNVPROC) (GLuint texture, GLuint memory, GLuint64 offset); +typedef void (GL_APIENTRYP PFNGLNAMEDBUFFERATTACHMEMORYNVPROC) (GLuint buffer, GLuint memory, GLuint64 offset); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glGetMemoryObjectDetachedResourcesuivNV (GLuint memory, GLenum pname, GLint first, GLsizei count, GLuint *params); +GL_APICALL void GL_APIENTRY glResetMemoryObjectParameterNV (GLuint memory, GLenum pname); +GL_APICALL void GL_APIENTRY glTexAttachMemoryNV (GLenum target, GLuint memory, GLuint64 offset); +GL_APICALL void GL_APIENTRY glBufferAttachMemoryNV (GLenum target, GLuint memory, GLuint64 offset); +GL_APICALL void GL_APIENTRY glTextureAttachMemoryNV (GLuint texture, GLuint memory, GLuint64 offset); +GL_APICALL void GL_APIENTRY glNamedBufferAttachMemoryNV (GLuint buffer, GLuint memory, GLuint64 offset); +#endif +#endif /* GL_NV_memory_attachment */ + +#ifndef GL_NV_memory_object_sparse +#define GL_NV_memory_object_sparse 1 +typedef void (GL_APIENTRYP PFNGLBUFFERPAGECOMMITMENTMEMNVPROC) (GLenum target, GLintptr offset, GLsizeiptr size, GLuint memory, GLuint64 memOffset, GLboolean commit); +typedef void (GL_APIENTRYP PFNGLTEXPAGECOMMITMENTMEMNVPROC) (GLenum target, GLint layer, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset, GLboolean commit); +typedef void (GL_APIENTRYP PFNGLNAMEDBUFFERPAGECOMMITMENTMEMNVPROC) (GLuint buffer, GLintptr offset, GLsizeiptr size, GLuint memory, GLuint64 memOffset, GLboolean commit); +typedef void (GL_APIENTRYP PFNGLTEXTUREPAGECOMMITMENTMEMNVPROC) (GLuint texture, GLint layer, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset, GLboolean commit); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glBufferPageCommitmentMemNV (GLenum target, GLintptr offset, GLsizeiptr size, GLuint memory, GLuint64 memOffset, GLboolean commit); +GL_APICALL void GL_APIENTRY glTexPageCommitmentMemNV (GLenum target, GLint layer, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset, GLboolean commit); +GL_APICALL void GL_APIENTRY glNamedBufferPageCommitmentMemNV (GLuint buffer, GLintptr offset, GLsizeiptr size, GLuint memory, GLuint64 memOffset, GLboolean commit); +GL_APICALL void GL_APIENTRY glTexturePageCommitmentMemNV (GLuint texture, GLint layer, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLuint memory, GLuint64 offset, GLboolean commit); +#endif +#endif /* GL_NV_memory_object_sparse */ + +#ifndef GL_NV_mesh_shader +#define GL_NV_mesh_shader 1 +#define GL_MESH_SHADER_NV 0x9559 +#define GL_TASK_SHADER_NV 0x955A +#define GL_MAX_MESH_UNIFORM_BLOCKS_NV 0x8E60 +#define GL_MAX_MESH_TEXTURE_IMAGE_UNITS_NV 0x8E61 +#define GL_MAX_MESH_IMAGE_UNIFORMS_NV 0x8E62 +#define GL_MAX_MESH_UNIFORM_COMPONENTS_NV 0x8E63 +#define GL_MAX_MESH_ATOMIC_COUNTER_BUFFERS_NV 0x8E64 +#define GL_MAX_MESH_ATOMIC_COUNTERS_NV 0x8E65 +#define GL_MAX_MESH_SHADER_STORAGE_BLOCKS_NV 0x8E66 +#define GL_MAX_COMBINED_MESH_UNIFORM_COMPONENTS_NV 0x8E67 +#define GL_MAX_TASK_UNIFORM_BLOCKS_NV 0x8E68 +#define GL_MAX_TASK_TEXTURE_IMAGE_UNITS_NV 0x8E69 +#define GL_MAX_TASK_IMAGE_UNIFORMS_NV 0x8E6A +#define GL_MAX_TASK_UNIFORM_COMPONENTS_NV 0x8E6B +#define GL_MAX_TASK_ATOMIC_COUNTER_BUFFERS_NV 0x8E6C +#define GL_MAX_TASK_ATOMIC_COUNTERS_NV 0x8E6D +#define GL_MAX_TASK_SHADER_STORAGE_BLOCKS_NV 0x8E6E +#define GL_MAX_COMBINED_TASK_UNIFORM_COMPONENTS_NV 0x8E6F +#define GL_MAX_MESH_WORK_GROUP_INVOCATIONS_NV 0x95A2 +#define GL_MAX_TASK_WORK_GROUP_INVOCATIONS_NV 0x95A3 +#define GL_MAX_MESH_TOTAL_MEMORY_SIZE_NV 0x9536 +#define GL_MAX_TASK_TOTAL_MEMORY_SIZE_NV 0x9537 +#define GL_MAX_MESH_OUTPUT_VERTICES_NV 0x9538 +#define GL_MAX_MESH_OUTPUT_PRIMITIVES_NV 0x9539 +#define GL_MAX_TASK_OUTPUT_COUNT_NV 0x953A +#define GL_MAX_DRAW_MESH_TASKS_COUNT_NV 0x953D +#define GL_MAX_MESH_VIEWS_NV 0x9557 +#define GL_MESH_OUTPUT_PER_VERTEX_GRANULARITY_NV 0x92DF +#define GL_MESH_OUTPUT_PER_PRIMITIVE_GRANULARITY_NV 0x9543 +#define GL_MAX_MESH_WORK_GROUP_SIZE_NV 0x953B +#define GL_MAX_TASK_WORK_GROUP_SIZE_NV 0x953C +#define GL_MESH_WORK_GROUP_SIZE_NV 0x953E +#define GL_TASK_WORK_GROUP_SIZE_NV 0x953F +#define GL_MESH_VERTICES_OUT_NV 0x9579 +#define GL_MESH_PRIMITIVES_OUT_NV 0x957A +#define GL_MESH_OUTPUT_TYPE_NV 0x957B +#define GL_UNIFORM_BLOCK_REFERENCED_BY_MESH_SHADER_NV 0x959C +#define GL_UNIFORM_BLOCK_REFERENCED_BY_TASK_SHADER_NV 0x959D +#define GL_REFERENCED_BY_MESH_SHADER_NV 0x95A0 +#define GL_REFERENCED_BY_TASK_SHADER_NV 0x95A1 +#define GL_MESH_SHADER_BIT_NV 0x00000040 +#define GL_TASK_SHADER_BIT_NV 0x00000080 +#define GL_MESH_SUBROUTINE_NV 0x957C +#define GL_TASK_SUBROUTINE_NV 0x957D +#define GL_MESH_SUBROUTINE_UNIFORM_NV 0x957E +#define GL_TASK_SUBROUTINE_UNIFORM_NV 0x957F +#define GL_ATOMIC_COUNTER_BUFFER_REFERENCED_BY_MESH_SHADER_NV 0x959E +#define GL_ATOMIC_COUNTER_BUFFER_REFERENCED_BY_TASK_SHADER_NV 0x959F +typedef void (GL_APIENTRYP PFNGLDRAWMESHTASKSNVPROC) (GLuint first, GLuint count); +typedef void (GL_APIENTRYP PFNGLDRAWMESHTASKSINDIRECTNVPROC) (GLintptr indirect); +typedef void (GL_APIENTRYP PFNGLMULTIDRAWMESHTASKSINDIRECTNVPROC) (GLintptr indirect, GLsizei drawcount, GLsizei stride); +typedef void (GL_APIENTRYP PFNGLMULTIDRAWMESHTASKSINDIRECTCOUNTNVPROC) (GLintptr indirect, GLintptr drawcount, GLsizei maxdrawcount, GLsizei stride); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glDrawMeshTasksNV (GLuint first, GLuint count); +GL_APICALL void GL_APIENTRY glDrawMeshTasksIndirectNV (GLintptr indirect); +GL_APICALL void GL_APIENTRY glMultiDrawMeshTasksIndirectNV (GLintptr indirect, GLsizei drawcount, GLsizei stride); +GL_APICALL void GL_APIENTRY glMultiDrawMeshTasksIndirectCountNV (GLintptr indirect, GLintptr drawcount, GLsizei maxdrawcount, GLsizei stride); +#endif +#endif /* GL_NV_mesh_shader */ + +#ifndef GL_NV_non_square_matrices +#define GL_NV_non_square_matrices 1 +#define GL_FLOAT_MAT2x3_NV 0x8B65 +#define GL_FLOAT_MAT2x4_NV 0x8B66 +#define GL_FLOAT_MAT3x2_NV 0x8B67 +#define GL_FLOAT_MAT3x4_NV 0x8B68 +#define GL_FLOAT_MAT4x2_NV 0x8B69 +#define GL_FLOAT_MAT4x3_NV 0x8B6A +typedef void (GL_APIENTRYP PFNGLUNIFORMMATRIX2X3FVNVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLUNIFORMMATRIX3X2FVNVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLUNIFORMMATRIX2X4FVNVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLUNIFORMMATRIX4X2FVNVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLUNIFORMMATRIX3X4FVNVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLUNIFORMMATRIX4X3FVNVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glUniformMatrix2x3fvNV (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glUniformMatrix3x2fvNV (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glUniformMatrix2x4fvNV (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glUniformMatrix4x2fvNV (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glUniformMatrix3x4fvNV (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +GL_APICALL void GL_APIENTRY glUniformMatrix4x3fvNV (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +#endif +#endif /* GL_NV_non_square_matrices */ + +#ifndef GL_NV_path_rendering +#define GL_NV_path_rendering 1 +typedef double GLdouble; +#define GL_PATH_FORMAT_SVG_NV 0x9070 +#define GL_PATH_FORMAT_PS_NV 0x9071 +#define GL_STANDARD_FONT_NAME_NV 0x9072 +#define GL_SYSTEM_FONT_NAME_NV 0x9073 +#define GL_FILE_NAME_NV 0x9074 +#define GL_PATH_STROKE_WIDTH_NV 0x9075 +#define GL_PATH_END_CAPS_NV 0x9076 +#define GL_PATH_INITIAL_END_CAP_NV 0x9077 +#define GL_PATH_TERMINAL_END_CAP_NV 0x9078 +#define GL_PATH_JOIN_STYLE_NV 0x9079 +#define GL_PATH_MITER_LIMIT_NV 0x907A +#define GL_PATH_DASH_CAPS_NV 0x907B +#define GL_PATH_INITIAL_DASH_CAP_NV 0x907C +#define GL_PATH_TERMINAL_DASH_CAP_NV 0x907D +#define GL_PATH_DASH_OFFSET_NV 0x907E +#define GL_PATH_CLIENT_LENGTH_NV 0x907F +#define GL_PATH_FILL_MODE_NV 0x9080 +#define GL_PATH_FILL_MASK_NV 0x9081 +#define GL_PATH_FILL_COVER_MODE_NV 0x9082 +#define GL_PATH_STROKE_COVER_MODE_NV 0x9083 +#define GL_PATH_STROKE_MASK_NV 0x9084 +#define GL_COUNT_UP_NV 0x9088 +#define GL_COUNT_DOWN_NV 0x9089 +#define GL_PATH_OBJECT_BOUNDING_BOX_NV 0x908A +#define GL_CONVEX_HULL_NV 0x908B +#define GL_BOUNDING_BOX_NV 0x908D +#define GL_TRANSLATE_X_NV 0x908E +#define GL_TRANSLATE_Y_NV 0x908F +#define GL_TRANSLATE_2D_NV 0x9090 +#define GL_TRANSLATE_3D_NV 0x9091 +#define GL_AFFINE_2D_NV 0x9092 +#define GL_AFFINE_3D_NV 0x9094 +#define GL_TRANSPOSE_AFFINE_2D_NV 0x9096 +#define GL_TRANSPOSE_AFFINE_3D_NV 0x9098 +#define GL_UTF8_NV 0x909A +#define GL_UTF16_NV 0x909B +#define GL_BOUNDING_BOX_OF_BOUNDING_BOXES_NV 0x909C +#define GL_PATH_COMMAND_COUNT_NV 0x909D +#define GL_PATH_COORD_COUNT_NV 0x909E +#define GL_PATH_DASH_ARRAY_COUNT_NV 0x909F +#define GL_PATH_COMPUTED_LENGTH_NV 0x90A0 +#define GL_PATH_FILL_BOUNDING_BOX_NV 0x90A1 +#define GL_PATH_STROKE_BOUNDING_BOX_NV 0x90A2 +#define GL_SQUARE_NV 0x90A3 +#define GL_ROUND_NV 0x90A4 +#define GL_TRIANGULAR_NV 0x90A5 +#define GL_BEVEL_NV 0x90A6 +#define GL_MITER_REVERT_NV 0x90A7 +#define GL_MITER_TRUNCATE_NV 0x90A8 +#define GL_SKIP_MISSING_GLYPH_NV 0x90A9 +#define GL_USE_MISSING_GLYPH_NV 0x90AA +#define GL_PATH_ERROR_POSITION_NV 0x90AB +#define GL_ACCUM_ADJACENT_PAIRS_NV 0x90AD +#define GL_ADJACENT_PAIRS_NV 0x90AE +#define GL_FIRST_TO_REST_NV 0x90AF +#define GL_PATH_GEN_MODE_NV 0x90B0 +#define GL_PATH_GEN_COEFF_NV 0x90B1 +#define GL_PATH_GEN_COMPONENTS_NV 0x90B3 +#define GL_PATH_STENCIL_FUNC_NV 0x90B7 +#define GL_PATH_STENCIL_REF_NV 0x90B8 +#define GL_PATH_STENCIL_VALUE_MASK_NV 0x90B9 +#define GL_PATH_STENCIL_DEPTH_OFFSET_FACTOR_NV 0x90BD +#define GL_PATH_STENCIL_DEPTH_OFFSET_UNITS_NV 0x90BE +#define GL_PATH_COVER_DEPTH_FUNC_NV 0x90BF +#define GL_PATH_DASH_OFFSET_RESET_NV 0x90B4 +#define GL_MOVE_TO_RESETS_NV 0x90B5 +#define GL_MOVE_TO_CONTINUES_NV 0x90B6 +#define GL_CLOSE_PATH_NV 0x00 +#define GL_MOVE_TO_NV 0x02 +#define GL_RELATIVE_MOVE_TO_NV 0x03 +#define GL_LINE_TO_NV 0x04 +#define GL_RELATIVE_LINE_TO_NV 0x05 +#define GL_HORIZONTAL_LINE_TO_NV 0x06 +#define GL_RELATIVE_HORIZONTAL_LINE_TO_NV 0x07 +#define GL_VERTICAL_LINE_TO_NV 0x08 +#define GL_RELATIVE_VERTICAL_LINE_TO_NV 0x09 +#define GL_QUADRATIC_CURVE_TO_NV 0x0A +#define GL_RELATIVE_QUADRATIC_CURVE_TO_NV 0x0B +#define GL_CUBIC_CURVE_TO_NV 0x0C +#define GL_RELATIVE_CUBIC_CURVE_TO_NV 0x0D +#define GL_SMOOTH_QUADRATIC_CURVE_TO_NV 0x0E +#define GL_RELATIVE_SMOOTH_QUADRATIC_CURVE_TO_NV 0x0F +#define GL_SMOOTH_CUBIC_CURVE_TO_NV 0x10 +#define GL_RELATIVE_SMOOTH_CUBIC_CURVE_TO_NV 0x11 +#define GL_SMALL_CCW_ARC_TO_NV 0x12 +#define GL_RELATIVE_SMALL_CCW_ARC_TO_NV 0x13 +#define GL_SMALL_CW_ARC_TO_NV 0x14 +#define GL_RELATIVE_SMALL_CW_ARC_TO_NV 0x15 +#define GL_LARGE_CCW_ARC_TO_NV 0x16 +#define GL_RELATIVE_LARGE_CCW_ARC_TO_NV 0x17 +#define GL_LARGE_CW_ARC_TO_NV 0x18 +#define GL_RELATIVE_LARGE_CW_ARC_TO_NV 0x19 +#define GL_RESTART_PATH_NV 0xF0 +#define GL_DUP_FIRST_CUBIC_CURVE_TO_NV 0xF2 +#define GL_DUP_LAST_CUBIC_CURVE_TO_NV 0xF4 +#define GL_RECT_NV 0xF6 +#define GL_CIRCULAR_CCW_ARC_TO_NV 0xF8 +#define GL_CIRCULAR_CW_ARC_TO_NV 0xFA +#define GL_CIRCULAR_TANGENT_ARC_TO_NV 0xFC +#define GL_ARC_TO_NV 0xFE +#define GL_RELATIVE_ARC_TO_NV 0xFF +#define GL_BOLD_BIT_NV 0x01 +#define GL_ITALIC_BIT_NV 0x02 +#define GL_GLYPH_WIDTH_BIT_NV 0x01 +#define GL_GLYPH_HEIGHT_BIT_NV 0x02 +#define GL_GLYPH_HORIZONTAL_BEARING_X_BIT_NV 0x04 +#define GL_GLYPH_HORIZONTAL_BEARING_Y_BIT_NV 0x08 +#define GL_GLYPH_HORIZONTAL_BEARING_ADVANCE_BIT_NV 0x10 +#define GL_GLYPH_VERTICAL_BEARING_X_BIT_NV 0x20 +#define GL_GLYPH_VERTICAL_BEARING_Y_BIT_NV 0x40 +#define GL_GLYPH_VERTICAL_BEARING_ADVANCE_BIT_NV 0x80 +#define GL_GLYPH_HAS_KERNING_BIT_NV 0x100 +#define GL_FONT_X_MIN_BOUNDS_BIT_NV 0x00010000 +#define GL_FONT_Y_MIN_BOUNDS_BIT_NV 0x00020000 +#define GL_FONT_X_MAX_BOUNDS_BIT_NV 0x00040000 +#define GL_FONT_Y_MAX_BOUNDS_BIT_NV 0x00080000 +#define GL_FONT_UNITS_PER_EM_BIT_NV 0x00100000 +#define GL_FONT_ASCENDER_BIT_NV 0x00200000 +#define GL_FONT_DESCENDER_BIT_NV 0x00400000 +#define GL_FONT_HEIGHT_BIT_NV 0x00800000 +#define GL_FONT_MAX_ADVANCE_WIDTH_BIT_NV 0x01000000 +#define GL_FONT_MAX_ADVANCE_HEIGHT_BIT_NV 0x02000000 +#define GL_FONT_UNDERLINE_POSITION_BIT_NV 0x04000000 +#define GL_FONT_UNDERLINE_THICKNESS_BIT_NV 0x08000000 +#define GL_FONT_HAS_KERNING_BIT_NV 0x10000000 +#define GL_ROUNDED_RECT_NV 0xE8 +#define GL_RELATIVE_ROUNDED_RECT_NV 0xE9 +#define GL_ROUNDED_RECT2_NV 0xEA +#define GL_RELATIVE_ROUNDED_RECT2_NV 0xEB +#define GL_ROUNDED_RECT4_NV 0xEC +#define GL_RELATIVE_ROUNDED_RECT4_NV 0xED +#define GL_ROUNDED_RECT8_NV 0xEE +#define GL_RELATIVE_ROUNDED_RECT8_NV 0xEF +#define GL_RELATIVE_RECT_NV 0xF7 +#define GL_FONT_GLYPHS_AVAILABLE_NV 0x9368 +#define GL_FONT_TARGET_UNAVAILABLE_NV 0x9369 +#define GL_FONT_UNAVAILABLE_NV 0x936A +#define GL_FONT_UNINTELLIGIBLE_NV 0x936B +#define GL_CONIC_CURVE_TO_NV 0x1A +#define GL_RELATIVE_CONIC_CURVE_TO_NV 0x1B +#define GL_FONT_NUM_GLYPH_INDICES_BIT_NV 0x20000000 +#define GL_STANDARD_FONT_FORMAT_NV 0x936C +#define GL_PATH_PROJECTION_NV 0x1701 +#define GL_PATH_MODELVIEW_NV 0x1700 +#define GL_PATH_MODELVIEW_STACK_DEPTH_NV 0x0BA3 +#define GL_PATH_MODELVIEW_MATRIX_NV 0x0BA6 +#define GL_PATH_MAX_MODELVIEW_STACK_DEPTH_NV 0x0D36 +#define GL_PATH_TRANSPOSE_MODELVIEW_MATRIX_NV 0x84E3 +#define GL_PATH_PROJECTION_STACK_DEPTH_NV 0x0BA4 +#define GL_PATH_PROJECTION_MATRIX_NV 0x0BA7 +#define GL_PATH_MAX_PROJECTION_STACK_DEPTH_NV 0x0D38 +#define GL_PATH_TRANSPOSE_PROJECTION_MATRIX_NV 0x84E4 +#define GL_FRAGMENT_INPUT_NV 0x936D +typedef GLuint (GL_APIENTRYP PFNGLGENPATHSNVPROC) (GLsizei range); +typedef void (GL_APIENTRYP PFNGLDELETEPATHSNVPROC) (GLuint path, GLsizei range); +typedef GLboolean (GL_APIENTRYP PFNGLISPATHNVPROC) (GLuint path); +typedef void (GL_APIENTRYP PFNGLPATHCOMMANDSNVPROC) (GLuint path, GLsizei numCommands, const GLubyte *commands, GLsizei numCoords, GLenum coordType, const void *coords); +typedef void (GL_APIENTRYP PFNGLPATHCOORDSNVPROC) (GLuint path, GLsizei numCoords, GLenum coordType, const void *coords); +typedef void (GL_APIENTRYP PFNGLPATHSUBCOMMANDSNVPROC) (GLuint path, GLsizei commandStart, GLsizei commandsToDelete, GLsizei numCommands, const GLubyte *commands, GLsizei numCoords, GLenum coordType, const void *coords); +typedef void (GL_APIENTRYP PFNGLPATHSUBCOORDSNVPROC) (GLuint path, GLsizei coordStart, GLsizei numCoords, GLenum coordType, const void *coords); +typedef void (GL_APIENTRYP PFNGLPATHSTRINGNVPROC) (GLuint path, GLenum format, GLsizei length, const void *pathString); +typedef void (GL_APIENTRYP PFNGLPATHGLYPHSNVPROC) (GLuint firstPathName, GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLsizei numGlyphs, GLenum type, const void *charcodes, GLenum handleMissingGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +typedef void (GL_APIENTRYP PFNGLPATHGLYPHRANGENVPROC) (GLuint firstPathName, GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLuint firstGlyph, GLsizei numGlyphs, GLenum handleMissingGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +typedef void (GL_APIENTRYP PFNGLWEIGHTPATHSNVPROC) (GLuint resultPath, GLsizei numPaths, const GLuint *paths, const GLfloat *weights); +typedef void (GL_APIENTRYP PFNGLCOPYPATHNVPROC) (GLuint resultPath, GLuint srcPath); +typedef void (GL_APIENTRYP PFNGLINTERPOLATEPATHSNVPROC) (GLuint resultPath, GLuint pathA, GLuint pathB, GLfloat weight); +typedef void (GL_APIENTRYP PFNGLTRANSFORMPATHNVPROC) (GLuint resultPath, GLuint srcPath, GLenum transformType, const GLfloat *transformValues); +typedef void (GL_APIENTRYP PFNGLPATHPARAMETERIVNVPROC) (GLuint path, GLenum pname, const GLint *value); +typedef void (GL_APIENTRYP PFNGLPATHPARAMETERINVPROC) (GLuint path, GLenum pname, GLint value); +typedef void (GL_APIENTRYP PFNGLPATHPARAMETERFVNVPROC) (GLuint path, GLenum pname, const GLfloat *value); +typedef void (GL_APIENTRYP PFNGLPATHPARAMETERFNVPROC) (GLuint path, GLenum pname, GLfloat value); +typedef void (GL_APIENTRYP PFNGLPATHDASHARRAYNVPROC) (GLuint path, GLsizei dashCount, const GLfloat *dashArray); +typedef void (GL_APIENTRYP PFNGLPATHSTENCILFUNCNVPROC) (GLenum func, GLint ref, GLuint mask); +typedef void (GL_APIENTRYP PFNGLPATHSTENCILDEPTHOFFSETNVPROC) (GLfloat factor, GLfloat units); +typedef void (GL_APIENTRYP PFNGLSTENCILFILLPATHNVPROC) (GLuint path, GLenum fillMode, GLuint mask); +typedef void (GL_APIENTRYP PFNGLSTENCILSTROKEPATHNVPROC) (GLuint path, GLint reference, GLuint mask); +typedef void (GL_APIENTRYP PFNGLSTENCILFILLPATHINSTANCEDNVPROC) (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum fillMode, GLuint mask, GLenum transformType, const GLfloat *transformValues); +typedef void (GL_APIENTRYP PFNGLSTENCILSTROKEPATHINSTANCEDNVPROC) (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLint reference, GLuint mask, GLenum transformType, const GLfloat *transformValues); +typedef void (GL_APIENTRYP PFNGLPATHCOVERDEPTHFUNCNVPROC) (GLenum func); +typedef void (GL_APIENTRYP PFNGLCOVERFILLPATHNVPROC) (GLuint path, GLenum coverMode); +typedef void (GL_APIENTRYP PFNGLCOVERSTROKEPATHNVPROC) (GLuint path, GLenum coverMode); +typedef void (GL_APIENTRYP PFNGLCOVERFILLPATHINSTANCEDNVPROC) (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +typedef void (GL_APIENTRYP PFNGLCOVERSTROKEPATHINSTANCEDNVPROC) (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +typedef void (GL_APIENTRYP PFNGLGETPATHPARAMETERIVNVPROC) (GLuint path, GLenum pname, GLint *value); +typedef void (GL_APIENTRYP PFNGLGETPATHPARAMETERFVNVPROC) (GLuint path, GLenum pname, GLfloat *value); +typedef void (GL_APIENTRYP PFNGLGETPATHCOMMANDSNVPROC) (GLuint path, GLubyte *commands); +typedef void (GL_APIENTRYP PFNGLGETPATHCOORDSNVPROC) (GLuint path, GLfloat *coords); +typedef void (GL_APIENTRYP PFNGLGETPATHDASHARRAYNVPROC) (GLuint path, GLfloat *dashArray); +typedef void (GL_APIENTRYP PFNGLGETPATHMETRICSNVPROC) (GLbitfield metricQueryMask, GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLsizei stride, GLfloat *metrics); +typedef void (GL_APIENTRYP PFNGLGETPATHMETRICRANGENVPROC) (GLbitfield metricQueryMask, GLuint firstPathName, GLsizei numPaths, GLsizei stride, GLfloat *metrics); +typedef void (GL_APIENTRYP PFNGLGETPATHSPACINGNVPROC) (GLenum pathListMode, GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLfloat advanceScale, GLfloat kerningScale, GLenum transformType, GLfloat *returnedSpacing); +typedef GLboolean (GL_APIENTRYP PFNGLISPOINTINFILLPATHNVPROC) (GLuint path, GLuint mask, GLfloat x, GLfloat y); +typedef GLboolean (GL_APIENTRYP PFNGLISPOINTINSTROKEPATHNVPROC) (GLuint path, GLfloat x, GLfloat y); +typedef GLfloat (GL_APIENTRYP PFNGLGETPATHLENGTHNVPROC) (GLuint path, GLsizei startSegment, GLsizei numSegments); +typedef GLboolean (GL_APIENTRYP PFNGLPOINTALONGPATHNVPROC) (GLuint path, GLsizei startSegment, GLsizei numSegments, GLfloat distance, GLfloat *x, GLfloat *y, GLfloat *tangentX, GLfloat *tangentY); +typedef void (GL_APIENTRYP PFNGLMATRIXLOAD3X2FNVPROC) (GLenum matrixMode, const GLfloat *m); +typedef void (GL_APIENTRYP PFNGLMATRIXLOAD3X3FNVPROC) (GLenum matrixMode, const GLfloat *m); +typedef void (GL_APIENTRYP PFNGLMATRIXLOADTRANSPOSE3X3FNVPROC) (GLenum matrixMode, const GLfloat *m); +typedef void (GL_APIENTRYP PFNGLMATRIXMULT3X2FNVPROC) (GLenum matrixMode, const GLfloat *m); +typedef void (GL_APIENTRYP PFNGLMATRIXMULT3X3FNVPROC) (GLenum matrixMode, const GLfloat *m); +typedef void (GL_APIENTRYP PFNGLMATRIXMULTTRANSPOSE3X3FNVPROC) (GLenum matrixMode, const GLfloat *m); +typedef void (GL_APIENTRYP PFNGLSTENCILTHENCOVERFILLPATHNVPROC) (GLuint path, GLenum fillMode, GLuint mask, GLenum coverMode); +typedef void (GL_APIENTRYP PFNGLSTENCILTHENCOVERSTROKEPATHNVPROC) (GLuint path, GLint reference, GLuint mask, GLenum coverMode); +typedef void (GL_APIENTRYP PFNGLSTENCILTHENCOVERFILLPATHINSTANCEDNVPROC) (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum fillMode, GLuint mask, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +typedef void (GL_APIENTRYP PFNGLSTENCILTHENCOVERSTROKEPATHINSTANCEDNVPROC) (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLint reference, GLuint mask, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +typedef GLenum (GL_APIENTRYP PFNGLPATHGLYPHINDEXRANGENVPROC) (GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLuint pathParameterTemplate, GLfloat emScale, GLuint *baseAndCount); +typedef GLenum (GL_APIENTRYP PFNGLPATHGLYPHINDEXARRAYNVPROC) (GLuint firstPathName, GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLuint firstGlyphIndex, GLsizei numGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +typedef GLenum (GL_APIENTRYP PFNGLPATHMEMORYGLYPHINDEXARRAYNVPROC) (GLuint firstPathName, GLenum fontTarget, GLsizeiptr fontSize, const void *fontData, GLsizei faceIndex, GLuint firstGlyphIndex, GLsizei numGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +typedef void (GL_APIENTRYP PFNGLPROGRAMPATHFRAGMENTINPUTGENNVPROC) (GLuint program, GLint location, GLenum genMode, GLint components, const GLfloat *coeffs); +typedef void (GL_APIENTRYP PFNGLGETPROGRAMRESOURCEFVNVPROC) (GLuint program, GLenum programInterface, GLuint index, GLsizei propCount, const GLenum *props, GLsizei count, GLsizei *length, GLfloat *params); +typedef void (GL_APIENTRYP PFNGLMATRIXFRUSTUMEXTPROC) (GLenum mode, GLdouble left, GLdouble right, GLdouble bottom, GLdouble top, GLdouble zNear, GLdouble zFar); +typedef void (GL_APIENTRYP PFNGLMATRIXLOADIDENTITYEXTPROC) (GLenum mode); +typedef void (GL_APIENTRYP PFNGLMATRIXLOADTRANSPOSEFEXTPROC) (GLenum mode, const GLfloat *m); +typedef void (GL_APIENTRYP PFNGLMATRIXLOADTRANSPOSEDEXTPROC) (GLenum mode, const GLdouble *m); +typedef void (GL_APIENTRYP PFNGLMATRIXLOADFEXTPROC) (GLenum mode, const GLfloat *m); +typedef void (GL_APIENTRYP PFNGLMATRIXLOADDEXTPROC) (GLenum mode, const GLdouble *m); +typedef void (GL_APIENTRYP PFNGLMATRIXMULTTRANSPOSEFEXTPROC) (GLenum mode, const GLfloat *m); +typedef void (GL_APIENTRYP PFNGLMATRIXMULTTRANSPOSEDEXTPROC) (GLenum mode, const GLdouble *m); +typedef void (GL_APIENTRYP PFNGLMATRIXMULTFEXTPROC) (GLenum mode, const GLfloat *m); +typedef void (GL_APIENTRYP PFNGLMATRIXMULTDEXTPROC) (GLenum mode, const GLdouble *m); +typedef void (GL_APIENTRYP PFNGLMATRIXORTHOEXTPROC) (GLenum mode, GLdouble left, GLdouble right, GLdouble bottom, GLdouble top, GLdouble zNear, GLdouble zFar); +typedef void (GL_APIENTRYP PFNGLMATRIXPOPEXTPROC) (GLenum mode); +typedef void (GL_APIENTRYP PFNGLMATRIXPUSHEXTPROC) (GLenum mode); +typedef void (GL_APIENTRYP PFNGLMATRIXROTATEFEXTPROC) (GLenum mode, GLfloat angle, GLfloat x, GLfloat y, GLfloat z); +typedef void (GL_APIENTRYP PFNGLMATRIXROTATEDEXTPROC) (GLenum mode, GLdouble angle, GLdouble x, GLdouble y, GLdouble z); +typedef void (GL_APIENTRYP PFNGLMATRIXSCALEFEXTPROC) (GLenum mode, GLfloat x, GLfloat y, GLfloat z); +typedef void (GL_APIENTRYP PFNGLMATRIXSCALEDEXTPROC) (GLenum mode, GLdouble x, GLdouble y, GLdouble z); +typedef void (GL_APIENTRYP PFNGLMATRIXTRANSLATEFEXTPROC) (GLenum mode, GLfloat x, GLfloat y, GLfloat z); +typedef void (GL_APIENTRYP PFNGLMATRIXTRANSLATEDEXTPROC) (GLenum mode, GLdouble x, GLdouble y, GLdouble z); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL GLuint GL_APIENTRY glGenPathsNV (GLsizei range); +GL_APICALL void GL_APIENTRY glDeletePathsNV (GLuint path, GLsizei range); +GL_APICALL GLboolean GL_APIENTRY glIsPathNV (GLuint path); +GL_APICALL void GL_APIENTRY glPathCommandsNV (GLuint path, GLsizei numCommands, const GLubyte *commands, GLsizei numCoords, GLenum coordType, const void *coords); +GL_APICALL void GL_APIENTRY glPathCoordsNV (GLuint path, GLsizei numCoords, GLenum coordType, const void *coords); +GL_APICALL void GL_APIENTRY glPathSubCommandsNV (GLuint path, GLsizei commandStart, GLsizei commandsToDelete, GLsizei numCommands, const GLubyte *commands, GLsizei numCoords, GLenum coordType, const void *coords); +GL_APICALL void GL_APIENTRY glPathSubCoordsNV (GLuint path, GLsizei coordStart, GLsizei numCoords, GLenum coordType, const void *coords); +GL_APICALL void GL_APIENTRY glPathStringNV (GLuint path, GLenum format, GLsizei length, const void *pathString); +GL_APICALL void GL_APIENTRY glPathGlyphsNV (GLuint firstPathName, GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLsizei numGlyphs, GLenum type, const void *charcodes, GLenum handleMissingGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +GL_APICALL void GL_APIENTRY glPathGlyphRangeNV (GLuint firstPathName, GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLuint firstGlyph, GLsizei numGlyphs, GLenum handleMissingGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +GL_APICALL void GL_APIENTRY glWeightPathsNV (GLuint resultPath, GLsizei numPaths, const GLuint *paths, const GLfloat *weights); +GL_APICALL void GL_APIENTRY glCopyPathNV (GLuint resultPath, GLuint srcPath); +GL_APICALL void GL_APIENTRY glInterpolatePathsNV (GLuint resultPath, GLuint pathA, GLuint pathB, GLfloat weight); +GL_APICALL void GL_APIENTRY glTransformPathNV (GLuint resultPath, GLuint srcPath, GLenum transformType, const GLfloat *transformValues); +GL_APICALL void GL_APIENTRY glPathParameterivNV (GLuint path, GLenum pname, const GLint *value); +GL_APICALL void GL_APIENTRY glPathParameteriNV (GLuint path, GLenum pname, GLint value); +GL_APICALL void GL_APIENTRY glPathParameterfvNV (GLuint path, GLenum pname, const GLfloat *value); +GL_APICALL void GL_APIENTRY glPathParameterfNV (GLuint path, GLenum pname, GLfloat value); +GL_APICALL void GL_APIENTRY glPathDashArrayNV (GLuint path, GLsizei dashCount, const GLfloat *dashArray); +GL_APICALL void GL_APIENTRY glPathStencilFuncNV (GLenum func, GLint ref, GLuint mask); +GL_APICALL void GL_APIENTRY glPathStencilDepthOffsetNV (GLfloat factor, GLfloat units); +GL_APICALL void GL_APIENTRY glStencilFillPathNV (GLuint path, GLenum fillMode, GLuint mask); +GL_APICALL void GL_APIENTRY glStencilStrokePathNV (GLuint path, GLint reference, GLuint mask); +GL_APICALL void GL_APIENTRY glStencilFillPathInstancedNV (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum fillMode, GLuint mask, GLenum transformType, const GLfloat *transformValues); +GL_APICALL void GL_APIENTRY glStencilStrokePathInstancedNV (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLint reference, GLuint mask, GLenum transformType, const GLfloat *transformValues); +GL_APICALL void GL_APIENTRY glPathCoverDepthFuncNV (GLenum func); +GL_APICALL void GL_APIENTRY glCoverFillPathNV (GLuint path, GLenum coverMode); +GL_APICALL void GL_APIENTRY glCoverStrokePathNV (GLuint path, GLenum coverMode); +GL_APICALL void GL_APIENTRY glCoverFillPathInstancedNV (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +GL_APICALL void GL_APIENTRY glCoverStrokePathInstancedNV (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +GL_APICALL void GL_APIENTRY glGetPathParameterivNV (GLuint path, GLenum pname, GLint *value); +GL_APICALL void GL_APIENTRY glGetPathParameterfvNV (GLuint path, GLenum pname, GLfloat *value); +GL_APICALL void GL_APIENTRY glGetPathCommandsNV (GLuint path, GLubyte *commands); +GL_APICALL void GL_APIENTRY glGetPathCoordsNV (GLuint path, GLfloat *coords); +GL_APICALL void GL_APIENTRY glGetPathDashArrayNV (GLuint path, GLfloat *dashArray); +GL_APICALL void GL_APIENTRY glGetPathMetricsNV (GLbitfield metricQueryMask, GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLsizei stride, GLfloat *metrics); +GL_APICALL void GL_APIENTRY glGetPathMetricRangeNV (GLbitfield metricQueryMask, GLuint firstPathName, GLsizei numPaths, GLsizei stride, GLfloat *metrics); +GL_APICALL void GL_APIENTRY glGetPathSpacingNV (GLenum pathListMode, GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLfloat advanceScale, GLfloat kerningScale, GLenum transformType, GLfloat *returnedSpacing); +GL_APICALL GLboolean GL_APIENTRY glIsPointInFillPathNV (GLuint path, GLuint mask, GLfloat x, GLfloat y); +GL_APICALL GLboolean GL_APIENTRY glIsPointInStrokePathNV (GLuint path, GLfloat x, GLfloat y); +GL_APICALL GLfloat GL_APIENTRY glGetPathLengthNV (GLuint path, GLsizei startSegment, GLsizei numSegments); +GL_APICALL GLboolean GL_APIENTRY glPointAlongPathNV (GLuint path, GLsizei startSegment, GLsizei numSegments, GLfloat distance, GLfloat *x, GLfloat *y, GLfloat *tangentX, GLfloat *tangentY); +GL_APICALL void GL_APIENTRY glMatrixLoad3x2fNV (GLenum matrixMode, const GLfloat *m); +GL_APICALL void GL_APIENTRY glMatrixLoad3x3fNV (GLenum matrixMode, const GLfloat *m); +GL_APICALL void GL_APIENTRY glMatrixLoadTranspose3x3fNV (GLenum matrixMode, const GLfloat *m); +GL_APICALL void GL_APIENTRY glMatrixMult3x2fNV (GLenum matrixMode, const GLfloat *m); +GL_APICALL void GL_APIENTRY glMatrixMult3x3fNV (GLenum matrixMode, const GLfloat *m); +GL_APICALL void GL_APIENTRY glMatrixMultTranspose3x3fNV (GLenum matrixMode, const GLfloat *m); +GL_APICALL void GL_APIENTRY glStencilThenCoverFillPathNV (GLuint path, GLenum fillMode, GLuint mask, GLenum coverMode); +GL_APICALL void GL_APIENTRY glStencilThenCoverStrokePathNV (GLuint path, GLint reference, GLuint mask, GLenum coverMode); +GL_APICALL void GL_APIENTRY glStencilThenCoverFillPathInstancedNV (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLenum fillMode, GLuint mask, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +GL_APICALL void GL_APIENTRY glStencilThenCoverStrokePathInstancedNV (GLsizei numPaths, GLenum pathNameType, const void *paths, GLuint pathBase, GLint reference, GLuint mask, GLenum coverMode, GLenum transformType, const GLfloat *transformValues); +GL_APICALL GLenum GL_APIENTRY glPathGlyphIndexRangeNV (GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLuint pathParameterTemplate, GLfloat emScale, GLuint *baseAndCount); +GL_APICALL GLenum GL_APIENTRY glPathGlyphIndexArrayNV (GLuint firstPathName, GLenum fontTarget, const void *fontName, GLbitfield fontStyle, GLuint firstGlyphIndex, GLsizei numGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +GL_APICALL GLenum GL_APIENTRY glPathMemoryGlyphIndexArrayNV (GLuint firstPathName, GLenum fontTarget, GLsizeiptr fontSize, const void *fontData, GLsizei faceIndex, GLuint firstGlyphIndex, GLsizei numGlyphs, GLuint pathParameterTemplate, GLfloat emScale); +GL_APICALL void GL_APIENTRY glProgramPathFragmentInputGenNV (GLuint program, GLint location, GLenum genMode, GLint components, const GLfloat *coeffs); +GL_APICALL void GL_APIENTRY glGetProgramResourcefvNV (GLuint program, GLenum programInterface, GLuint index, GLsizei propCount, const GLenum *props, GLsizei count, GLsizei *length, GLfloat *params); +GL_APICALL void GL_APIENTRY glMatrixFrustumEXT (GLenum mode, GLdouble left, GLdouble right, GLdouble bottom, GLdouble top, GLdouble zNear, GLdouble zFar); +GL_APICALL void GL_APIENTRY glMatrixLoadIdentityEXT (GLenum mode); +GL_APICALL void GL_APIENTRY glMatrixLoadTransposefEXT (GLenum mode, const GLfloat *m); +GL_APICALL void GL_APIENTRY glMatrixLoadTransposedEXT (GLenum mode, const GLdouble *m); +GL_APICALL void GL_APIENTRY glMatrixLoadfEXT (GLenum mode, const GLfloat *m); +GL_APICALL void GL_APIENTRY glMatrixLoaddEXT (GLenum mode, const GLdouble *m); +GL_APICALL void GL_APIENTRY glMatrixMultTransposefEXT (GLenum mode, const GLfloat *m); +GL_APICALL void GL_APIENTRY glMatrixMultTransposedEXT (GLenum mode, const GLdouble *m); +GL_APICALL void GL_APIENTRY glMatrixMultfEXT (GLenum mode, const GLfloat *m); +GL_APICALL void GL_APIENTRY glMatrixMultdEXT (GLenum mode, const GLdouble *m); +GL_APICALL void GL_APIENTRY glMatrixOrthoEXT (GLenum mode, GLdouble left, GLdouble right, GLdouble bottom, GLdouble top, GLdouble zNear, GLdouble zFar); +GL_APICALL void GL_APIENTRY glMatrixPopEXT (GLenum mode); +GL_APICALL void GL_APIENTRY glMatrixPushEXT (GLenum mode); +GL_APICALL void GL_APIENTRY glMatrixRotatefEXT (GLenum mode, GLfloat angle, GLfloat x, GLfloat y, GLfloat z); +GL_APICALL void GL_APIENTRY glMatrixRotatedEXT (GLenum mode, GLdouble angle, GLdouble x, GLdouble y, GLdouble z); +GL_APICALL void GL_APIENTRY glMatrixScalefEXT (GLenum mode, GLfloat x, GLfloat y, GLfloat z); +GL_APICALL void GL_APIENTRY glMatrixScaledEXT (GLenum mode, GLdouble x, GLdouble y, GLdouble z); +GL_APICALL void GL_APIENTRY glMatrixTranslatefEXT (GLenum mode, GLfloat x, GLfloat y, GLfloat z); +GL_APICALL void GL_APIENTRY glMatrixTranslatedEXT (GLenum mode, GLdouble x, GLdouble y, GLdouble z); +#endif +#endif /* GL_NV_path_rendering */ + +#ifndef GL_NV_path_rendering_shared_edge +#define GL_NV_path_rendering_shared_edge 1 +#define GL_SHARED_EDGE_NV 0xC0 +#endif /* GL_NV_path_rendering_shared_edge */ + +#ifndef GL_NV_pixel_buffer_object +#define GL_NV_pixel_buffer_object 1 +#define GL_PIXEL_PACK_BUFFER_NV 0x88EB +#define GL_PIXEL_UNPACK_BUFFER_NV 0x88EC +#define GL_PIXEL_PACK_BUFFER_BINDING_NV 0x88ED +#define GL_PIXEL_UNPACK_BUFFER_BINDING_NV 0x88EF +#endif /* GL_NV_pixel_buffer_object */ + +#ifndef GL_NV_polygon_mode +#define GL_NV_polygon_mode 1 +#define GL_POLYGON_MODE_NV 0x0B40 +#define GL_POLYGON_OFFSET_POINT_NV 0x2A01 +#define GL_POLYGON_OFFSET_LINE_NV 0x2A02 +#define GL_POINT_NV 0x1B00 +#define GL_LINE_NV 0x1B01 +#define GL_FILL_NV 0x1B02 +typedef void (GL_APIENTRYP PFNGLPOLYGONMODENVPROC) (GLenum face, GLenum mode); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glPolygonModeNV (GLenum face, GLenum mode); +#endif +#endif /* GL_NV_polygon_mode */ + +#ifndef GL_NV_primitive_shading_rate +#define GL_NV_primitive_shading_rate 1 +#define GL_SHADING_RATE_IMAGE_PER_PRIMITIVE_NV 0x95B1 +#define GL_SHADING_RATE_IMAGE_PALETTE_COUNT_NV 0x95B2 +#endif /* GL_NV_primitive_shading_rate */ + +#ifndef GL_NV_read_buffer +#define GL_NV_read_buffer 1 +#define GL_READ_BUFFER_NV 0x0C02 +typedef void (GL_APIENTRYP PFNGLREADBUFFERNVPROC) (GLenum mode); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glReadBufferNV (GLenum mode); +#endif +#endif /* GL_NV_read_buffer */ + +#ifndef GL_NV_read_buffer_front +#define GL_NV_read_buffer_front 1 +#endif /* GL_NV_read_buffer_front */ + +#ifndef GL_NV_read_depth +#define GL_NV_read_depth 1 +#endif /* GL_NV_read_depth */ + +#ifndef GL_NV_read_depth_stencil +#define GL_NV_read_depth_stencil 1 +#endif /* GL_NV_read_depth_stencil */ + +#ifndef GL_NV_read_stencil +#define GL_NV_read_stencil 1 +#endif /* GL_NV_read_stencil */ + +#ifndef GL_NV_representative_fragment_test +#define GL_NV_representative_fragment_test 1 +#define GL_REPRESENTATIVE_FRAGMENT_TEST_NV 0x937F +#endif /* GL_NV_representative_fragment_test */ + +#ifndef GL_NV_sRGB_formats +#define GL_NV_sRGB_formats 1 +#define GL_SLUMINANCE_NV 0x8C46 +#define GL_SLUMINANCE_ALPHA_NV 0x8C44 +#define GL_SRGB8_NV 0x8C41 +#define GL_SLUMINANCE8_NV 0x8C47 +#define GL_SLUMINANCE8_ALPHA8_NV 0x8C45 +#define GL_COMPRESSED_SRGB_S3TC_DXT1_NV 0x8C4C +#define GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT1_NV 0x8C4D +#define GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT3_NV 0x8C4E +#define GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT5_NV 0x8C4F +#define GL_ETC1_SRGB8_NV 0x88EE +#endif /* GL_NV_sRGB_formats */ + +#ifndef GL_NV_sample_locations +#define GL_NV_sample_locations 1 +#define GL_SAMPLE_LOCATION_SUBPIXEL_BITS_NV 0x933D +#define GL_SAMPLE_LOCATION_PIXEL_GRID_WIDTH_NV 0x933E +#define GL_SAMPLE_LOCATION_PIXEL_GRID_HEIGHT_NV 0x933F +#define GL_PROGRAMMABLE_SAMPLE_LOCATION_TABLE_SIZE_NV 0x9340 +#define GL_SAMPLE_LOCATION_NV 0x8E50 +#define GL_PROGRAMMABLE_SAMPLE_LOCATION_NV 0x9341 +#define GL_FRAMEBUFFER_PROGRAMMABLE_SAMPLE_LOCATIONS_NV 0x9342 +#define GL_FRAMEBUFFER_SAMPLE_LOCATION_PIXEL_GRID_NV 0x9343 +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERSAMPLELOCATIONSFVNVPROC) (GLenum target, GLuint start, GLsizei count, const GLfloat *v); +typedef void (GL_APIENTRYP PFNGLNAMEDFRAMEBUFFERSAMPLELOCATIONSFVNVPROC) (GLuint framebuffer, GLuint start, GLsizei count, const GLfloat *v); +typedef void (GL_APIENTRYP PFNGLRESOLVEDEPTHVALUESNVPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glFramebufferSampleLocationsfvNV (GLenum target, GLuint start, GLsizei count, const GLfloat *v); +GL_APICALL void GL_APIENTRY glNamedFramebufferSampleLocationsfvNV (GLuint framebuffer, GLuint start, GLsizei count, const GLfloat *v); +GL_APICALL void GL_APIENTRY glResolveDepthValuesNV (void); +#endif +#endif /* GL_NV_sample_locations */ + +#ifndef GL_NV_sample_mask_override_coverage +#define GL_NV_sample_mask_override_coverage 1 +#endif /* GL_NV_sample_mask_override_coverage */ + +#ifndef GL_NV_scissor_exclusive +#define GL_NV_scissor_exclusive 1 +#define GL_SCISSOR_TEST_EXCLUSIVE_NV 0x9555 +#define GL_SCISSOR_BOX_EXCLUSIVE_NV 0x9556 +typedef void (GL_APIENTRYP PFNGLSCISSOREXCLUSIVENVPROC) (GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (GL_APIENTRYP PFNGLSCISSOREXCLUSIVEARRAYVNVPROC) (GLuint first, GLsizei count, const GLint *v); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glScissorExclusiveNV (GLint x, GLint y, GLsizei width, GLsizei height); +GL_APICALL void GL_APIENTRY glScissorExclusiveArrayvNV (GLuint first, GLsizei count, const GLint *v); +#endif +#endif /* GL_NV_scissor_exclusive */ + +#ifndef GL_NV_shader_atomic_fp16_vector +#define GL_NV_shader_atomic_fp16_vector 1 +#endif /* GL_NV_shader_atomic_fp16_vector */ + +#ifndef GL_NV_shader_noperspective_interpolation +#define GL_NV_shader_noperspective_interpolation 1 +#endif /* GL_NV_shader_noperspective_interpolation */ + +#ifndef GL_NV_shader_subgroup_partitioned +#define GL_NV_shader_subgroup_partitioned 1 +#define GL_SUBGROUP_FEATURE_PARTITIONED_BIT_NV 0x00000100 +#endif /* GL_NV_shader_subgroup_partitioned */ + +#ifndef GL_NV_shader_texture_footprint +#define GL_NV_shader_texture_footprint 1 +#endif /* GL_NV_shader_texture_footprint */ + +#ifndef GL_NV_shading_rate_image +#define GL_NV_shading_rate_image 1 +#define GL_SHADING_RATE_IMAGE_NV 0x9563 +#define GL_SHADING_RATE_NO_INVOCATIONS_NV 0x9564 +#define GL_SHADING_RATE_1_INVOCATION_PER_PIXEL_NV 0x9565 +#define GL_SHADING_RATE_1_INVOCATION_PER_1X2_PIXELS_NV 0x9566 +#define GL_SHADING_RATE_1_INVOCATION_PER_2X1_PIXELS_NV 0x9567 +#define GL_SHADING_RATE_1_INVOCATION_PER_2X2_PIXELS_NV 0x9568 +#define GL_SHADING_RATE_1_INVOCATION_PER_2X4_PIXELS_NV 0x9569 +#define GL_SHADING_RATE_1_INVOCATION_PER_4X2_PIXELS_NV 0x956A +#define GL_SHADING_RATE_1_INVOCATION_PER_4X4_PIXELS_NV 0x956B +#define GL_SHADING_RATE_2_INVOCATIONS_PER_PIXEL_NV 0x956C +#define GL_SHADING_RATE_4_INVOCATIONS_PER_PIXEL_NV 0x956D +#define GL_SHADING_RATE_8_INVOCATIONS_PER_PIXEL_NV 0x956E +#define GL_SHADING_RATE_16_INVOCATIONS_PER_PIXEL_NV 0x956F +#define GL_SHADING_RATE_IMAGE_BINDING_NV 0x955B +#define GL_SHADING_RATE_IMAGE_TEXEL_WIDTH_NV 0x955C +#define GL_SHADING_RATE_IMAGE_TEXEL_HEIGHT_NV 0x955D +#define GL_SHADING_RATE_IMAGE_PALETTE_SIZE_NV 0x955E +#define GL_MAX_COARSE_FRAGMENT_SAMPLES_NV 0x955F +#define GL_SHADING_RATE_SAMPLE_ORDER_DEFAULT_NV 0x95AE +#define GL_SHADING_RATE_SAMPLE_ORDER_PIXEL_MAJOR_NV 0x95AF +#define GL_SHADING_RATE_SAMPLE_ORDER_SAMPLE_MAJOR_NV 0x95B0 +typedef void (GL_APIENTRYP PFNGLBINDSHADINGRATEIMAGENVPROC) (GLuint texture); +typedef void (GL_APIENTRYP PFNGLGETSHADINGRATEIMAGEPALETTENVPROC) (GLuint viewport, GLuint entry, GLenum *rate); +typedef void (GL_APIENTRYP PFNGLGETSHADINGRATESAMPLELOCATIONIVNVPROC) (GLenum rate, GLuint samples, GLuint index, GLint *location); +typedef void (GL_APIENTRYP PFNGLSHADINGRATEIMAGEBARRIERNVPROC) (GLboolean synchronize); +typedef void (GL_APIENTRYP PFNGLSHADINGRATEIMAGEPALETTENVPROC) (GLuint viewport, GLuint first, GLsizei count, const GLenum *rates); +typedef void (GL_APIENTRYP PFNGLSHADINGRATESAMPLEORDERNVPROC) (GLenum order); +typedef void (GL_APIENTRYP PFNGLSHADINGRATESAMPLEORDERCUSTOMNVPROC) (GLenum rate, GLuint samples, const GLint *locations); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glBindShadingRateImageNV (GLuint texture); +GL_APICALL void GL_APIENTRY glGetShadingRateImagePaletteNV (GLuint viewport, GLuint entry, GLenum *rate); +GL_APICALL void GL_APIENTRY glGetShadingRateSampleLocationivNV (GLenum rate, GLuint samples, GLuint index, GLint *location); +GL_APICALL void GL_APIENTRY glShadingRateImageBarrierNV (GLboolean synchronize); +GL_APICALL void GL_APIENTRY glShadingRateImagePaletteNV (GLuint viewport, GLuint first, GLsizei count, const GLenum *rates); +GL_APICALL void GL_APIENTRY glShadingRateSampleOrderNV (GLenum order); +GL_APICALL void GL_APIENTRY glShadingRateSampleOrderCustomNV (GLenum rate, GLuint samples, const GLint *locations); +#endif +#endif /* GL_NV_shading_rate_image */ + +#ifndef GL_NV_shadow_samplers_array +#define GL_NV_shadow_samplers_array 1 +#define GL_SAMPLER_2D_ARRAY_SHADOW_NV 0x8DC4 +#endif /* GL_NV_shadow_samplers_array */ + +#ifndef GL_NV_shadow_samplers_cube +#define GL_NV_shadow_samplers_cube 1 +#define GL_SAMPLER_CUBE_SHADOW_NV 0x8DC5 +#endif /* GL_NV_shadow_samplers_cube */ + +#ifndef GL_NV_stereo_view_rendering +#define GL_NV_stereo_view_rendering 1 +#endif /* GL_NV_stereo_view_rendering */ + +#ifndef GL_NV_texture_border_clamp +#define GL_NV_texture_border_clamp 1 +#define GL_TEXTURE_BORDER_COLOR_NV 0x1004 +#define GL_CLAMP_TO_BORDER_NV 0x812D +#endif /* GL_NV_texture_border_clamp */ + +#ifndef GL_NV_texture_compression_s3tc_update +#define GL_NV_texture_compression_s3tc_update 1 +#endif /* GL_NV_texture_compression_s3tc_update */ + +#ifndef GL_NV_texture_npot_2D_mipmap +#define GL_NV_texture_npot_2D_mipmap 1 +#endif /* GL_NV_texture_npot_2D_mipmap */ + +#ifndef GL_NV_timeline_semaphore +#define GL_NV_timeline_semaphore 1 +#define GL_TIMELINE_SEMAPHORE_VALUE_NV 0x9595 +#define GL_SEMAPHORE_TYPE_NV 0x95B3 +#define GL_SEMAPHORE_TYPE_BINARY_NV 0x95B4 +#define GL_SEMAPHORE_TYPE_TIMELINE_NV 0x95B5 +#define GL_MAX_TIMELINE_SEMAPHORE_VALUE_DIFFERENCE_NV 0x95B6 +typedef void (GL_APIENTRYP PFNGLCREATESEMAPHORESNVPROC) (GLsizei n, GLuint *semaphores); +typedef void (GL_APIENTRYP PFNGLSEMAPHOREPARAMETERIVNVPROC) (GLuint semaphore, GLenum pname, const GLint *params); +typedef void (GL_APIENTRYP PFNGLGETSEMAPHOREPARAMETERIVNVPROC) (GLuint semaphore, GLenum pname, GLint *params); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glCreateSemaphoresNV (GLsizei n, GLuint *semaphores); +GL_APICALL void GL_APIENTRY glSemaphoreParameterivNV (GLuint semaphore, GLenum pname, const GLint *params); +GL_APICALL void GL_APIENTRY glGetSemaphoreParameterivNV (GLuint semaphore, GLenum pname, GLint *params); +#endif +#endif /* GL_NV_timeline_semaphore */ + +#ifndef GL_NV_viewport_array +#define GL_NV_viewport_array 1 +#define GL_MAX_VIEWPORTS_NV 0x825B +#define GL_VIEWPORT_SUBPIXEL_BITS_NV 0x825C +#define GL_VIEWPORT_BOUNDS_RANGE_NV 0x825D +#define GL_VIEWPORT_INDEX_PROVOKING_VERTEX_NV 0x825F +typedef void (GL_APIENTRYP PFNGLVIEWPORTARRAYVNVPROC) (GLuint first, GLsizei count, const GLfloat *v); +typedef void (GL_APIENTRYP PFNGLVIEWPORTINDEXEDFNVPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat w, GLfloat h); +typedef void (GL_APIENTRYP PFNGLVIEWPORTINDEXEDFVNVPROC) (GLuint index, const GLfloat *v); +typedef void (GL_APIENTRYP PFNGLSCISSORARRAYVNVPROC) (GLuint first, GLsizei count, const GLint *v); +typedef void (GL_APIENTRYP PFNGLSCISSORINDEXEDNVPROC) (GLuint index, GLint left, GLint bottom, GLsizei width, GLsizei height); +typedef void (GL_APIENTRYP PFNGLSCISSORINDEXEDVNVPROC) (GLuint index, const GLint *v); +typedef void (GL_APIENTRYP PFNGLDEPTHRANGEARRAYFVNVPROC) (GLuint first, GLsizei count, const GLfloat *v); +typedef void (GL_APIENTRYP PFNGLDEPTHRANGEINDEXEDFNVPROC) (GLuint index, GLfloat n, GLfloat f); +typedef void (GL_APIENTRYP PFNGLGETFLOATI_VNVPROC) (GLenum target, GLuint index, GLfloat *data); +typedef void (GL_APIENTRYP PFNGLENABLEINVPROC) (GLenum target, GLuint index); +typedef void (GL_APIENTRYP PFNGLDISABLEINVPROC) (GLenum target, GLuint index); +typedef GLboolean (GL_APIENTRYP PFNGLISENABLEDINVPROC) (GLenum target, GLuint index); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glViewportArrayvNV (GLuint first, GLsizei count, const GLfloat *v); +GL_APICALL void GL_APIENTRY glViewportIndexedfNV (GLuint index, GLfloat x, GLfloat y, GLfloat w, GLfloat h); +GL_APICALL void GL_APIENTRY glViewportIndexedfvNV (GLuint index, const GLfloat *v); +GL_APICALL void GL_APIENTRY glScissorArrayvNV (GLuint first, GLsizei count, const GLint *v); +GL_APICALL void GL_APIENTRY glScissorIndexedNV (GLuint index, GLint left, GLint bottom, GLsizei width, GLsizei height); +GL_APICALL void GL_APIENTRY glScissorIndexedvNV (GLuint index, const GLint *v); +GL_APICALL void GL_APIENTRY glDepthRangeArrayfvNV (GLuint first, GLsizei count, const GLfloat *v); +GL_APICALL void GL_APIENTRY glDepthRangeIndexedfNV (GLuint index, GLfloat n, GLfloat f); +GL_APICALL void GL_APIENTRY glGetFloati_vNV (GLenum target, GLuint index, GLfloat *data); +GL_APICALL void GL_APIENTRY glEnableiNV (GLenum target, GLuint index); +GL_APICALL void GL_APIENTRY glDisableiNV (GLenum target, GLuint index); +GL_APICALL GLboolean GL_APIENTRY glIsEnablediNV (GLenum target, GLuint index); +#endif +#endif /* GL_NV_viewport_array */ + +#ifndef GL_NV_viewport_array2 +#define GL_NV_viewport_array2 1 +#endif /* GL_NV_viewport_array2 */ + +#ifndef GL_NV_viewport_swizzle +#define GL_NV_viewport_swizzle 1 +#define GL_VIEWPORT_SWIZZLE_POSITIVE_X_NV 0x9350 +#define GL_VIEWPORT_SWIZZLE_NEGATIVE_X_NV 0x9351 +#define GL_VIEWPORT_SWIZZLE_POSITIVE_Y_NV 0x9352 +#define GL_VIEWPORT_SWIZZLE_NEGATIVE_Y_NV 0x9353 +#define GL_VIEWPORT_SWIZZLE_POSITIVE_Z_NV 0x9354 +#define GL_VIEWPORT_SWIZZLE_NEGATIVE_Z_NV 0x9355 +#define GL_VIEWPORT_SWIZZLE_POSITIVE_W_NV 0x9356 +#define GL_VIEWPORT_SWIZZLE_NEGATIVE_W_NV 0x9357 +#define GL_VIEWPORT_SWIZZLE_X_NV 0x9358 +#define GL_VIEWPORT_SWIZZLE_Y_NV 0x9359 +#define GL_VIEWPORT_SWIZZLE_Z_NV 0x935A +#define GL_VIEWPORT_SWIZZLE_W_NV 0x935B +typedef void (GL_APIENTRYP PFNGLVIEWPORTSWIZZLENVPROC) (GLuint index, GLenum swizzlex, GLenum swizzley, GLenum swizzlez, GLenum swizzlew); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glViewportSwizzleNV (GLuint index, GLenum swizzlex, GLenum swizzley, GLenum swizzlez, GLenum swizzlew); +#endif +#endif /* GL_NV_viewport_swizzle */ + +#ifndef GL_OVR_multiview +#define GL_OVR_multiview 1 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_NUM_VIEWS_OVR 0x9630 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_BASE_VIEW_INDEX_OVR 0x9632 +#define GL_MAX_VIEWS_OVR 0x9631 +#define GL_FRAMEBUFFER_INCOMPLETE_VIEW_TARGETS_OVR 0x9633 +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERTEXTUREMULTIVIEWOVRPROC) (GLenum target, GLenum attachment, GLuint texture, GLint level, GLint baseViewIndex, GLsizei numViews); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glFramebufferTextureMultiviewOVR (GLenum target, GLenum attachment, GLuint texture, GLint level, GLint baseViewIndex, GLsizei numViews); +#endif +#endif /* GL_OVR_multiview */ + +#ifndef GL_OVR_multiview2 +#define GL_OVR_multiview2 1 +#endif /* GL_OVR_multiview2 */ + +#ifndef GL_OVR_multiview_multisampled_render_to_texture +#define GL_OVR_multiview_multisampled_render_to_texture 1 +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERTEXTUREMULTISAMPLEMULTIVIEWOVRPROC) (GLenum target, GLenum attachment, GLuint texture, GLint level, GLsizei samples, GLint baseViewIndex, GLsizei numViews); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glFramebufferTextureMultisampleMultiviewOVR (GLenum target, GLenum attachment, GLuint texture, GLint level, GLsizei samples, GLint baseViewIndex, GLsizei numViews); +#endif +#endif /* GL_OVR_multiview_multisampled_render_to_texture */ + +#ifndef GL_QCOM_YUV_texture_gather +#define GL_QCOM_YUV_texture_gather 1 +#endif /* GL_QCOM_YUV_texture_gather */ + +#ifndef GL_QCOM_alpha_test +#define GL_QCOM_alpha_test 1 +#define GL_ALPHA_TEST_QCOM 0x0BC0 +#define GL_ALPHA_TEST_FUNC_QCOM 0x0BC1 +#define GL_ALPHA_TEST_REF_QCOM 0x0BC2 +typedef void (GL_APIENTRYP PFNGLALPHAFUNCQCOMPROC) (GLenum func, GLclampf ref); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glAlphaFuncQCOM (GLenum func, GLclampf ref); +#endif +#endif /* GL_QCOM_alpha_test */ + +#ifndef GL_QCOM_binning_control +#define GL_QCOM_binning_control 1 +#define GL_BINNING_CONTROL_HINT_QCOM 0x8FB0 +#define GL_CPU_OPTIMIZED_QCOM 0x8FB1 +#define GL_GPU_OPTIMIZED_QCOM 0x8FB2 +#define GL_RENDER_DIRECT_TO_FRAMEBUFFER_QCOM 0x8FB3 +#endif /* GL_QCOM_binning_control */ + +#ifndef GL_QCOM_driver_control +#define GL_QCOM_driver_control 1 +typedef void (GL_APIENTRYP PFNGLGETDRIVERCONTROLSQCOMPROC) (GLint *num, GLsizei size, GLuint *driverControls); +typedef void (GL_APIENTRYP PFNGLGETDRIVERCONTROLSTRINGQCOMPROC) (GLuint driverControl, GLsizei bufSize, GLsizei *length, GLchar *driverControlString); +typedef void (GL_APIENTRYP PFNGLENABLEDRIVERCONTROLQCOMPROC) (GLuint driverControl); +typedef void (GL_APIENTRYP PFNGLDISABLEDRIVERCONTROLQCOMPROC) (GLuint driverControl); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glGetDriverControlsQCOM (GLint *num, GLsizei size, GLuint *driverControls); +GL_APICALL void GL_APIENTRY glGetDriverControlStringQCOM (GLuint driverControl, GLsizei bufSize, GLsizei *length, GLchar *driverControlString); +GL_APICALL void GL_APIENTRY glEnableDriverControlQCOM (GLuint driverControl); +GL_APICALL void GL_APIENTRY glDisableDriverControlQCOM (GLuint driverControl); +#endif +#endif /* GL_QCOM_driver_control */ + +#ifndef GL_QCOM_extended_get +#define GL_QCOM_extended_get 1 +#define GL_TEXTURE_WIDTH_QCOM 0x8BD2 +#define GL_TEXTURE_HEIGHT_QCOM 0x8BD3 +#define GL_TEXTURE_DEPTH_QCOM 0x8BD4 +#define GL_TEXTURE_INTERNAL_FORMAT_QCOM 0x8BD5 +#define GL_TEXTURE_FORMAT_QCOM 0x8BD6 +#define GL_TEXTURE_TYPE_QCOM 0x8BD7 +#define GL_TEXTURE_IMAGE_VALID_QCOM 0x8BD8 +#define GL_TEXTURE_NUM_LEVELS_QCOM 0x8BD9 +#define GL_TEXTURE_TARGET_QCOM 0x8BDA +#define GL_TEXTURE_OBJECT_VALID_QCOM 0x8BDB +#define GL_STATE_RESTORE 0x8BDC +typedef void (GL_APIENTRYP PFNGLEXTGETTEXTURESQCOMPROC) (GLuint *textures, GLint maxTextures, GLint *numTextures); +typedef void (GL_APIENTRYP PFNGLEXTGETBUFFERSQCOMPROC) (GLuint *buffers, GLint maxBuffers, GLint *numBuffers); +typedef void (GL_APIENTRYP PFNGLEXTGETRENDERBUFFERSQCOMPROC) (GLuint *renderbuffers, GLint maxRenderbuffers, GLint *numRenderbuffers); +typedef void (GL_APIENTRYP PFNGLEXTGETFRAMEBUFFERSQCOMPROC) (GLuint *framebuffers, GLint maxFramebuffers, GLint *numFramebuffers); +typedef void (GL_APIENTRYP PFNGLEXTGETTEXLEVELPARAMETERIVQCOMPROC) (GLuint texture, GLenum face, GLint level, GLenum pname, GLint *params); +typedef void (GL_APIENTRYP PFNGLEXTTEXOBJECTSTATEOVERRIDEIQCOMPROC) (GLenum target, GLenum pname, GLint param); +typedef void (GL_APIENTRYP PFNGLEXTGETTEXSUBIMAGEQCOMPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, void *texels); +typedef void (GL_APIENTRYP PFNGLEXTGETBUFFERPOINTERVQCOMPROC) (GLenum target, void **params); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glExtGetTexturesQCOM (GLuint *textures, GLint maxTextures, GLint *numTextures); +GL_APICALL void GL_APIENTRY glExtGetBuffersQCOM (GLuint *buffers, GLint maxBuffers, GLint *numBuffers); +GL_APICALL void GL_APIENTRY glExtGetRenderbuffersQCOM (GLuint *renderbuffers, GLint maxRenderbuffers, GLint *numRenderbuffers); +GL_APICALL void GL_APIENTRY glExtGetFramebuffersQCOM (GLuint *framebuffers, GLint maxFramebuffers, GLint *numFramebuffers); +GL_APICALL void GL_APIENTRY glExtGetTexLevelParameterivQCOM (GLuint texture, GLenum face, GLint level, GLenum pname, GLint *params); +GL_APICALL void GL_APIENTRY glExtTexObjectStateOverrideiQCOM (GLenum target, GLenum pname, GLint param); +GL_APICALL void GL_APIENTRY glExtGetTexSubImageQCOM (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, void *texels); +GL_APICALL void GL_APIENTRY glExtGetBufferPointervQCOM (GLenum target, void **params); +#endif +#endif /* GL_QCOM_extended_get */ + +#ifndef GL_QCOM_extended_get2 +#define GL_QCOM_extended_get2 1 +typedef void (GL_APIENTRYP PFNGLEXTGETSHADERSQCOMPROC) (GLuint *shaders, GLint maxShaders, GLint *numShaders); +typedef void (GL_APIENTRYP PFNGLEXTGETPROGRAMSQCOMPROC) (GLuint *programs, GLint maxPrograms, GLint *numPrograms); +typedef GLboolean (GL_APIENTRYP PFNGLEXTISPROGRAMBINARYQCOMPROC) (GLuint program); +typedef void (GL_APIENTRYP PFNGLEXTGETPROGRAMBINARYSOURCEQCOMPROC) (GLuint program, GLenum shadertype, GLchar *source, GLint *length); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glExtGetShadersQCOM (GLuint *shaders, GLint maxShaders, GLint *numShaders); +GL_APICALL void GL_APIENTRY glExtGetProgramsQCOM (GLuint *programs, GLint maxPrograms, GLint *numPrograms); +GL_APICALL GLboolean GL_APIENTRY glExtIsProgramBinaryQCOM (GLuint program); +GL_APICALL void GL_APIENTRY glExtGetProgramBinarySourceQCOM (GLuint program, GLenum shadertype, GLchar *source, GLint *length); +#endif +#endif /* GL_QCOM_extended_get2 */ + +#ifndef GL_QCOM_frame_extrapolation +#define GL_QCOM_frame_extrapolation 1 +typedef void (GL_APIENTRYP PFNGLEXTRAPOLATETEX2DQCOMPROC) (GLuint src1, GLuint src2, GLuint output, GLfloat scaleFactor); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glExtrapolateTex2DQCOM (GLuint src1, GLuint src2, GLuint output, GLfloat scaleFactor); +#endif +#endif /* GL_QCOM_frame_extrapolation */ + +#ifndef GL_QCOM_framebuffer_foveated +#define GL_QCOM_framebuffer_foveated 1 +#define GL_FOVEATION_ENABLE_BIT_QCOM 0x00000001 +#define GL_FOVEATION_SCALED_BIN_METHOD_BIT_QCOM 0x00000002 +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERFOVEATIONCONFIGQCOMPROC) (GLuint framebuffer, GLuint numLayers, GLuint focalPointsPerLayer, GLuint requestedFeatures, GLuint *providedFeatures); +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERFOVEATIONPARAMETERSQCOMPROC) (GLuint framebuffer, GLuint layer, GLuint focalPoint, GLfloat focalX, GLfloat focalY, GLfloat gainX, GLfloat gainY, GLfloat foveaArea); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glFramebufferFoveationConfigQCOM (GLuint framebuffer, GLuint numLayers, GLuint focalPointsPerLayer, GLuint requestedFeatures, GLuint *providedFeatures); +GL_APICALL void GL_APIENTRY glFramebufferFoveationParametersQCOM (GLuint framebuffer, GLuint layer, GLuint focalPoint, GLfloat focalX, GLfloat focalY, GLfloat gainX, GLfloat gainY, GLfloat foveaArea); +#endif +#endif /* GL_QCOM_framebuffer_foveated */ + +#ifndef GL_QCOM_motion_estimation +#define GL_QCOM_motion_estimation 1 +#define GL_MOTION_ESTIMATION_SEARCH_BLOCK_X_QCOM 0x8C90 +#define GL_MOTION_ESTIMATION_SEARCH_BLOCK_Y_QCOM 0x8C91 +typedef void (GL_APIENTRYP PFNGLTEXESTIMATEMOTIONQCOMPROC) (GLuint ref, GLuint target, GLuint output); +typedef void (GL_APIENTRYP PFNGLTEXESTIMATEMOTIONREGIONSQCOMPROC) (GLuint ref, GLuint target, GLuint output, GLuint mask); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glTexEstimateMotionQCOM (GLuint ref, GLuint target, GLuint output); +GL_APICALL void GL_APIENTRY glTexEstimateMotionRegionsQCOM (GLuint ref, GLuint target, GLuint output, GLuint mask); +#endif +#endif /* GL_QCOM_motion_estimation */ + +#ifndef GL_QCOM_perfmon_global_mode +#define GL_QCOM_perfmon_global_mode 1 +#define GL_PERFMON_GLOBAL_MODE_QCOM 0x8FA0 +#endif /* GL_QCOM_perfmon_global_mode */ + +#ifndef GL_QCOM_render_shared_exponent +#define GL_QCOM_render_shared_exponent 1 +#endif /* GL_QCOM_render_shared_exponent */ + +#ifndef GL_QCOM_shader_framebuffer_fetch_noncoherent +#define GL_QCOM_shader_framebuffer_fetch_noncoherent 1 +#define GL_FRAMEBUFFER_FETCH_NONCOHERENT_QCOM 0x96A2 +typedef void (GL_APIENTRYP PFNGLFRAMEBUFFERFETCHBARRIERQCOMPROC) (void); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glFramebufferFetchBarrierQCOM (void); +#endif +#endif /* GL_QCOM_shader_framebuffer_fetch_noncoherent */ + +#ifndef GL_QCOM_shader_framebuffer_fetch_rate +#define GL_QCOM_shader_framebuffer_fetch_rate 1 +#endif /* GL_QCOM_shader_framebuffer_fetch_rate */ + +#ifndef GL_QCOM_shading_rate +#define GL_QCOM_shading_rate 1 +#define GL_SHADING_RATE_QCOM 0x96A4 +#define GL_SHADING_RATE_PRESERVE_ASPECT_RATIO_QCOM 0x96A5 +#define GL_SHADING_RATE_1X1_PIXELS_QCOM 0x96A6 +#define GL_SHADING_RATE_1X2_PIXELS_QCOM 0x96A7 +#define GL_SHADING_RATE_2X1_PIXELS_QCOM 0x96A8 +#define GL_SHADING_RATE_2X2_PIXELS_QCOM 0x96A9 +#define GL_SHADING_RATE_4X2_PIXELS_QCOM 0x96AC +#define GL_SHADING_RATE_4X4_PIXELS_QCOM 0x96AE +typedef void (GL_APIENTRYP PFNGLSHADINGRATEQCOMPROC) (GLenum rate); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glShadingRateQCOM (GLenum rate); +#endif +#endif /* GL_QCOM_shading_rate */ + +#ifndef GL_QCOM_texture_foveated +#define GL_QCOM_texture_foveated 1 +#define GL_TEXTURE_FOVEATED_FEATURE_BITS_QCOM 0x8BFB +#define GL_TEXTURE_FOVEATED_MIN_PIXEL_DENSITY_QCOM 0x8BFC +#define GL_TEXTURE_FOVEATED_FEATURE_QUERY_QCOM 0x8BFD +#define GL_TEXTURE_FOVEATED_NUM_FOCAL_POINTS_QUERY_QCOM 0x8BFE +#define GL_FRAMEBUFFER_INCOMPLETE_FOVEATION_QCOM 0x8BFF +typedef void (GL_APIENTRYP PFNGLTEXTUREFOVEATIONPARAMETERSQCOMPROC) (GLuint texture, GLuint layer, GLuint focalPoint, GLfloat focalX, GLfloat focalY, GLfloat gainX, GLfloat gainY, GLfloat foveaArea); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glTextureFoveationParametersQCOM (GLuint texture, GLuint layer, GLuint focalPoint, GLfloat focalX, GLfloat focalY, GLfloat gainX, GLfloat gainY, GLfloat foveaArea); +#endif +#endif /* GL_QCOM_texture_foveated */ + +#ifndef GL_QCOM_texture_foveated2 +#define GL_QCOM_texture_foveated2 1 +#define GL_TEXTURE_FOVEATED_CUTOFF_DENSITY_QCOM 0x96A0 +#endif /* GL_QCOM_texture_foveated2 */ + +#ifndef GL_QCOM_texture_foveated_subsampled_layout +#define GL_QCOM_texture_foveated_subsampled_layout 1 +#define GL_FOVEATION_SUBSAMPLED_LAYOUT_METHOD_BIT_QCOM 0x00000004 +#define GL_MAX_SHADER_SUBSAMPLED_IMAGE_UNITS_QCOM 0x8FA1 +#endif /* GL_QCOM_texture_foveated_subsampled_layout */ + +#ifndef GL_QCOM_tiled_rendering +#define GL_QCOM_tiled_rendering 1 +#define GL_COLOR_BUFFER_BIT0_QCOM 0x00000001 +#define GL_COLOR_BUFFER_BIT1_QCOM 0x00000002 +#define GL_COLOR_BUFFER_BIT2_QCOM 0x00000004 +#define GL_COLOR_BUFFER_BIT3_QCOM 0x00000008 +#define GL_COLOR_BUFFER_BIT4_QCOM 0x00000010 +#define GL_COLOR_BUFFER_BIT5_QCOM 0x00000020 +#define GL_COLOR_BUFFER_BIT6_QCOM 0x00000040 +#define GL_COLOR_BUFFER_BIT7_QCOM 0x00000080 +#define GL_DEPTH_BUFFER_BIT0_QCOM 0x00000100 +#define GL_DEPTH_BUFFER_BIT1_QCOM 0x00000200 +#define GL_DEPTH_BUFFER_BIT2_QCOM 0x00000400 +#define GL_DEPTH_BUFFER_BIT3_QCOM 0x00000800 +#define GL_DEPTH_BUFFER_BIT4_QCOM 0x00001000 +#define GL_DEPTH_BUFFER_BIT5_QCOM 0x00002000 +#define GL_DEPTH_BUFFER_BIT6_QCOM 0x00004000 +#define GL_DEPTH_BUFFER_BIT7_QCOM 0x00008000 +#define GL_STENCIL_BUFFER_BIT0_QCOM 0x00010000 +#define GL_STENCIL_BUFFER_BIT1_QCOM 0x00020000 +#define GL_STENCIL_BUFFER_BIT2_QCOM 0x00040000 +#define GL_STENCIL_BUFFER_BIT3_QCOM 0x00080000 +#define GL_STENCIL_BUFFER_BIT4_QCOM 0x00100000 +#define GL_STENCIL_BUFFER_BIT5_QCOM 0x00200000 +#define GL_STENCIL_BUFFER_BIT6_QCOM 0x00400000 +#define GL_STENCIL_BUFFER_BIT7_QCOM 0x00800000 +#define GL_MULTISAMPLE_BUFFER_BIT0_QCOM 0x01000000 +#define GL_MULTISAMPLE_BUFFER_BIT1_QCOM 0x02000000 +#define GL_MULTISAMPLE_BUFFER_BIT2_QCOM 0x04000000 +#define GL_MULTISAMPLE_BUFFER_BIT3_QCOM 0x08000000 +#define GL_MULTISAMPLE_BUFFER_BIT4_QCOM 0x10000000 +#define GL_MULTISAMPLE_BUFFER_BIT5_QCOM 0x20000000 +#define GL_MULTISAMPLE_BUFFER_BIT6_QCOM 0x40000000 +#define GL_MULTISAMPLE_BUFFER_BIT7_QCOM 0x80000000 +typedef void (GL_APIENTRYP PFNGLSTARTTILINGQCOMPROC) (GLuint x, GLuint y, GLuint width, GLuint height, GLbitfield preserveMask); +typedef void (GL_APIENTRYP PFNGLENDTILINGQCOMPROC) (GLbitfield preserveMask); +#ifdef GL_GLEXT_PROTOTYPES +GL_APICALL void GL_APIENTRY glStartTilingQCOM (GLuint x, GLuint y, GLuint width, GLuint height, GLbitfield preserveMask); +GL_APICALL void GL_APIENTRY glEndTilingQCOM (GLbitfield preserveMask); +#endif +#endif /* GL_QCOM_tiled_rendering */ + +#ifndef GL_QCOM_writeonly_rendering +#define GL_QCOM_writeonly_rendering 1 +#define GL_WRITEONLY_RENDERING_QCOM 0x8823 +#endif /* GL_QCOM_writeonly_rendering */ + +#ifndef GL_VIV_shader_binary +#define GL_VIV_shader_binary 1 +#define GL_SHADER_BINARY_VIV 0x8FC4 +#endif /* GL_VIV_shader_binary */ + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_gl2platform.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_gl2platform.h new file mode 100644 index 00000000..426796ef --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_gl2platform.h @@ -0,0 +1,27 @@ +#ifndef __gl2platform_h_ +#define __gl2platform_h_ + +/* +** Copyright 2017-2020 The Khronos Group Inc. +** SPDX-License-Identifier: Apache-2.0 +*/ + +/* Platform-specific types and definitions for OpenGL ES 2.X gl2.h + * + * Adopters may modify khrplatform.h and this file to suit their platform. + * Please contribute modifications back to Khronos as pull requests on the + * public github repository: + * https://github.com/KhronosGroup/OpenGL-Registry + */ + +/*#include */ + +#ifndef GL_APICALL +#define GL_APICALL KHRONOS_APICALL +#endif + +#ifndef GL_APIENTRY +#define GL_APIENTRY KHRONOS_APIENTRY +#endif + +#endif /* __gl2platform_h_ */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_khrplatform.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_khrplatform.h new file mode 100644 index 00000000..01646449 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_opengles2_khrplatform.h @@ -0,0 +1,311 @@ +#ifndef __khrplatform_h_ +#define __khrplatform_h_ + +/* +** Copyright (c) 2008-2018 The Khronos Group Inc. +** +** Permission is hereby granted, free of charge, to any person obtaining a +** copy of this software and/or associated documentation files (the +** "Materials"), to deal in the Materials without restriction, including +** without limitation the rights to use, copy, modify, merge, publish, +** distribute, sublicense, and/or sell copies of the Materials, and to +** permit persons to whom the Materials are furnished to do so, subject to +** the following conditions: +** +** The above copyright notice and this permission notice shall be included +** in all copies or substantial portions of the Materials. +** +** THE MATERIALS ARE PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. +** IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY +** CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, +** TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE +** MATERIALS OR THE USE OR OTHER DEALINGS IN THE MATERIALS. +*/ + +/* Khronos platform-specific types and definitions. + * + * The master copy of khrplatform.h is maintained in the Khronos EGL + * Registry repository at https://github.com/KhronosGroup/EGL-Registry + * The last semantic modification to khrplatform.h was at commit ID: + * 67a3e0864c2d75ea5287b9f3d2eb74a745936692 + * + * Adopters may modify this file to suit their platform. Adopters are + * encouraged to submit platform specific modifications to the Khronos + * group so that they can be included in future versions of this file. + * Please submit changes by filing pull requests or issues on + * the EGL Registry repository linked above. + * + * + * See the Implementer's Guidelines for information about where this file + * should be located on your system and for more details of its use: + * http://www.khronos.org/registry/implementers_guide.pdf + * + * This file should be included as + * #include + * by Khronos client API header files that use its types and defines. + * + * The types in khrplatform.h should only be used to define API-specific types. + * + * Types defined in khrplatform.h: + * khronos_int8_t signed 8 bit + * khronos_uint8_t unsigned 8 bit + * khronos_int16_t signed 16 bit + * khronos_uint16_t unsigned 16 bit + * khronos_int32_t signed 32 bit + * khronos_uint32_t unsigned 32 bit + * khronos_int64_t signed 64 bit + * khronos_uint64_t unsigned 64 bit + * khronos_intptr_t signed same number of bits as a pointer + * khronos_uintptr_t unsigned same number of bits as a pointer + * khronos_ssize_t signed size + * khronos_usize_t unsigned size + * khronos_float_t signed 32 bit floating point + * khronos_time_ns_t unsigned 64 bit time in nanoseconds + * khronos_utime_nanoseconds_t unsigned time interval or absolute time in + * nanoseconds + * khronos_stime_nanoseconds_t signed time interval in nanoseconds + * khronos_boolean_enum_t enumerated boolean type. This should + * only be used as a base type when a client API's boolean type is + * an enum. Client APIs which use an integer or other type for + * booleans cannot use this as the base type for their boolean. + * + * Tokens defined in khrplatform.h: + * + * KHRONOS_FALSE, KHRONOS_TRUE Enumerated boolean false/true values. + * + * KHRONOS_SUPPORT_INT64 is 1 if 64 bit integers are supported; otherwise 0. + * KHRONOS_SUPPORT_FLOAT is 1 if floats are supported; otherwise 0. + * + * Calling convention macros defined in this file: + * KHRONOS_APICALL + * KHRONOS_APIENTRY + * KHRONOS_APIATTRIBUTES + * + * These may be used in function prototypes as: + * + * KHRONOS_APICALL void KHRONOS_APIENTRY funcname( + * int arg1, + * int arg2) KHRONOS_APIATTRIBUTES; + */ + +#if defined(__SCITECH_SNAP__) && !defined(KHRONOS_STATIC) +# define KHRONOS_STATIC 1 +#endif + +/*------------------------------------------------------------------------- + * Definition of KHRONOS_APICALL + *------------------------------------------------------------------------- + * This precedes the return type of the function in the function prototype. + */ +#if defined(KHRONOS_STATIC) + /* If the preprocessor constant KHRONOS_STATIC is defined, make the + * header compatible with static linking. */ +# define KHRONOS_APICALL +#elif defined(_WIN32) +# define KHRONOS_APICALL __declspec(dllimport) +#elif defined (__SYMBIAN32__) +# define KHRONOS_APICALL IMPORT_C +#elif defined(__ANDROID__) +# define KHRONOS_APICALL __attribute__((visibility("default"))) +#else +# define KHRONOS_APICALL +#endif + +/*------------------------------------------------------------------------- + * Definition of KHRONOS_APIENTRY + *------------------------------------------------------------------------- + * This follows the return type of the function and precedes the function + * name in the function prototype. + */ +#if defined(_WIN32) && !defined(_WIN32_WCE) && !defined(__SCITECH_SNAP__) + /* Win32 but not WinCE */ +# define KHRONOS_APIENTRY __stdcall +#else +# define KHRONOS_APIENTRY +#endif + +/*------------------------------------------------------------------------- + * Definition of KHRONOS_APIATTRIBUTES + *------------------------------------------------------------------------- + * This follows the closing parenthesis of the function prototype arguments. + */ +#if defined (__ARMCC_2__) +#define KHRONOS_APIATTRIBUTES __softfp +#else +#define KHRONOS_APIATTRIBUTES +#endif + +/*------------------------------------------------------------------------- + * basic type definitions + *-----------------------------------------------------------------------*/ +#if (defined(__STDC_VERSION__) && __STDC_VERSION__ >= 199901L) || defined(__GNUC__) || defined(__SCO__) || defined(__USLC__) + + +/* + * Using + */ +#include +typedef int32_t khronos_int32_t; +typedef uint32_t khronos_uint32_t; +typedef int64_t khronos_int64_t; +typedef uint64_t khronos_uint64_t; +#define KHRONOS_SUPPORT_INT64 1 +#define KHRONOS_SUPPORT_FLOAT 1 +/* + * To support platform where unsigned long cannot be used interchangeably with + * inptr_t (e.g. CHERI-extended ISAs), we can use the stdint.h intptr_t. + * Ideally, we could just use (u)intptr_t everywhere, but this could result in + * ABI breakage if khronos_uintptr_t is changed from unsigned long to + * unsigned long long or similar (this results in different C++ name mangling). + * To avoid changes for existing platforms, we restrict usage of intptr_t to + * platforms where the size of a pointer is larger than the size of long. + */ +#if defined(__SIZEOF_LONG__) && defined(__SIZEOF_POINTER__) +#if __SIZEOF_POINTER__ > __SIZEOF_LONG__ +#define KHRONOS_USE_INTPTR_T +#endif +#endif + +#elif defined(__VMS ) || defined(__sgi) + +/* + * Using + */ +#include +typedef int32_t khronos_int32_t; +typedef uint32_t khronos_uint32_t; +typedef int64_t khronos_int64_t; +typedef uint64_t khronos_uint64_t; +#define KHRONOS_SUPPORT_INT64 1 +#define KHRONOS_SUPPORT_FLOAT 1 + +#elif defined(_WIN32) && !defined(__SCITECH_SNAP__) + +/* + * Win32 + */ +typedef __int32 khronos_int32_t; +typedef unsigned __int32 khronos_uint32_t; +typedef __int64 khronos_int64_t; +typedef unsigned __int64 khronos_uint64_t; +#define KHRONOS_SUPPORT_INT64 1 +#define KHRONOS_SUPPORT_FLOAT 1 + +#elif defined(__sun__) || defined(__digital__) + +/* + * Sun or Digital + */ +typedef int khronos_int32_t; +typedef unsigned int khronos_uint32_t; +#if defined(__arch64__) || defined(_LP64) +typedef long int khronos_int64_t; +typedef unsigned long int khronos_uint64_t; +#else +typedef long long int khronos_int64_t; +typedef unsigned long long int khronos_uint64_t; +#endif /* __arch64__ */ +#define KHRONOS_SUPPORT_INT64 1 +#define KHRONOS_SUPPORT_FLOAT 1 + +#elif 0 + +/* + * Hypothetical platform with no float or int64 support + */ +typedef int khronos_int32_t; +typedef unsigned int khronos_uint32_t; +#define KHRONOS_SUPPORT_INT64 0 +#define KHRONOS_SUPPORT_FLOAT 0 + +#else + +/* + * Generic fallback + */ +#include +typedef int32_t khronos_int32_t; +typedef uint32_t khronos_uint32_t; +typedef int64_t khronos_int64_t; +typedef uint64_t khronos_uint64_t; +#define KHRONOS_SUPPORT_INT64 1 +#define KHRONOS_SUPPORT_FLOAT 1 + +#endif + + +/* + * Types that are (so far) the same on all platforms + */ +typedef signed char khronos_int8_t; +typedef unsigned char khronos_uint8_t; +typedef signed short int khronos_int16_t; +typedef unsigned short int khronos_uint16_t; + +/* + * Types that differ between LLP64 and LP64 architectures - in LLP64, + * pointers are 64 bits, but 'long' is still 32 bits. Win64 appears + * to be the only LLP64 architecture in current use. + */ +#ifdef KHRONOS_USE_INTPTR_T +typedef intptr_t khronos_intptr_t; +typedef uintptr_t khronos_uintptr_t; +#elif defined(_WIN64) +typedef signed long long int khronos_intptr_t; +typedef unsigned long long int khronos_uintptr_t; +#else +typedef signed long int khronos_intptr_t; +typedef unsigned long int khronos_uintptr_t; +#endif + +#if defined(_WIN64) +typedef signed long long int khronos_ssize_t; +typedef unsigned long long int khronos_usize_t; +#else +typedef signed long int khronos_ssize_t; +typedef unsigned long int khronos_usize_t; +#endif + +#if KHRONOS_SUPPORT_FLOAT +/* + * Float type + */ +typedef float khronos_float_t; +#endif + +#if KHRONOS_SUPPORT_INT64 +/* Time types + * + * These types can be used to represent a time interval in nanoseconds or + * an absolute Unadjusted System Time. Unadjusted System Time is the number + * of nanoseconds since some arbitrary system event (e.g. since the last + * time the system booted). The Unadjusted System Time is an unsigned + * 64 bit value that wraps back to 0 every 584 years. Time intervals + * may be either signed or unsigned. + */ +typedef khronos_uint64_t khronos_utime_nanoseconds_t; +typedef khronos_int64_t khronos_stime_nanoseconds_t; +#endif + +/* + * Dummy value used to pad enum types to 32 bits. + */ +#ifndef KHRONOS_MAX_ENUM +#define KHRONOS_MAX_ENUM 0x7FFFFFFF +#endif + +/* + * Enumerated boolean type + * + * Values other than zero should be considered to be true. Therefore + * comparisons should not be made against KHRONOS_TRUE. + */ +typedef enum { + KHRONOS_FALSE = 0, + KHRONOS_TRUE = 1, + KHRONOS_BOOLEAN_ENUM_FORCE_SIZE = KHRONOS_MAX_ENUM +} khronos_boolean_enum_t; + +#endif /* __khrplatform_h_ */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_pixels.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_pixels.h new file mode 100644 index 00000000..9abd57b4 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_pixels.h @@ -0,0 +1,644 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_pixels.h + * + * Header for the enumerated pixel format definitions. + */ + +#ifndef SDL_pixels_h_ +#define SDL_pixels_h_ + +#include "SDL_stdinc.h" +#include "SDL_endian.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \name Transparency definitions + * + * These define alpha as the opacity of a surface. + */ +/* @{ */ +#define SDL_ALPHA_OPAQUE 255 +#define SDL_ALPHA_TRANSPARENT 0 +/* @} */ + +/** Pixel type. */ +typedef enum +{ + SDL_PIXELTYPE_UNKNOWN, + SDL_PIXELTYPE_INDEX1, + SDL_PIXELTYPE_INDEX4, + SDL_PIXELTYPE_INDEX8, + SDL_PIXELTYPE_PACKED8, + SDL_PIXELTYPE_PACKED16, + SDL_PIXELTYPE_PACKED32, + SDL_PIXELTYPE_ARRAYU8, + SDL_PIXELTYPE_ARRAYU16, + SDL_PIXELTYPE_ARRAYU32, + SDL_PIXELTYPE_ARRAYF16, + SDL_PIXELTYPE_ARRAYF32 +} SDL_PixelType; + +/** Bitmap pixel order, high bit -> low bit. */ +typedef enum +{ + SDL_BITMAPORDER_NONE, + SDL_BITMAPORDER_4321, + SDL_BITMAPORDER_1234 +} SDL_BitmapOrder; + +/** Packed component order, high bit -> low bit. */ +typedef enum +{ + SDL_PACKEDORDER_NONE, + SDL_PACKEDORDER_XRGB, + SDL_PACKEDORDER_RGBX, + SDL_PACKEDORDER_ARGB, + SDL_PACKEDORDER_RGBA, + SDL_PACKEDORDER_XBGR, + SDL_PACKEDORDER_BGRX, + SDL_PACKEDORDER_ABGR, + SDL_PACKEDORDER_BGRA +} SDL_PackedOrder; + +/** Array component order, low byte -> high byte. */ +/* !!! FIXME: in 2.1, make these not overlap differently with + !!! FIXME: SDL_PACKEDORDER_*, so we can simplify SDL_ISPIXELFORMAT_ALPHA */ +typedef enum +{ + SDL_ARRAYORDER_NONE, + SDL_ARRAYORDER_RGB, + SDL_ARRAYORDER_RGBA, + SDL_ARRAYORDER_ARGB, + SDL_ARRAYORDER_BGR, + SDL_ARRAYORDER_BGRA, + SDL_ARRAYORDER_ABGR +} SDL_ArrayOrder; + +/** Packed component layout. */ +typedef enum +{ + SDL_PACKEDLAYOUT_NONE, + SDL_PACKEDLAYOUT_332, + SDL_PACKEDLAYOUT_4444, + SDL_PACKEDLAYOUT_1555, + SDL_PACKEDLAYOUT_5551, + SDL_PACKEDLAYOUT_565, + SDL_PACKEDLAYOUT_8888, + SDL_PACKEDLAYOUT_2101010, + SDL_PACKEDLAYOUT_1010102 +} SDL_PackedLayout; + +#define SDL_DEFINE_PIXELFOURCC(A, B, C, D) SDL_FOURCC(A, B, C, D) + +#define SDL_DEFINE_PIXELFORMAT(type, order, layout, bits, bytes) \ + ((1 << 28) | ((type) << 24) | ((order) << 20) | ((layout) << 16) | \ + ((bits) << 8) | ((bytes) << 0)) + +#define SDL_PIXELFLAG(X) (((X) >> 28) & 0x0F) +#define SDL_PIXELTYPE(X) (((X) >> 24) & 0x0F) +#define SDL_PIXELORDER(X) (((X) >> 20) & 0x0F) +#define SDL_PIXELLAYOUT(X) (((X) >> 16) & 0x0F) +#define SDL_BITSPERPIXEL(X) (((X) >> 8) & 0xFF) +#define SDL_BYTESPERPIXEL(X) \ + (SDL_ISPIXELFORMAT_FOURCC(X) ? \ + ((((X) == SDL_PIXELFORMAT_YUY2) || \ + ((X) == SDL_PIXELFORMAT_UYVY) || \ + ((X) == SDL_PIXELFORMAT_YVYU)) ? 2 : 1) : (((X) >> 0) & 0xFF)) + +#define SDL_ISPIXELFORMAT_INDEXED(format) \ + (!SDL_ISPIXELFORMAT_FOURCC(format) && \ + ((SDL_PIXELTYPE(format) == SDL_PIXELTYPE_INDEX1) || \ + (SDL_PIXELTYPE(format) == SDL_PIXELTYPE_INDEX4) || \ + (SDL_PIXELTYPE(format) == SDL_PIXELTYPE_INDEX8))) + +#define SDL_ISPIXELFORMAT_PACKED(format) \ + (!SDL_ISPIXELFORMAT_FOURCC(format) && \ + ((SDL_PIXELTYPE(format) == SDL_PIXELTYPE_PACKED8) || \ + (SDL_PIXELTYPE(format) == SDL_PIXELTYPE_PACKED16) || \ + (SDL_PIXELTYPE(format) == SDL_PIXELTYPE_PACKED32))) + +#define SDL_ISPIXELFORMAT_ARRAY(format) \ + (!SDL_ISPIXELFORMAT_FOURCC(format) && \ + ((SDL_PIXELTYPE(format) == SDL_PIXELTYPE_ARRAYU8) || \ + (SDL_PIXELTYPE(format) == SDL_PIXELTYPE_ARRAYU16) || \ + (SDL_PIXELTYPE(format) == SDL_PIXELTYPE_ARRAYU32) || \ + (SDL_PIXELTYPE(format) == SDL_PIXELTYPE_ARRAYF16) || \ + (SDL_PIXELTYPE(format) == SDL_PIXELTYPE_ARRAYF32))) + +#define SDL_ISPIXELFORMAT_ALPHA(format) \ + ((SDL_ISPIXELFORMAT_PACKED(format) && \ + ((SDL_PIXELORDER(format) == SDL_PACKEDORDER_ARGB) || \ + (SDL_PIXELORDER(format) == SDL_PACKEDORDER_RGBA) || \ + (SDL_PIXELORDER(format) == SDL_PACKEDORDER_ABGR) || \ + (SDL_PIXELORDER(format) == SDL_PACKEDORDER_BGRA))) || \ + (SDL_ISPIXELFORMAT_ARRAY(format) && \ + ((SDL_PIXELORDER(format) == SDL_ARRAYORDER_ARGB) || \ + (SDL_PIXELORDER(format) == SDL_ARRAYORDER_RGBA) || \ + (SDL_PIXELORDER(format) == SDL_ARRAYORDER_ABGR) || \ + (SDL_PIXELORDER(format) == SDL_ARRAYORDER_BGRA)))) + +/* The flag is set to 1 because 0x1? is not in the printable ASCII range */ +#define SDL_ISPIXELFORMAT_FOURCC(format) \ + ((format) && (SDL_PIXELFLAG(format) != 1)) + +/* Note: If you modify this list, update SDL_GetPixelFormatName() */ +typedef enum +{ + SDL_PIXELFORMAT_UNKNOWN, + SDL_PIXELFORMAT_INDEX1LSB = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_INDEX1, SDL_BITMAPORDER_4321, 0, + 1, 0), + SDL_PIXELFORMAT_INDEX1MSB = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_INDEX1, SDL_BITMAPORDER_1234, 0, + 1, 0), + SDL_PIXELFORMAT_INDEX4LSB = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_INDEX4, SDL_BITMAPORDER_4321, 0, + 4, 0), + SDL_PIXELFORMAT_INDEX4MSB = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_INDEX4, SDL_BITMAPORDER_1234, 0, + 4, 0), + SDL_PIXELFORMAT_INDEX8 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_INDEX8, 0, 0, 8, 1), + SDL_PIXELFORMAT_RGB332 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED8, SDL_PACKEDORDER_XRGB, + SDL_PACKEDLAYOUT_332, 8, 1), + SDL_PIXELFORMAT_XRGB4444 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED16, SDL_PACKEDORDER_XRGB, + SDL_PACKEDLAYOUT_4444, 12, 2), + SDL_PIXELFORMAT_RGB444 = SDL_PIXELFORMAT_XRGB4444, + SDL_PIXELFORMAT_XBGR4444 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED16, SDL_PACKEDORDER_XBGR, + SDL_PACKEDLAYOUT_4444, 12, 2), + SDL_PIXELFORMAT_BGR444 = SDL_PIXELFORMAT_XBGR4444, + SDL_PIXELFORMAT_XRGB1555 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED16, SDL_PACKEDORDER_XRGB, + SDL_PACKEDLAYOUT_1555, 15, 2), + SDL_PIXELFORMAT_RGB555 = SDL_PIXELFORMAT_XRGB1555, + SDL_PIXELFORMAT_XBGR1555 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED16, SDL_PACKEDORDER_XBGR, + SDL_PACKEDLAYOUT_1555, 15, 2), + SDL_PIXELFORMAT_BGR555 = SDL_PIXELFORMAT_XBGR1555, + SDL_PIXELFORMAT_ARGB4444 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED16, SDL_PACKEDORDER_ARGB, + SDL_PACKEDLAYOUT_4444, 16, 2), + SDL_PIXELFORMAT_RGBA4444 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED16, SDL_PACKEDORDER_RGBA, + SDL_PACKEDLAYOUT_4444, 16, 2), + SDL_PIXELFORMAT_ABGR4444 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED16, SDL_PACKEDORDER_ABGR, + SDL_PACKEDLAYOUT_4444, 16, 2), + SDL_PIXELFORMAT_BGRA4444 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED16, SDL_PACKEDORDER_BGRA, + SDL_PACKEDLAYOUT_4444, 16, 2), + SDL_PIXELFORMAT_ARGB1555 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED16, SDL_PACKEDORDER_ARGB, + SDL_PACKEDLAYOUT_1555, 16, 2), + SDL_PIXELFORMAT_RGBA5551 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED16, SDL_PACKEDORDER_RGBA, + SDL_PACKEDLAYOUT_5551, 16, 2), + SDL_PIXELFORMAT_ABGR1555 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED16, SDL_PACKEDORDER_ABGR, + SDL_PACKEDLAYOUT_1555, 16, 2), + SDL_PIXELFORMAT_BGRA5551 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED16, SDL_PACKEDORDER_BGRA, + SDL_PACKEDLAYOUT_5551, 16, 2), + SDL_PIXELFORMAT_RGB565 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED16, SDL_PACKEDORDER_XRGB, + SDL_PACKEDLAYOUT_565, 16, 2), + SDL_PIXELFORMAT_BGR565 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED16, SDL_PACKEDORDER_XBGR, + SDL_PACKEDLAYOUT_565, 16, 2), + SDL_PIXELFORMAT_RGB24 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_ARRAYU8, SDL_ARRAYORDER_RGB, 0, + 24, 3), + SDL_PIXELFORMAT_BGR24 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_ARRAYU8, SDL_ARRAYORDER_BGR, 0, + 24, 3), + SDL_PIXELFORMAT_XRGB8888 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED32, SDL_PACKEDORDER_XRGB, + SDL_PACKEDLAYOUT_8888, 24, 4), + SDL_PIXELFORMAT_RGB888 = SDL_PIXELFORMAT_XRGB8888, + SDL_PIXELFORMAT_RGBX8888 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED32, SDL_PACKEDORDER_RGBX, + SDL_PACKEDLAYOUT_8888, 24, 4), + SDL_PIXELFORMAT_XBGR8888 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED32, SDL_PACKEDORDER_XBGR, + SDL_PACKEDLAYOUT_8888, 24, 4), + SDL_PIXELFORMAT_BGR888 = SDL_PIXELFORMAT_XBGR8888, + SDL_PIXELFORMAT_BGRX8888 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED32, SDL_PACKEDORDER_BGRX, + SDL_PACKEDLAYOUT_8888, 24, 4), + SDL_PIXELFORMAT_ARGB8888 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED32, SDL_PACKEDORDER_ARGB, + SDL_PACKEDLAYOUT_8888, 32, 4), + SDL_PIXELFORMAT_RGBA8888 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED32, SDL_PACKEDORDER_RGBA, + SDL_PACKEDLAYOUT_8888, 32, 4), + SDL_PIXELFORMAT_ABGR8888 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED32, SDL_PACKEDORDER_ABGR, + SDL_PACKEDLAYOUT_8888, 32, 4), + SDL_PIXELFORMAT_BGRA8888 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED32, SDL_PACKEDORDER_BGRA, + SDL_PACKEDLAYOUT_8888, 32, 4), + SDL_PIXELFORMAT_ARGB2101010 = + SDL_DEFINE_PIXELFORMAT(SDL_PIXELTYPE_PACKED32, SDL_PACKEDORDER_ARGB, + SDL_PACKEDLAYOUT_2101010, 32, 4), + + /* Aliases for RGBA byte arrays of color data, for the current platform */ +#if SDL_BYTEORDER == SDL_BIG_ENDIAN + SDL_PIXELFORMAT_RGBA32 = SDL_PIXELFORMAT_RGBA8888, + SDL_PIXELFORMAT_ARGB32 = SDL_PIXELFORMAT_ARGB8888, + SDL_PIXELFORMAT_BGRA32 = SDL_PIXELFORMAT_BGRA8888, + SDL_PIXELFORMAT_ABGR32 = SDL_PIXELFORMAT_ABGR8888, +#else + SDL_PIXELFORMAT_RGBA32 = SDL_PIXELFORMAT_ABGR8888, + SDL_PIXELFORMAT_ARGB32 = SDL_PIXELFORMAT_BGRA8888, + SDL_PIXELFORMAT_BGRA32 = SDL_PIXELFORMAT_ARGB8888, + SDL_PIXELFORMAT_ABGR32 = SDL_PIXELFORMAT_RGBA8888, +#endif + + SDL_PIXELFORMAT_YV12 = /**< Planar mode: Y + V + U (3 planes) */ + SDL_DEFINE_PIXELFOURCC('Y', 'V', '1', '2'), + SDL_PIXELFORMAT_IYUV = /**< Planar mode: Y + U + V (3 planes) */ + SDL_DEFINE_PIXELFOURCC('I', 'Y', 'U', 'V'), + SDL_PIXELFORMAT_YUY2 = /**< Packed mode: Y0+U0+Y1+V0 (1 plane) */ + SDL_DEFINE_PIXELFOURCC('Y', 'U', 'Y', '2'), + SDL_PIXELFORMAT_UYVY = /**< Packed mode: U0+Y0+V0+Y1 (1 plane) */ + SDL_DEFINE_PIXELFOURCC('U', 'Y', 'V', 'Y'), + SDL_PIXELFORMAT_YVYU = /**< Packed mode: Y0+V0+Y1+U0 (1 plane) */ + SDL_DEFINE_PIXELFOURCC('Y', 'V', 'Y', 'U'), + SDL_PIXELFORMAT_NV12 = /**< Planar mode: Y + U/V interleaved (2 planes) */ + SDL_DEFINE_PIXELFOURCC('N', 'V', '1', '2'), + SDL_PIXELFORMAT_NV21 = /**< Planar mode: Y + V/U interleaved (2 planes) */ + SDL_DEFINE_PIXELFOURCC('N', 'V', '2', '1'), + SDL_PIXELFORMAT_EXTERNAL_OES = /**< Android video texture format */ + SDL_DEFINE_PIXELFOURCC('O', 'E', 'S', ' ') +} SDL_PixelFormatEnum; + +/** + * The bits of this structure can be directly reinterpreted as an integer-packed + * color which uses the SDL_PIXELFORMAT_RGBA32 format (SDL_PIXELFORMAT_ABGR8888 + * on little-endian systems and SDL_PIXELFORMAT_RGBA8888 on big-endian systems). + */ +typedef struct SDL_Color +{ + Uint8 r; + Uint8 g; + Uint8 b; + Uint8 a; +} SDL_Color; +#define SDL_Colour SDL_Color + +typedef struct SDL_Palette +{ + int ncolors; + SDL_Color *colors; + Uint32 version; + int refcount; +} SDL_Palette; + +/** + * \note Everything in the pixel format structure is read-only. + */ +typedef struct SDL_PixelFormat +{ + Uint32 format; + SDL_Palette *palette; + Uint8 BitsPerPixel; + Uint8 BytesPerPixel; + Uint8 padding[2]; + Uint32 Rmask; + Uint32 Gmask; + Uint32 Bmask; + Uint32 Amask; + Uint8 Rloss; + Uint8 Gloss; + Uint8 Bloss; + Uint8 Aloss; + Uint8 Rshift; + Uint8 Gshift; + Uint8 Bshift; + Uint8 Ashift; + int refcount; + struct SDL_PixelFormat *next; +} SDL_PixelFormat; + +/** + * Get the human readable name of a pixel format. + * + * \param format the pixel format to query + * \returns the human readable name of the specified pixel format or + * `SDL_PIXELFORMAT_UNKNOWN` if the format isn't recognized. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC const char* SDLCALL SDL_GetPixelFormatName(Uint32 format); + +/** + * Convert one of the enumerated pixel formats to a bpp value and RGBA masks. + * + * \param format one of the SDL_PixelFormatEnum values + * \param bpp a bits per pixel value; usually 15, 16, or 32 + * \param Rmask a pointer filled in with the red mask for the format + * \param Gmask a pointer filled in with the green mask for the format + * \param Bmask a pointer filled in with the blue mask for the format + * \param Amask a pointer filled in with the alpha mask for the format + * \returns SDL_TRUE on success or SDL_FALSE if the conversion wasn't + * possible; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_MasksToPixelFormatEnum + */ +extern DECLSPEC SDL_bool SDLCALL SDL_PixelFormatEnumToMasks(Uint32 format, + int *bpp, + Uint32 * Rmask, + Uint32 * Gmask, + Uint32 * Bmask, + Uint32 * Amask); + +/** + * Convert a bpp value and RGBA masks to an enumerated pixel format. + * + * This will return `SDL_PIXELFORMAT_UNKNOWN` if the conversion wasn't + * possible. + * + * \param bpp a bits per pixel value; usually 15, 16, or 32 + * \param Rmask the red mask for the format + * \param Gmask the green mask for the format + * \param Bmask the blue mask for the format + * \param Amask the alpha mask for the format + * \returns one of the SDL_PixelFormatEnum values + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_PixelFormatEnumToMasks + */ +extern DECLSPEC Uint32 SDLCALL SDL_MasksToPixelFormatEnum(int bpp, + Uint32 Rmask, + Uint32 Gmask, + Uint32 Bmask, + Uint32 Amask); + +/** + * Create an SDL_PixelFormat structure corresponding to a pixel format. + * + * Returned structure may come from a shared global cache (i.e. not newly + * allocated), and hence should not be modified, especially the palette. Weird + * errors such as `Blit combination not supported` may occur. + * + * \param pixel_format one of the SDL_PixelFormatEnum values + * \returns the new SDL_PixelFormat structure or NULL on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_FreeFormat + */ +extern DECLSPEC SDL_PixelFormat * SDLCALL SDL_AllocFormat(Uint32 pixel_format); + +/** + * Free an SDL_PixelFormat structure allocated by SDL_AllocFormat(). + * + * \param format the SDL_PixelFormat structure to free + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AllocFormat + */ +extern DECLSPEC void SDLCALL SDL_FreeFormat(SDL_PixelFormat *format); + +/** + * Create a palette structure with the specified number of color entries. + * + * The palette entries are initialized to white. + * + * \param ncolors represents the number of color entries in the color palette + * \returns a new SDL_Palette structure on success or NULL on failure (e.g. if + * there wasn't enough memory); call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_FreePalette + */ +extern DECLSPEC SDL_Palette *SDLCALL SDL_AllocPalette(int ncolors); + +/** + * Set the palette for a pixel format structure. + * + * \param format the SDL_PixelFormat structure that will use the palette + * \param palette the SDL_Palette structure that will be used + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AllocPalette + * \sa SDL_FreePalette + */ +extern DECLSPEC int SDLCALL SDL_SetPixelFormatPalette(SDL_PixelFormat * format, + SDL_Palette *palette); + +/** + * Set a range of colors in a palette. + * + * \param palette the SDL_Palette structure to modify + * \param colors an array of SDL_Color structures to copy into the palette + * \param firstcolor the index of the first palette entry to modify + * \param ncolors the number of entries to modify + * \returns 0 on success or a negative error code if not all of the colors + * could be set; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AllocPalette + * \sa SDL_CreateRGBSurface + */ +extern DECLSPEC int SDLCALL SDL_SetPaletteColors(SDL_Palette * palette, + const SDL_Color * colors, + int firstcolor, int ncolors); + +/** + * Free a palette created with SDL_AllocPalette(). + * + * \param palette the SDL_Palette structure to be freed + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AllocPalette + */ +extern DECLSPEC void SDLCALL SDL_FreePalette(SDL_Palette * palette); + +/** + * Map an RGB triple to an opaque pixel value for a given pixel format. + * + * This function maps the RGB color value to the specified pixel format and + * returns the pixel value best approximating the given RGB color value for + * the given pixel format. + * + * If the format has a palette (8-bit) the index of the closest matching color + * in the palette will be returned. + * + * If the specified pixel format has an alpha component it will be returned as + * all 1 bits (fully opaque). + * + * If the pixel format bpp (color depth) is less than 32-bpp then the unused + * upper bits of the return value can safely be ignored (e.g., with a 16-bpp + * format the return value can be assigned to a Uint16, and similarly a Uint8 + * for an 8-bpp format). + * + * \param format an SDL_PixelFormat structure describing the pixel format + * \param r the red component of the pixel in the range 0-255 + * \param g the green component of the pixel in the range 0-255 + * \param b the blue component of the pixel in the range 0-255 + * \returns a pixel value + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetRGB + * \sa SDL_GetRGBA + * \sa SDL_MapRGBA + */ +extern DECLSPEC Uint32 SDLCALL SDL_MapRGB(const SDL_PixelFormat * format, + Uint8 r, Uint8 g, Uint8 b); + +/** + * Map an RGBA quadruple to a pixel value for a given pixel format. + * + * This function maps the RGBA color value to the specified pixel format and + * returns the pixel value best approximating the given RGBA color value for + * the given pixel format. + * + * If the specified pixel format has no alpha component the alpha value will + * be ignored (as it will be in formats with a palette). + * + * If the format has a palette (8-bit) the index of the closest matching color + * in the palette will be returned. + * + * If the pixel format bpp (color depth) is less than 32-bpp then the unused + * upper bits of the return value can safely be ignored (e.g., with a 16-bpp + * format the return value can be assigned to a Uint16, and similarly a Uint8 + * for an 8-bpp format). + * + * \param format an SDL_PixelFormat structure describing the format of the + * pixel + * \param r the red component of the pixel in the range 0-255 + * \param g the green component of the pixel in the range 0-255 + * \param b the blue component of the pixel in the range 0-255 + * \param a the alpha component of the pixel in the range 0-255 + * \returns a pixel value + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetRGB + * \sa SDL_GetRGBA + * \sa SDL_MapRGB + */ +extern DECLSPEC Uint32 SDLCALL SDL_MapRGBA(const SDL_PixelFormat * format, + Uint8 r, Uint8 g, Uint8 b, + Uint8 a); + +/** + * Get RGB values from a pixel in the specified format. + * + * This function uses the entire 8-bit [0..255] range when converting color + * components from pixel formats with less than 8-bits per RGB component + * (e.g., a completely white pixel in 16-bit RGB565 format would return [0xff, + * 0xff, 0xff] not [0xf8, 0xfc, 0xf8]). + * + * \param pixel a pixel value + * \param format an SDL_PixelFormat structure describing the format of the + * pixel + * \param r a pointer filled in with the red component + * \param g a pointer filled in with the green component + * \param b a pointer filled in with the blue component + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetRGBA + * \sa SDL_MapRGB + * \sa SDL_MapRGBA + */ +extern DECLSPEC void SDLCALL SDL_GetRGB(Uint32 pixel, + const SDL_PixelFormat * format, + Uint8 * r, Uint8 * g, Uint8 * b); + +/** + * Get RGBA values from a pixel in the specified format. + * + * This function uses the entire 8-bit [0..255] range when converting color + * components from pixel formats with less than 8-bits per RGB component + * (e.g., a completely white pixel in 16-bit RGB565 format would return [0xff, + * 0xff, 0xff] not [0xf8, 0xfc, 0xf8]). + * + * If the surface has no alpha component, the alpha will be returned as 0xff + * (100% opaque). + * + * \param pixel a pixel value + * \param format an SDL_PixelFormat structure describing the format of the + * pixel + * \param r a pointer filled in with the red component + * \param g a pointer filled in with the green component + * \param b a pointer filled in with the blue component + * \param a a pointer filled in with the alpha component + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetRGB + * \sa SDL_MapRGB + * \sa SDL_MapRGBA + */ +extern DECLSPEC void SDLCALL SDL_GetRGBA(Uint32 pixel, + const SDL_PixelFormat * format, + Uint8 * r, Uint8 * g, Uint8 * b, + Uint8 * a); + +/** + * Calculate a 256 entry gamma ramp for a gamma value. + * + * \param gamma a gamma value where 0.0 is black and 1.0 is identity + * \param ramp an array of 256 values filled in with the gamma ramp + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetWindowGammaRamp + */ +extern DECLSPEC void SDLCALL SDL_CalculateGammaRamp(float gamma, Uint16 * ramp); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_pixels_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_platform.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_platform.h new file mode 100644 index 00000000..d2a7e052 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_platform.h @@ -0,0 +1,261 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_platform.h + * + * Try to get a standard set of platform defines. + */ + +#ifndef SDL_platform_h_ +#define SDL_platform_h_ + +#if defined(_AIX) +#undef __AIX__ +#define __AIX__ 1 +#endif +#if defined(__HAIKU__) +#undef __HAIKU__ +#define __HAIKU__ 1 +#endif +#if defined(bsdi) || defined(__bsdi) || defined(__bsdi__) +#undef __BSDI__ +#define __BSDI__ 1 +#endif +#if defined(_arch_dreamcast) +#undef __DREAMCAST__ +#define __DREAMCAST__ 1 +#endif +#if defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || defined(__DragonFly__) +#undef __FREEBSD__ +#define __FREEBSD__ 1 +#endif +#if defined(hpux) || defined(__hpux) || defined(__hpux__) +#undef __HPUX__ +#define __HPUX__ 1 +#endif +#if defined(sgi) || defined(__sgi) || defined(__sgi__) || defined(_SGI_SOURCE) +#undef __IRIX__ +#define __IRIX__ 1 +#endif +#if (defined(linux) || defined(__linux) || defined(__linux__)) +#undef __LINUX__ +#define __LINUX__ 1 +#endif +#if defined(ANDROID) || defined(__ANDROID__) +#undef __ANDROID__ +#undef __LINUX__ /* do we need to do this? */ +#define __ANDROID__ 1 +#endif +#if defined(__NGAGE__) +#undef __NGAGE__ +#define __NGAGE__ 1 +#endif + +#if defined(__APPLE__) +/* lets us know what version of Mac OS X we're compiling on */ +#include +#include + +/* Fix building with older SDKs that don't define these + See this for more information: + https://stackoverflow.com/questions/12132933/preprocessor-macro-for-os-x-targets +*/ +#ifndef TARGET_OS_MACCATALYST +#define TARGET_OS_MACCATALYST 0 +#endif +#ifndef TARGET_OS_IOS +#define TARGET_OS_IOS 0 +#endif +#ifndef TARGET_OS_IPHONE +#define TARGET_OS_IPHONE 0 +#endif +#ifndef TARGET_OS_TV +#define TARGET_OS_TV 0 +#endif +#ifndef TARGET_OS_SIMULATOR +#define TARGET_OS_SIMULATOR 0 +#endif + +#if TARGET_OS_TV +#undef __TVOS__ +#define __TVOS__ 1 +#endif +#if TARGET_OS_IPHONE +/* if compiling for iOS */ +#undef __IPHONEOS__ +#define __IPHONEOS__ 1 +#undef __MACOSX__ +#else +/* if not compiling for iOS */ +#undef __MACOSX__ +#define __MACOSX__ 1 +#if MAC_OS_X_VERSION_MIN_REQUIRED < 1070 +# error SDL for Mac OS X only supports deploying on 10.7 and above. +#endif /* MAC_OS_X_VERSION_MIN_REQUIRED < 1070 */ +#endif /* TARGET_OS_IPHONE */ +#endif /* defined(__APPLE__) */ + +#if defined(__NetBSD__) +#undef __NETBSD__ +#define __NETBSD__ 1 +#endif +#if defined(__OpenBSD__) +#undef __OPENBSD__ +#define __OPENBSD__ 1 +#endif +#if defined(__OS2__) || defined(__EMX__) +#undef __OS2__ +#define __OS2__ 1 +#endif +#if defined(osf) || defined(__osf) || defined(__osf__) || defined(_OSF_SOURCE) +#undef __OSF__ +#define __OSF__ 1 +#endif +#if defined(__QNXNTO__) +#undef __QNXNTO__ +#define __QNXNTO__ 1 +#endif +#if defined(riscos) || defined(__riscos) || defined(__riscos__) +#undef __RISCOS__ +#define __RISCOS__ 1 +#endif +#if defined(__sun) && defined(__SVR4) +#undef __SOLARIS__ +#define __SOLARIS__ 1 +#endif + +#if defined(WIN32) || defined(_WIN32) || defined(__CYGWIN__) || defined(__MINGW32__) +/* Try to find out if we're compiling for WinRT, GDK or non-WinRT/GDK */ +#if defined(_MSC_VER) && defined(__has_include) +#if __has_include() +#define HAVE_WINAPIFAMILY_H 1 +#else +#define HAVE_WINAPIFAMILY_H 0 +#endif + +/* If _USING_V110_SDK71_ is defined it means we are using the Windows XP toolset. */ +#elif defined(_MSC_VER) && (_MSC_VER >= 1700 && !_USING_V110_SDK71_) /* _MSC_VER == 1700 for Visual Studio 2012 */ +#define HAVE_WINAPIFAMILY_H 1 +#else +#define HAVE_WINAPIFAMILY_H 0 +#endif + +#if HAVE_WINAPIFAMILY_H +#include +#define WINAPI_FAMILY_WINRT (!WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP) && WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_APP)) +#else +#define WINAPI_FAMILY_WINRT 0 +#endif /* HAVE_WINAPIFAMILY_H */ + +#if WINAPI_FAMILY_WINRT +#undef __WINRT__ +#define __WINRT__ 1 +#elif defined(_GAMING_DESKTOP) /* GDK project configuration always defines _GAMING_XXX */ +#undef __WINGDK__ +#define __WINGDK__ 1 +#elif defined(_GAMING_XBOX_XBOXONE) +#undef __XBOXONE__ +#define __XBOXONE__ 1 +#elif defined(_GAMING_XBOX_SCARLETT) +#undef __XBOXSERIES__ +#define __XBOXSERIES__ 1 +#else +#undef __WINDOWS__ +#define __WINDOWS__ 1 +#endif +#endif /* defined(WIN32) || defined(_WIN32) || defined(__CYGWIN__) */ + +#if defined(__WINDOWS__) +#undef __WIN32__ +#define __WIN32__ 1 +#endif +/* This is to support generic "any GDK" separate from a platform-specific GDK */ +#if defined(__WINGDK__) || defined(__XBOXONE__) || defined(__XBOXSERIES__) +#undef __GDK__ +#define __GDK__ 1 +#endif +#if defined(__PSP__) +#undef __PSP__ +#define __PSP__ 1 +#endif +#if defined(PS2) +#define __PS2__ 1 +#endif + +/* The NACL compiler defines __native_client__ and __pnacl__ + * Ref: http://www.chromium.org/nativeclient/pnacl/stability-of-the-pnacl-bitcode-abi + */ +#if defined(__native_client__) +#undef __LINUX__ +#undef __NACL__ +#define __NACL__ 1 +#endif +#if defined(__pnacl__) +#undef __LINUX__ +#undef __PNACL__ +#define __PNACL__ 1 +/* PNACL with newlib supports static linking only */ +#define __SDL_NOGETPROCADDR__ +#endif + +#if defined(__vita__) +#define __VITA__ 1 +#endif + +#if defined(__3DS__) +#undef __3DS__ +#define __3DS__ 1 +#endif + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * Get the name of the platform. + * + * Here are the names returned for some (but not all) supported platforms: + * + * - "Windows" + * - "Mac OS X" + * - "Linux" + * - "iOS" + * - "Android" + * + * \returns the name of the platform. If the correct platform name is not + * available, returns a string beginning with the text "Unknown". + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC const char * SDLCALL SDL_GetPlatform (void); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_platform_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_power.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_power.h new file mode 100644 index 00000000..1d75704c --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_power.h @@ -0,0 +1,87 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +#ifndef SDL_power_h_ +#define SDL_power_h_ + +/** + * \file SDL_power.h + * + * Header for the SDL power management routines. + */ + +#include "SDL_stdinc.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * The basic state for the system's power supply. + */ +typedef enum +{ + SDL_POWERSTATE_UNKNOWN, /**< cannot determine power status */ + SDL_POWERSTATE_ON_BATTERY, /**< Not plugged in, running on the battery */ + SDL_POWERSTATE_NO_BATTERY, /**< Plugged in, no battery available */ + SDL_POWERSTATE_CHARGING, /**< Plugged in, charging battery */ + SDL_POWERSTATE_CHARGED /**< Plugged in, battery charged */ +} SDL_PowerState; + +/** + * Get the current power supply details. + * + * You should never take a battery status as absolute truth. Batteries + * (especially failing batteries) are delicate hardware, and the values + * reported here are best estimates based on what that hardware reports. It's + * not uncommon for older batteries to lose stored power much faster than it + * reports, or completely drain when reporting it has 20 percent left, etc. + * + * Battery status can change at any time; if you are concerned with power + * state, you should call this function frequently, and perhaps ignore changes + * until they seem to be stable for a few seconds. + * + * It's possible a platform can only report battery percentage or time left + * but not both. + * + * \param seconds seconds of battery life left, you can pass a NULL here if + * you don't care, will return -1 if we can't determine a + * value, or we're not running on a battery + * \param percent percentage of battery life left, between 0 and 100, you can + * pass a NULL here if you don't care, will return -1 if we + * can't determine a value, or we're not running on a battery + * \returns an SDL_PowerState enum representing the current battery state. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC SDL_PowerState SDLCALL SDL_GetPowerInfo(int *seconds, int *percent); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_power_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_quit.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_quit.h new file mode 100644 index 00000000..d8ceb894 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_quit.h @@ -0,0 +1,58 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_quit.h + * + * Include file for SDL quit event handling. + */ + +#ifndef SDL_quit_h_ +#define SDL_quit_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +/** + * \file SDL_quit.h + * + * An ::SDL_QUIT event is generated when the user tries to close the application + * window. If it is ignored or filtered out, the window will remain open. + * If it is not ignored or filtered, it is queued normally and the window + * is allowed to close. When the window is closed, screen updates will + * complete, but have no effect. + * + * SDL_Init() installs signal handlers for SIGINT (keyboard interrupt) + * and SIGTERM (system termination request), if handlers do not already + * exist, that generate ::SDL_QUIT events as well. There is no way + * to determine the cause of an ::SDL_QUIT event, but setting a signal + * handler in your application will override the default generation of + * quit events for that signal. + * + * \sa SDL_Quit() + */ + +/* There are no functions directly affecting the quit event */ + +#define SDL_QuitRequested() \ + (SDL_PumpEvents(), (SDL_PeepEvents(NULL,0,SDL_PEEKEVENT,SDL_QUIT,SDL_QUIT) > 0)) + +#endif /* SDL_quit_h_ */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_rect.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_rect.h new file mode 100644 index 00000000..9611a311 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_rect.h @@ -0,0 +1,376 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_rect.h + * + * Header file for SDL_rect definition and management functions. + */ + +#ifndef SDL_rect_h_ +#define SDL_rect_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_pixels.h" +#include "SDL_rwops.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * The structure that defines a point (integer) + * + * \sa SDL_EnclosePoints + * \sa SDL_PointInRect + */ +typedef struct SDL_Point +{ + int x; + int y; +} SDL_Point; + +/** + * The structure that defines a point (floating point) + * + * \sa SDL_EncloseFPoints + * \sa SDL_PointInFRect + */ +typedef struct SDL_FPoint +{ + float x; + float y; +} SDL_FPoint; + + +/** + * A rectangle, with the origin at the upper left (integer). + * + * \sa SDL_RectEmpty + * \sa SDL_RectEquals + * \sa SDL_HasIntersection + * \sa SDL_IntersectRect + * \sa SDL_IntersectRectAndLine + * \sa SDL_UnionRect + * \sa SDL_EnclosePoints + */ +typedef struct SDL_Rect +{ + int x, y; + int w, h; +} SDL_Rect; + + +/** + * A rectangle, with the origin at the upper left (floating point). + * + * \sa SDL_FRectEmpty + * \sa SDL_FRectEquals + * \sa SDL_FRectEqualsEpsilon + * \sa SDL_HasIntersectionF + * \sa SDL_IntersectFRect + * \sa SDL_IntersectFRectAndLine + * \sa SDL_UnionFRect + * \sa SDL_EncloseFPoints + * \sa SDL_PointInFRect + */ +typedef struct SDL_FRect +{ + float x; + float y; + float w; + float h; +} SDL_FRect; + + +/** + * Returns true if point resides inside a rectangle. + */ +SDL_FORCE_INLINE SDL_bool SDL_PointInRect(const SDL_Point *p, const SDL_Rect *r) +{ + return ( (p->x >= r->x) && (p->x < (r->x + r->w)) && + (p->y >= r->y) && (p->y < (r->y + r->h)) ) ? SDL_TRUE : SDL_FALSE; +} + +/** + * Returns true if the rectangle has no area. + */ +SDL_FORCE_INLINE SDL_bool SDL_RectEmpty(const SDL_Rect *r) +{ + return ((!r) || (r->w <= 0) || (r->h <= 0)) ? SDL_TRUE : SDL_FALSE; +} + +/** + * Returns true if the two rectangles are equal. + */ +SDL_FORCE_INLINE SDL_bool SDL_RectEquals(const SDL_Rect *a, const SDL_Rect *b) +{ + return (a && b && (a->x == b->x) && (a->y == b->y) && + (a->w == b->w) && (a->h == b->h)) ? SDL_TRUE : SDL_FALSE; +} + +/** + * Determine whether two rectangles intersect. + * + * If either pointer is NULL the function will return SDL_FALSE. + * + * \param A an SDL_Rect structure representing the first rectangle + * \param B an SDL_Rect structure representing the second rectangle + * \returns SDL_TRUE if there is an intersection, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_IntersectRect + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasIntersection(const SDL_Rect * A, + const SDL_Rect * B); + +/** + * Calculate the intersection of two rectangles. + * + * If `result` is NULL then this function will return SDL_FALSE. + * + * \param A an SDL_Rect structure representing the first rectangle + * \param B an SDL_Rect structure representing the second rectangle + * \param result an SDL_Rect structure filled in with the intersection of + * rectangles `A` and `B` + * \returns SDL_TRUE if there is an intersection, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HasIntersection + */ +extern DECLSPEC SDL_bool SDLCALL SDL_IntersectRect(const SDL_Rect * A, + const SDL_Rect * B, + SDL_Rect * result); + +/** + * Calculate the union of two rectangles. + * + * \param A an SDL_Rect structure representing the first rectangle + * \param B an SDL_Rect structure representing the second rectangle + * \param result an SDL_Rect structure filled in with the union of rectangles + * `A` and `B` + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC void SDLCALL SDL_UnionRect(const SDL_Rect * A, + const SDL_Rect * B, + SDL_Rect * result); + +/** + * Calculate a minimal rectangle enclosing a set of points. + * + * If `clip` is not NULL then only points inside of the clipping rectangle are + * considered. + * + * \param points an array of SDL_Point structures representing points to be + * enclosed + * \param count the number of structures in the `points` array + * \param clip an SDL_Rect used for clipping or NULL to enclose all points + * \param result an SDL_Rect structure filled in with the minimal enclosing + * rectangle + * \returns SDL_TRUE if any points were enclosed or SDL_FALSE if all the + * points were outside of the clipping rectangle. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_EnclosePoints(const SDL_Point * points, + int count, + const SDL_Rect * clip, + SDL_Rect * result); + +/** + * Calculate the intersection of a rectangle and line segment. + * + * This function is used to clip a line segment to a rectangle. A line segment + * contained entirely within the rectangle or that does not intersect will + * remain unchanged. A line segment that crosses the rectangle at either or + * both ends will be clipped to the boundary of the rectangle and the new + * coordinates saved in `X1`, `Y1`, `X2`, and/or `Y2` as necessary. + * + * \param rect an SDL_Rect structure representing the rectangle to intersect + * \param X1 a pointer to the starting X-coordinate of the line + * \param Y1 a pointer to the starting Y-coordinate of the line + * \param X2 a pointer to the ending X-coordinate of the line + * \param Y2 a pointer to the ending Y-coordinate of the line + * \returns SDL_TRUE if there is an intersection, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_IntersectRectAndLine(const SDL_Rect * + rect, int *X1, + int *Y1, int *X2, + int *Y2); + + +/* SDL_FRect versions... */ + +/** + * Returns true if point resides inside a rectangle. + */ +SDL_FORCE_INLINE SDL_bool SDL_PointInFRect(const SDL_FPoint *p, const SDL_FRect *r) +{ + return ( (p->x >= r->x) && (p->x < (r->x + r->w)) && + (p->y >= r->y) && (p->y < (r->y + r->h)) ) ? SDL_TRUE : SDL_FALSE; +} + +/** + * Returns true if the rectangle has no area. + */ +SDL_FORCE_INLINE SDL_bool SDL_FRectEmpty(const SDL_FRect *r) +{ + return ((!r) || (r->w <= 0.0f) || (r->h <= 0.0f)) ? SDL_TRUE : SDL_FALSE; +} + +/** + * Returns true if the two rectangles are equal, within some given epsilon. + * + * \since This function is available since SDL 2.0.22. + */ +SDL_FORCE_INLINE SDL_bool SDL_FRectEqualsEpsilon(const SDL_FRect *a, const SDL_FRect *b, const float epsilon) +{ + return (a && b && ((a == b) || + ((SDL_fabsf(a->x - b->x) <= epsilon) && + (SDL_fabsf(a->y - b->y) <= epsilon) && + (SDL_fabsf(a->w - b->w) <= epsilon) && + (SDL_fabsf(a->h - b->h) <= epsilon)))) + ? SDL_TRUE : SDL_FALSE; +} + +/** + * Returns true if the two rectangles are equal, using a default epsilon. + * + * \since This function is available since SDL 2.0.22. + */ +SDL_FORCE_INLINE SDL_bool SDL_FRectEquals(const SDL_FRect *a, const SDL_FRect *b) +{ + return SDL_FRectEqualsEpsilon(a, b, SDL_FLT_EPSILON); +} + +/** + * Determine whether two rectangles intersect with float precision. + * + * If either pointer is NULL the function will return SDL_FALSE. + * + * \param A an SDL_FRect structure representing the first rectangle + * \param B an SDL_FRect structure representing the second rectangle + * \returns SDL_TRUE if there is an intersection, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.22. + * + * \sa SDL_IntersectRect + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasIntersectionF(const SDL_FRect * A, + const SDL_FRect * B); + +/** + * Calculate the intersection of two rectangles with float precision. + * + * If `result` is NULL then this function will return SDL_FALSE. + * + * \param A an SDL_FRect structure representing the first rectangle + * \param B an SDL_FRect structure representing the second rectangle + * \param result an SDL_FRect structure filled in with the intersection of + * rectangles `A` and `B` + * \returns SDL_TRUE if there is an intersection, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.22. + * + * \sa SDL_HasIntersectionF + */ +extern DECLSPEC SDL_bool SDLCALL SDL_IntersectFRect(const SDL_FRect * A, + const SDL_FRect * B, + SDL_FRect * result); + +/** + * Calculate the union of two rectangles with float precision. + * + * \param A an SDL_FRect structure representing the first rectangle + * \param B an SDL_FRect structure representing the second rectangle + * \param result an SDL_FRect structure filled in with the union of rectangles + * `A` and `B` + * + * \since This function is available since SDL 2.0.22. + */ +extern DECLSPEC void SDLCALL SDL_UnionFRect(const SDL_FRect * A, + const SDL_FRect * B, + SDL_FRect * result); + +/** + * Calculate a minimal rectangle enclosing a set of points with float + * precision. + * + * If `clip` is not NULL then only points inside of the clipping rectangle are + * considered. + * + * \param points an array of SDL_FPoint structures representing points to be + * enclosed + * \param count the number of structures in the `points` array + * \param clip an SDL_FRect used for clipping or NULL to enclose all points + * \param result an SDL_FRect structure filled in with the minimal enclosing + * rectangle + * \returns SDL_TRUE if any points were enclosed or SDL_FALSE if all the + * points were outside of the clipping rectangle. + * + * \since This function is available since SDL 2.0.22. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_EncloseFPoints(const SDL_FPoint * points, + int count, + const SDL_FRect * clip, + SDL_FRect * result); + +/** + * Calculate the intersection of a rectangle and line segment with float + * precision. + * + * This function is used to clip a line segment to a rectangle. A line segment + * contained entirely within the rectangle or that does not intersect will + * remain unchanged. A line segment that crosses the rectangle at either or + * both ends will be clipped to the boundary of the rectangle and the new + * coordinates saved in `X1`, `Y1`, `X2`, and/or `Y2` as necessary. + * + * \param rect an SDL_FRect structure representing the rectangle to intersect + * \param X1 a pointer to the starting X-coordinate of the line + * \param Y1 a pointer to the starting Y-coordinate of the line + * \param X2 a pointer to the ending X-coordinate of the line + * \param Y2 a pointer to the ending Y-coordinate of the line + * \returns SDL_TRUE if there is an intersection, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.22. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_IntersectFRectAndLine(const SDL_FRect * + rect, float *X1, + float *Y1, float *X2, + float *Y2); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_rect_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_render.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_render.h new file mode 100644 index 00000000..2d3f0736 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_render.h @@ -0,0 +1,1924 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_render.h + * + * Header file for SDL 2D rendering functions. + * + * This API supports the following features: + * * single pixel points + * * single pixel lines + * * filled rectangles + * * texture images + * + * The primitives may be drawn in opaque, blended, or additive modes. + * + * The texture images may be drawn in opaque, blended, or additive modes. + * They can have an additional color tint or alpha modulation applied to + * them, and may also be stretched with linear interpolation. + * + * This API is designed to accelerate simple 2D operations. You may + * want more functionality such as polygons and particle effects and + * in that case you should use SDL's OpenGL/Direct3D support or one + * of the many good 3D engines. + * + * These functions must be called from the main thread. + * See this bug for details: http://bugzilla.libsdl.org/show_bug.cgi?id=1995 + */ + +#ifndef SDL_render_h_ +#define SDL_render_h_ + +#include "SDL_stdinc.h" +#include "SDL_rect.h" +#include "SDL_video.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * Flags used when creating a rendering context + */ +typedef enum +{ + SDL_RENDERER_SOFTWARE = 0x00000001, /**< The renderer is a software fallback */ + SDL_RENDERER_ACCELERATED = 0x00000002, /**< The renderer uses hardware + acceleration */ + SDL_RENDERER_PRESENTVSYNC = 0x00000004, /**< Present is synchronized + with the refresh rate */ + SDL_RENDERER_TARGETTEXTURE = 0x00000008 /**< The renderer supports + rendering to texture */ +} SDL_RendererFlags; + +/** + * Information on the capabilities of a render driver or context. + */ +typedef struct SDL_RendererInfo +{ + const char *name; /**< The name of the renderer */ + Uint32 flags; /**< Supported ::SDL_RendererFlags */ + Uint32 num_texture_formats; /**< The number of available texture formats */ + Uint32 texture_formats[16]; /**< The available texture formats */ + int max_texture_width; /**< The maximum texture width */ + int max_texture_height; /**< The maximum texture height */ +} SDL_RendererInfo; + +/** + * Vertex structure + */ +typedef struct SDL_Vertex +{ + SDL_FPoint position; /**< Vertex position, in SDL_Renderer coordinates */ + SDL_Color color; /**< Vertex color */ + SDL_FPoint tex_coord; /**< Normalized texture coordinates, if needed */ +} SDL_Vertex; + +/** + * The scaling mode for a texture. + */ +typedef enum +{ + SDL_ScaleModeNearest, /**< nearest pixel sampling */ + SDL_ScaleModeLinear, /**< linear filtering */ + SDL_ScaleModeBest /**< anisotropic filtering */ +} SDL_ScaleMode; + +/** + * The access pattern allowed for a texture. + */ +typedef enum +{ + SDL_TEXTUREACCESS_STATIC, /**< Changes rarely, not lockable */ + SDL_TEXTUREACCESS_STREAMING, /**< Changes frequently, lockable */ + SDL_TEXTUREACCESS_TARGET /**< Texture can be used as a render target */ +} SDL_TextureAccess; + +/** + * The texture channel modulation used in SDL_RenderCopy(). + */ +typedef enum +{ + SDL_TEXTUREMODULATE_NONE = 0x00000000, /**< No modulation */ + SDL_TEXTUREMODULATE_COLOR = 0x00000001, /**< srcC = srcC * color */ + SDL_TEXTUREMODULATE_ALPHA = 0x00000002 /**< srcA = srcA * alpha */ +} SDL_TextureModulate; + +/** + * Flip constants for SDL_RenderCopyEx + */ +typedef enum +{ + SDL_FLIP_NONE = 0x00000000, /**< Do not flip */ + SDL_FLIP_HORIZONTAL = 0x00000001, /**< flip horizontally */ + SDL_FLIP_VERTICAL = 0x00000002 /**< flip vertically */ +} SDL_RendererFlip; + +/** + * A structure representing rendering state + */ +struct SDL_Renderer; +typedef struct SDL_Renderer SDL_Renderer; + +/** + * An efficient driver-specific representation of pixel data + */ +struct SDL_Texture; +typedef struct SDL_Texture SDL_Texture; + +/* Function prototypes */ + +/** + * Get the number of 2D rendering drivers available for the current display. + * + * A render driver is a set of code that handles rendering and texture + * management on a particular display. Normally there is only one, but some + * drivers may have several available with different capabilities. + * + * There may be none if SDL was compiled without render support. + * + * \returns a number >= 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateRenderer + * \sa SDL_GetRenderDriverInfo + */ +extern DECLSPEC int SDLCALL SDL_GetNumRenderDrivers(void); + +/** + * Get info about a specific 2D rendering driver for the current display. + * + * \param index the index of the driver to query information about + * \param info an SDL_RendererInfo structure to be filled with information on + * the rendering driver + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateRenderer + * \sa SDL_GetNumRenderDrivers + */ +extern DECLSPEC int SDLCALL SDL_GetRenderDriverInfo(int index, + SDL_RendererInfo * info); + +/** + * Create a window and default renderer. + * + * \param width the width of the window + * \param height the height of the window + * \param window_flags the flags used to create the window (see + * SDL_CreateWindow()) + * \param window a pointer filled with the window, or NULL on error + * \param renderer a pointer filled with the renderer, or NULL on error + * \returns 0 on success, or -1 on error; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateRenderer + * \sa SDL_CreateWindow + */ +extern DECLSPEC int SDLCALL SDL_CreateWindowAndRenderer( + int width, int height, Uint32 window_flags, + SDL_Window **window, SDL_Renderer **renderer); + + +/** + * Create a 2D rendering context for a window. + * + * \param window the window where rendering is displayed + * \param index the index of the rendering driver to initialize, or -1 to + * initialize the first one supporting the requested flags + * \param flags 0, or one or more SDL_RendererFlags OR'd together + * \returns a valid rendering context or NULL if there was an error; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateSoftwareRenderer + * \sa SDL_DestroyRenderer + * \sa SDL_GetNumRenderDrivers + * \sa SDL_GetRendererInfo + */ +extern DECLSPEC SDL_Renderer * SDLCALL SDL_CreateRenderer(SDL_Window * window, + int index, Uint32 flags); + +/** + * Create a 2D software rendering context for a surface. + * + * Two other API which can be used to create SDL_Renderer: + * SDL_CreateRenderer() and SDL_CreateWindowAndRenderer(). These can _also_ + * create a software renderer, but they are intended to be used with an + * SDL_Window as the final destination and not an SDL_Surface. + * + * \param surface the SDL_Surface structure representing the surface where + * rendering is done + * \returns a valid rendering context or NULL if there was an error; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateRenderer + * \sa SDL_CreateWindowRenderer + * \sa SDL_DestroyRenderer + */ +extern DECLSPEC SDL_Renderer * SDLCALL SDL_CreateSoftwareRenderer(SDL_Surface * surface); + +/** + * Get the renderer associated with a window. + * + * \param window the window to query + * \returns the rendering context on success or NULL on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateRenderer + */ +extern DECLSPEC SDL_Renderer * SDLCALL SDL_GetRenderer(SDL_Window * window); + +/** + * Get the window associated with a renderer. + * + * \param renderer the renderer to query + * \returns the window on success or NULL on failure; call SDL_GetError() for + * more information. + * + * \since This function is available since SDL 2.0.22. + */ +extern DECLSPEC SDL_Window * SDLCALL SDL_RenderGetWindow(SDL_Renderer *renderer); + +/** + * Get information about a rendering context. + * + * \param renderer the rendering context + * \param info an SDL_RendererInfo structure filled with information about the + * current renderer + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateRenderer + */ +extern DECLSPEC int SDLCALL SDL_GetRendererInfo(SDL_Renderer * renderer, + SDL_RendererInfo * info); + +/** + * Get the output size in pixels of a rendering context. + * + * Due to high-dpi displays, you might end up with a rendering context that + * has more pixels than the window that contains it, so use this instead of + * SDL_GetWindowSize() to decide how much drawing area you have. + * + * \param renderer the rendering context + * \param w an int filled with the width + * \param h an int filled with the height + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetRenderer + */ +extern DECLSPEC int SDLCALL SDL_GetRendererOutputSize(SDL_Renderer * renderer, + int *w, int *h); + +/** + * Create a texture for a rendering context. + * + * You can set the texture scaling method by setting + * `SDL_HINT_RENDER_SCALE_QUALITY` before creating the texture. + * + * \param renderer the rendering context + * \param format one of the enumerated values in SDL_PixelFormatEnum + * \param access one of the enumerated values in SDL_TextureAccess + * \param w the width of the texture in pixels + * \param h the height of the texture in pixels + * \returns a pointer to the created texture or NULL if no rendering context + * was active, the format was unsupported, or the width or height + * were out of range; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateTextureFromSurface + * \sa SDL_DestroyTexture + * \sa SDL_QueryTexture + * \sa SDL_UpdateTexture + */ +extern DECLSPEC SDL_Texture * SDLCALL SDL_CreateTexture(SDL_Renderer * renderer, + Uint32 format, + int access, int w, + int h); + +/** + * Create a texture from an existing surface. + * + * The surface is not modified or freed by this function. + * + * The SDL_TextureAccess hint for the created texture is + * `SDL_TEXTUREACCESS_STATIC`. + * + * The pixel format of the created texture may be different from the pixel + * format of the surface. Use SDL_QueryTexture() to query the pixel format of + * the texture. + * + * \param renderer the rendering context + * \param surface the SDL_Surface structure containing pixel data used to fill + * the texture + * \returns the created texture or NULL on failure; call SDL_GetError() for + * more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateTexture + * \sa SDL_DestroyTexture + * \sa SDL_QueryTexture + */ +extern DECLSPEC SDL_Texture * SDLCALL SDL_CreateTextureFromSurface(SDL_Renderer * renderer, SDL_Surface * surface); + +/** + * Query the attributes of a texture. + * + * \param texture the texture to query + * \param format a pointer filled in with the raw format of the texture; the + * actual format may differ, but pixel transfers will use this + * format (one of the SDL_PixelFormatEnum values). This argument + * can be NULL if you don't need this information. + * \param access a pointer filled in with the actual access to the texture + * (one of the SDL_TextureAccess values). This argument can be + * NULL if you don't need this information. + * \param w a pointer filled in with the width of the texture in pixels. This + * argument can be NULL if you don't need this information. + * \param h a pointer filled in with the height of the texture in pixels. This + * argument can be NULL if you don't need this information. + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateTexture + */ +extern DECLSPEC int SDLCALL SDL_QueryTexture(SDL_Texture * texture, + Uint32 * format, int *access, + int *w, int *h); + +/** + * Set an additional color value multiplied into render copy operations. + * + * When this texture is rendered, during the copy operation each source color + * channel is modulated by the appropriate color value according to the + * following formula: + * + * `srcC = srcC * (color / 255)` + * + * Color modulation is not always supported by the renderer; it will return -1 + * if color modulation is not supported. + * + * \param texture the texture to update + * \param r the red color value multiplied into copy operations + * \param g the green color value multiplied into copy operations + * \param b the blue color value multiplied into copy operations + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetTextureColorMod + * \sa SDL_SetTextureAlphaMod + */ +extern DECLSPEC int SDLCALL SDL_SetTextureColorMod(SDL_Texture * texture, + Uint8 r, Uint8 g, Uint8 b); + + +/** + * Get the additional color value multiplied into render copy operations. + * + * \param texture the texture to query + * \param r a pointer filled in with the current red color value + * \param g a pointer filled in with the current green color value + * \param b a pointer filled in with the current blue color value + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetTextureAlphaMod + * \sa SDL_SetTextureColorMod + */ +extern DECLSPEC int SDLCALL SDL_GetTextureColorMod(SDL_Texture * texture, + Uint8 * r, Uint8 * g, + Uint8 * b); + +/** + * Set an additional alpha value multiplied into render copy operations. + * + * When this texture is rendered, during the copy operation the source alpha + * value is modulated by this alpha value according to the following formula: + * + * `srcA = srcA * (alpha / 255)` + * + * Alpha modulation is not always supported by the renderer; it will return -1 + * if alpha modulation is not supported. + * + * \param texture the texture to update + * \param alpha the source alpha value multiplied into copy operations + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetTextureAlphaMod + * \sa SDL_SetTextureColorMod + */ +extern DECLSPEC int SDLCALL SDL_SetTextureAlphaMod(SDL_Texture * texture, + Uint8 alpha); + +/** + * Get the additional alpha value multiplied into render copy operations. + * + * \param texture the texture to query + * \param alpha a pointer filled in with the current alpha value + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetTextureColorMod + * \sa SDL_SetTextureAlphaMod + */ +extern DECLSPEC int SDLCALL SDL_GetTextureAlphaMod(SDL_Texture * texture, + Uint8 * alpha); + +/** + * Set the blend mode for a texture, used by SDL_RenderCopy(). + * + * If the blend mode is not supported, the closest supported mode is chosen + * and this function returns -1. + * + * \param texture the texture to update + * \param blendMode the SDL_BlendMode to use for texture blending + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetTextureBlendMode + * \sa SDL_RenderCopy + */ +extern DECLSPEC int SDLCALL SDL_SetTextureBlendMode(SDL_Texture * texture, + SDL_BlendMode blendMode); + +/** + * Get the blend mode used for texture copy operations. + * + * \param texture the texture to query + * \param blendMode a pointer filled in with the current SDL_BlendMode + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetTextureBlendMode + */ +extern DECLSPEC int SDLCALL SDL_GetTextureBlendMode(SDL_Texture * texture, + SDL_BlendMode *blendMode); + +/** + * Set the scale mode used for texture scale operations. + * + * If the scale mode is not supported, the closest supported mode is chosen. + * + * \param texture The texture to update. + * \param scaleMode the SDL_ScaleMode to use for texture scaling. + * \returns 0 on success, or -1 if the texture is not valid. + * + * \since This function is available since SDL 2.0.12. + * + * \sa SDL_GetTextureScaleMode + */ +extern DECLSPEC int SDLCALL SDL_SetTextureScaleMode(SDL_Texture * texture, + SDL_ScaleMode scaleMode); + +/** + * Get the scale mode used for texture scale operations. + * + * \param texture the texture to query. + * \param scaleMode a pointer filled in with the current scale mode. + * \return 0 on success, or -1 if the texture is not valid. + * + * \since This function is available since SDL 2.0.12. + * + * \sa SDL_SetTextureScaleMode + */ +extern DECLSPEC int SDLCALL SDL_GetTextureScaleMode(SDL_Texture * texture, + SDL_ScaleMode *scaleMode); + +/** + * Associate a user-specified pointer with a texture. + * + * \param texture the texture to update. + * \param userdata the pointer to associate with the texture. + * \returns 0 on success, or -1 if the texture is not valid. + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_GetTextureUserData + */ +extern DECLSPEC int SDLCALL SDL_SetTextureUserData(SDL_Texture * texture, + void *userdata); + +/** + * Get the user-specified pointer associated with a texture + * + * \param texture the texture to query. + * \return the pointer associated with the texture, or NULL if the texture is + * not valid. + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_SetTextureUserData + */ +extern DECLSPEC void * SDLCALL SDL_GetTextureUserData(SDL_Texture * texture); + +/** + * Update the given texture rectangle with new pixel data. + * + * The pixel data must be in the pixel format of the texture. Use + * SDL_QueryTexture() to query the pixel format of the texture. + * + * This is a fairly slow function, intended for use with static textures that + * do not change often. + * + * If the texture is intended to be updated often, it is preferred to create + * the texture as streaming and use the locking functions referenced below. + * While this function will work with streaming textures, for optimization + * reasons you may not get the pixels back if you lock the texture afterward. + * + * \param texture the texture to update + * \param rect an SDL_Rect structure representing the area to update, or NULL + * to update the entire texture + * \param pixels the raw pixel data in the format of the texture + * \param pitch the number of bytes in a row of pixel data, including padding + * between lines + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateTexture + * \sa SDL_LockTexture + * \sa SDL_UnlockTexture + */ +extern DECLSPEC int SDLCALL SDL_UpdateTexture(SDL_Texture * texture, + const SDL_Rect * rect, + const void *pixels, int pitch); + +/** + * Update a rectangle within a planar YV12 or IYUV texture with new pixel + * data. + * + * You can use SDL_UpdateTexture() as long as your pixel data is a contiguous + * block of Y and U/V planes in the proper order, but this function is + * available if your pixel data is not contiguous. + * + * \param texture the texture to update + * \param rect a pointer to the rectangle of pixels to update, or NULL to + * update the entire texture + * \param Yplane the raw pixel data for the Y plane + * \param Ypitch the number of bytes between rows of pixel data for the Y + * plane + * \param Uplane the raw pixel data for the U plane + * \param Upitch the number of bytes between rows of pixel data for the U + * plane + * \param Vplane the raw pixel data for the V plane + * \param Vpitch the number of bytes between rows of pixel data for the V + * plane + * \returns 0 on success or -1 if the texture is not valid; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.1. + * + * \sa SDL_UpdateTexture + */ +extern DECLSPEC int SDLCALL SDL_UpdateYUVTexture(SDL_Texture * texture, + const SDL_Rect * rect, + const Uint8 *Yplane, int Ypitch, + const Uint8 *Uplane, int Upitch, + const Uint8 *Vplane, int Vpitch); + +/** + * Update a rectangle within a planar NV12 or NV21 texture with new pixels. + * + * You can use SDL_UpdateTexture() as long as your pixel data is a contiguous + * block of NV12/21 planes in the proper order, but this function is available + * if your pixel data is not contiguous. + * + * \param texture the texture to update + * \param rect a pointer to the rectangle of pixels to update, or NULL to + * update the entire texture. + * \param Yplane the raw pixel data for the Y plane. + * \param Ypitch the number of bytes between rows of pixel data for the Y + * plane. + * \param UVplane the raw pixel data for the UV plane. + * \param UVpitch the number of bytes between rows of pixel data for the UV + * plane. + * \return 0 on success, or -1 if the texture is not valid. + * + * \since This function is available since SDL 2.0.16. + */ +extern DECLSPEC int SDLCALL SDL_UpdateNVTexture(SDL_Texture * texture, + const SDL_Rect * rect, + const Uint8 *Yplane, int Ypitch, + const Uint8 *UVplane, int UVpitch); + +/** + * Lock a portion of the texture for **write-only** pixel access. + * + * As an optimization, the pixels made available for editing don't necessarily + * contain the old texture data. This is a write-only operation, and if you + * need to keep a copy of the texture data you should do that at the + * application level. + * + * You must use SDL_UnlockTexture() to unlock the pixels and apply any + * changes. + * + * \param texture the texture to lock for access, which was created with + * `SDL_TEXTUREACCESS_STREAMING` + * \param rect an SDL_Rect structure representing the area to lock for access; + * NULL to lock the entire texture + * \param pixels this is filled in with a pointer to the locked pixels, + * appropriately offset by the locked area + * \param pitch this is filled in with the pitch of the locked pixels; the + * pitch is the length of one row in bytes + * \returns 0 on success or a negative error code if the texture is not valid + * or was not created with `SDL_TEXTUREACCESS_STREAMING`; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_UnlockTexture + */ +extern DECLSPEC int SDLCALL SDL_LockTexture(SDL_Texture * texture, + const SDL_Rect * rect, + void **pixels, int *pitch); + +/** + * Lock a portion of the texture for **write-only** pixel access, and expose + * it as a SDL surface. + * + * Besides providing an SDL_Surface instead of raw pixel data, this function + * operates like SDL_LockTexture. + * + * As an optimization, the pixels made available for editing don't necessarily + * contain the old texture data. This is a write-only operation, and if you + * need to keep a copy of the texture data you should do that at the + * application level. + * + * You must use SDL_UnlockTexture() to unlock the pixels and apply any + * changes. + * + * The returned surface is freed internally after calling SDL_UnlockTexture() + * or SDL_DestroyTexture(). The caller should not free it. + * + * \param texture the texture to lock for access, which was created with + * `SDL_TEXTUREACCESS_STREAMING` + * \param rect a pointer to the rectangle to lock for access. If the rect is + * NULL, the entire texture will be locked + * \param surface this is filled in with an SDL surface representing the + * locked area + * \returns 0 on success, or -1 if the texture is not valid or was not created + * with `SDL_TEXTUREACCESS_STREAMING` + * + * \since This function is available since SDL 2.0.12. + * + * \sa SDL_LockTexture + * \sa SDL_UnlockTexture + */ +extern DECLSPEC int SDLCALL SDL_LockTextureToSurface(SDL_Texture *texture, + const SDL_Rect *rect, + SDL_Surface **surface); + +/** + * Unlock a texture, uploading the changes to video memory, if needed. + * + * **Warning**: Please note that SDL_LockTexture() is intended to be + * write-only; it will not guarantee the previous contents of the texture will + * be provided. You must fully initialize any area of a texture that you lock + * before unlocking it, as the pixels might otherwise be uninitialized memory. + * + * Which is to say: locking and immediately unlocking a texture can result in + * corrupted textures, depending on the renderer in use. + * + * \param texture a texture locked by SDL_LockTexture() + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LockTexture + */ +extern DECLSPEC void SDLCALL SDL_UnlockTexture(SDL_Texture * texture); + +/** + * Determine whether a renderer supports the use of render targets. + * + * \param renderer the renderer that will be checked + * \returns SDL_TRUE if supported or SDL_FALSE if not. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetRenderTarget + */ +extern DECLSPEC SDL_bool SDLCALL SDL_RenderTargetSupported(SDL_Renderer *renderer); + +/** + * Set a texture as the current rendering target. + * + * Before using this function, you should check the + * `SDL_RENDERER_TARGETTEXTURE` bit in the flags of SDL_RendererInfo to see if + * render targets are supported. + * + * The default render target is the window for which the renderer was created. + * To stop rendering to a texture and render to the window again, call this + * function with a NULL `texture`. + * + * \param renderer the rendering context + * \param texture the targeted texture, which must be created with the + * `SDL_TEXTUREACCESS_TARGET` flag, or NULL to render to the + * window instead of a texture. + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetRenderTarget + */ +extern DECLSPEC int SDLCALL SDL_SetRenderTarget(SDL_Renderer *renderer, + SDL_Texture *texture); + +/** + * Get the current render target. + * + * The default render target is the window for which the renderer was created, + * and is reported a NULL here. + * + * \param renderer the rendering context + * \returns the current render target or NULL for the default render target. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetRenderTarget + */ +extern DECLSPEC SDL_Texture * SDLCALL SDL_GetRenderTarget(SDL_Renderer *renderer); + +/** + * Set a device independent resolution for rendering. + * + * This function uses the viewport and scaling functionality to allow a fixed + * logical resolution for rendering, regardless of the actual output + * resolution. If the actual output resolution doesn't have the same aspect + * ratio the output rendering will be centered within the output display. + * + * If the output display is a window, mouse and touch events in the window + * will be filtered and scaled so they seem to arrive within the logical + * resolution. The SDL_HINT_MOUSE_RELATIVE_SCALING hint controls whether + * relative motion events are also scaled. + * + * If this function results in scaling or subpixel drawing by the rendering + * backend, it will be handled using the appropriate quality hints. + * + * \param renderer the renderer for which resolution should be set + * \param w the width of the logical resolution + * \param h the height of the logical resolution + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderGetLogicalSize + */ +extern DECLSPEC int SDLCALL SDL_RenderSetLogicalSize(SDL_Renderer * renderer, int w, int h); + +/** + * Get device independent resolution for rendering. + * + * When using the main rendering target (eg no target texture is set): this + * may return 0 for `w` and `h` if the SDL_Renderer has never had its logical + * size set by SDL_RenderSetLogicalSize(). Otherwise it returns the logical + * width and height. + * + * When using a target texture: Never return 0 for `w` and `h` at first. Then + * it returns the logical width and height that are set. + * + * \param renderer a rendering context + * \param w an int to be filled with the width + * \param h an int to be filled with the height + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderSetLogicalSize + */ +extern DECLSPEC void SDLCALL SDL_RenderGetLogicalSize(SDL_Renderer * renderer, int *w, int *h); + +/** + * Set whether to force integer scales for resolution-independent rendering. + * + * This function restricts the logical viewport to integer values - that is, + * when a resolution is between two multiples of a logical size, the viewport + * size is rounded down to the lower multiple. + * + * \param renderer the renderer for which integer scaling should be set + * \param enable enable or disable the integer scaling for rendering + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.5. + * + * \sa SDL_RenderGetIntegerScale + * \sa SDL_RenderSetLogicalSize + */ +extern DECLSPEC int SDLCALL SDL_RenderSetIntegerScale(SDL_Renderer * renderer, + SDL_bool enable); + +/** + * Get whether integer scales are forced for resolution-independent rendering. + * + * \param renderer the renderer from which integer scaling should be queried + * \returns SDL_TRUE if integer scales are forced or SDL_FALSE if not and on + * failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.5. + * + * \sa SDL_RenderSetIntegerScale + */ +extern DECLSPEC SDL_bool SDLCALL SDL_RenderGetIntegerScale(SDL_Renderer * renderer); + +/** + * Set the drawing area for rendering on the current target. + * + * When the window is resized, the viewport is reset to fill the entire new + * window size. + * + * \param renderer the rendering context + * \param rect the SDL_Rect structure representing the drawing area, or NULL + * to set the viewport to the entire target + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderGetViewport + */ +extern DECLSPEC int SDLCALL SDL_RenderSetViewport(SDL_Renderer * renderer, + const SDL_Rect * rect); + +/** + * Get the drawing area for the current target. + * + * \param renderer the rendering context + * \param rect an SDL_Rect structure filled in with the current drawing area + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderSetViewport + */ +extern DECLSPEC void SDLCALL SDL_RenderGetViewport(SDL_Renderer * renderer, + SDL_Rect * rect); + +/** + * Set the clip rectangle for rendering on the specified target. + * + * \param renderer the rendering context for which clip rectangle should be + * set + * \param rect an SDL_Rect structure representing the clip area, relative to + * the viewport, or NULL to disable clipping + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderGetClipRect + * \sa SDL_RenderIsClipEnabled + */ +extern DECLSPEC int SDLCALL SDL_RenderSetClipRect(SDL_Renderer * renderer, + const SDL_Rect * rect); + +/** + * Get the clip rectangle for the current target. + * + * \param renderer the rendering context from which clip rectangle should be + * queried + * \param rect an SDL_Rect structure filled in with the current clipping area + * or an empty rectangle if clipping is disabled + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderIsClipEnabled + * \sa SDL_RenderSetClipRect + */ +extern DECLSPEC void SDLCALL SDL_RenderGetClipRect(SDL_Renderer * renderer, + SDL_Rect * rect); + +/** + * Get whether clipping is enabled on the given renderer. + * + * \param renderer the renderer from which clip state should be queried + * \returns SDL_TRUE if clipping is enabled or SDL_FALSE if not; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.4. + * + * \sa SDL_RenderGetClipRect + * \sa SDL_RenderSetClipRect + */ +extern DECLSPEC SDL_bool SDLCALL SDL_RenderIsClipEnabled(SDL_Renderer * renderer); + + +/** + * Set the drawing scale for rendering on the current target. + * + * The drawing coordinates are scaled by the x/y scaling factors before they + * are used by the renderer. This allows resolution independent drawing with a + * single coordinate system. + * + * If this results in scaling or subpixel drawing by the rendering backend, it + * will be handled using the appropriate quality hints. For best results use + * integer scaling factors. + * + * \param renderer a rendering context + * \param scaleX the horizontal scaling factor + * \param scaleY the vertical scaling factor + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderGetScale + * \sa SDL_RenderSetLogicalSize + */ +extern DECLSPEC int SDLCALL SDL_RenderSetScale(SDL_Renderer * renderer, + float scaleX, float scaleY); + +/** + * Get the drawing scale for the current target. + * + * \param renderer the renderer from which drawing scale should be queried + * \param scaleX a pointer filled in with the horizontal scaling factor + * \param scaleY a pointer filled in with the vertical scaling factor + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderSetScale + */ +extern DECLSPEC void SDLCALL SDL_RenderGetScale(SDL_Renderer * renderer, + float *scaleX, float *scaleY); + +/** + * Get logical coordinates of point in renderer when given real coordinates of + * point in window. + * + * Logical coordinates will differ from real coordinates when render is scaled + * and logical renderer size set + * + * \param renderer the renderer from which the logical coordinates should be + * calculated + * \param windowX the real X coordinate in the window + * \param windowY the real Y coordinate in the window + * \param logicalX the pointer filled with the logical x coordinate + * \param logicalY the pointer filled with the logical y coordinate + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_RenderGetScale + * \sa SDL_RenderSetScale + * \sa SDL_RenderGetLogicalSize + * \sa SDL_RenderSetLogicalSize + */ +extern DECLSPEC void SDLCALL SDL_RenderWindowToLogical(SDL_Renderer * renderer, + int windowX, int windowY, + float *logicalX, float *logicalY); + + +/** + * Get real coordinates of point in window when given logical coordinates of + * point in renderer. + * + * Logical coordinates will differ from real coordinates when render is scaled + * and logical renderer size set + * + * \param renderer the renderer from which the window coordinates should be + * calculated + * \param logicalX the logical x coordinate + * \param logicalY the logical y coordinate + * \param windowX the pointer filled with the real X coordinate in the window + * \param windowY the pointer filled with the real Y coordinate in the window + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_RenderGetScale + * \sa SDL_RenderSetScale + * \sa SDL_RenderGetLogicalSize + * \sa SDL_RenderSetLogicalSize + */ +extern DECLSPEC void SDLCALL SDL_RenderLogicalToWindow(SDL_Renderer * renderer, + float logicalX, float logicalY, + int *windowX, int *windowY); + +/** + * Set the color used for drawing operations (Rect, Line and Clear). + * + * Set the color for drawing or filling rectangles, lines, and points, and for + * SDL_RenderClear(). + * + * \param renderer the rendering context + * \param r the red value used to draw on the rendering target + * \param g the green value used to draw on the rendering target + * \param b the blue value used to draw on the rendering target + * \param a the alpha value used to draw on the rendering target; usually + * `SDL_ALPHA_OPAQUE` (255). Use SDL_SetRenderDrawBlendMode to + * specify how the alpha channel is used + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetRenderDrawColor + * \sa SDL_RenderClear + * \sa SDL_RenderDrawLine + * \sa SDL_RenderDrawLines + * \sa SDL_RenderDrawPoint + * \sa SDL_RenderDrawPoints + * \sa SDL_RenderDrawRect + * \sa SDL_RenderDrawRects + * \sa SDL_RenderFillRect + * \sa SDL_RenderFillRects + */ +extern DECLSPEC int SDLCALL SDL_SetRenderDrawColor(SDL_Renderer * renderer, + Uint8 r, Uint8 g, Uint8 b, + Uint8 a); + +/** + * Get the color used for drawing operations (Rect, Line and Clear). + * + * \param renderer the rendering context + * \param r a pointer filled in with the red value used to draw on the + * rendering target + * \param g a pointer filled in with the green value used to draw on the + * rendering target + * \param b a pointer filled in with the blue value used to draw on the + * rendering target + * \param a a pointer filled in with the alpha value used to draw on the + * rendering target; usually `SDL_ALPHA_OPAQUE` (255) + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetRenderDrawColor + */ +extern DECLSPEC int SDLCALL SDL_GetRenderDrawColor(SDL_Renderer * renderer, + Uint8 * r, Uint8 * g, Uint8 * b, + Uint8 * a); + +/** + * Set the blend mode used for drawing operations (Fill and Line). + * + * If the blend mode is not supported, the closest supported mode is chosen. + * + * \param renderer the rendering context + * \param blendMode the SDL_BlendMode to use for blending + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetRenderDrawBlendMode + * \sa SDL_RenderDrawLine + * \sa SDL_RenderDrawLines + * \sa SDL_RenderDrawPoint + * \sa SDL_RenderDrawPoints + * \sa SDL_RenderDrawRect + * \sa SDL_RenderDrawRects + * \sa SDL_RenderFillRect + * \sa SDL_RenderFillRects + */ +extern DECLSPEC int SDLCALL SDL_SetRenderDrawBlendMode(SDL_Renderer * renderer, + SDL_BlendMode blendMode); + +/** + * Get the blend mode used for drawing operations. + * + * \param renderer the rendering context + * \param blendMode a pointer filled in with the current SDL_BlendMode + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetRenderDrawBlendMode + */ +extern DECLSPEC int SDLCALL SDL_GetRenderDrawBlendMode(SDL_Renderer * renderer, + SDL_BlendMode *blendMode); + +/** + * Clear the current rendering target with the drawing color. + * + * This function clears the entire rendering target, ignoring the viewport and + * the clip rectangle. + * + * \param renderer the rendering context + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetRenderDrawColor + */ +extern DECLSPEC int SDLCALL SDL_RenderClear(SDL_Renderer * renderer); + +/** + * Draw a point on the current rendering target. + * + * SDL_RenderDrawPoint() draws a single point. If you want to draw multiple, + * use SDL_RenderDrawPoints() instead. + * + * \param renderer the rendering context + * \param x the x coordinate of the point + * \param y the y coordinate of the point + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderDrawLine + * \sa SDL_RenderDrawLines + * \sa SDL_RenderDrawPoints + * \sa SDL_RenderDrawRect + * \sa SDL_RenderDrawRects + * \sa SDL_RenderFillRect + * \sa SDL_RenderFillRects + * \sa SDL_RenderPresent + * \sa SDL_SetRenderDrawBlendMode + * \sa SDL_SetRenderDrawColor + */ +extern DECLSPEC int SDLCALL SDL_RenderDrawPoint(SDL_Renderer * renderer, + int x, int y); + +/** + * Draw multiple points on the current rendering target. + * + * \param renderer the rendering context + * \param points an array of SDL_Point structures that represent the points to + * draw + * \param count the number of points to draw + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderDrawLine + * \sa SDL_RenderDrawLines + * \sa SDL_RenderDrawPoint + * \sa SDL_RenderDrawRect + * \sa SDL_RenderDrawRects + * \sa SDL_RenderFillRect + * \sa SDL_RenderFillRects + * \sa SDL_RenderPresent + * \sa SDL_SetRenderDrawBlendMode + * \sa SDL_SetRenderDrawColor + */ +extern DECLSPEC int SDLCALL SDL_RenderDrawPoints(SDL_Renderer * renderer, + const SDL_Point * points, + int count); + +/** + * Draw a line on the current rendering target. + * + * SDL_RenderDrawLine() draws the line to include both end points. If you want + * to draw multiple, connecting lines use SDL_RenderDrawLines() instead. + * + * \param renderer the rendering context + * \param x1 the x coordinate of the start point + * \param y1 the y coordinate of the start point + * \param x2 the x coordinate of the end point + * \param y2 the y coordinate of the end point + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderDrawLines + * \sa SDL_RenderDrawPoint + * \sa SDL_RenderDrawPoints + * \sa SDL_RenderDrawRect + * \sa SDL_RenderDrawRects + * \sa SDL_RenderFillRect + * \sa SDL_RenderFillRects + * \sa SDL_RenderPresent + * \sa SDL_SetRenderDrawBlendMode + * \sa SDL_SetRenderDrawColor + */ +extern DECLSPEC int SDLCALL SDL_RenderDrawLine(SDL_Renderer * renderer, + int x1, int y1, int x2, int y2); + +/** + * Draw a series of connected lines on the current rendering target. + * + * \param renderer the rendering context + * \param points an array of SDL_Point structures representing points along + * the lines + * \param count the number of points, drawing count-1 lines + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderDrawLine + * \sa SDL_RenderDrawPoint + * \sa SDL_RenderDrawPoints + * \sa SDL_RenderDrawRect + * \sa SDL_RenderDrawRects + * \sa SDL_RenderFillRect + * \sa SDL_RenderFillRects + * \sa SDL_RenderPresent + * \sa SDL_SetRenderDrawBlendMode + * \sa SDL_SetRenderDrawColor + */ +extern DECLSPEC int SDLCALL SDL_RenderDrawLines(SDL_Renderer * renderer, + const SDL_Point * points, + int count); + +/** + * Draw a rectangle on the current rendering target. + * + * \param renderer the rendering context + * \param rect an SDL_Rect structure representing the rectangle to draw, or + * NULL to outline the entire rendering target + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderDrawLine + * \sa SDL_RenderDrawLines + * \sa SDL_RenderDrawPoint + * \sa SDL_RenderDrawPoints + * \sa SDL_RenderDrawRects + * \sa SDL_RenderFillRect + * \sa SDL_RenderFillRects + * \sa SDL_RenderPresent + * \sa SDL_SetRenderDrawBlendMode + * \sa SDL_SetRenderDrawColor + */ +extern DECLSPEC int SDLCALL SDL_RenderDrawRect(SDL_Renderer * renderer, + const SDL_Rect * rect); + +/** + * Draw some number of rectangles on the current rendering target. + * + * \param renderer the rendering context + * \param rects an array of SDL_Rect structures representing the rectangles to + * be drawn + * \param count the number of rectangles + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderDrawLine + * \sa SDL_RenderDrawLines + * \sa SDL_RenderDrawPoint + * \sa SDL_RenderDrawPoints + * \sa SDL_RenderDrawRect + * \sa SDL_RenderFillRect + * \sa SDL_RenderFillRects + * \sa SDL_RenderPresent + * \sa SDL_SetRenderDrawBlendMode + * \sa SDL_SetRenderDrawColor + */ +extern DECLSPEC int SDLCALL SDL_RenderDrawRects(SDL_Renderer * renderer, + const SDL_Rect * rects, + int count); + +/** + * Fill a rectangle on the current rendering target with the drawing color. + * + * The current drawing color is set by SDL_SetRenderDrawColor(), and the + * color's alpha value is ignored unless blending is enabled with the + * appropriate call to SDL_SetRenderDrawBlendMode(). + * + * \param renderer the rendering context + * \param rect the SDL_Rect structure representing the rectangle to fill, or + * NULL for the entire rendering target + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderDrawLine + * \sa SDL_RenderDrawLines + * \sa SDL_RenderDrawPoint + * \sa SDL_RenderDrawPoints + * \sa SDL_RenderDrawRect + * \sa SDL_RenderDrawRects + * \sa SDL_RenderFillRects + * \sa SDL_RenderPresent + * \sa SDL_SetRenderDrawBlendMode + * \sa SDL_SetRenderDrawColor + */ +extern DECLSPEC int SDLCALL SDL_RenderFillRect(SDL_Renderer * renderer, + const SDL_Rect * rect); + +/** + * Fill some number of rectangles on the current rendering target with the + * drawing color. + * + * \param renderer the rendering context + * \param rects an array of SDL_Rect structures representing the rectangles to + * be filled + * \param count the number of rectangles + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderDrawLine + * \sa SDL_RenderDrawLines + * \sa SDL_RenderDrawPoint + * \sa SDL_RenderDrawPoints + * \sa SDL_RenderDrawRect + * \sa SDL_RenderDrawRects + * \sa SDL_RenderFillRect + * \sa SDL_RenderPresent + */ +extern DECLSPEC int SDLCALL SDL_RenderFillRects(SDL_Renderer * renderer, + const SDL_Rect * rects, + int count); + +/** + * Copy a portion of the texture to the current rendering target. + * + * The texture is blended with the destination based on its blend mode set + * with SDL_SetTextureBlendMode(). + * + * The texture color is affected based on its color modulation set by + * SDL_SetTextureColorMod(). + * + * The texture alpha is affected based on its alpha modulation set by + * SDL_SetTextureAlphaMod(). + * + * \param renderer the rendering context + * \param texture the source texture + * \param srcrect the source SDL_Rect structure or NULL for the entire texture + * \param dstrect the destination SDL_Rect structure or NULL for the entire + * rendering target; the texture will be stretched to fill the + * given rectangle + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderCopyEx + * \sa SDL_SetTextureAlphaMod + * \sa SDL_SetTextureBlendMode + * \sa SDL_SetTextureColorMod + */ +extern DECLSPEC int SDLCALL SDL_RenderCopy(SDL_Renderer * renderer, + SDL_Texture * texture, + const SDL_Rect * srcrect, + const SDL_Rect * dstrect); + +/** + * Copy a portion of the texture to the current rendering, with optional + * rotation and flipping. + * + * Copy a portion of the texture to the current rendering target, optionally + * rotating it by angle around the given center and also flipping it + * top-bottom and/or left-right. + * + * The texture is blended with the destination based on its blend mode set + * with SDL_SetTextureBlendMode(). + * + * The texture color is affected based on its color modulation set by + * SDL_SetTextureColorMod(). + * + * The texture alpha is affected based on its alpha modulation set by + * SDL_SetTextureAlphaMod(). + * + * \param renderer the rendering context + * \param texture the source texture + * \param srcrect the source SDL_Rect structure or NULL for the entire texture + * \param dstrect the destination SDL_Rect structure or NULL for the entire + * rendering target + * \param angle an angle in degrees that indicates the rotation that will be + * applied to dstrect, rotating it in a clockwise direction + * \param center a pointer to a point indicating the point around which + * dstrect will be rotated (if NULL, rotation will be done + * around `dstrect.w / 2`, `dstrect.h / 2`) + * \param flip a SDL_RendererFlip value stating which flipping actions should + * be performed on the texture + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderCopy + * \sa SDL_SetTextureAlphaMod + * \sa SDL_SetTextureBlendMode + * \sa SDL_SetTextureColorMod + */ +extern DECLSPEC int SDLCALL SDL_RenderCopyEx(SDL_Renderer * renderer, + SDL_Texture * texture, + const SDL_Rect * srcrect, + const SDL_Rect * dstrect, + const double angle, + const SDL_Point *center, + const SDL_RendererFlip flip); + + +/** + * Draw a point on the current rendering target at subpixel precision. + * + * \param renderer The renderer which should draw a point. + * \param x The x coordinate of the point. + * \param y The y coordinate of the point. + * \return 0 on success, or -1 on error + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC int SDLCALL SDL_RenderDrawPointF(SDL_Renderer * renderer, + float x, float y); + +/** + * Draw multiple points on the current rendering target at subpixel precision. + * + * \param renderer The renderer which should draw multiple points. + * \param points The points to draw + * \param count The number of points to draw + * \return 0 on success, or -1 on error + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC int SDLCALL SDL_RenderDrawPointsF(SDL_Renderer * renderer, + const SDL_FPoint * points, + int count); + +/** + * Draw a line on the current rendering target at subpixel precision. + * + * \param renderer The renderer which should draw a line. + * \param x1 The x coordinate of the start point. + * \param y1 The y coordinate of the start point. + * \param x2 The x coordinate of the end point. + * \param y2 The y coordinate of the end point. + * \return 0 on success, or -1 on error + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC int SDLCALL SDL_RenderDrawLineF(SDL_Renderer * renderer, + float x1, float y1, float x2, float y2); + +/** + * Draw a series of connected lines on the current rendering target at + * subpixel precision. + * + * \param renderer The renderer which should draw multiple lines. + * \param points The points along the lines + * \param count The number of points, drawing count-1 lines + * \return 0 on success, or -1 on error + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC int SDLCALL SDL_RenderDrawLinesF(SDL_Renderer * renderer, + const SDL_FPoint * points, + int count); + +/** + * Draw a rectangle on the current rendering target at subpixel precision. + * + * \param renderer The renderer which should draw a rectangle. + * \param rect A pointer to the destination rectangle, or NULL to outline the + * entire rendering target. + * \return 0 on success, or -1 on error + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC int SDLCALL SDL_RenderDrawRectF(SDL_Renderer * renderer, + const SDL_FRect * rect); + +/** + * Draw some number of rectangles on the current rendering target at subpixel + * precision. + * + * \param renderer The renderer which should draw multiple rectangles. + * \param rects A pointer to an array of destination rectangles. + * \param count The number of rectangles. + * \return 0 on success, or -1 on error + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC int SDLCALL SDL_RenderDrawRectsF(SDL_Renderer * renderer, + const SDL_FRect * rects, + int count); + +/** + * Fill a rectangle on the current rendering target with the drawing color at + * subpixel precision. + * + * \param renderer The renderer which should fill a rectangle. + * \param rect A pointer to the destination rectangle, or NULL for the entire + * rendering target. + * \return 0 on success, or -1 on error + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC int SDLCALL SDL_RenderFillRectF(SDL_Renderer * renderer, + const SDL_FRect * rect); + +/** + * Fill some number of rectangles on the current rendering target with the + * drawing color at subpixel precision. + * + * \param renderer The renderer which should fill multiple rectangles. + * \param rects A pointer to an array of destination rectangles. + * \param count The number of rectangles. + * \return 0 on success, or -1 on error + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC int SDLCALL SDL_RenderFillRectsF(SDL_Renderer * renderer, + const SDL_FRect * rects, + int count); + +/** + * Copy a portion of the texture to the current rendering target at subpixel + * precision. + * + * \param renderer The renderer which should copy parts of a texture. + * \param texture The source texture. + * \param srcrect A pointer to the source rectangle, or NULL for the entire + * texture. + * \param dstrect A pointer to the destination rectangle, or NULL for the + * entire rendering target. + * \return 0 on success, or -1 on error + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC int SDLCALL SDL_RenderCopyF(SDL_Renderer * renderer, + SDL_Texture * texture, + const SDL_Rect * srcrect, + const SDL_FRect * dstrect); + +/** + * Copy a portion of the source texture to the current rendering target, with + * rotation and flipping, at subpixel precision. + * + * \param renderer The renderer which should copy parts of a texture. + * \param texture The source texture. + * \param srcrect A pointer to the source rectangle, or NULL for the entire + * texture. + * \param dstrect A pointer to the destination rectangle, or NULL for the + * entire rendering target. + * \param angle An angle in degrees that indicates the rotation that will be + * applied to dstrect, rotating it in a clockwise direction + * \param center A pointer to a point indicating the point around which + * dstrect will be rotated (if NULL, rotation will be done + * around dstrect.w/2, dstrect.h/2). + * \param flip An SDL_RendererFlip value stating which flipping actions should + * be performed on the texture + * \return 0 on success, or -1 on error + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC int SDLCALL SDL_RenderCopyExF(SDL_Renderer * renderer, + SDL_Texture * texture, + const SDL_Rect * srcrect, + const SDL_FRect * dstrect, + const double angle, + const SDL_FPoint *center, + const SDL_RendererFlip flip); + +/** + * Render a list of triangles, optionally using a texture and indices into the + * vertex array Color and alpha modulation is done per vertex + * (SDL_SetTextureColorMod and SDL_SetTextureAlphaMod are ignored). + * + * \param renderer The rendering context. + * \param texture (optional) The SDL texture to use. + * \param vertices Vertices. + * \param num_vertices Number of vertices. + * \param indices (optional) An array of integer indices into the 'vertices' + * array, if NULL all vertices will be rendered in sequential + * order. + * \param num_indices Number of indices. + * \return 0 on success, or -1 if the operation is not supported + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_RenderGeometryRaw + * \sa SDL_Vertex + */ +extern DECLSPEC int SDLCALL SDL_RenderGeometry(SDL_Renderer *renderer, + SDL_Texture *texture, + const SDL_Vertex *vertices, int num_vertices, + const int *indices, int num_indices); + +/** + * Render a list of triangles, optionally using a texture and indices into the + * vertex arrays Color and alpha modulation is done per vertex + * (SDL_SetTextureColorMod and SDL_SetTextureAlphaMod are ignored). + * + * \param renderer The rendering context. + * \param texture (optional) The SDL texture to use. + * \param xy Vertex positions + * \param xy_stride Byte size to move from one element to the next element + * \param color Vertex colors (as SDL_Color) + * \param color_stride Byte size to move from one element to the next element + * \param uv Vertex normalized texture coordinates + * \param uv_stride Byte size to move from one element to the next element + * \param num_vertices Number of vertices. + * \param indices (optional) An array of indices into the 'vertices' arrays, + * if NULL all vertices will be rendered in sequential order. + * \param num_indices Number of indices. + * \param size_indices Index size: 1 (byte), 2 (short), 4 (int) + * \return 0 on success, or -1 if the operation is not supported + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_RenderGeometry + * \sa SDL_Vertex + */ +extern DECLSPEC int SDLCALL SDL_RenderGeometryRaw(SDL_Renderer *renderer, + SDL_Texture *texture, + const float *xy, int xy_stride, + const SDL_Color *color, int color_stride, + const float *uv, int uv_stride, + int num_vertices, + const void *indices, int num_indices, int size_indices); + +/** + * Read pixels from the current rendering target to an array of pixels. + * + * **WARNING**: This is a very slow operation, and should not be used + * frequently. If you're using this on the main rendering target, it should be + * called after rendering and before SDL_RenderPresent(). + * + * `pitch` specifies the number of bytes between rows in the destination + * `pixels` data. This allows you to write to a subrectangle or have padded + * rows in the destination. Generally, `pitch` should equal the number of + * pixels per row in the `pixels` data times the number of bytes per pixel, + * but it might contain additional padding (for example, 24bit RGB Windows + * Bitmap data pads all rows to multiples of 4 bytes). + * + * \param renderer the rendering context + * \param rect an SDL_Rect structure representing the area to read, or NULL + * for the entire render target + * \param format an SDL_PixelFormatEnum value of the desired format of the + * pixel data, or 0 to use the format of the rendering target + * \param pixels a pointer to the pixel data to copy into + * \param pitch the pitch of the `pixels` parameter + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC int SDLCALL SDL_RenderReadPixels(SDL_Renderer * renderer, + const SDL_Rect * rect, + Uint32 format, + void *pixels, int pitch); + +/** + * Update the screen with any rendering performed since the previous call. + * + * SDL's rendering functions operate on a backbuffer; that is, calling a + * rendering function such as SDL_RenderDrawLine() does not directly put a + * line on the screen, but rather updates the backbuffer. As such, you compose + * your entire scene and *present* the composed backbuffer to the screen as a + * complete picture. + * + * Therefore, when using SDL's rendering API, one does all drawing intended + * for the frame, and then calls this function once per frame to present the + * final drawing to the user. + * + * The backbuffer should be considered invalidated after each present; do not + * assume that previous contents will exist between frames. You are strongly + * encouraged to call SDL_RenderClear() to initialize the backbuffer before + * starting each new frame's drawing, even if you plan to overwrite every + * pixel. + * + * \param renderer the rendering context + * + * \threadsafety You may only call this function on the main thread. If this + * happens to work on a background thread on any given platform + * or backend, it's purely by luck and you should not rely on it + * to work next time. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RenderClear + * \sa SDL_RenderDrawLine + * \sa SDL_RenderDrawLines + * \sa SDL_RenderDrawPoint + * \sa SDL_RenderDrawPoints + * \sa SDL_RenderDrawRect + * \sa SDL_RenderDrawRects + * \sa SDL_RenderFillRect + * \sa SDL_RenderFillRects + * \sa SDL_SetRenderDrawBlendMode + * \sa SDL_SetRenderDrawColor + */ +extern DECLSPEC void SDLCALL SDL_RenderPresent(SDL_Renderer * renderer); + +/** + * Destroy the specified texture. + * + * Passing NULL or an otherwise invalid texture will set the SDL error message + * to "Invalid texture". + * + * \param texture the texture to destroy + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateTexture + * \sa SDL_CreateTextureFromSurface + */ +extern DECLSPEC void SDLCALL SDL_DestroyTexture(SDL_Texture * texture); + +/** + * Destroy the rendering context for a window and free associated textures. + * + * If `renderer` is NULL, this function will return immediately after setting + * the SDL error message to "Invalid renderer". See SDL_GetError(). + * + * \param renderer the rendering context + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateRenderer + */ +extern DECLSPEC void SDLCALL SDL_DestroyRenderer(SDL_Renderer * renderer); + +/** + * Force the rendering context to flush any pending commands to the underlying + * rendering API. + * + * You do not need to (and in fact, shouldn't) call this function unless you + * are planning to call into OpenGL/Direct3D/Metal/whatever directly in + * addition to using an SDL_Renderer. + * + * This is for a very-specific case: if you are using SDL's render API, you + * asked for a specific renderer backend (OpenGL, Direct3D, etc), you set + * SDL_HINT_RENDER_BATCHING to "1", and you plan to make OpenGL/D3D/whatever + * calls in addition to SDL render API calls. If all of this applies, you + * should call SDL_RenderFlush() between calls to SDL's render API and the + * low-level API you're using in cooperation. + * + * In all other cases, you can ignore this function. This is only here to get + * maximum performance out of a specific situation. In all other cases, SDL + * will do the right thing, perhaps at a performance loss. + * + * This function is first available in SDL 2.0.10, and is not needed in 2.0.9 + * and earlier, as earlier versions did not queue rendering commands at all, + * instead flushing them to the OS immediately. + * + * \param renderer the rendering context + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC int SDLCALL SDL_RenderFlush(SDL_Renderer * renderer); + + +/** + * Bind an OpenGL/ES/ES2 texture to the current context. + * + * This is for use with OpenGL instructions when rendering OpenGL primitives + * directly. + * + * If not NULL, `texw` and `texh` will be filled with the width and height + * values suitable for the provided texture. In most cases, both will be 1.0, + * however, on systems that support the GL_ARB_texture_rectangle extension, + * these values will actually be the pixel width and height used to create the + * texture, so this factor needs to be taken into account when providing + * texture coordinates to OpenGL. + * + * You need a renderer to create an SDL_Texture, therefore you can only use + * this function with an implicit OpenGL context from SDL_CreateRenderer(), + * not with your own OpenGL context. If you need control over your OpenGL + * context, you need to write your own texture-loading methods. + * + * Also note that SDL may upload RGB textures as BGR (or vice-versa), and + * re-order the color channels in the shaders phase, so the uploaded texture + * may have swapped color channels. + * + * \param texture the texture to bind to the current OpenGL/ES/ES2 context + * \param texw a pointer to a float value which will be filled with the + * texture width or NULL if you don't need that value + * \param texh a pointer to a float value which will be filled with the + * texture height or NULL if you don't need that value + * \returns 0 on success, or -1 if the operation is not supported; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GL_MakeCurrent + * \sa SDL_GL_UnbindTexture + */ +extern DECLSPEC int SDLCALL SDL_GL_BindTexture(SDL_Texture *texture, float *texw, float *texh); + +/** + * Unbind an OpenGL/ES/ES2 texture from the current context. + * + * See SDL_GL_BindTexture() for examples on how to use these functions + * + * \param texture the texture to unbind from the current OpenGL/ES/ES2 context + * \returns 0 on success, or -1 if the operation is not supported + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GL_BindTexture + * \sa SDL_GL_MakeCurrent + */ +extern DECLSPEC int SDLCALL SDL_GL_UnbindTexture(SDL_Texture *texture); + +/** + * Get the CAMetalLayer associated with the given Metal renderer. + * + * This function returns `void *`, so SDL doesn't have to include Metal's + * headers, but it can be safely cast to a `CAMetalLayer *`. + * + * \param renderer The renderer to query + * \returns a `CAMetalLayer *` on success, or NULL if the renderer isn't a + * Metal renderer + * + * \since This function is available since SDL 2.0.8. + * + * \sa SDL_RenderGetMetalCommandEncoder + */ +extern DECLSPEC void *SDLCALL SDL_RenderGetMetalLayer(SDL_Renderer * renderer); + +/** + * Get the Metal command encoder for the current frame + * + * This function returns `void *`, so SDL doesn't have to include Metal's + * headers, but it can be safely cast to an `id`. + * + * Note that as of SDL 2.0.18, this will return NULL if Metal refuses to give + * SDL a drawable to render to, which might happen if the window is + * hidden/minimized/offscreen. This doesn't apply to command encoders for + * render targets, just the window's backbuffer. Check your return values! + * + * \param renderer The renderer to query + * \returns an `id` on success, or NULL if the + * renderer isn't a Metal renderer or there was an error. + * + * \since This function is available since SDL 2.0.8. + * + * \sa SDL_RenderGetMetalLayer + */ +extern DECLSPEC void *SDLCALL SDL_RenderGetMetalCommandEncoder(SDL_Renderer * renderer); + +/** + * Toggle VSync of the given renderer. + * + * \param renderer The renderer to toggle + * \param vsync 1 for on, 0 for off. All other values are reserved + * \returns a 0 int on success, or non-zero on failure + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC int SDLCALL SDL_RenderSetVSync(SDL_Renderer* renderer, int vsync); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_render_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_revision.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_revision.h new file mode 100644 index 00000000..4455a08b --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_revision.h @@ -0,0 +1,7 @@ +/* Generated by updaterev.sh, do not edit */ +#ifdef SDL_VENDOR_INFO +#define SDL_REVISION "SDL-release-2.28.5-0-g15ead9a40 (" SDL_VENDOR_INFO ")" +#else +#define SDL_REVISION "SDL-release-2.28.5-0-g15ead9a40" +#endif +#define SDL_REVISION_NUMBER 0 diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_rwops.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_rwops.h new file mode 100644 index 00000000..8615cb54 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_rwops.h @@ -0,0 +1,841 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_rwops.h + * + * This file provides a general interface for SDL to read and write + * data streams. It can easily be extended to files, memory, etc. + */ + +#ifndef SDL_rwops_h_ +#define SDL_rwops_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* RWops Types */ +#define SDL_RWOPS_UNKNOWN 0U /**< Unknown stream type */ +#define SDL_RWOPS_WINFILE 1U /**< Win32 file */ +#define SDL_RWOPS_STDFILE 2U /**< Stdio file */ +#define SDL_RWOPS_JNIFILE 3U /**< Android asset */ +#define SDL_RWOPS_MEMORY 4U /**< Memory stream */ +#define SDL_RWOPS_MEMORY_RO 5U /**< Read-Only memory stream */ + +/** + * This is the read/write operation structure -- very basic. + */ +typedef struct SDL_RWops +{ + /** + * Return the size of the file in this rwops, or -1 if unknown + */ + Sint64 (SDLCALL * size) (struct SDL_RWops * context); + + /** + * Seek to \c offset relative to \c whence, one of stdio's whence values: + * RW_SEEK_SET, RW_SEEK_CUR, RW_SEEK_END + * + * \return the final offset in the data stream, or -1 on error. + */ + Sint64 (SDLCALL * seek) (struct SDL_RWops * context, Sint64 offset, + int whence); + + /** + * Read up to \c maxnum objects each of size \c size from the data + * stream to the area pointed at by \c ptr. + * + * \return the number of objects read, or 0 at error or end of file. + */ + size_t (SDLCALL * read) (struct SDL_RWops * context, void *ptr, + size_t size, size_t maxnum); + + /** + * Write exactly \c num objects each of size \c size from the area + * pointed at by \c ptr to data stream. + * + * \return the number of objects written, or 0 at error or end of file. + */ + size_t (SDLCALL * write) (struct SDL_RWops * context, const void *ptr, + size_t size, size_t num); + + /** + * Close and free an allocated SDL_RWops structure. + * + * \return 0 if successful or -1 on write error when flushing data. + */ + int (SDLCALL * close) (struct SDL_RWops * context); + + Uint32 type; + union + { +#if defined(__ANDROID__) + struct + { + void *asset; + } androidio; +#elif defined(__WIN32__) || defined(__GDK__) + struct + { + SDL_bool append; + void *h; + struct + { + void *data; + size_t size; + size_t left; + } buffer; + } windowsio; +#endif + +#ifdef HAVE_STDIO_H + struct + { + SDL_bool autoclose; + FILE *fp; + } stdio; +#endif + struct + { + Uint8 *base; + Uint8 *here; + Uint8 *stop; + } mem; + struct + { + void *data1; + void *data2; + } unknown; + } hidden; + +} SDL_RWops; + + +/** + * \name RWFrom functions + * + * Functions to create SDL_RWops structures from various data streams. + */ +/* @{ */ + +/** + * Use this function to create a new SDL_RWops structure for reading from + * and/or writing to a named file. + * + * The `mode` string is treated roughly the same as in a call to the C + * library's fopen(), even if SDL doesn't happen to use fopen() behind the + * scenes. + * + * Available `mode` strings: + * + * - "r": Open a file for reading. The file must exist. + * - "w": Create an empty file for writing. If a file with the same name + * already exists its content is erased and the file is treated as a new + * empty file. + * - "a": Append to a file. Writing operations append data at the end of the + * file. The file is created if it does not exist. + * - "r+": Open a file for update both reading and writing. The file must + * exist. + * - "w+": Create an empty file for both reading and writing. If a file with + * the same name already exists its content is erased and the file is + * treated as a new empty file. + * - "a+": Open a file for reading and appending. All writing operations are + * performed at the end of the file, protecting the previous content to be + * overwritten. You can reposition (fseek, rewind) the internal pointer to + * anywhere in the file for reading, but writing operations will move it + * back to the end of file. The file is created if it does not exist. + * + * **NOTE**: In order to open a file as a binary file, a "b" character has to + * be included in the `mode` string. This additional "b" character can either + * be appended at the end of the string (thus making the following compound + * modes: "rb", "wb", "ab", "r+b", "w+b", "a+b") or be inserted between the + * letter and the "+" sign for the mixed modes ("rb+", "wb+", "ab+"). + * Additional characters may follow the sequence, although they should have no + * effect. For example, "t" is sometimes appended to make explicit the file is + * a text file. + * + * This function supports Unicode filenames, but they must be encoded in UTF-8 + * format, regardless of the underlying operating system. + * + * As a fallback, SDL_RWFromFile() will transparently open a matching filename + * in an Android app's `assets`. + * + * Closing the SDL_RWops will close the file handle SDL is holding internally. + * + * \param file a UTF-8 string representing the filename to open + * \param mode an ASCII string representing the mode to be used for opening + * the file. + * \returns a pointer to the SDL_RWops structure that is created, or NULL on + * failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RWclose + * \sa SDL_RWFromConstMem + * \sa SDL_RWFromFP + * \sa SDL_RWFromMem + * \sa SDL_RWread + * \sa SDL_RWseek + * \sa SDL_RWtell + * \sa SDL_RWwrite + */ +extern DECLSPEC SDL_RWops *SDLCALL SDL_RWFromFile(const char *file, + const char *mode); + +#ifdef HAVE_STDIO_H + +extern DECLSPEC SDL_RWops *SDLCALL SDL_RWFromFP(FILE * fp, SDL_bool autoclose); + +#else + +/** + * Use this function to create an SDL_RWops structure from a standard I/O file + * pointer (stdio.h's `FILE*`). + * + * This function is not available on Windows, since files opened in an + * application on that platform cannot be used by a dynamically linked + * library. + * + * On some platforms, the first parameter is a `void*`, on others, it's a + * `FILE*`, depending on what system headers are available to SDL. It is + * always intended to be the `FILE*` type from the C runtime's stdio.h. + * + * \param fp the `FILE*` that feeds the SDL_RWops stream + * \param autoclose SDL_TRUE to close the `FILE*` when closing the SDL_RWops, + * SDL_FALSE to leave the `FILE*` open when the RWops is + * closed + * \returns a pointer to the SDL_RWops structure that is created, or NULL on + * failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RWclose + * \sa SDL_RWFromConstMem + * \sa SDL_RWFromFile + * \sa SDL_RWFromMem + * \sa SDL_RWread + * \sa SDL_RWseek + * \sa SDL_RWtell + * \sa SDL_RWwrite + */ +extern DECLSPEC SDL_RWops *SDLCALL SDL_RWFromFP(void * fp, + SDL_bool autoclose); +#endif + +/** + * Use this function to prepare a read-write memory buffer for use with + * SDL_RWops. + * + * This function sets up an SDL_RWops struct based on a memory area of a + * certain size, for both read and write access. + * + * This memory buffer is not copied by the RWops; the pointer you provide must + * remain valid until you close the stream. Closing the stream will not free + * the original buffer. + * + * If you need to make sure the RWops never writes to the memory buffer, you + * should use SDL_RWFromConstMem() with a read-only buffer of memory instead. + * + * \param mem a pointer to a buffer to feed an SDL_RWops stream + * \param size the buffer size, in bytes + * \returns a pointer to a new SDL_RWops structure, or NULL if it fails; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RWclose + * \sa SDL_RWFromConstMem + * \sa SDL_RWFromFile + * \sa SDL_RWFromFP + * \sa SDL_RWFromMem + * \sa SDL_RWread + * \sa SDL_RWseek + * \sa SDL_RWtell + * \sa SDL_RWwrite + */ +extern DECLSPEC SDL_RWops *SDLCALL SDL_RWFromMem(void *mem, int size); + +/** + * Use this function to prepare a read-only memory buffer for use with RWops. + * + * This function sets up an SDL_RWops struct based on a memory area of a + * certain size. It assumes the memory area is not writable. + * + * Attempting to write to this RWops stream will report an error without + * writing to the memory buffer. + * + * This memory buffer is not copied by the RWops; the pointer you provide must + * remain valid until you close the stream. Closing the stream will not free + * the original buffer. + * + * If you need to write to a memory buffer, you should use SDL_RWFromMem() + * with a writable buffer of memory instead. + * + * \param mem a pointer to a read-only buffer to feed an SDL_RWops stream + * \param size the buffer size, in bytes + * \returns a pointer to a new SDL_RWops structure, or NULL if it fails; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RWclose + * \sa SDL_RWFromConstMem + * \sa SDL_RWFromFile + * \sa SDL_RWFromFP + * \sa SDL_RWFromMem + * \sa SDL_RWread + * \sa SDL_RWseek + * \sa SDL_RWtell + */ +extern DECLSPEC SDL_RWops *SDLCALL SDL_RWFromConstMem(const void *mem, + int size); + +/* @} *//* RWFrom functions */ + + +/** + * Use this function to allocate an empty, unpopulated SDL_RWops structure. + * + * Applications do not need to use this function unless they are providing + * their own SDL_RWops implementation. If you just need a SDL_RWops to + * read/write a common data source, you should use the built-in + * implementations in SDL, like SDL_RWFromFile() or SDL_RWFromMem(), etc. + * + * You must free the returned pointer with SDL_FreeRW(). Depending on your + * operating system and compiler, there may be a difference between the + * malloc() and free() your program uses and the versions SDL calls + * internally. Trying to mix the two can cause crashing such as segmentation + * faults. Since all SDL_RWops must free themselves when their **close** + * method is called, all SDL_RWops must be allocated through this function, so + * they can all be freed correctly with SDL_FreeRW(). + * + * \returns a pointer to the allocated memory on success, or NULL on failure; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_FreeRW + */ +extern DECLSPEC SDL_RWops *SDLCALL SDL_AllocRW(void); + +/** + * Use this function to free an SDL_RWops structure allocated by + * SDL_AllocRW(). + * + * Applications do not need to use this function unless they are providing + * their own SDL_RWops implementation. If you just need a SDL_RWops to + * read/write a common data source, you should use the built-in + * implementations in SDL, like SDL_RWFromFile() or SDL_RWFromMem(), etc, and + * call the **close** method on those SDL_RWops pointers when you are done + * with them. + * + * Only use SDL_FreeRW() on pointers returned by SDL_AllocRW(). The pointer is + * invalid as soon as this function returns. Any extra memory allocated during + * creation of the SDL_RWops is not freed by SDL_FreeRW(); the programmer must + * be responsible for managing that memory in their **close** method. + * + * \param area the SDL_RWops structure to be freed + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AllocRW + */ +extern DECLSPEC void SDLCALL SDL_FreeRW(SDL_RWops * area); + +#define RW_SEEK_SET 0 /**< Seek from the beginning of data */ +#define RW_SEEK_CUR 1 /**< Seek relative to current read point */ +#define RW_SEEK_END 2 /**< Seek relative to the end of data */ + +/** + * Use this function to get the size of the data stream in an SDL_RWops. + * + * Prior to SDL 2.0.10, this function was a macro. + * + * \param context the SDL_RWops to get the size of the data stream from + * \returns the size of the data stream in the SDL_RWops on success, -1 if + * unknown or a negative error code on failure; call SDL_GetError() + * for more information. + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC Sint64 SDLCALL SDL_RWsize(SDL_RWops *context); + +/** + * Seek within an SDL_RWops data stream. + * + * This function seeks to byte `offset`, relative to `whence`. + * + * `whence` may be any of the following values: + * + * - `RW_SEEK_SET`: seek from the beginning of data + * - `RW_SEEK_CUR`: seek relative to current read point + * - `RW_SEEK_END`: seek relative to the end of data + * + * If this stream can not seek, it will return -1. + * + * SDL_RWseek() is actually a wrapper function that calls the SDL_RWops's + * `seek` method appropriately, to simplify application development. + * + * Prior to SDL 2.0.10, this function was a macro. + * + * \param context a pointer to an SDL_RWops structure + * \param offset an offset in bytes, relative to **whence** location; can be + * negative + * \param whence any of `RW_SEEK_SET`, `RW_SEEK_CUR`, `RW_SEEK_END` + * \returns the final offset in the data stream after the seek or -1 on error. + * + * \since This function is available since SDL 2.0.10. + * + * \sa SDL_RWclose + * \sa SDL_RWFromConstMem + * \sa SDL_RWFromFile + * \sa SDL_RWFromFP + * \sa SDL_RWFromMem + * \sa SDL_RWread + * \sa SDL_RWtell + * \sa SDL_RWwrite + */ +extern DECLSPEC Sint64 SDLCALL SDL_RWseek(SDL_RWops *context, + Sint64 offset, int whence); + +/** + * Determine the current read/write offset in an SDL_RWops data stream. + * + * SDL_RWtell is actually a wrapper function that calls the SDL_RWops's `seek` + * method, with an offset of 0 bytes from `RW_SEEK_CUR`, to simplify + * application development. + * + * Prior to SDL 2.0.10, this function was a macro. + * + * \param context a SDL_RWops data stream object from which to get the current + * offset + * \returns the current offset in the stream, or -1 if the information can not + * be determined. + * + * \since This function is available since SDL 2.0.10. + * + * \sa SDL_RWclose + * \sa SDL_RWFromConstMem + * \sa SDL_RWFromFile + * \sa SDL_RWFromFP + * \sa SDL_RWFromMem + * \sa SDL_RWread + * \sa SDL_RWseek + * \sa SDL_RWwrite + */ +extern DECLSPEC Sint64 SDLCALL SDL_RWtell(SDL_RWops *context); + +/** + * Read from a data source. + * + * This function reads up to `maxnum` objects each of size `size` from the + * data source to the area pointed at by `ptr`. This function may read less + * objects than requested. It will return zero when there has been an error or + * the data stream is completely read. + * + * SDL_RWread() is actually a function wrapper that calls the SDL_RWops's + * `read` method appropriately, to simplify application development. + * + * Prior to SDL 2.0.10, this function was a macro. + * + * \param context a pointer to an SDL_RWops structure + * \param ptr a pointer to a buffer to read data into + * \param size the size of each object to read, in bytes + * \param maxnum the maximum number of objects to be read + * \returns the number of objects read, or 0 at error or end of file; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.10. + * + * \sa SDL_RWclose + * \sa SDL_RWFromConstMem + * \sa SDL_RWFromFile + * \sa SDL_RWFromFP + * \sa SDL_RWFromMem + * \sa SDL_RWseek + * \sa SDL_RWwrite + */ +extern DECLSPEC size_t SDLCALL SDL_RWread(SDL_RWops *context, + void *ptr, size_t size, + size_t maxnum); + +/** + * Write to an SDL_RWops data stream. + * + * This function writes exactly `num` objects each of size `size` from the + * area pointed at by `ptr` to the stream. If this fails for any reason, it'll + * return less than `num` to demonstrate how far the write progressed. On + * success, it returns `num`. + * + * SDL_RWwrite is actually a function wrapper that calls the SDL_RWops's + * `write` method appropriately, to simplify application development. + * + * Prior to SDL 2.0.10, this function was a macro. + * + * \param context a pointer to an SDL_RWops structure + * \param ptr a pointer to a buffer containing data to write + * \param size the size of an object to write, in bytes + * \param num the number of objects to write + * \returns the number of objects written, which will be less than **num** on + * error; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.10. + * + * \sa SDL_RWclose + * \sa SDL_RWFromConstMem + * \sa SDL_RWFromFile + * \sa SDL_RWFromFP + * \sa SDL_RWFromMem + * \sa SDL_RWread + * \sa SDL_RWseek + */ +extern DECLSPEC size_t SDLCALL SDL_RWwrite(SDL_RWops *context, + const void *ptr, size_t size, + size_t num); + +/** + * Close and free an allocated SDL_RWops structure. + * + * SDL_RWclose() closes and cleans up the SDL_RWops stream. It releases any + * resources used by the stream and frees the SDL_RWops itself with + * SDL_FreeRW(). This returns 0 on success, or -1 if the stream failed to + * flush to its output (e.g. to disk). + * + * Note that if this fails to flush the stream to disk, this function reports + * an error, but the SDL_RWops is still invalid once this function returns. + * + * Prior to SDL 2.0.10, this function was a macro. + * + * \param context SDL_RWops structure to close + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.10. + * + * \sa SDL_RWFromConstMem + * \sa SDL_RWFromFile + * \sa SDL_RWFromFP + * \sa SDL_RWFromMem + * \sa SDL_RWread + * \sa SDL_RWseek + * \sa SDL_RWwrite + */ +extern DECLSPEC int SDLCALL SDL_RWclose(SDL_RWops *context); + +/** + * Load all the data from an SDL data stream. + * + * The data is allocated with a zero byte at the end (null terminated) for + * convenience. This extra byte is not included in the value reported via + * `datasize`. + * + * The data should be freed with SDL_free(). + * + * \param src the SDL_RWops to read all available data from + * \param datasize if not NULL, will store the number of bytes read + * \param freesrc if non-zero, calls SDL_RWclose() on `src` before returning + * \returns the data, or NULL if there was an error. + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC void *SDLCALL SDL_LoadFile_RW(SDL_RWops *src, + size_t *datasize, + int freesrc); + +/** + * Load all the data from a file path. + * + * The data is allocated with a zero byte at the end (null terminated) for + * convenience. This extra byte is not included in the value reported via + * `datasize`. + * + * The data should be freed with SDL_free(). + * + * Prior to SDL 2.0.10, this function was a macro wrapping around + * SDL_LoadFile_RW. + * + * \param file the path to read all available data from + * \param datasize if not NULL, will store the number of bytes read + * \returns the data, or NULL if there was an error. + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC void *SDLCALL SDL_LoadFile(const char *file, size_t *datasize); + +/** + * \name Read endian functions + * + * Read an item of the specified endianness and return in native format. + */ +/* @{ */ + +/** + * Use this function to read a byte from an SDL_RWops. + * + * \param src the SDL_RWops to read from + * \returns the read byte on success or 0 on failure; call SDL_GetError() for + * more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_WriteU8 + */ +extern DECLSPEC Uint8 SDLCALL SDL_ReadU8(SDL_RWops * src); + +/** + * Use this function to read 16 bits of little-endian data from an SDL_RWops + * and return in native format. + * + * SDL byteswaps the data only if necessary, so the data returned will be in + * the native byte order. + * + * \param src the stream from which to read data + * \returns 16 bits of data in the native byte order of the platform. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ReadBE16 + */ +extern DECLSPEC Uint16 SDLCALL SDL_ReadLE16(SDL_RWops * src); + +/** + * Use this function to read 16 bits of big-endian data from an SDL_RWops and + * return in native format. + * + * SDL byteswaps the data only if necessary, so the data returned will be in + * the native byte order. + * + * \param src the stream from which to read data + * \returns 16 bits of data in the native byte order of the platform. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ReadLE16 + */ +extern DECLSPEC Uint16 SDLCALL SDL_ReadBE16(SDL_RWops * src); + +/** + * Use this function to read 32 bits of little-endian data from an SDL_RWops + * and return in native format. + * + * SDL byteswaps the data only if necessary, so the data returned will be in + * the native byte order. + * + * \param src the stream from which to read data + * \returns 32 bits of data in the native byte order of the platform. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ReadBE32 + */ +extern DECLSPEC Uint32 SDLCALL SDL_ReadLE32(SDL_RWops * src); + +/** + * Use this function to read 32 bits of big-endian data from an SDL_RWops and + * return in native format. + * + * SDL byteswaps the data only if necessary, so the data returned will be in + * the native byte order. + * + * \param src the stream from which to read data + * \returns 32 bits of data in the native byte order of the platform. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ReadLE32 + */ +extern DECLSPEC Uint32 SDLCALL SDL_ReadBE32(SDL_RWops * src); + +/** + * Use this function to read 64 bits of little-endian data from an SDL_RWops + * and return in native format. + * + * SDL byteswaps the data only if necessary, so the data returned will be in + * the native byte order. + * + * \param src the stream from which to read data + * \returns 64 bits of data in the native byte order of the platform. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ReadBE64 + */ +extern DECLSPEC Uint64 SDLCALL SDL_ReadLE64(SDL_RWops * src); + +/** + * Use this function to read 64 bits of big-endian data from an SDL_RWops and + * return in native format. + * + * SDL byteswaps the data only if necessary, so the data returned will be in + * the native byte order. + * + * \param src the stream from which to read data + * \returns 64 bits of data in the native byte order of the platform. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ReadLE64 + */ +extern DECLSPEC Uint64 SDLCALL SDL_ReadBE64(SDL_RWops * src); +/* @} *//* Read endian functions */ + +/** + * \name Write endian functions + * + * Write an item of native format to the specified endianness. + */ +/* @{ */ + +/** + * Use this function to write a byte to an SDL_RWops. + * + * \param dst the SDL_RWops to write to + * \param value the byte value to write + * \returns 1 on success or 0 on failure; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ReadU8 + */ +extern DECLSPEC size_t SDLCALL SDL_WriteU8(SDL_RWops * dst, Uint8 value); + +/** + * Use this function to write 16 bits in native format to a SDL_RWops as + * little-endian data. + * + * SDL byteswaps the data only if necessary, so the application always + * specifies native format, and the data written will be in little-endian + * format. + * + * \param dst the stream to which data will be written + * \param value the data to be written, in native format + * \returns 1 on successful write, 0 on error. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_WriteBE16 + */ +extern DECLSPEC size_t SDLCALL SDL_WriteLE16(SDL_RWops * dst, Uint16 value); + +/** + * Use this function to write 16 bits in native format to a SDL_RWops as + * big-endian data. + * + * SDL byteswaps the data only if necessary, so the application always + * specifies native format, and the data written will be in big-endian format. + * + * \param dst the stream to which data will be written + * \param value the data to be written, in native format + * \returns 1 on successful write, 0 on error. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_WriteLE16 + */ +extern DECLSPEC size_t SDLCALL SDL_WriteBE16(SDL_RWops * dst, Uint16 value); + +/** + * Use this function to write 32 bits in native format to a SDL_RWops as + * little-endian data. + * + * SDL byteswaps the data only if necessary, so the application always + * specifies native format, and the data written will be in little-endian + * format. + * + * \param dst the stream to which data will be written + * \param value the data to be written, in native format + * \returns 1 on successful write, 0 on error. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_WriteBE32 + */ +extern DECLSPEC size_t SDLCALL SDL_WriteLE32(SDL_RWops * dst, Uint32 value); + +/** + * Use this function to write 32 bits in native format to a SDL_RWops as + * big-endian data. + * + * SDL byteswaps the data only if necessary, so the application always + * specifies native format, and the data written will be in big-endian format. + * + * \param dst the stream to which data will be written + * \param value the data to be written, in native format + * \returns 1 on successful write, 0 on error. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_WriteLE32 + */ +extern DECLSPEC size_t SDLCALL SDL_WriteBE32(SDL_RWops * dst, Uint32 value); + +/** + * Use this function to write 64 bits in native format to a SDL_RWops as + * little-endian data. + * + * SDL byteswaps the data only if necessary, so the application always + * specifies native format, and the data written will be in little-endian + * format. + * + * \param dst the stream to which data will be written + * \param value the data to be written, in native format + * \returns 1 on successful write, 0 on error. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_WriteBE64 + */ +extern DECLSPEC size_t SDLCALL SDL_WriteLE64(SDL_RWops * dst, Uint64 value); + +/** + * Use this function to write 64 bits in native format to a SDL_RWops as + * big-endian data. + * + * SDL byteswaps the data only if necessary, so the application always + * specifies native format, and the data written will be in big-endian format. + * + * \param dst the stream to which data will be written + * \param value the data to be written, in native format + * \returns 1 on successful write, 0 on error. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_WriteLE64 + */ +extern DECLSPEC size_t SDLCALL SDL_WriteBE64(SDL_RWops * dst, Uint64 value); +/* @} *//* Write endian functions */ + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_rwops_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_scancode.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_scancode.h new file mode 100644 index 00000000..a960a799 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_scancode.h @@ -0,0 +1,438 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_scancode.h + * + * Defines keyboard scancodes. + */ + +#ifndef SDL_scancode_h_ +#define SDL_scancode_h_ + +#include "SDL_stdinc.h" + +/** + * \brief The SDL keyboard scancode representation. + * + * Values of this type are used to represent keyboard keys, among other places + * in the \link SDL_Keysym::scancode key.keysym.scancode \endlink field of the + * SDL_Event structure. + * + * The values in this enumeration are based on the USB usage page standard: + * https://www.usb.org/sites/default/files/documents/hut1_12v2.pdf + */ +typedef enum +{ + SDL_SCANCODE_UNKNOWN = 0, + + /** + * \name Usage page 0x07 + * + * These values are from usage page 0x07 (USB keyboard page). + */ + /* @{ */ + + SDL_SCANCODE_A = 4, + SDL_SCANCODE_B = 5, + SDL_SCANCODE_C = 6, + SDL_SCANCODE_D = 7, + SDL_SCANCODE_E = 8, + SDL_SCANCODE_F = 9, + SDL_SCANCODE_G = 10, + SDL_SCANCODE_H = 11, + SDL_SCANCODE_I = 12, + SDL_SCANCODE_J = 13, + SDL_SCANCODE_K = 14, + SDL_SCANCODE_L = 15, + SDL_SCANCODE_M = 16, + SDL_SCANCODE_N = 17, + SDL_SCANCODE_O = 18, + SDL_SCANCODE_P = 19, + SDL_SCANCODE_Q = 20, + SDL_SCANCODE_R = 21, + SDL_SCANCODE_S = 22, + SDL_SCANCODE_T = 23, + SDL_SCANCODE_U = 24, + SDL_SCANCODE_V = 25, + SDL_SCANCODE_W = 26, + SDL_SCANCODE_X = 27, + SDL_SCANCODE_Y = 28, + SDL_SCANCODE_Z = 29, + + SDL_SCANCODE_1 = 30, + SDL_SCANCODE_2 = 31, + SDL_SCANCODE_3 = 32, + SDL_SCANCODE_4 = 33, + SDL_SCANCODE_5 = 34, + SDL_SCANCODE_6 = 35, + SDL_SCANCODE_7 = 36, + SDL_SCANCODE_8 = 37, + SDL_SCANCODE_9 = 38, + SDL_SCANCODE_0 = 39, + + SDL_SCANCODE_RETURN = 40, + SDL_SCANCODE_ESCAPE = 41, + SDL_SCANCODE_BACKSPACE = 42, + SDL_SCANCODE_TAB = 43, + SDL_SCANCODE_SPACE = 44, + + SDL_SCANCODE_MINUS = 45, + SDL_SCANCODE_EQUALS = 46, + SDL_SCANCODE_LEFTBRACKET = 47, + SDL_SCANCODE_RIGHTBRACKET = 48, + SDL_SCANCODE_BACKSLASH = 49, /**< Located at the lower left of the return + * key on ISO keyboards and at the right end + * of the QWERTY row on ANSI keyboards. + * Produces REVERSE SOLIDUS (backslash) and + * VERTICAL LINE in a US layout, REVERSE + * SOLIDUS and VERTICAL LINE in a UK Mac + * layout, NUMBER SIGN and TILDE in a UK + * Windows layout, DOLLAR SIGN and POUND SIGN + * in a Swiss German layout, NUMBER SIGN and + * APOSTROPHE in a German layout, GRAVE + * ACCENT and POUND SIGN in a French Mac + * layout, and ASTERISK and MICRO SIGN in a + * French Windows layout. + */ + SDL_SCANCODE_NONUSHASH = 50, /**< ISO USB keyboards actually use this code + * instead of 49 for the same key, but all + * OSes I've seen treat the two codes + * identically. So, as an implementor, unless + * your keyboard generates both of those + * codes and your OS treats them differently, + * you should generate SDL_SCANCODE_BACKSLASH + * instead of this code. As a user, you + * should not rely on this code because SDL + * will never generate it with most (all?) + * keyboards. + */ + SDL_SCANCODE_SEMICOLON = 51, + SDL_SCANCODE_APOSTROPHE = 52, + SDL_SCANCODE_GRAVE = 53, /**< Located in the top left corner (on both ANSI + * and ISO keyboards). Produces GRAVE ACCENT and + * TILDE in a US Windows layout and in US and UK + * Mac layouts on ANSI keyboards, GRAVE ACCENT + * and NOT SIGN in a UK Windows layout, SECTION + * SIGN and PLUS-MINUS SIGN in US and UK Mac + * layouts on ISO keyboards, SECTION SIGN and + * DEGREE SIGN in a Swiss German layout (Mac: + * only on ISO keyboards), CIRCUMFLEX ACCENT and + * DEGREE SIGN in a German layout (Mac: only on + * ISO keyboards), SUPERSCRIPT TWO and TILDE in a + * French Windows layout, COMMERCIAL AT and + * NUMBER SIGN in a French Mac layout on ISO + * keyboards, and LESS-THAN SIGN and GREATER-THAN + * SIGN in a Swiss German, German, or French Mac + * layout on ANSI keyboards. + */ + SDL_SCANCODE_COMMA = 54, + SDL_SCANCODE_PERIOD = 55, + SDL_SCANCODE_SLASH = 56, + + SDL_SCANCODE_CAPSLOCK = 57, + + SDL_SCANCODE_F1 = 58, + SDL_SCANCODE_F2 = 59, + SDL_SCANCODE_F3 = 60, + SDL_SCANCODE_F4 = 61, + SDL_SCANCODE_F5 = 62, + SDL_SCANCODE_F6 = 63, + SDL_SCANCODE_F7 = 64, + SDL_SCANCODE_F8 = 65, + SDL_SCANCODE_F9 = 66, + SDL_SCANCODE_F10 = 67, + SDL_SCANCODE_F11 = 68, + SDL_SCANCODE_F12 = 69, + + SDL_SCANCODE_PRINTSCREEN = 70, + SDL_SCANCODE_SCROLLLOCK = 71, + SDL_SCANCODE_PAUSE = 72, + SDL_SCANCODE_INSERT = 73, /**< insert on PC, help on some Mac keyboards (but + does send code 73, not 117) */ + SDL_SCANCODE_HOME = 74, + SDL_SCANCODE_PAGEUP = 75, + SDL_SCANCODE_DELETE = 76, + SDL_SCANCODE_END = 77, + SDL_SCANCODE_PAGEDOWN = 78, + SDL_SCANCODE_RIGHT = 79, + SDL_SCANCODE_LEFT = 80, + SDL_SCANCODE_DOWN = 81, + SDL_SCANCODE_UP = 82, + + SDL_SCANCODE_NUMLOCKCLEAR = 83, /**< num lock on PC, clear on Mac keyboards + */ + SDL_SCANCODE_KP_DIVIDE = 84, + SDL_SCANCODE_KP_MULTIPLY = 85, + SDL_SCANCODE_KP_MINUS = 86, + SDL_SCANCODE_KP_PLUS = 87, + SDL_SCANCODE_KP_ENTER = 88, + SDL_SCANCODE_KP_1 = 89, + SDL_SCANCODE_KP_2 = 90, + SDL_SCANCODE_KP_3 = 91, + SDL_SCANCODE_KP_4 = 92, + SDL_SCANCODE_KP_5 = 93, + SDL_SCANCODE_KP_6 = 94, + SDL_SCANCODE_KP_7 = 95, + SDL_SCANCODE_KP_8 = 96, + SDL_SCANCODE_KP_9 = 97, + SDL_SCANCODE_KP_0 = 98, + SDL_SCANCODE_KP_PERIOD = 99, + + SDL_SCANCODE_NONUSBACKSLASH = 100, /**< This is the additional key that ISO + * keyboards have over ANSI ones, + * located between left shift and Y. + * Produces GRAVE ACCENT and TILDE in a + * US or UK Mac layout, REVERSE SOLIDUS + * (backslash) and VERTICAL LINE in a + * US or UK Windows layout, and + * LESS-THAN SIGN and GREATER-THAN SIGN + * in a Swiss German, German, or French + * layout. */ + SDL_SCANCODE_APPLICATION = 101, /**< windows contextual menu, compose */ + SDL_SCANCODE_POWER = 102, /**< The USB document says this is a status flag, + * not a physical key - but some Mac keyboards + * do have a power key. */ + SDL_SCANCODE_KP_EQUALS = 103, + SDL_SCANCODE_F13 = 104, + SDL_SCANCODE_F14 = 105, + SDL_SCANCODE_F15 = 106, + SDL_SCANCODE_F16 = 107, + SDL_SCANCODE_F17 = 108, + SDL_SCANCODE_F18 = 109, + SDL_SCANCODE_F19 = 110, + SDL_SCANCODE_F20 = 111, + SDL_SCANCODE_F21 = 112, + SDL_SCANCODE_F22 = 113, + SDL_SCANCODE_F23 = 114, + SDL_SCANCODE_F24 = 115, + SDL_SCANCODE_EXECUTE = 116, + SDL_SCANCODE_HELP = 117, /**< AL Integrated Help Center */ + SDL_SCANCODE_MENU = 118, /**< Menu (show menu) */ + SDL_SCANCODE_SELECT = 119, + SDL_SCANCODE_STOP = 120, /**< AC Stop */ + SDL_SCANCODE_AGAIN = 121, /**< AC Redo/Repeat */ + SDL_SCANCODE_UNDO = 122, /**< AC Undo */ + SDL_SCANCODE_CUT = 123, /**< AC Cut */ + SDL_SCANCODE_COPY = 124, /**< AC Copy */ + SDL_SCANCODE_PASTE = 125, /**< AC Paste */ + SDL_SCANCODE_FIND = 126, /**< AC Find */ + SDL_SCANCODE_MUTE = 127, + SDL_SCANCODE_VOLUMEUP = 128, + SDL_SCANCODE_VOLUMEDOWN = 129, +/* not sure whether there's a reason to enable these */ +/* SDL_SCANCODE_LOCKINGCAPSLOCK = 130, */ +/* SDL_SCANCODE_LOCKINGNUMLOCK = 131, */ +/* SDL_SCANCODE_LOCKINGSCROLLLOCK = 132, */ + SDL_SCANCODE_KP_COMMA = 133, + SDL_SCANCODE_KP_EQUALSAS400 = 134, + + SDL_SCANCODE_INTERNATIONAL1 = 135, /**< used on Asian keyboards, see + footnotes in USB doc */ + SDL_SCANCODE_INTERNATIONAL2 = 136, + SDL_SCANCODE_INTERNATIONAL3 = 137, /**< Yen */ + SDL_SCANCODE_INTERNATIONAL4 = 138, + SDL_SCANCODE_INTERNATIONAL5 = 139, + SDL_SCANCODE_INTERNATIONAL6 = 140, + SDL_SCANCODE_INTERNATIONAL7 = 141, + SDL_SCANCODE_INTERNATIONAL8 = 142, + SDL_SCANCODE_INTERNATIONAL9 = 143, + SDL_SCANCODE_LANG1 = 144, /**< Hangul/English toggle */ + SDL_SCANCODE_LANG2 = 145, /**< Hanja conversion */ + SDL_SCANCODE_LANG3 = 146, /**< Katakana */ + SDL_SCANCODE_LANG4 = 147, /**< Hiragana */ + SDL_SCANCODE_LANG5 = 148, /**< Zenkaku/Hankaku */ + SDL_SCANCODE_LANG6 = 149, /**< reserved */ + SDL_SCANCODE_LANG7 = 150, /**< reserved */ + SDL_SCANCODE_LANG8 = 151, /**< reserved */ + SDL_SCANCODE_LANG9 = 152, /**< reserved */ + + SDL_SCANCODE_ALTERASE = 153, /**< Erase-Eaze */ + SDL_SCANCODE_SYSREQ = 154, + SDL_SCANCODE_CANCEL = 155, /**< AC Cancel */ + SDL_SCANCODE_CLEAR = 156, + SDL_SCANCODE_PRIOR = 157, + SDL_SCANCODE_RETURN2 = 158, + SDL_SCANCODE_SEPARATOR = 159, + SDL_SCANCODE_OUT = 160, + SDL_SCANCODE_OPER = 161, + SDL_SCANCODE_CLEARAGAIN = 162, + SDL_SCANCODE_CRSEL = 163, + SDL_SCANCODE_EXSEL = 164, + + SDL_SCANCODE_KP_00 = 176, + SDL_SCANCODE_KP_000 = 177, + SDL_SCANCODE_THOUSANDSSEPARATOR = 178, + SDL_SCANCODE_DECIMALSEPARATOR = 179, + SDL_SCANCODE_CURRENCYUNIT = 180, + SDL_SCANCODE_CURRENCYSUBUNIT = 181, + SDL_SCANCODE_KP_LEFTPAREN = 182, + SDL_SCANCODE_KP_RIGHTPAREN = 183, + SDL_SCANCODE_KP_LEFTBRACE = 184, + SDL_SCANCODE_KP_RIGHTBRACE = 185, + SDL_SCANCODE_KP_TAB = 186, + SDL_SCANCODE_KP_BACKSPACE = 187, + SDL_SCANCODE_KP_A = 188, + SDL_SCANCODE_KP_B = 189, + SDL_SCANCODE_KP_C = 190, + SDL_SCANCODE_KP_D = 191, + SDL_SCANCODE_KP_E = 192, + SDL_SCANCODE_KP_F = 193, + SDL_SCANCODE_KP_XOR = 194, + SDL_SCANCODE_KP_POWER = 195, + SDL_SCANCODE_KP_PERCENT = 196, + SDL_SCANCODE_KP_LESS = 197, + SDL_SCANCODE_KP_GREATER = 198, + SDL_SCANCODE_KP_AMPERSAND = 199, + SDL_SCANCODE_KP_DBLAMPERSAND = 200, + SDL_SCANCODE_KP_VERTICALBAR = 201, + SDL_SCANCODE_KP_DBLVERTICALBAR = 202, + SDL_SCANCODE_KP_COLON = 203, + SDL_SCANCODE_KP_HASH = 204, + SDL_SCANCODE_KP_SPACE = 205, + SDL_SCANCODE_KP_AT = 206, + SDL_SCANCODE_KP_EXCLAM = 207, + SDL_SCANCODE_KP_MEMSTORE = 208, + SDL_SCANCODE_KP_MEMRECALL = 209, + SDL_SCANCODE_KP_MEMCLEAR = 210, + SDL_SCANCODE_KP_MEMADD = 211, + SDL_SCANCODE_KP_MEMSUBTRACT = 212, + SDL_SCANCODE_KP_MEMMULTIPLY = 213, + SDL_SCANCODE_KP_MEMDIVIDE = 214, + SDL_SCANCODE_KP_PLUSMINUS = 215, + SDL_SCANCODE_KP_CLEAR = 216, + SDL_SCANCODE_KP_CLEARENTRY = 217, + SDL_SCANCODE_KP_BINARY = 218, + SDL_SCANCODE_KP_OCTAL = 219, + SDL_SCANCODE_KP_DECIMAL = 220, + SDL_SCANCODE_KP_HEXADECIMAL = 221, + + SDL_SCANCODE_LCTRL = 224, + SDL_SCANCODE_LSHIFT = 225, + SDL_SCANCODE_LALT = 226, /**< alt, option */ + SDL_SCANCODE_LGUI = 227, /**< windows, command (apple), meta */ + SDL_SCANCODE_RCTRL = 228, + SDL_SCANCODE_RSHIFT = 229, + SDL_SCANCODE_RALT = 230, /**< alt gr, option */ + SDL_SCANCODE_RGUI = 231, /**< windows, command (apple), meta */ + + SDL_SCANCODE_MODE = 257, /**< I'm not sure if this is really not covered + * by any of the above, but since there's a + * special KMOD_MODE for it I'm adding it here + */ + + /* @} *//* Usage page 0x07 */ + + /** + * \name Usage page 0x0C + * + * These values are mapped from usage page 0x0C (USB consumer page). + * See https://usb.org/sites/default/files/hut1_2.pdf + * + * There are way more keys in the spec than we can represent in the + * current scancode range, so pick the ones that commonly come up in + * real world usage. + */ + /* @{ */ + + SDL_SCANCODE_AUDIONEXT = 258, + SDL_SCANCODE_AUDIOPREV = 259, + SDL_SCANCODE_AUDIOSTOP = 260, + SDL_SCANCODE_AUDIOPLAY = 261, + SDL_SCANCODE_AUDIOMUTE = 262, + SDL_SCANCODE_MEDIASELECT = 263, + SDL_SCANCODE_WWW = 264, /**< AL Internet Browser */ + SDL_SCANCODE_MAIL = 265, + SDL_SCANCODE_CALCULATOR = 266, /**< AL Calculator */ + SDL_SCANCODE_COMPUTER = 267, + SDL_SCANCODE_AC_SEARCH = 268, /**< AC Search */ + SDL_SCANCODE_AC_HOME = 269, /**< AC Home */ + SDL_SCANCODE_AC_BACK = 270, /**< AC Back */ + SDL_SCANCODE_AC_FORWARD = 271, /**< AC Forward */ + SDL_SCANCODE_AC_STOP = 272, /**< AC Stop */ + SDL_SCANCODE_AC_REFRESH = 273, /**< AC Refresh */ + SDL_SCANCODE_AC_BOOKMARKS = 274, /**< AC Bookmarks */ + + /* @} *//* Usage page 0x0C */ + + /** + * \name Walther keys + * + * These are values that Christian Walther added (for mac keyboard?). + */ + /* @{ */ + + SDL_SCANCODE_BRIGHTNESSDOWN = 275, + SDL_SCANCODE_BRIGHTNESSUP = 276, + SDL_SCANCODE_DISPLAYSWITCH = 277, /**< display mirroring/dual display + switch, video mode switch */ + SDL_SCANCODE_KBDILLUMTOGGLE = 278, + SDL_SCANCODE_KBDILLUMDOWN = 279, + SDL_SCANCODE_KBDILLUMUP = 280, + SDL_SCANCODE_EJECT = 281, + SDL_SCANCODE_SLEEP = 282, /**< SC System Sleep */ + + SDL_SCANCODE_APP1 = 283, + SDL_SCANCODE_APP2 = 284, + + /* @} *//* Walther keys */ + + /** + * \name Usage page 0x0C (additional media keys) + * + * These values are mapped from usage page 0x0C (USB consumer page). + */ + /* @{ */ + + SDL_SCANCODE_AUDIOREWIND = 285, + SDL_SCANCODE_AUDIOFASTFORWARD = 286, + + /* @} *//* Usage page 0x0C (additional media keys) */ + + /** + * \name Mobile keys + * + * These are values that are often used on mobile phones. + */ + /* @{ */ + + SDL_SCANCODE_SOFTLEFT = 287, /**< Usually situated below the display on phones and + used as a multi-function feature key for selecting + a software defined function shown on the bottom left + of the display. */ + SDL_SCANCODE_SOFTRIGHT = 288, /**< Usually situated below the display on phones and + used as a multi-function feature key for selecting + a software defined function shown on the bottom right + of the display. */ + SDL_SCANCODE_CALL = 289, /**< Used for accepting phone calls. */ + SDL_SCANCODE_ENDCALL = 290, /**< Used for rejecting phone calls. */ + + /* @} *//* Mobile keys */ + + /* Add any other keys here. */ + + SDL_NUM_SCANCODES = 512 /**< not a key, just marks the number of scancodes + for array bounds */ +} SDL_Scancode; + +#endif /* SDL_scancode_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_sensor.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_sensor.h new file mode 100644 index 00000000..9ecce44b --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_sensor.h @@ -0,0 +1,322 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_sensor.h + * + * Include file for SDL sensor event handling + * + */ + +#ifndef SDL_sensor_h_ +#define SDL_sensor_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +/* *INDENT-OFF* */ +extern "C" { +/* *INDENT-ON* */ +#endif + +/** + * \brief SDL_sensor.h + * + * In order to use these functions, SDL_Init() must have been called + * with the ::SDL_INIT_SENSOR flag. This causes SDL to scan the system + * for sensors, and load appropriate drivers. + */ + +struct _SDL_Sensor; +typedef struct _SDL_Sensor SDL_Sensor; + +/** + * This is a unique ID for a sensor for the time it is connected to the system, + * and is never reused for the lifetime of the application. + * + * The ID value starts at 0 and increments from there. The value -1 is an invalid ID. + */ +typedef Sint32 SDL_SensorID; + +/* The different sensors defined by SDL + * + * Additional sensors may be available, using platform dependent semantics. + * + * Hare are the additional Android sensors: + * https://developer.android.com/reference/android/hardware/SensorEvent.html#values + */ +typedef enum +{ + SDL_SENSOR_INVALID = -1, /**< Returned for an invalid sensor */ + SDL_SENSOR_UNKNOWN, /**< Unknown sensor type */ + SDL_SENSOR_ACCEL, /**< Accelerometer */ + SDL_SENSOR_GYRO, /**< Gyroscope */ + SDL_SENSOR_ACCEL_L, /**< Accelerometer for left Joy-Con controller and Wii nunchuk */ + SDL_SENSOR_GYRO_L, /**< Gyroscope for left Joy-Con controller */ + SDL_SENSOR_ACCEL_R, /**< Accelerometer for right Joy-Con controller */ + SDL_SENSOR_GYRO_R /**< Gyroscope for right Joy-Con controller */ +} SDL_SensorType; + +/** + * Accelerometer sensor + * + * The accelerometer returns the current acceleration in SI meters per + * second squared. This measurement includes the force of gravity, so + * a device at rest will have an value of SDL_STANDARD_GRAVITY away + * from the center of the earth, which is a positive Y value. + * + * values[0]: Acceleration on the x axis + * values[1]: Acceleration on the y axis + * values[2]: Acceleration on the z axis + * + * For phones held in portrait mode and game controllers held in front of you, + * the axes are defined as follows: + * -X ... +X : left ... right + * -Y ... +Y : bottom ... top + * -Z ... +Z : farther ... closer + * + * The axis data is not changed when the phone is rotated. + * + * \sa SDL_GetDisplayOrientation() + */ +#define SDL_STANDARD_GRAVITY 9.80665f + +/** + * Gyroscope sensor + * + * The gyroscope returns the current rate of rotation in radians per second. + * The rotation is positive in the counter-clockwise direction. That is, + * an observer looking from a positive location on one of the axes would + * see positive rotation on that axis when it appeared to be rotating + * counter-clockwise. + * + * values[0]: Angular speed around the x axis (pitch) + * values[1]: Angular speed around the y axis (yaw) + * values[2]: Angular speed around the z axis (roll) + * + * For phones held in portrait mode and game controllers held in front of you, + * the axes are defined as follows: + * -X ... +X : left ... right + * -Y ... +Y : bottom ... top + * -Z ... +Z : farther ... closer + * + * The axis data is not changed when the phone or controller is rotated. + * + * \sa SDL_GetDisplayOrientation() + */ + +/* Function prototypes */ + +/** + * Locking for multi-threaded access to the sensor API + * + * If you are using the sensor API or handling events from multiple threads + * you should use these locking functions to protect access to the sensors. + * + * In particular, you are guaranteed that the sensor list won't change, so the + * API functions that take a sensor index will be valid, and sensor events + * will not be delivered. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC void SDLCALL SDL_LockSensors(void); +extern DECLSPEC void SDLCALL SDL_UnlockSensors(void); + +/** + * Count the number of sensors attached to the system right now. + * + * \returns the number of sensors detected. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC int SDLCALL SDL_NumSensors(void); + +/** + * Get the implementation dependent name of a sensor. + * + * \param device_index The sensor to obtain name from + * \returns the sensor name, or NULL if `device_index` is out of range. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC const char *SDLCALL SDL_SensorGetDeviceName(int device_index); + +/** + * Get the type of a sensor. + * + * \param device_index The sensor to get the type from + * \returns the SDL_SensorType, or `SDL_SENSOR_INVALID` if `device_index` is + * out of range. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC SDL_SensorType SDLCALL SDL_SensorGetDeviceType(int device_index); + +/** + * Get the platform dependent type of a sensor. + * + * \param device_index The sensor to check + * \returns the sensor platform dependent type, or -1 if `device_index` is out + * of range. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC int SDLCALL SDL_SensorGetDeviceNonPortableType(int device_index); + +/** + * Get the instance ID of a sensor. + * + * \param device_index The sensor to get instance id from + * \returns the sensor instance ID, or -1 if `device_index` is out of range. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC SDL_SensorID SDLCALL SDL_SensorGetDeviceInstanceID(int device_index); + +/** + * Open a sensor for use. + * + * \param device_index The sensor to open + * \returns an SDL_Sensor sensor object, or NULL if an error occurred. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC SDL_Sensor *SDLCALL SDL_SensorOpen(int device_index); + +/** + * Return the SDL_Sensor associated with an instance id. + * + * \param instance_id The sensor from instance id + * \returns an SDL_Sensor object. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC SDL_Sensor *SDLCALL SDL_SensorFromInstanceID(SDL_SensorID instance_id); + +/** + * Get the implementation dependent name of a sensor + * + * \param sensor The SDL_Sensor object + * \returns the sensor name, or NULL if `sensor` is NULL. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC const char *SDLCALL SDL_SensorGetName(SDL_Sensor *sensor); + +/** + * Get the type of a sensor. + * + * \param sensor The SDL_Sensor object to inspect + * \returns the SDL_SensorType type, or `SDL_SENSOR_INVALID` if `sensor` is + * NULL. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC SDL_SensorType SDLCALL SDL_SensorGetType(SDL_Sensor *sensor); + +/** + * Get the platform dependent type of a sensor. + * + * \param sensor The SDL_Sensor object to inspect + * \returns the sensor platform dependent type, or -1 if `sensor` is NULL. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC int SDLCALL SDL_SensorGetNonPortableType(SDL_Sensor *sensor); + +/** + * Get the instance ID of a sensor. + * + * \param sensor The SDL_Sensor object to inspect + * \returns the sensor instance ID, or -1 if `sensor` is NULL. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC SDL_SensorID SDLCALL SDL_SensorGetInstanceID(SDL_Sensor *sensor); + +/** + * Get the current state of an opened sensor. + * + * The number of values and interpretation of the data is sensor dependent. + * + * \param sensor The SDL_Sensor object to query + * \param data A pointer filled with the current sensor state + * \param num_values The number of values to write to data + * \returns 0 or -1 if an error occurred. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC int SDLCALL SDL_SensorGetData(SDL_Sensor *sensor, float *data, int num_values); + +/** + * Get the current state of an opened sensor with the timestamp of the last + * update. + * + * The number of values and interpretation of the data is sensor dependent. + * + * \param sensor The SDL_Sensor object to query + * \param timestamp A pointer filled with the timestamp in microseconds of the + * current sensor reading if available, or 0 if not + * \param data A pointer filled with the current sensor state + * \param num_values The number of values to write to data + * \returns 0 or -1 if an error occurred. + * + * \since This function is available since SDL 2.26.0. + */ +extern DECLSPEC int SDLCALL SDL_SensorGetDataWithTimestamp(SDL_Sensor *sensor, Uint64 *timestamp, float *data, int num_values); + +/** + * Close a sensor previously opened with SDL_SensorOpen(). + * + * \param sensor The SDL_Sensor object to close + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC void SDLCALL SDL_SensorClose(SDL_Sensor *sensor); + +/** + * Update the current state of the open sensors. + * + * This is called automatically by the event loop if sensor events are + * enabled. + * + * This needs to be called from the thread that initialized the sensor + * subsystem. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC void SDLCALL SDL_SensorUpdate(void); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +/* *INDENT-OFF* */ +} +/* *INDENT-ON* */ +#endif +#include "close_code.h" + +#endif /* SDL_sensor_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_shape.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_shape.h new file mode 100644 index 00000000..f66babc0 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_shape.h @@ -0,0 +1,155 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +#ifndef SDL_shape_h_ +#define SDL_shape_h_ + +#include "SDL_stdinc.h" +#include "SDL_pixels.h" +#include "SDL_rect.h" +#include "SDL_surface.h" +#include "SDL_video.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** \file SDL_shape.h + * + * Header file for the shaped window API. + */ + +#define SDL_NONSHAPEABLE_WINDOW -1 +#define SDL_INVALID_SHAPE_ARGUMENT -2 +#define SDL_WINDOW_LACKS_SHAPE -3 + +/** + * Create a window that can be shaped with the specified position, dimensions, + * and flags. + * + * \param title The title of the window, in UTF-8 encoding. + * \param x The x position of the window, ::SDL_WINDOWPOS_CENTERED, or + * ::SDL_WINDOWPOS_UNDEFINED. + * \param y The y position of the window, ::SDL_WINDOWPOS_CENTERED, or + * ::SDL_WINDOWPOS_UNDEFINED. + * \param w The width of the window. + * \param h The height of the window. + * \param flags The flags for the window, a mask of SDL_WINDOW_BORDERLESS with + * any of the following: ::SDL_WINDOW_OPENGL, + * ::SDL_WINDOW_INPUT_GRABBED, ::SDL_WINDOW_HIDDEN, + * ::SDL_WINDOW_RESIZABLE, ::SDL_WINDOW_MAXIMIZED, + * ::SDL_WINDOW_MINIMIZED, ::SDL_WINDOW_BORDERLESS is always set, + * and ::SDL_WINDOW_FULLSCREEN is always unset. + * \return the window created, or NULL if window creation failed. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_DestroyWindow + */ +extern DECLSPEC SDL_Window * SDLCALL SDL_CreateShapedWindow(const char *title,unsigned int x,unsigned int y,unsigned int w,unsigned int h,Uint32 flags); + +/** + * Return whether the given window is a shaped window. + * + * \param window The window to query for being shaped. + * \return SDL_TRUE if the window is a window that can be shaped, SDL_FALSE if + * the window is unshaped or NULL. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateShapedWindow + */ +extern DECLSPEC SDL_bool SDLCALL SDL_IsShapedWindow(const SDL_Window *window); + +/** \brief An enum denoting the specific type of contents present in an SDL_WindowShapeParams union. */ +typedef enum { + /** \brief The default mode, a binarized alpha cutoff of 1. */ + ShapeModeDefault, + /** \brief A binarized alpha cutoff with a given integer value. */ + ShapeModeBinarizeAlpha, + /** \brief A binarized alpha cutoff with a given integer value, but with the opposite comparison. */ + ShapeModeReverseBinarizeAlpha, + /** \brief A color key is applied. */ + ShapeModeColorKey +} WindowShapeMode; + +#define SDL_SHAPEMODEALPHA(mode) (mode == ShapeModeDefault || mode == ShapeModeBinarizeAlpha || mode == ShapeModeReverseBinarizeAlpha) + +/** \brief A union containing parameters for shaped windows. */ +typedef union { + /** \brief A cutoff alpha value for binarization of the window shape's alpha channel. */ + Uint8 binarizationCutoff; + SDL_Color colorKey; +} SDL_WindowShapeParams; + +/** \brief A struct that tags the SDL_WindowShapeParams union with an enum describing the type of its contents. */ +typedef struct SDL_WindowShapeMode { + /** \brief The mode of these window-shape parameters. */ + WindowShapeMode mode; + /** \brief Window-shape parameters. */ + SDL_WindowShapeParams parameters; +} SDL_WindowShapeMode; + +/** + * Set the shape and parameters of a shaped window. + * + * \param window The shaped window whose parameters should be set. + * \param shape A surface encoding the desired shape for the window. + * \param shape_mode The parameters to set for the shaped window. + * \return 0 on success, SDL_INVALID_SHAPE_ARGUMENT on an invalid shape + * argument, or SDL_NONSHAPEABLE_WINDOW if the SDL_Window given does + * not reference a valid shaped window. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_WindowShapeMode + * \sa SDL_GetShapedWindowMode + */ +extern DECLSPEC int SDLCALL SDL_SetWindowShape(SDL_Window *window,SDL_Surface *shape,SDL_WindowShapeMode *shape_mode); + +/** + * Get the shape parameters of a shaped window. + * + * \param window The shaped window whose parameters should be retrieved. + * \param shape_mode An empty shape-mode structure to fill, or NULL to check + * whether the window has a shape. + * \return 0 if the window has a shape and, provided shape_mode was not NULL, + * shape_mode has been filled with the mode data, + * SDL_NONSHAPEABLE_WINDOW if the SDL_Window given is not a shaped + * window, or SDL_WINDOW_LACKS_SHAPE if the SDL_Window given is a + * shapeable window currently lacking a shape. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_WindowShapeMode + * \sa SDL_SetWindowShape + */ +extern DECLSPEC int SDLCALL SDL_GetShapedWindowMode(SDL_Window *window,SDL_WindowShapeMode *shape_mode); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_shape_h_ */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_stdinc.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_stdinc.h new file mode 100644 index 00000000..182ed86e --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_stdinc.h @@ -0,0 +1,838 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_stdinc.h + * + * This is a general header that includes C language support. + */ + +#ifndef SDL_stdinc_h_ +#define SDL_stdinc_h_ + +#include "SDL_config.h" + +#ifdef HAVE_SYS_TYPES_H +#include +#endif +#ifdef HAVE_STDIO_H +#include +#endif +#if defined(STDC_HEADERS) +# include +# include +# include +#else +# if defined(HAVE_STDLIB_H) +# include +# elif defined(HAVE_MALLOC_H) +# include +# endif +# if defined(HAVE_STDDEF_H) +# include +# endif +# if defined(HAVE_STDARG_H) +# include +# endif +#endif +#ifdef HAVE_STRING_H +# if !defined(STDC_HEADERS) && defined(HAVE_MEMORY_H) +# include +# endif +# include +#endif +#ifdef HAVE_STRINGS_H +# include +#endif +#ifdef HAVE_WCHAR_H +# include +#endif +#if defined(HAVE_INTTYPES_H) +# include +#elif defined(HAVE_STDINT_H) +# include +#endif +#ifdef HAVE_CTYPE_H +# include +#endif +#ifdef HAVE_MATH_H +# if defined(_MSC_VER) +/* Defining _USE_MATH_DEFINES is required to get M_PI to be defined on + Visual Studio. See http://msdn.microsoft.com/en-us/library/4hwaceh6.aspx + for more information. +*/ +# ifndef _USE_MATH_DEFINES +# define _USE_MATH_DEFINES +# endif +# endif +# include +#endif +#ifdef HAVE_FLOAT_H +# include +#endif +#if defined(HAVE_ALLOCA) && !defined(alloca) +# if defined(HAVE_ALLOCA_H) +# include +# elif defined(__GNUC__) +# define alloca __builtin_alloca +# elif defined(_MSC_VER) +# include +# define alloca _alloca +# elif defined(__WATCOMC__) +# include +# elif defined(__BORLANDC__) +# include +# elif defined(__DMC__) +# include +# elif defined(__AIX__) +#pragma alloca +# elif defined(__MRC__) +void *alloca(unsigned); +# else +char *alloca(); +# endif +#endif + +#ifdef SIZE_MAX +# define SDL_SIZE_MAX SIZE_MAX +#else +# define SDL_SIZE_MAX ((size_t) -1) +#endif + +/** + * Check if the compiler supports a given builtin. + * Supported by virtually all clang versions and recent gcc. Use this + * instead of checking the clang version if possible. + */ +#ifdef __has_builtin +#define _SDL_HAS_BUILTIN(x) __has_builtin(x) +#else +#define _SDL_HAS_BUILTIN(x) 0 +#endif + +/** + * The number of elements in an array. + */ +#define SDL_arraysize(array) (sizeof(array)/sizeof(array[0])) +#define SDL_TABLESIZE(table) SDL_arraysize(table) + +/** + * Macro useful for building other macros with strings in them + * + * e.g. #define LOG_ERROR(X) OutputDebugString(SDL_STRINGIFY_ARG(__FUNCTION__) ": " X "\n") + */ +#define SDL_STRINGIFY_ARG(arg) #arg + +/** + * \name Cast operators + * + * Use proper C++ casts when compiled as C++ to be compatible with the option + * -Wold-style-cast of GCC (and -Werror=old-style-cast in GCC 4.2 and above). + */ +/* @{ */ +#ifdef __cplusplus +#define SDL_reinterpret_cast(type, expression) reinterpret_cast(expression) +#define SDL_static_cast(type, expression) static_cast(expression) +#define SDL_const_cast(type, expression) const_cast(expression) +#else +#define SDL_reinterpret_cast(type, expression) ((type)(expression)) +#define SDL_static_cast(type, expression) ((type)(expression)) +#define SDL_const_cast(type, expression) ((type)(expression)) +#endif +/* @} *//* Cast operators */ + +/* Define a four character code as a Uint32 */ +#define SDL_FOURCC(A, B, C, D) \ + ((SDL_static_cast(Uint32, SDL_static_cast(Uint8, (A))) << 0) | \ + (SDL_static_cast(Uint32, SDL_static_cast(Uint8, (B))) << 8) | \ + (SDL_static_cast(Uint32, SDL_static_cast(Uint8, (C))) << 16) | \ + (SDL_static_cast(Uint32, SDL_static_cast(Uint8, (D))) << 24)) + +/** + * \name Basic data types + */ +/* @{ */ + +#ifdef __CC_ARM +/* ARM's compiler throws warnings if we use an enum: like "SDL_bool x = a < b;" */ +#define SDL_FALSE 0 +#define SDL_TRUE 1 +typedef int SDL_bool; +#else +typedef enum +{ + SDL_FALSE = 0, + SDL_TRUE = 1 +} SDL_bool; +#endif + +/** + * \brief A signed 8-bit integer type. + */ +#define SDL_MAX_SINT8 ((Sint8)0x7F) /* 127 */ +#define SDL_MIN_SINT8 ((Sint8)(~0x7F)) /* -128 */ +typedef int8_t Sint8; +/** + * \brief An unsigned 8-bit integer type. + */ +#define SDL_MAX_UINT8 ((Uint8)0xFF) /* 255 */ +#define SDL_MIN_UINT8 ((Uint8)0x00) /* 0 */ +typedef uint8_t Uint8; +/** + * \brief A signed 16-bit integer type. + */ +#define SDL_MAX_SINT16 ((Sint16)0x7FFF) /* 32767 */ +#define SDL_MIN_SINT16 ((Sint16)(~0x7FFF)) /* -32768 */ +typedef int16_t Sint16; +/** + * \brief An unsigned 16-bit integer type. + */ +#define SDL_MAX_UINT16 ((Uint16)0xFFFF) /* 65535 */ +#define SDL_MIN_UINT16 ((Uint16)0x0000) /* 0 */ +typedef uint16_t Uint16; +/** + * \brief A signed 32-bit integer type. + */ +#define SDL_MAX_SINT32 ((Sint32)0x7FFFFFFF) /* 2147483647 */ +#define SDL_MIN_SINT32 ((Sint32)(~0x7FFFFFFF)) /* -2147483648 */ +typedef int32_t Sint32; +/** + * \brief An unsigned 32-bit integer type. + */ +#define SDL_MAX_UINT32 ((Uint32)0xFFFFFFFFu) /* 4294967295 */ +#define SDL_MIN_UINT32 ((Uint32)0x00000000) /* 0 */ +typedef uint32_t Uint32; + +/** + * \brief A signed 64-bit integer type. + */ +#define SDL_MAX_SINT64 ((Sint64)0x7FFFFFFFFFFFFFFFll) /* 9223372036854775807 */ +#define SDL_MIN_SINT64 ((Sint64)(~0x7FFFFFFFFFFFFFFFll)) /* -9223372036854775808 */ +typedef int64_t Sint64; +/** + * \brief An unsigned 64-bit integer type. + */ +#define SDL_MAX_UINT64 ((Uint64)0xFFFFFFFFFFFFFFFFull) /* 18446744073709551615 */ +#define SDL_MIN_UINT64 ((Uint64)(0x0000000000000000ull)) /* 0 */ +typedef uint64_t Uint64; + +/* @} *//* Basic data types */ + +/** + * \name Floating-point constants + */ +/* @{ */ + +#ifdef FLT_EPSILON +#define SDL_FLT_EPSILON FLT_EPSILON +#else +#define SDL_FLT_EPSILON 1.1920928955078125e-07F /* 0x0.000002p0 */ +#endif + +/* @} *//* Floating-point constants */ + +/* Make sure we have macros for printing width-based integers. + * should define these but this is not true all platforms. + * (for example win32) */ +#ifndef SDL_PRIs64 +#ifdef PRIs64 +#define SDL_PRIs64 PRIs64 +#elif defined(__WIN32__) || defined(__GDK__) +#define SDL_PRIs64 "I64d" +#elif defined(__LINUX__) && defined(__LP64__) +#define SDL_PRIs64 "ld" +#else +#define SDL_PRIs64 "lld" +#endif +#endif +#ifndef SDL_PRIu64 +#ifdef PRIu64 +#define SDL_PRIu64 PRIu64 +#elif defined(__WIN32__) || defined(__GDK__) +#define SDL_PRIu64 "I64u" +#elif defined(__LINUX__) && defined(__LP64__) +#define SDL_PRIu64 "lu" +#else +#define SDL_PRIu64 "llu" +#endif +#endif +#ifndef SDL_PRIx64 +#ifdef PRIx64 +#define SDL_PRIx64 PRIx64 +#elif defined(__WIN32__) || defined(__GDK__) +#define SDL_PRIx64 "I64x" +#elif defined(__LINUX__) && defined(__LP64__) +#define SDL_PRIx64 "lx" +#else +#define SDL_PRIx64 "llx" +#endif +#endif +#ifndef SDL_PRIX64 +#ifdef PRIX64 +#define SDL_PRIX64 PRIX64 +#elif defined(__WIN32__) || defined(__GDK__) +#define SDL_PRIX64 "I64X" +#elif defined(__LINUX__) && defined(__LP64__) +#define SDL_PRIX64 "lX" +#else +#define SDL_PRIX64 "llX" +#endif +#endif +#ifndef SDL_PRIs32 +#ifdef PRId32 +#define SDL_PRIs32 PRId32 +#else +#define SDL_PRIs32 "d" +#endif +#endif +#ifndef SDL_PRIu32 +#ifdef PRIu32 +#define SDL_PRIu32 PRIu32 +#else +#define SDL_PRIu32 "u" +#endif +#endif +#ifndef SDL_PRIx32 +#ifdef PRIx32 +#define SDL_PRIx32 PRIx32 +#else +#define SDL_PRIx32 "x" +#endif +#endif +#ifndef SDL_PRIX32 +#ifdef PRIX32 +#define SDL_PRIX32 PRIX32 +#else +#define SDL_PRIX32 "X" +#endif +#endif + +/* Annotations to help code analysis tools */ +#ifdef SDL_DISABLE_ANALYZE_MACROS +#define SDL_IN_BYTECAP(x) +#define SDL_INOUT_Z_CAP(x) +#define SDL_OUT_Z_CAP(x) +#define SDL_OUT_CAP(x) +#define SDL_OUT_BYTECAP(x) +#define SDL_OUT_Z_BYTECAP(x) +#define SDL_PRINTF_FORMAT_STRING +#define SDL_SCANF_FORMAT_STRING +#define SDL_PRINTF_VARARG_FUNC( fmtargnumber ) +#define SDL_SCANF_VARARG_FUNC( fmtargnumber ) +#else +#if defined(_MSC_VER) && (_MSC_VER >= 1600) /* VS 2010 and above */ +#include + +#define SDL_IN_BYTECAP(x) _In_bytecount_(x) +#define SDL_INOUT_Z_CAP(x) _Inout_z_cap_(x) +#define SDL_OUT_Z_CAP(x) _Out_z_cap_(x) +#define SDL_OUT_CAP(x) _Out_cap_(x) +#define SDL_OUT_BYTECAP(x) _Out_bytecap_(x) +#define SDL_OUT_Z_BYTECAP(x) _Out_z_bytecap_(x) + +#define SDL_PRINTF_FORMAT_STRING _Printf_format_string_ +#define SDL_SCANF_FORMAT_STRING _Scanf_format_string_impl_ +#else +#define SDL_IN_BYTECAP(x) +#define SDL_INOUT_Z_CAP(x) +#define SDL_OUT_Z_CAP(x) +#define SDL_OUT_CAP(x) +#define SDL_OUT_BYTECAP(x) +#define SDL_OUT_Z_BYTECAP(x) +#define SDL_PRINTF_FORMAT_STRING +#define SDL_SCANF_FORMAT_STRING +#endif +#if defined(__GNUC__) +#define SDL_PRINTF_VARARG_FUNC( fmtargnumber ) __attribute__ (( format( __printf__, fmtargnumber, fmtargnumber+1 ))) +#define SDL_SCANF_VARARG_FUNC( fmtargnumber ) __attribute__ (( format( __scanf__, fmtargnumber, fmtargnumber+1 ))) +#else +#define SDL_PRINTF_VARARG_FUNC( fmtargnumber ) +#define SDL_SCANF_VARARG_FUNC( fmtargnumber ) +#endif +#endif /* SDL_DISABLE_ANALYZE_MACROS */ + +#ifndef SDL_COMPILE_TIME_ASSERT +#if defined(__cplusplus) +#if (__cplusplus >= 201103L) +#define SDL_COMPILE_TIME_ASSERT(name, x) static_assert(x, #x) +#endif +#elif defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 201112L) +#define SDL_COMPILE_TIME_ASSERT(name, x) _Static_assert(x, #x) +#endif +#endif /* !SDL_COMPILE_TIME_ASSERT */ + +#ifndef SDL_COMPILE_TIME_ASSERT +/* universal, but may trigger -Wunused-local-typedefs */ +#define SDL_COMPILE_TIME_ASSERT(name, x) \ + typedef int SDL_compile_time_assert_ ## name[(x) * 2 - 1] +#endif + +/** \cond */ +#ifndef DOXYGEN_SHOULD_IGNORE_THIS +SDL_COMPILE_TIME_ASSERT(uint8, sizeof(Uint8) == 1); +SDL_COMPILE_TIME_ASSERT(sint8, sizeof(Sint8) == 1); +SDL_COMPILE_TIME_ASSERT(uint16, sizeof(Uint16) == 2); +SDL_COMPILE_TIME_ASSERT(sint16, sizeof(Sint16) == 2); +SDL_COMPILE_TIME_ASSERT(uint32, sizeof(Uint32) == 4); +SDL_COMPILE_TIME_ASSERT(sint32, sizeof(Sint32) == 4); +SDL_COMPILE_TIME_ASSERT(uint64, sizeof(Uint64) == 8); +SDL_COMPILE_TIME_ASSERT(sint64, sizeof(Sint64) == 8); +#endif /* DOXYGEN_SHOULD_IGNORE_THIS */ +/** \endcond */ + +/* Check to make sure enums are the size of ints, for structure packing. + For both Watcom C/C++ and Borland C/C++ the compiler option that makes + enums having the size of an int must be enabled. + This is "-b" for Borland C/C++ and "-ei" for Watcom C/C++ (v11). +*/ + +/** \cond */ +#ifndef DOXYGEN_SHOULD_IGNORE_THIS +#if !defined(__ANDROID__) && !defined(__VITA__) && !defined(__3DS__) + /* TODO: include/SDL_stdinc.h:174: error: size of array 'SDL_dummy_enum' is negative */ +typedef enum +{ + DUMMY_ENUM_VALUE +} SDL_DUMMY_ENUM; + +SDL_COMPILE_TIME_ASSERT(enum, sizeof(SDL_DUMMY_ENUM) == sizeof(int)); +#endif +#endif /* DOXYGEN_SHOULD_IGNORE_THIS */ +/** \endcond */ + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +#ifdef HAVE_ALLOCA +#define SDL_stack_alloc(type, count) (type*)alloca(sizeof(type)*(count)) +#define SDL_stack_free(data) +#else +#define SDL_stack_alloc(type, count) (type*)SDL_malloc(sizeof(type)*(count)) +#define SDL_stack_free(data) SDL_free(data) +#endif + +extern DECLSPEC void *SDLCALL SDL_malloc(size_t size); +extern DECLSPEC void *SDLCALL SDL_calloc(size_t nmemb, size_t size); +extern DECLSPEC void *SDLCALL SDL_realloc(void *mem, size_t size); +extern DECLSPEC void SDLCALL SDL_free(void *mem); + +typedef void *(SDLCALL *SDL_malloc_func)(size_t size); +typedef void *(SDLCALL *SDL_calloc_func)(size_t nmemb, size_t size); +typedef void *(SDLCALL *SDL_realloc_func)(void *mem, size_t size); +typedef void (SDLCALL *SDL_free_func)(void *mem); + +/** + * Get the original set of SDL memory functions + * + * \since This function is available since SDL 2.24.0. + */ +extern DECLSPEC void SDLCALL SDL_GetOriginalMemoryFunctions(SDL_malloc_func *malloc_func, + SDL_calloc_func *calloc_func, + SDL_realloc_func *realloc_func, + SDL_free_func *free_func); + +/** + * Get the current set of SDL memory functions + * + * \since This function is available since SDL 2.0.7. + */ +extern DECLSPEC void SDLCALL SDL_GetMemoryFunctions(SDL_malloc_func *malloc_func, + SDL_calloc_func *calloc_func, + SDL_realloc_func *realloc_func, + SDL_free_func *free_func); + +/** + * Replace SDL's memory allocation functions with a custom set + * + * \since This function is available since SDL 2.0.7. + */ +extern DECLSPEC int SDLCALL SDL_SetMemoryFunctions(SDL_malloc_func malloc_func, + SDL_calloc_func calloc_func, + SDL_realloc_func realloc_func, + SDL_free_func free_func); + +/** + * Get the number of outstanding (unfreed) allocations + * + * \since This function is available since SDL 2.0.7. + */ +extern DECLSPEC int SDLCALL SDL_GetNumAllocations(void); + +extern DECLSPEC char *SDLCALL SDL_getenv(const char *name); +extern DECLSPEC int SDLCALL SDL_setenv(const char *name, const char *value, int overwrite); + +extern DECLSPEC void SDLCALL SDL_qsort(void *base, size_t nmemb, size_t size, int (SDLCALL *compare) (const void *, const void *)); +extern DECLSPEC void * SDLCALL SDL_bsearch(const void *key, const void *base, size_t nmemb, size_t size, int (SDLCALL *compare) (const void *, const void *)); + +extern DECLSPEC int SDLCALL SDL_abs(int x); + +/* NOTE: these double-evaluate their arguments, so you should never have side effects in the parameters */ +#define SDL_min(x, y) (((x) < (y)) ? (x) : (y)) +#define SDL_max(x, y) (((x) > (y)) ? (x) : (y)) +#define SDL_clamp(x, a, b) (((x) < (a)) ? (a) : (((x) > (b)) ? (b) : (x))) + +extern DECLSPEC int SDLCALL SDL_isalpha(int x); +extern DECLSPEC int SDLCALL SDL_isalnum(int x); +extern DECLSPEC int SDLCALL SDL_isblank(int x); +extern DECLSPEC int SDLCALL SDL_iscntrl(int x); +extern DECLSPEC int SDLCALL SDL_isdigit(int x); +extern DECLSPEC int SDLCALL SDL_isxdigit(int x); +extern DECLSPEC int SDLCALL SDL_ispunct(int x); +extern DECLSPEC int SDLCALL SDL_isspace(int x); +extern DECLSPEC int SDLCALL SDL_isupper(int x); +extern DECLSPEC int SDLCALL SDL_islower(int x); +extern DECLSPEC int SDLCALL SDL_isprint(int x); +extern DECLSPEC int SDLCALL SDL_isgraph(int x); +extern DECLSPEC int SDLCALL SDL_toupper(int x); +extern DECLSPEC int SDLCALL SDL_tolower(int x); + +extern DECLSPEC Uint16 SDLCALL SDL_crc16(Uint16 crc, const void *data, size_t len); +extern DECLSPEC Uint32 SDLCALL SDL_crc32(Uint32 crc, const void *data, size_t len); + +extern DECLSPEC void *SDLCALL SDL_memset(SDL_OUT_BYTECAP(len) void *dst, int c, size_t len); + +#define SDL_zero(x) SDL_memset(&(x), 0, sizeof((x))) +#define SDL_zerop(x) SDL_memset((x), 0, sizeof(*(x))) +#define SDL_zeroa(x) SDL_memset((x), 0, sizeof((x))) + +#define SDL_copyp(dst, src) \ + { SDL_COMPILE_TIME_ASSERT(SDL_copyp, sizeof (*(dst)) == sizeof (*(src))); } \ + SDL_memcpy((dst), (src), sizeof (*(src))) + + +/* Note that memset() is a byte assignment and this is a 32-bit assignment, so they're not directly equivalent. */ +SDL_FORCE_INLINE void SDL_memset4(void *dst, Uint32 val, size_t dwords) +{ +#if defined(__GNUC__) && defined(__i386__) + int u0, u1, u2; + __asm__ __volatile__ ( + "cld \n\t" + "rep ; stosl \n\t" + : "=&D" (u0), "=&a" (u1), "=&c" (u2) + : "0" (dst), "1" (val), "2" (SDL_static_cast(Uint32, dwords)) + : "memory" + ); +#else + size_t _n = (dwords + 3) / 4; + Uint32 *_p = SDL_static_cast(Uint32 *, dst); + Uint32 _val = (val); + if (dwords == 0) { + return; + } + switch (dwords % 4) { + case 0: do { *_p++ = _val; SDL_FALLTHROUGH; + case 3: *_p++ = _val; SDL_FALLTHROUGH; + case 2: *_p++ = _val; SDL_FALLTHROUGH; + case 1: *_p++ = _val; + } while ( --_n ); + } +#endif +} + +extern DECLSPEC void *SDLCALL SDL_memcpy(SDL_OUT_BYTECAP(len) void *dst, SDL_IN_BYTECAP(len) const void *src, size_t len); + +extern DECLSPEC void *SDLCALL SDL_memmove(SDL_OUT_BYTECAP(len) void *dst, SDL_IN_BYTECAP(len) const void *src, size_t len); +extern DECLSPEC int SDLCALL SDL_memcmp(const void *s1, const void *s2, size_t len); + +extern DECLSPEC size_t SDLCALL SDL_wcslen(const wchar_t *wstr); +extern DECLSPEC size_t SDLCALL SDL_wcslcpy(SDL_OUT_Z_CAP(maxlen) wchar_t *dst, const wchar_t *src, size_t maxlen); +extern DECLSPEC size_t SDLCALL SDL_wcslcat(SDL_INOUT_Z_CAP(maxlen) wchar_t *dst, const wchar_t *src, size_t maxlen); +extern DECLSPEC wchar_t *SDLCALL SDL_wcsdup(const wchar_t *wstr); +extern DECLSPEC wchar_t *SDLCALL SDL_wcsstr(const wchar_t *haystack, const wchar_t *needle); + +extern DECLSPEC int SDLCALL SDL_wcscmp(const wchar_t *str1, const wchar_t *str2); +extern DECLSPEC int SDLCALL SDL_wcsncmp(const wchar_t *str1, const wchar_t *str2, size_t maxlen); +extern DECLSPEC int SDLCALL SDL_wcscasecmp(const wchar_t *str1, const wchar_t *str2); +extern DECLSPEC int SDLCALL SDL_wcsncasecmp(const wchar_t *str1, const wchar_t *str2, size_t len); + +extern DECLSPEC size_t SDLCALL SDL_strlen(const char *str); +extern DECLSPEC size_t SDLCALL SDL_strlcpy(SDL_OUT_Z_CAP(maxlen) char *dst, const char *src, size_t maxlen); +extern DECLSPEC size_t SDLCALL SDL_utf8strlcpy(SDL_OUT_Z_CAP(dst_bytes) char *dst, const char *src, size_t dst_bytes); +extern DECLSPEC size_t SDLCALL SDL_strlcat(SDL_INOUT_Z_CAP(maxlen) char *dst, const char *src, size_t maxlen); +extern DECLSPEC char *SDLCALL SDL_strdup(const char *str); +extern DECLSPEC char *SDLCALL SDL_strrev(char *str); +extern DECLSPEC char *SDLCALL SDL_strupr(char *str); +extern DECLSPEC char *SDLCALL SDL_strlwr(char *str); +extern DECLSPEC char *SDLCALL SDL_strchr(const char *str, int c); +extern DECLSPEC char *SDLCALL SDL_strrchr(const char *str, int c); +extern DECLSPEC char *SDLCALL SDL_strstr(const char *haystack, const char *needle); +extern DECLSPEC char *SDLCALL SDL_strcasestr(const char *haystack, const char *needle); +extern DECLSPEC char *SDLCALL SDL_strtokr(char *s1, const char *s2, char **saveptr); +extern DECLSPEC size_t SDLCALL SDL_utf8strlen(const char *str); +extern DECLSPEC size_t SDLCALL SDL_utf8strnlen(const char *str, size_t bytes); + +extern DECLSPEC char *SDLCALL SDL_itoa(int value, char *str, int radix); +extern DECLSPEC char *SDLCALL SDL_uitoa(unsigned int value, char *str, int radix); +extern DECLSPEC char *SDLCALL SDL_ltoa(long value, char *str, int radix); +extern DECLSPEC char *SDLCALL SDL_ultoa(unsigned long value, char *str, int radix); +extern DECLSPEC char *SDLCALL SDL_lltoa(Sint64 value, char *str, int radix); +extern DECLSPEC char *SDLCALL SDL_ulltoa(Uint64 value, char *str, int radix); + +extern DECLSPEC int SDLCALL SDL_atoi(const char *str); +extern DECLSPEC double SDLCALL SDL_atof(const char *str); +extern DECLSPEC long SDLCALL SDL_strtol(const char *str, char **endp, int base); +extern DECLSPEC unsigned long SDLCALL SDL_strtoul(const char *str, char **endp, int base); +extern DECLSPEC Sint64 SDLCALL SDL_strtoll(const char *str, char **endp, int base); +extern DECLSPEC Uint64 SDLCALL SDL_strtoull(const char *str, char **endp, int base); +extern DECLSPEC double SDLCALL SDL_strtod(const char *str, char **endp); + +extern DECLSPEC int SDLCALL SDL_strcmp(const char *str1, const char *str2); +extern DECLSPEC int SDLCALL SDL_strncmp(const char *str1, const char *str2, size_t maxlen); +extern DECLSPEC int SDLCALL SDL_strcasecmp(const char *str1, const char *str2); +extern DECLSPEC int SDLCALL SDL_strncasecmp(const char *str1, const char *str2, size_t len); + +extern DECLSPEC int SDLCALL SDL_sscanf(const char *text, SDL_SCANF_FORMAT_STRING const char *fmt, ...) SDL_SCANF_VARARG_FUNC(2); +extern DECLSPEC int SDLCALL SDL_vsscanf(const char *text, const char *fmt, va_list ap); +extern DECLSPEC int SDLCALL SDL_snprintf(SDL_OUT_Z_CAP(maxlen) char *text, size_t maxlen, SDL_PRINTF_FORMAT_STRING const char *fmt, ... ) SDL_PRINTF_VARARG_FUNC(3); +extern DECLSPEC int SDLCALL SDL_vsnprintf(SDL_OUT_Z_CAP(maxlen) char *text, size_t maxlen, const char *fmt, va_list ap); +extern DECLSPEC int SDLCALL SDL_asprintf(char **strp, SDL_PRINTF_FORMAT_STRING const char *fmt, ...) SDL_PRINTF_VARARG_FUNC(2); +extern DECLSPEC int SDLCALL SDL_vasprintf(char **strp, const char *fmt, va_list ap); + +#ifndef HAVE_M_PI +#ifndef M_PI +#define M_PI 3.14159265358979323846264338327950288 /**< pi */ +#endif +#endif + +/** + * Use this function to compute arc cosine of `x`. + * + * The definition of `y = acos(x)` is `x = cos(y)`. + * + * Domain: `-1 <= x <= 1` + * + * Range: `0 <= y <= Pi` + * + * \param x floating point value, in radians. + * \returns arc cosine of `x`. + * + * \since This function is available since SDL 2.0.2. + */ +extern DECLSPEC double SDLCALL SDL_acos(double x); +extern DECLSPEC float SDLCALL SDL_acosf(float x); +extern DECLSPEC double SDLCALL SDL_asin(double x); +extern DECLSPEC float SDLCALL SDL_asinf(float x); +extern DECLSPEC double SDLCALL SDL_atan(double x); +extern DECLSPEC float SDLCALL SDL_atanf(float x); +extern DECLSPEC double SDLCALL SDL_atan2(double y, double x); +extern DECLSPEC float SDLCALL SDL_atan2f(float y, float x); +extern DECLSPEC double SDLCALL SDL_ceil(double x); +extern DECLSPEC float SDLCALL SDL_ceilf(float x); +extern DECLSPEC double SDLCALL SDL_copysign(double x, double y); +extern DECLSPEC float SDLCALL SDL_copysignf(float x, float y); +extern DECLSPEC double SDLCALL SDL_cos(double x); +extern DECLSPEC float SDLCALL SDL_cosf(float x); +extern DECLSPEC double SDLCALL SDL_exp(double x); +extern DECLSPEC float SDLCALL SDL_expf(float x); +extern DECLSPEC double SDLCALL SDL_fabs(double x); +extern DECLSPEC float SDLCALL SDL_fabsf(float x); +extern DECLSPEC double SDLCALL SDL_floor(double x); +extern DECLSPEC float SDLCALL SDL_floorf(float x); +extern DECLSPEC double SDLCALL SDL_trunc(double x); +extern DECLSPEC float SDLCALL SDL_truncf(float x); +extern DECLSPEC double SDLCALL SDL_fmod(double x, double y); +extern DECLSPEC float SDLCALL SDL_fmodf(float x, float y); +extern DECLSPEC double SDLCALL SDL_log(double x); +extern DECLSPEC float SDLCALL SDL_logf(float x); +extern DECLSPEC double SDLCALL SDL_log10(double x); +extern DECLSPEC float SDLCALL SDL_log10f(float x); +extern DECLSPEC double SDLCALL SDL_pow(double x, double y); +extern DECLSPEC float SDLCALL SDL_powf(float x, float y); +extern DECLSPEC double SDLCALL SDL_round(double x); +extern DECLSPEC float SDLCALL SDL_roundf(float x); +extern DECLSPEC long SDLCALL SDL_lround(double x); +extern DECLSPEC long SDLCALL SDL_lroundf(float x); +extern DECLSPEC double SDLCALL SDL_scalbn(double x, int n); +extern DECLSPEC float SDLCALL SDL_scalbnf(float x, int n); +extern DECLSPEC double SDLCALL SDL_sin(double x); +extern DECLSPEC float SDLCALL SDL_sinf(float x); +extern DECLSPEC double SDLCALL SDL_sqrt(double x); +extern DECLSPEC float SDLCALL SDL_sqrtf(float x); +extern DECLSPEC double SDLCALL SDL_tan(double x); +extern DECLSPEC float SDLCALL SDL_tanf(float x); + +/* The SDL implementation of iconv() returns these error codes */ +#define SDL_ICONV_ERROR (size_t)-1 +#define SDL_ICONV_E2BIG (size_t)-2 +#define SDL_ICONV_EILSEQ (size_t)-3 +#define SDL_ICONV_EINVAL (size_t)-4 + +/* SDL_iconv_* are now always real symbols/types, not macros or inlined. */ +typedef struct _SDL_iconv_t *SDL_iconv_t; +extern DECLSPEC SDL_iconv_t SDLCALL SDL_iconv_open(const char *tocode, + const char *fromcode); +extern DECLSPEC int SDLCALL SDL_iconv_close(SDL_iconv_t cd); +extern DECLSPEC size_t SDLCALL SDL_iconv(SDL_iconv_t cd, const char **inbuf, + size_t * inbytesleft, char **outbuf, + size_t * outbytesleft); + +/** + * This function converts a buffer or string between encodings in one pass, returning a + * string that must be freed with SDL_free() or NULL on error. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC char *SDLCALL SDL_iconv_string(const char *tocode, + const char *fromcode, + const char *inbuf, + size_t inbytesleft); +#define SDL_iconv_utf8_locale(S) SDL_iconv_string("", "UTF-8", S, SDL_strlen(S)+1) +#define SDL_iconv_utf8_ucs2(S) (Uint16 *)SDL_iconv_string("UCS-2-INTERNAL", "UTF-8", S, SDL_strlen(S)+1) +#define SDL_iconv_utf8_ucs4(S) (Uint32 *)SDL_iconv_string("UCS-4-INTERNAL", "UTF-8", S, SDL_strlen(S)+1) +#define SDL_iconv_wchar_utf8(S) SDL_iconv_string("UTF-8", "WCHAR_T", (char *)S, (SDL_wcslen(S)+1)*sizeof(wchar_t)) + +/* force builds using Clang's static analysis tools to use literal C runtime + here, since there are possibly tests that are ineffective otherwise. */ +#if defined(__clang_analyzer__) && !defined(SDL_DISABLE_ANALYZE_MACROS) + +/* The analyzer knows about strlcpy even when the system doesn't provide it */ +#ifndef HAVE_STRLCPY +size_t strlcpy(char* dst, const char* src, size_t size); +#endif + +/* The analyzer knows about strlcat even when the system doesn't provide it */ +#ifndef HAVE_STRLCAT +size_t strlcat(char* dst, const char* src, size_t size); +#endif + +#ifndef HAVE_WCSLCPY +size_t wcslcpy(wchar_t *dst, const wchar_t *src, size_t size); +#endif + +#ifndef HAVE_WCSLCAT +size_t wcslcat(wchar_t *dst, const wchar_t *src, size_t size); +#endif + +/* Starting LLVM 16, the analyser errors out if these functions do not have + their prototype defined (clang-diagnostic-implicit-function-declaration) */ +#include +#include +#include + +#define SDL_malloc malloc +#define SDL_calloc calloc +#define SDL_realloc realloc +#define SDL_free free +#define SDL_memset memset +#define SDL_memcpy memcpy +#define SDL_memmove memmove +#define SDL_memcmp memcmp +#define SDL_strlcpy strlcpy +#define SDL_strlcat strlcat +#define SDL_strlen strlen +#define SDL_wcslen wcslen +#define SDL_wcslcpy wcslcpy +#define SDL_wcslcat wcslcat +#define SDL_strdup strdup +#define SDL_wcsdup wcsdup +#define SDL_strchr strchr +#define SDL_strrchr strrchr +#define SDL_strstr strstr +#define SDL_wcsstr wcsstr +#define SDL_strtokr strtok_r +#define SDL_strcmp strcmp +#define SDL_wcscmp wcscmp +#define SDL_strncmp strncmp +#define SDL_wcsncmp wcsncmp +#define SDL_strcasecmp strcasecmp +#define SDL_strncasecmp strncasecmp +#define SDL_sscanf sscanf +#define SDL_vsscanf vsscanf +#define SDL_snprintf snprintf +#define SDL_vsnprintf vsnprintf +#endif + +SDL_FORCE_INLINE void *SDL_memcpy4(SDL_OUT_BYTECAP(dwords*4) void *dst, SDL_IN_BYTECAP(dwords*4) const void *src, size_t dwords) +{ + return SDL_memcpy(dst, src, dwords * 4); +} + +/** + * If a * b would overflow, return -1. Otherwise store a * b via ret + * and return 0. + * + * \since This function is available since SDL 2.24.0. + */ +SDL_FORCE_INLINE int SDL_size_mul_overflow (size_t a, + size_t b, + size_t *ret) +{ + if (a != 0 && b > SDL_SIZE_MAX / a) { + return -1; + } + *ret = a * b; + return 0; +} + +#if _SDL_HAS_BUILTIN(__builtin_mul_overflow) +/* This needs to be wrapped in an inline rather than being a direct #define, + * because __builtin_mul_overflow() is type-generic, but we want to be + * consistent about interpreting a and b as size_t. */ +SDL_FORCE_INLINE int _SDL_size_mul_overflow_builtin (size_t a, + size_t b, + size_t *ret) +{ + return __builtin_mul_overflow(a, b, ret) == 0 ? 0 : -1; +} +#define SDL_size_mul_overflow(a, b, ret) (_SDL_size_mul_overflow_builtin(a, b, ret)) +#endif + +/** + * If a + b would overflow, return -1. Otherwise store a + b via ret + * and return 0. + * + * \since This function is available since SDL 2.24.0. + */ +SDL_FORCE_INLINE int SDL_size_add_overflow (size_t a, + size_t b, + size_t *ret) +{ + if (b > SDL_SIZE_MAX - a) { + return -1; + } + *ret = a + b; + return 0; +} + +#if _SDL_HAS_BUILTIN(__builtin_add_overflow) +/* This needs to be wrapped in an inline rather than being a direct #define, + * the same as the call to __builtin_mul_overflow() above. */ +SDL_FORCE_INLINE int _SDL_size_add_overflow_builtin (size_t a, + size_t b, + size_t *ret) +{ + return __builtin_add_overflow(a, b, ret) == 0 ? 0 : -1; +} +#define SDL_size_add_overflow(a, b, ret) (_SDL_size_add_overflow_builtin(a, b, ret)) +#endif + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_stdinc_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_surface.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_surface.h new file mode 100644 index 00000000..d6ee615c --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_surface.h @@ -0,0 +1,997 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_surface.h + * + * Header file for ::SDL_Surface definition and management functions. + */ + +#ifndef SDL_surface_h_ +#define SDL_surface_h_ + +#include "SDL_stdinc.h" +#include "SDL_pixels.h" +#include "SDL_rect.h" +#include "SDL_blendmode.h" +#include "SDL_rwops.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \name Surface flags + * + * These are the currently supported flags for the ::SDL_Surface. + * + * \internal + * Used internally (read-only). + */ +/* @{ */ +#define SDL_SWSURFACE 0 /**< Just here for compatibility */ +#define SDL_PREALLOC 0x00000001 /**< Surface uses preallocated memory */ +#define SDL_RLEACCEL 0x00000002 /**< Surface is RLE encoded */ +#define SDL_DONTFREE 0x00000004 /**< Surface is referenced internally */ +#define SDL_SIMD_ALIGNED 0x00000008 /**< Surface uses aligned memory */ +/* @} *//* Surface flags */ + +/** + * Evaluates to true if the surface needs to be locked before access. + */ +#define SDL_MUSTLOCK(S) (((S)->flags & SDL_RLEACCEL) != 0) + +typedef struct SDL_BlitMap SDL_BlitMap; /* this is an opaque type. */ + +/** + * \brief A collection of pixels used in software blitting. + * + * \note This structure should be treated as read-only, except for \c pixels, + * which, if not NULL, contains the raw pixel data for the surface. + */ +typedef struct SDL_Surface +{ + Uint32 flags; /**< Read-only */ + SDL_PixelFormat *format; /**< Read-only */ + int w, h; /**< Read-only */ + int pitch; /**< Read-only */ + void *pixels; /**< Read-write */ + + /** Application data associated with the surface */ + void *userdata; /**< Read-write */ + + /** information needed for surfaces requiring locks */ + int locked; /**< Read-only */ + + /** list of BlitMap that hold a reference to this surface */ + void *list_blitmap; /**< Private */ + + /** clipping information */ + SDL_Rect clip_rect; /**< Read-only */ + + /** info for fast blit mapping to other surfaces */ + SDL_BlitMap *map; /**< Private */ + + /** Reference count -- used when freeing surface */ + int refcount; /**< Read-mostly */ +} SDL_Surface; + +/** + * \brief The type of function used for surface blitting functions. + */ +typedef int (SDLCALL *SDL_blit) (struct SDL_Surface * src, SDL_Rect * srcrect, + struct SDL_Surface * dst, SDL_Rect * dstrect); + +/** + * \brief The formula used for converting between YUV and RGB + */ +typedef enum +{ + SDL_YUV_CONVERSION_JPEG, /**< Full range JPEG */ + SDL_YUV_CONVERSION_BT601, /**< BT.601 (the default) */ + SDL_YUV_CONVERSION_BT709, /**< BT.709 */ + SDL_YUV_CONVERSION_AUTOMATIC /**< BT.601 for SD content, BT.709 for HD content */ +} SDL_YUV_CONVERSION_MODE; + +/** + * Allocate a new RGB surface. + * + * If `depth` is 4 or 8 bits, an empty palette is allocated for the surface. + * If `depth` is greater than 8 bits, the pixel format is set using the + * [RGBA]mask parameters. + * + * The [RGBA]mask parameters are the bitmasks used to extract that color from + * a pixel. For instance, `Rmask` being 0xFF000000 means the red data is + * stored in the most significant byte. Using zeros for the RGB masks sets a + * default value, based on the depth. For example: + * + * ```c++ + * SDL_CreateRGBSurface(0,w,h,32,0,0,0,0); + * ``` + * + * However, using zero for the Amask results in an Amask of 0. + * + * By default surfaces with an alpha mask are set up for blending as with: + * + * ```c++ + * SDL_SetSurfaceBlendMode(surface, SDL_BLENDMODE_BLEND) + * ``` + * + * You can change this by calling SDL_SetSurfaceBlendMode() and selecting a + * different `blendMode`. + * + * \param flags the flags are unused and should be set to 0 + * \param width the width of the surface + * \param height the height of the surface + * \param depth the depth of the surface in bits + * \param Rmask the red mask for the pixels + * \param Gmask the green mask for the pixels + * \param Bmask the blue mask for the pixels + * \param Amask the alpha mask for the pixels + * \returns the new SDL_Surface structure that is created or NULL if it fails; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateRGBSurfaceFrom + * \sa SDL_CreateRGBSurfaceWithFormat + * \sa SDL_FreeSurface + */ +extern DECLSPEC SDL_Surface *SDLCALL SDL_CreateRGBSurface + (Uint32 flags, int width, int height, int depth, + Uint32 Rmask, Uint32 Gmask, Uint32 Bmask, Uint32 Amask); + + +/* !!! FIXME for 2.1: why does this ask for depth? Format provides that. */ + +/** + * Allocate a new RGB surface with a specific pixel format. + * + * This function operates mostly like SDL_CreateRGBSurface(), except instead + * of providing pixel color masks, you provide it with a predefined format + * from SDL_PixelFormatEnum. + * + * \param flags the flags are unused and should be set to 0 + * \param width the width of the surface + * \param height the height of the surface + * \param depth the depth of the surface in bits + * \param format the SDL_PixelFormatEnum for the new surface's pixel format. + * \returns the new SDL_Surface structure that is created or NULL if it fails; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.5. + * + * \sa SDL_CreateRGBSurface + * \sa SDL_CreateRGBSurfaceFrom + * \sa SDL_FreeSurface + */ +extern DECLSPEC SDL_Surface *SDLCALL SDL_CreateRGBSurfaceWithFormat + (Uint32 flags, int width, int height, int depth, Uint32 format); + +/** + * Allocate a new RGB surface with existing pixel data. + * + * This function operates mostly like SDL_CreateRGBSurface(), except it does + * not allocate memory for the pixel data, instead the caller provides an + * existing buffer of data for the surface to use. + * + * No copy is made of the pixel data. Pixel data is not managed automatically; + * you must free the surface before you free the pixel data. + * + * \param pixels a pointer to existing pixel data + * \param width the width of the surface + * \param height the height of the surface + * \param depth the depth of the surface in bits + * \param pitch the pitch of the surface in bytes + * \param Rmask the red mask for the pixels + * \param Gmask the green mask for the pixels + * \param Bmask the blue mask for the pixels + * \param Amask the alpha mask for the pixels + * \returns the new SDL_Surface structure that is created or NULL if it fails; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateRGBSurface + * \sa SDL_CreateRGBSurfaceWithFormat + * \sa SDL_FreeSurface + */ +extern DECLSPEC SDL_Surface *SDLCALL SDL_CreateRGBSurfaceFrom(void *pixels, + int width, + int height, + int depth, + int pitch, + Uint32 Rmask, + Uint32 Gmask, + Uint32 Bmask, + Uint32 Amask); + +/* !!! FIXME for 2.1: why does this ask for depth? Format provides that. */ + +/** + * Allocate a new RGB surface with with a specific pixel format and existing + * pixel data. + * + * This function operates mostly like SDL_CreateRGBSurfaceFrom(), except + * instead of providing pixel color masks, you provide it with a predefined + * format from SDL_PixelFormatEnum. + * + * No copy is made of the pixel data. Pixel data is not managed automatically; + * you must free the surface before you free the pixel data. + * + * \param pixels a pointer to existing pixel data + * \param width the width of the surface + * \param height the height of the surface + * \param depth the depth of the surface in bits + * \param pitch the pitch of the surface in bytes + * \param format the SDL_PixelFormatEnum for the new surface's pixel format. + * \returns the new SDL_Surface structure that is created or NULL if it fails; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.5. + * + * \sa SDL_CreateRGBSurfaceFrom + * \sa SDL_CreateRGBSurfaceWithFormat + * \sa SDL_FreeSurface + */ +extern DECLSPEC SDL_Surface *SDLCALL SDL_CreateRGBSurfaceWithFormatFrom + (void *pixels, int width, int height, int depth, int pitch, Uint32 format); + +/** + * Free an RGB surface. + * + * It is safe to pass NULL to this function. + * + * \param surface the SDL_Surface to free. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateRGBSurface + * \sa SDL_CreateRGBSurfaceFrom + * \sa SDL_LoadBMP + * \sa SDL_LoadBMP_RW + */ +extern DECLSPEC void SDLCALL SDL_FreeSurface(SDL_Surface * surface); + +/** + * Set the palette used by a surface. + * + * A single palette can be shared with many surfaces. + * + * \param surface the SDL_Surface structure to update + * \param palette the SDL_Palette structure to use + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC int SDLCALL SDL_SetSurfacePalette(SDL_Surface * surface, + SDL_Palette * palette); + +/** + * Set up a surface for directly accessing the pixels. + * + * Between calls to SDL_LockSurface() / SDL_UnlockSurface(), you can write to + * and read from `surface->pixels`, using the pixel format stored in + * `surface->format`. Once you are done accessing the surface, you should use + * SDL_UnlockSurface() to release it. + * + * Not all surfaces require locking. If `SDL_MUSTLOCK(surface)` evaluates to + * 0, then you can read and write to the surface at any time, and the pixel + * format of the surface will not change. + * + * \param surface the SDL_Surface structure to be locked + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_MUSTLOCK + * \sa SDL_UnlockSurface + */ +extern DECLSPEC int SDLCALL SDL_LockSurface(SDL_Surface * surface); + +/** + * Release a surface after directly accessing the pixels. + * + * \param surface the SDL_Surface structure to be unlocked + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LockSurface + */ +extern DECLSPEC void SDLCALL SDL_UnlockSurface(SDL_Surface * surface); + +/** + * Load a BMP image from a seekable SDL data stream. + * + * The new surface should be freed with SDL_FreeSurface(). Not doing so will + * result in a memory leak. + * + * src is an open SDL_RWops buffer, typically loaded with SDL_RWFromFile. + * Alternitavely, you might also use the macro SDL_LoadBMP to load a bitmap + * from a file, convert it to an SDL_Surface and then close the file. + * + * \param src the data stream for the surface + * \param freesrc non-zero to close the stream after being read + * \returns a pointer to a new SDL_Surface structure or NULL if there was an + * error; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_FreeSurface + * \sa SDL_RWFromFile + * \sa SDL_LoadBMP + * \sa SDL_SaveBMP_RW + */ +extern DECLSPEC SDL_Surface *SDLCALL SDL_LoadBMP_RW(SDL_RWops * src, + int freesrc); + +/** + * Load a surface from a file. + * + * Convenience macro. + */ +#define SDL_LoadBMP(file) SDL_LoadBMP_RW(SDL_RWFromFile(file, "rb"), 1) + +/** + * Save a surface to a seekable SDL data stream in BMP format. + * + * Surfaces with a 24-bit, 32-bit and paletted 8-bit format get saved in the + * BMP directly. Other RGB formats with 8-bit or higher get converted to a + * 24-bit surface or, if they have an alpha mask or a colorkey, to a 32-bit + * surface before they are saved. YUV and paletted 1-bit and 4-bit formats are + * not supported. + * + * \param surface the SDL_Surface structure containing the image to be saved + * \param dst a data stream to save to + * \param freedst non-zero to close the stream after being written + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_LoadBMP_RW + * \sa SDL_SaveBMP + */ +extern DECLSPEC int SDLCALL SDL_SaveBMP_RW + (SDL_Surface * surface, SDL_RWops * dst, int freedst); + +/** + * Save a surface to a file. + * + * Convenience macro. + */ +#define SDL_SaveBMP(surface, file) \ + SDL_SaveBMP_RW(surface, SDL_RWFromFile(file, "wb"), 1) + +/** + * Set the RLE acceleration hint for a surface. + * + * If RLE is enabled, color key and alpha blending blits are much faster, but + * the surface must be locked before directly accessing the pixels. + * + * \param surface the SDL_Surface structure to optimize + * \param flag 0 to disable, non-zero to enable RLE acceleration + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_BlitSurface + * \sa SDL_LockSurface + * \sa SDL_UnlockSurface + */ +extern DECLSPEC int SDLCALL SDL_SetSurfaceRLE(SDL_Surface * surface, + int flag); + +/** + * Returns whether the surface is RLE enabled + * + * It is safe to pass a NULL `surface` here; it will return SDL_FALSE. + * + * \param surface the SDL_Surface structure to query + * \returns SDL_TRUE if the surface is RLE enabled, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.14. + * + * \sa SDL_SetSurfaceRLE + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasSurfaceRLE(SDL_Surface * surface); + +/** + * Set the color key (transparent pixel) in a surface. + * + * The color key defines a pixel value that will be treated as transparent in + * a blit. For example, one can use this to specify that cyan pixels should be + * considered transparent, and therefore not rendered. + * + * It is a pixel of the format used by the surface, as generated by + * SDL_MapRGB(). + * + * RLE acceleration can substantially speed up blitting of images with large + * horizontal runs of transparent pixels. See SDL_SetSurfaceRLE() for details. + * + * \param surface the SDL_Surface structure to update + * \param flag SDL_TRUE to enable color key, SDL_FALSE to disable color key + * \param key the transparent pixel + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_BlitSurface + * \sa SDL_GetColorKey + */ +extern DECLSPEC int SDLCALL SDL_SetColorKey(SDL_Surface * surface, + int flag, Uint32 key); + +/** + * Returns whether the surface has a color key + * + * It is safe to pass a NULL `surface` here; it will return SDL_FALSE. + * + * \param surface the SDL_Surface structure to query + * \return SDL_TRUE if the surface has a color key, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.9. + * + * \sa SDL_SetColorKey + * \sa SDL_GetColorKey + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasColorKey(SDL_Surface * surface); + +/** + * Get the color key (transparent pixel) for a surface. + * + * The color key is a pixel of the format used by the surface, as generated by + * SDL_MapRGB(). + * + * If the surface doesn't have color key enabled this function returns -1. + * + * \param surface the SDL_Surface structure to query + * \param key a pointer filled in with the transparent pixel + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_BlitSurface + * \sa SDL_SetColorKey + */ +extern DECLSPEC int SDLCALL SDL_GetColorKey(SDL_Surface * surface, + Uint32 * key); + +/** + * Set an additional color value multiplied into blit operations. + * + * When this surface is blitted, during the blit operation each source color + * channel is modulated by the appropriate color value according to the + * following formula: + * + * `srcC = srcC * (color / 255)` + * + * \param surface the SDL_Surface structure to update + * \param r the red color value multiplied into blit operations + * \param g the green color value multiplied into blit operations + * \param b the blue color value multiplied into blit operations + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetSurfaceColorMod + * \sa SDL_SetSurfaceAlphaMod + */ +extern DECLSPEC int SDLCALL SDL_SetSurfaceColorMod(SDL_Surface * surface, + Uint8 r, Uint8 g, Uint8 b); + + +/** + * Get the additional color value multiplied into blit operations. + * + * \param surface the SDL_Surface structure to query + * \param r a pointer filled in with the current red color value + * \param g a pointer filled in with the current green color value + * \param b a pointer filled in with the current blue color value + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetSurfaceAlphaMod + * \sa SDL_SetSurfaceColorMod + */ +extern DECLSPEC int SDLCALL SDL_GetSurfaceColorMod(SDL_Surface * surface, + Uint8 * r, Uint8 * g, + Uint8 * b); + +/** + * Set an additional alpha value used in blit operations. + * + * When this surface is blitted, during the blit operation the source alpha + * value is modulated by this alpha value according to the following formula: + * + * `srcA = srcA * (alpha / 255)` + * + * \param surface the SDL_Surface structure to update + * \param alpha the alpha value multiplied into blit operations + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetSurfaceAlphaMod + * \sa SDL_SetSurfaceColorMod + */ +extern DECLSPEC int SDLCALL SDL_SetSurfaceAlphaMod(SDL_Surface * surface, + Uint8 alpha); + +/** + * Get the additional alpha value used in blit operations. + * + * \param surface the SDL_Surface structure to query + * \param alpha a pointer filled in with the current alpha value + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetSurfaceColorMod + * \sa SDL_SetSurfaceAlphaMod + */ +extern DECLSPEC int SDLCALL SDL_GetSurfaceAlphaMod(SDL_Surface * surface, + Uint8 * alpha); + +/** + * Set the blend mode used for blit operations. + * + * To copy a surface to another surface (or texture) without blending with the + * existing data, the blendmode of the SOURCE surface should be set to + * `SDL_BLENDMODE_NONE`. + * + * \param surface the SDL_Surface structure to update + * \param blendMode the SDL_BlendMode to use for blit blending + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetSurfaceBlendMode + */ +extern DECLSPEC int SDLCALL SDL_SetSurfaceBlendMode(SDL_Surface * surface, + SDL_BlendMode blendMode); + +/** + * Get the blend mode used for blit operations. + * + * \param surface the SDL_Surface structure to query + * \param blendMode a pointer filled in with the current SDL_BlendMode + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetSurfaceBlendMode + */ +extern DECLSPEC int SDLCALL SDL_GetSurfaceBlendMode(SDL_Surface * surface, + SDL_BlendMode *blendMode); + +/** + * Set the clipping rectangle for a surface. + * + * When `surface` is the destination of a blit, only the area within the clip + * rectangle is drawn into. + * + * Note that blits are automatically clipped to the edges of the source and + * destination surfaces. + * + * \param surface the SDL_Surface structure to be clipped + * \param rect the SDL_Rect structure representing the clipping rectangle, or + * NULL to disable clipping + * \returns SDL_TRUE if the rectangle intersects the surface, otherwise + * SDL_FALSE and blits will be completely clipped. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_BlitSurface + * \sa SDL_GetClipRect + */ +extern DECLSPEC SDL_bool SDLCALL SDL_SetClipRect(SDL_Surface * surface, + const SDL_Rect * rect); + +/** + * Get the clipping rectangle for a surface. + * + * When `surface` is the destination of a blit, only the area within the clip + * rectangle is drawn into. + * + * \param surface the SDL_Surface structure representing the surface to be + * clipped + * \param rect an SDL_Rect structure filled in with the clipping rectangle for + * the surface + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_BlitSurface + * \sa SDL_SetClipRect + */ +extern DECLSPEC void SDLCALL SDL_GetClipRect(SDL_Surface * surface, + SDL_Rect * rect); + +/* + * Creates a new surface identical to the existing surface. + * + * The returned surface should be freed with SDL_FreeSurface(). + * + * \param surface the surface to duplicate. + * \returns a copy of the surface, or NULL on failure; call SDL_GetError() for + * more information. + */ +extern DECLSPEC SDL_Surface *SDLCALL SDL_DuplicateSurface(SDL_Surface * surface); + +/** + * Copy an existing surface to a new surface of the specified format. + * + * This function is used to optimize images for faster *repeat* blitting. This + * is accomplished by converting the original and storing the result as a new + * surface. The new, optimized surface can then be used as the source for + * future blits, making them faster. + * + * \param src the existing SDL_Surface structure to convert + * \param fmt the SDL_PixelFormat structure that the new surface is optimized + * for + * \param flags the flags are unused and should be set to 0; this is a + * leftover from SDL 1.2's API + * \returns the new SDL_Surface structure that is created or NULL if it fails; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AllocFormat + * \sa SDL_ConvertSurfaceFormat + * \sa SDL_CreateRGBSurface + */ +extern DECLSPEC SDL_Surface *SDLCALL SDL_ConvertSurface + (SDL_Surface * src, const SDL_PixelFormat * fmt, Uint32 flags); + +/** + * Copy an existing surface to a new surface of the specified format enum. + * + * This function operates just like SDL_ConvertSurface(), but accepts an + * SDL_PixelFormatEnum value instead of an SDL_PixelFormat structure. As such, + * it might be easier to call but it doesn't have access to palette + * information for the destination surface, in case that would be important. + * + * \param src the existing SDL_Surface structure to convert + * \param pixel_format the SDL_PixelFormatEnum that the new surface is + * optimized for + * \param flags the flags are unused and should be set to 0; this is a + * leftover from SDL 1.2's API + * \returns the new SDL_Surface structure that is created or NULL if it fails; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AllocFormat + * \sa SDL_ConvertSurface + * \sa SDL_CreateRGBSurface + */ +extern DECLSPEC SDL_Surface *SDLCALL SDL_ConvertSurfaceFormat + (SDL_Surface * src, Uint32 pixel_format, Uint32 flags); + +/** + * Copy a block of pixels of one format to another format. + * + * \param width the width of the block to copy, in pixels + * \param height the height of the block to copy, in pixels + * \param src_format an SDL_PixelFormatEnum value of the `src` pixels format + * \param src a pointer to the source pixels + * \param src_pitch the pitch of the source pixels, in bytes + * \param dst_format an SDL_PixelFormatEnum value of the `dst` pixels format + * \param dst a pointer to be filled in with new pixel data + * \param dst_pitch the pitch of the destination pixels, in bytes + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC int SDLCALL SDL_ConvertPixels(int width, int height, + Uint32 src_format, + const void * src, int src_pitch, + Uint32 dst_format, + void * dst, int dst_pitch); + +/** + * Premultiply the alpha on a block of pixels. + * + * This is safe to use with src == dst, but not for other overlapping areas. + * + * This function is currently only implemented for SDL_PIXELFORMAT_ARGB8888. + * + * \param width the width of the block to convert, in pixels + * \param height the height of the block to convert, in pixels + * \param src_format an SDL_PixelFormatEnum value of the `src` pixels format + * \param src a pointer to the source pixels + * \param src_pitch the pitch of the source pixels, in bytes + * \param dst_format an SDL_PixelFormatEnum value of the `dst` pixels format + * \param dst a pointer to be filled in with premultiplied pixel data + * \param dst_pitch the pitch of the destination pixels, in bytes + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC int SDLCALL SDL_PremultiplyAlpha(int width, int height, + Uint32 src_format, + const void * src, int src_pitch, + Uint32 dst_format, + void * dst, int dst_pitch); + +/** + * Perform a fast fill of a rectangle with a specific color. + * + * `color` should be a pixel of the format used by the surface, and can be + * generated by SDL_MapRGB() or SDL_MapRGBA(). If the color value contains an + * alpha component then the destination is simply filled with that alpha + * information, no blending takes place. + * + * If there is a clip rectangle set on the destination (set via + * SDL_SetClipRect()), then this function will fill based on the intersection + * of the clip rectangle and `rect`. + * + * \param dst the SDL_Surface structure that is the drawing target + * \param rect the SDL_Rect structure representing the rectangle to fill, or + * NULL to fill the entire surface + * \param color the color to fill with + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_FillRects + */ +extern DECLSPEC int SDLCALL SDL_FillRect + (SDL_Surface * dst, const SDL_Rect * rect, Uint32 color); + +/** + * Perform a fast fill of a set of rectangles with a specific color. + * + * `color` should be a pixel of the format used by the surface, and can be + * generated by SDL_MapRGB() or SDL_MapRGBA(). If the color value contains an + * alpha component then the destination is simply filled with that alpha + * information, no blending takes place. + * + * If there is a clip rectangle set on the destination (set via + * SDL_SetClipRect()), then this function will fill based on the intersection + * of the clip rectangle and `rect`. + * + * \param dst the SDL_Surface structure that is the drawing target + * \param rects an array of SDL_Rects representing the rectangles to fill. + * \param count the number of rectangles in the array + * \param color the color to fill with + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_FillRect + */ +extern DECLSPEC int SDLCALL SDL_FillRects + (SDL_Surface * dst, const SDL_Rect * rects, int count, Uint32 color); + +/* !!! FIXME: merge this documentation with the wiki */ +/** + * Performs a fast blit from the source surface to the destination surface. + * + * This assumes that the source and destination rectangles are + * the same size. If either \c srcrect or \c dstrect are NULL, the entire + * surface (\c src or \c dst) is copied. The final blit rectangles are saved + * in \c srcrect and \c dstrect after all clipping is performed. + * + * \returns 0 if the blit is successful, otherwise it returns -1. + * + * The blit function should not be called on a locked surface. + * + * The blit semantics for surfaces with and without blending and colorkey + * are defined as follows: + * \verbatim + RGBA->RGB: + Source surface blend mode set to SDL_BLENDMODE_BLEND: + alpha-blend (using the source alpha-channel and per-surface alpha) + SDL_SRCCOLORKEY ignored. + Source surface blend mode set to SDL_BLENDMODE_NONE: + copy RGB. + if SDL_SRCCOLORKEY set, only copy the pixels matching the + RGB values of the source color key, ignoring alpha in the + comparison. + + RGB->RGBA: + Source surface blend mode set to SDL_BLENDMODE_BLEND: + alpha-blend (using the source per-surface alpha) + Source surface blend mode set to SDL_BLENDMODE_NONE: + copy RGB, set destination alpha to source per-surface alpha value. + both: + if SDL_SRCCOLORKEY set, only copy the pixels matching the + source color key. + + RGBA->RGBA: + Source surface blend mode set to SDL_BLENDMODE_BLEND: + alpha-blend (using the source alpha-channel and per-surface alpha) + SDL_SRCCOLORKEY ignored. + Source surface blend mode set to SDL_BLENDMODE_NONE: + copy all of RGBA to the destination. + if SDL_SRCCOLORKEY set, only copy the pixels matching the + RGB values of the source color key, ignoring alpha in the + comparison. + + RGB->RGB: + Source surface blend mode set to SDL_BLENDMODE_BLEND: + alpha-blend (using the source per-surface alpha) + Source surface blend mode set to SDL_BLENDMODE_NONE: + copy RGB. + both: + if SDL_SRCCOLORKEY set, only copy the pixels matching the + source color key. + \endverbatim + * + * You should call SDL_BlitSurface() unless you know exactly how SDL + * blitting works internally and how to use the other blit functions. + */ +#define SDL_BlitSurface SDL_UpperBlit + +/** + * Perform a fast blit from the source surface to the destination surface. + * + * SDL_UpperBlit() has been replaced by SDL_BlitSurface(), which is merely a + * macro for this function with a less confusing name. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_BlitSurface + */ +extern DECLSPEC int SDLCALL SDL_UpperBlit + (SDL_Surface * src, const SDL_Rect * srcrect, + SDL_Surface * dst, SDL_Rect * dstrect); + +/** + * Perform low-level surface blitting only. + * + * This is a semi-private blit function and it performs low-level surface + * blitting, assuming the input rectangles have already been clipped. + * + * Unless you know what you're doing, you should be using SDL_BlitSurface() + * instead. + * + * \param src the SDL_Surface structure to be copied from + * \param srcrect the SDL_Rect structure representing the rectangle to be + * copied, or NULL to copy the entire surface + * \param dst the SDL_Surface structure that is the blit target + * \param dstrect the SDL_Rect structure representing the rectangle that is + * copied into + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_BlitSurface + */ +extern DECLSPEC int SDLCALL SDL_LowerBlit + (SDL_Surface * src, SDL_Rect * srcrect, + SDL_Surface * dst, SDL_Rect * dstrect); + + +/** + * Perform a fast, low quality, stretch blit between two surfaces of the same + * format. + * + * Please use SDL_BlitScaled() instead. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC int SDLCALL SDL_SoftStretch(SDL_Surface * src, + const SDL_Rect * srcrect, + SDL_Surface * dst, + const SDL_Rect * dstrect); + +/** + * Perform bilinear scaling between two surfaces of the same format, 32BPP. + * + * \since This function is available since SDL 2.0.16. + */ +extern DECLSPEC int SDLCALL SDL_SoftStretchLinear(SDL_Surface * src, + const SDL_Rect * srcrect, + SDL_Surface * dst, + const SDL_Rect * dstrect); + + +#define SDL_BlitScaled SDL_UpperBlitScaled + +/** + * Perform a scaled surface copy to a destination surface. + * + * SDL_UpperBlitScaled() has been replaced by SDL_BlitScaled(), which is + * merely a macro for this function with a less confusing name. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_BlitScaled + */ +extern DECLSPEC int SDLCALL SDL_UpperBlitScaled + (SDL_Surface * src, const SDL_Rect * srcrect, + SDL_Surface * dst, SDL_Rect * dstrect); + +/** + * Perform low-level surface scaled blitting only. + * + * This is a semi-private function and it performs low-level surface blitting, + * assuming the input rectangles have already been clipped. + * + * \param src the SDL_Surface structure to be copied from + * \param srcrect the SDL_Rect structure representing the rectangle to be + * copied + * \param dst the SDL_Surface structure that is the blit target + * \param dstrect the SDL_Rect structure representing the rectangle that is + * copied into + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_BlitScaled + */ +extern DECLSPEC int SDLCALL SDL_LowerBlitScaled + (SDL_Surface * src, SDL_Rect * srcrect, + SDL_Surface * dst, SDL_Rect * dstrect); + +/** + * Set the YUV conversion mode + * + * \since This function is available since SDL 2.0.8. + */ +extern DECLSPEC void SDLCALL SDL_SetYUVConversionMode(SDL_YUV_CONVERSION_MODE mode); + +/** + * Get the YUV conversion mode + * + * \since This function is available since SDL 2.0.8. + */ +extern DECLSPEC SDL_YUV_CONVERSION_MODE SDLCALL SDL_GetYUVConversionMode(void); + +/** + * Get the YUV conversion mode, returning the correct mode for the resolution + * when the current conversion mode is SDL_YUV_CONVERSION_AUTOMATIC + * + * \since This function is available since SDL 2.0.8. + */ +extern DECLSPEC SDL_YUV_CONVERSION_MODE SDLCALL SDL_GetYUVConversionModeForResolution(int width, int height); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_surface_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_system.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_system.h new file mode 100644 index 00000000..4b7eaddc --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_system.h @@ -0,0 +1,623 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_system.h + * + * Include file for platform specific SDL API functions + */ + +#ifndef SDL_system_h_ +#define SDL_system_h_ + +#include "SDL_stdinc.h" +#include "SDL_keyboard.h" +#include "SDL_render.h" +#include "SDL_video.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + + +/* Platform specific functions for Windows */ +#if defined(__WIN32__) || defined(__GDK__) + +typedef void (SDLCALL * SDL_WindowsMessageHook)(void *userdata, void *hWnd, unsigned int message, Uint64 wParam, Sint64 lParam); + +/** + * Set a callback for every Windows message, run before TranslateMessage(). + * + * \param callback The SDL_WindowsMessageHook function to call. + * \param userdata a pointer to pass to every iteration of `callback` + * + * \since This function is available since SDL 2.0.4. + */ +extern DECLSPEC void SDLCALL SDL_SetWindowsMessageHook(SDL_WindowsMessageHook callback, void *userdata); + +#endif /* defined(__WIN32__) || defined(__GDK__) */ + +#if defined(__WIN32__) || defined(__WINGDK__) + +/** + * Get the D3D9 adapter index that matches the specified display index. + * + * The returned adapter index can be passed to `IDirect3D9::CreateDevice` and + * controls on which monitor a full screen application will appear. + * + * \param displayIndex the display index for which to get the D3D9 adapter + * index + * \returns the D3D9 adapter index on success or a negative error code on + * failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.1. + */ +extern DECLSPEC int SDLCALL SDL_Direct3D9GetAdapterIndex( int displayIndex ); + +typedef struct IDirect3DDevice9 IDirect3DDevice9; + +/** + * Get the D3D9 device associated with a renderer. + * + * Once you are done using the device, you should release it to avoid a + * resource leak. + * + * \param renderer the renderer from which to get the associated D3D device + * \returns the D3D9 device associated with given renderer or NULL if it is + * not a D3D9 renderer; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.1. + */ +extern DECLSPEC IDirect3DDevice9* SDLCALL SDL_RenderGetD3D9Device(SDL_Renderer * renderer); + +typedef struct ID3D11Device ID3D11Device; + +/** + * Get the D3D11 device associated with a renderer. + * + * Once you are done using the device, you should release it to avoid a + * resource leak. + * + * \param renderer the renderer from which to get the associated D3D11 device + * \returns the D3D11 device associated with given renderer or NULL if it is + * not a D3D11 renderer; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.16. + */ +extern DECLSPEC ID3D11Device* SDLCALL SDL_RenderGetD3D11Device(SDL_Renderer * renderer); + +#endif /* defined(__WIN32__) || defined(__WINGDK__) */ + +#if defined(__WIN32__) || defined(__GDK__) + +typedef struct ID3D12Device ID3D12Device; + +/** + * Get the D3D12 device associated with a renderer. + * + * Once you are done using the device, you should release it to avoid a + * resource leak. + * + * \param renderer the renderer from which to get the associated D3D12 device + * \returns the D3D12 device associated with given renderer or NULL if it is + * not a D3D12 renderer; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.24.0. + */ +extern DECLSPEC ID3D12Device* SDLCALL SDL_RenderGetD3D12Device(SDL_Renderer* renderer); + +#endif /* defined(__WIN32__) || defined(__GDK__) */ + +#if defined(__WIN32__) || defined(__WINGDK__) + +/** + * Get the DXGI Adapter and Output indices for the specified display index. + * + * The DXGI Adapter and Output indices can be passed to `EnumAdapters` and + * `EnumOutputs` respectively to get the objects required to create a DX10 or + * DX11 device and swap chain. + * + * Before SDL 2.0.4 this function did not return a value. Since SDL 2.0.4 it + * returns an SDL_bool. + * + * \param displayIndex the display index for which to get both indices + * \param adapterIndex a pointer to be filled in with the adapter index + * \param outputIndex a pointer to be filled in with the output index + * \returns SDL_TRUE on success or SDL_FALSE on failure; call SDL_GetError() + * for more information. + * + * \since This function is available since SDL 2.0.2. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_DXGIGetOutputInfo( int displayIndex, int *adapterIndex, int *outputIndex ); + +#endif /* defined(__WIN32__) || defined(__WINGDK__) */ + +/* Platform specific functions for Linux */ +#ifdef __LINUX__ + +/** + * Sets the UNIX nice value for a thread. + * + * This uses setpriority() if possible, and RealtimeKit if available. + * + * \param threadID the Unix thread ID to change priority of. + * \param priority The new, Unix-specific, priority value. + * \returns 0 on success, or -1 on error. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC int SDLCALL SDL_LinuxSetThreadPriority(Sint64 threadID, int priority); + +/** + * Sets the priority (not nice level) and scheduling policy for a thread. + * + * This uses setpriority() if possible, and RealtimeKit if available. + * + * \param threadID The Unix thread ID to change priority of. + * \param sdlPriority The new SDL_ThreadPriority value. + * \param schedPolicy The new scheduling policy (SCHED_FIFO, SCHED_RR, + * SCHED_OTHER, etc...) + * \returns 0 on success, or -1 on error. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC int SDLCALL SDL_LinuxSetThreadPriorityAndPolicy(Sint64 threadID, int sdlPriority, int schedPolicy); + +#endif /* __LINUX__ */ + +/* Platform specific functions for iOS */ +#ifdef __IPHONEOS__ + +#define SDL_iOSSetAnimationCallback(window, interval, callback, callbackParam) SDL_iPhoneSetAnimationCallback(window, interval, callback, callbackParam) + +/** + * Use this function to set the animation callback on Apple iOS. + * + * The function prototype for `callback` is: + * + * ```c + * void callback(void* callbackParam); + * ``` + * + * Where its parameter, `callbackParam`, is what was passed as `callbackParam` + * to SDL_iPhoneSetAnimationCallback(). + * + * This function is only available on Apple iOS. + * + * For more information see: + * https://github.com/libsdl-org/SDL/blob/main/docs/README-ios.md + * + * This functions is also accessible using the macro + * SDL_iOSSetAnimationCallback() since SDL 2.0.4. + * + * \param window the window for which the animation callback should be set + * \param interval the number of frames after which **callback** will be + * called + * \param callback the function to call for every frame. + * \param callbackParam a pointer that is passed to `callback`. + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_iPhoneSetEventPump + */ +extern DECLSPEC int SDLCALL SDL_iPhoneSetAnimationCallback(SDL_Window * window, int interval, void (SDLCALL *callback)(void*), void *callbackParam); + +#define SDL_iOSSetEventPump(enabled) SDL_iPhoneSetEventPump(enabled) + +/** + * Use this function to enable or disable the SDL event pump on Apple iOS. + * + * This function is only available on Apple iOS. + * + * This functions is also accessible using the macro SDL_iOSSetEventPump() + * since SDL 2.0.4. + * + * \param enabled SDL_TRUE to enable the event pump, SDL_FALSE to disable it + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_iPhoneSetAnimationCallback + */ +extern DECLSPEC void SDLCALL SDL_iPhoneSetEventPump(SDL_bool enabled); + +#endif /* __IPHONEOS__ */ + + +/* Platform specific functions for Android */ +#ifdef __ANDROID__ + +/** + * Get the Android Java Native Interface Environment of the current thread. + * + * This is the JNIEnv one needs to access the Java virtual machine from native + * code, and is needed for many Android APIs to be usable from C. + * + * The prototype of the function in SDL's code actually declare a void* return + * type, even if the implementation returns a pointer to a JNIEnv. The + * rationale being that the SDL headers can avoid including jni.h. + * + * \returns a pointer to Java native interface object (JNIEnv) to which the + * current thread is attached, or 0 on error. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AndroidGetActivity + */ +extern DECLSPEC void * SDLCALL SDL_AndroidGetJNIEnv(void); + +/** + * Retrieve the Java instance of the Android activity class. + * + * The prototype of the function in SDL's code actually declares a void* + * return type, even if the implementation returns a jobject. The rationale + * being that the SDL headers can avoid including jni.h. + * + * The jobject returned by the function is a local reference and must be + * released by the caller. See the PushLocalFrame() and PopLocalFrame() or + * DeleteLocalRef() functions of the Java native interface: + * + * https://docs.oracle.com/javase/1.5.0/docs/guide/jni/spec/functions.html + * + * \returns the jobject representing the instance of the Activity class of the + * Android application, or NULL on error. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AndroidGetJNIEnv + */ +extern DECLSPEC void * SDLCALL SDL_AndroidGetActivity(void); + +/** + * Query Android API level of the current device. + * + * - API level 31: Android 12 + * - API level 30: Android 11 + * - API level 29: Android 10 + * - API level 28: Android 9 + * - API level 27: Android 8.1 + * - API level 26: Android 8.0 + * - API level 25: Android 7.1 + * - API level 24: Android 7.0 + * - API level 23: Android 6.0 + * - API level 22: Android 5.1 + * - API level 21: Android 5.0 + * - API level 20: Android 4.4W + * - API level 19: Android 4.4 + * - API level 18: Android 4.3 + * - API level 17: Android 4.2 + * - API level 16: Android 4.1 + * - API level 15: Android 4.0.3 + * - API level 14: Android 4.0 + * - API level 13: Android 3.2 + * - API level 12: Android 3.1 + * - API level 11: Android 3.0 + * - API level 10: Android 2.3.3 + * + * \returns the Android API level. + * + * \since This function is available since SDL 2.0.12. + */ +extern DECLSPEC int SDLCALL SDL_GetAndroidSDKVersion(void); + +/** + * Query if the application is running on Android TV. + * + * \returns SDL_TRUE if this is Android TV, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.8. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_IsAndroidTV(void); + +/** + * Query if the application is running on a Chromebook. + * + * \returns SDL_TRUE if this is a Chromebook, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_IsChromebook(void); + +/** + * Query if the application is running on a Samsung DeX docking station. + * + * \returns SDL_TRUE if this is a DeX docking station, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_IsDeXMode(void); + +/** + * Trigger the Android system back button behavior. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC void SDLCALL SDL_AndroidBackButton(void); + +/** + See the official Android developer guide for more information: + http://developer.android.com/guide/topics/data/data-storage.html +*/ +#define SDL_ANDROID_EXTERNAL_STORAGE_READ 0x01 +#define SDL_ANDROID_EXTERNAL_STORAGE_WRITE 0x02 + +/** + * Get the path used for internal storage for this application. + * + * This path is unique to your application and cannot be written to by other + * applications. + * + * Your internal storage path is typically: + * `/data/data/your.app.package/files`. + * + * \returns the path used for internal storage or NULL on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AndroidGetExternalStorageState + */ +extern DECLSPEC const char * SDLCALL SDL_AndroidGetInternalStoragePath(void); + +/** + * Get the current state of external storage. + * + * The current state of external storage, a bitmask of these values: + * `SDL_ANDROID_EXTERNAL_STORAGE_READ`, `SDL_ANDROID_EXTERNAL_STORAGE_WRITE`. + * + * If external storage is currently unavailable, this will return 0. + * + * \returns the current state of external storage on success or 0 on failure; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AndroidGetExternalStoragePath + */ +extern DECLSPEC int SDLCALL SDL_AndroidGetExternalStorageState(void); + +/** + * Get the path used for external storage for this application. + * + * This path is unique to your application, but is public and can be written + * to by other applications. + * + * Your external storage path is typically: + * `/storage/sdcard0/Android/data/your.app.package/files`. + * + * \returns the path used for external storage for this application on success + * or NULL on failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AndroidGetExternalStorageState + */ +extern DECLSPEC const char * SDLCALL SDL_AndroidGetExternalStoragePath(void); + +/** + * Request permissions at runtime. + * + * This blocks the calling thread until the permission is granted or denied. + * + * \param permission The permission to request. + * \returns SDL_TRUE if the permission was granted, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.14. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_AndroidRequestPermission(const char *permission); + +/** + * Shows an Android toast notification. + * + * Toasts are a sort of lightweight notification that are unique to Android. + * + * https://developer.android.com/guide/topics/ui/notifiers/toasts + * + * Shows toast in UI thread. + * + * For the `gravity` parameter, choose a value from here, or -1 if you don't + * have a preference: + * + * https://developer.android.com/reference/android/view/Gravity + * + * \param message text message to be shown + * \param duration 0=short, 1=long + * \param gravity where the notification should appear on the screen. + * \param xoffset set this parameter only when gravity >=0 + * \param yoffset set this parameter only when gravity >=0 + * \returns 0 if success, -1 if any error occurs. + * + * \since This function is available since SDL 2.0.16. + */ +extern DECLSPEC int SDLCALL SDL_AndroidShowToast(const char* message, int duration, int gravity, int xoffset, int yoffset); + +/** + * Send a user command to SDLActivity. + * + * Override "boolean onUnhandledMessage(Message msg)" to handle the message. + * + * \param command user command that must be greater or equal to 0x8000 + * \param param user parameter + * + * \since This function is available since SDL 2.0.22. + */ +extern DECLSPEC int SDLCALL SDL_AndroidSendMessage(Uint32 command, int param); + +#endif /* __ANDROID__ */ + +/* Platform specific functions for WinRT */ +#ifdef __WINRT__ + +/** + * \brief WinRT / Windows Phone path types + */ +typedef enum +{ + /** \brief The installed app's root directory. + Files here are likely to be read-only. */ + SDL_WINRT_PATH_INSTALLED_LOCATION, + + /** \brief The app's local data store. Files may be written here */ + SDL_WINRT_PATH_LOCAL_FOLDER, + + /** \brief The app's roaming data store. Unsupported on Windows Phone. + Files written here may be copied to other machines via a network + connection. + */ + SDL_WINRT_PATH_ROAMING_FOLDER, + + /** \brief The app's temporary data store. Unsupported on Windows Phone. + Files written here may be deleted at any time. */ + SDL_WINRT_PATH_TEMP_FOLDER +} SDL_WinRT_Path; + + +/** + * \brief WinRT Device Family + */ +typedef enum +{ + /** \brief Unknown family */ + SDL_WINRT_DEVICEFAMILY_UNKNOWN, + + /** \brief Desktop family*/ + SDL_WINRT_DEVICEFAMILY_DESKTOP, + + /** \brief Mobile family (for example smartphone) */ + SDL_WINRT_DEVICEFAMILY_MOBILE, + + /** \brief XBox family */ + SDL_WINRT_DEVICEFAMILY_XBOX, +} SDL_WinRT_DeviceFamily; + + +/** + * Retrieve a WinRT defined path on the local file system. + * + * Not all paths are available on all versions of Windows. This is especially + * true on Windows Phone. Check the documentation for the given SDL_WinRT_Path + * for more information on which path types are supported where. + * + * Documentation on most app-specific path types on WinRT can be found on + * MSDN, at the URL: + * + * https://msdn.microsoft.com/en-us/library/windows/apps/hh464917.aspx + * + * \param pathType the type of path to retrieve, one of SDL_WinRT_Path + * \returns a UCS-2 string (16-bit, wide-char) containing the path, or NULL if + * the path is not available for any reason; call SDL_GetError() for + * more information. + * + * \since This function is available since SDL 2.0.3. + * + * \sa SDL_WinRTGetFSPathUTF8 + */ +extern DECLSPEC const wchar_t * SDLCALL SDL_WinRTGetFSPathUNICODE(SDL_WinRT_Path pathType); + +/** + * Retrieve a WinRT defined path on the local file system. + * + * Not all paths are available on all versions of Windows. This is especially + * true on Windows Phone. Check the documentation for the given SDL_WinRT_Path + * for more information on which path types are supported where. + * + * Documentation on most app-specific path types on WinRT can be found on + * MSDN, at the URL: + * + * https://msdn.microsoft.com/en-us/library/windows/apps/hh464917.aspx + * + * \param pathType the type of path to retrieve, one of SDL_WinRT_Path + * \returns a UTF-8 string (8-bit, multi-byte) containing the path, or NULL if + * the path is not available for any reason; call SDL_GetError() for + * more information. + * + * \since This function is available since SDL 2.0.3. + * + * \sa SDL_WinRTGetFSPathUNICODE + */ +extern DECLSPEC const char * SDLCALL SDL_WinRTGetFSPathUTF8(SDL_WinRT_Path pathType); + +/** + * Detects the device family of WinRT platform at runtime. + * + * \returns a value from the SDL_WinRT_DeviceFamily enum. + * + * \since This function is available since SDL 2.0.8. + */ +extern DECLSPEC SDL_WinRT_DeviceFamily SDLCALL SDL_WinRTGetDeviceFamily(); + +#endif /* __WINRT__ */ + +/** + * Query if the current device is a tablet. + * + * If SDL can't determine this, it will return SDL_FALSE. + * + * \returns SDL_TRUE if the device is a tablet, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.9. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_IsTablet(void); + +/* Functions used by iOS application delegates to notify SDL about state changes */ +extern DECLSPEC void SDLCALL SDL_OnApplicationWillTerminate(void); +extern DECLSPEC void SDLCALL SDL_OnApplicationDidReceiveMemoryWarning(void); +extern DECLSPEC void SDLCALL SDL_OnApplicationWillResignActive(void); +extern DECLSPEC void SDLCALL SDL_OnApplicationDidEnterBackground(void); +extern DECLSPEC void SDLCALL SDL_OnApplicationWillEnterForeground(void); +extern DECLSPEC void SDLCALL SDL_OnApplicationDidBecomeActive(void); +#ifdef __IPHONEOS__ +extern DECLSPEC void SDLCALL SDL_OnApplicationDidChangeStatusBarOrientation(void); +#endif + +/* Functions used only by GDK */ +#if defined(__GDK__) +typedef struct XTaskQueueObject * XTaskQueueHandle; + +/** + * Gets a reference to the global async task queue handle for GDK, + * initializing if needed. + * + * Once you are done with the task queue, you should call + * XTaskQueueCloseHandle to reduce the reference count to avoid a resource + * leak. + * + * \param outTaskQueue a pointer to be filled in with task queue handle. + * \returns 0 if success, -1 if any error occurs. + * + * \since This function is available since SDL 2.24.0. + */ +extern DECLSPEC int SDLCALL SDL_GDKGetTaskQueue(XTaskQueueHandle * outTaskQueue); + +#endif + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_system_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_syswm.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_syswm.h new file mode 100644 index 00000000..b35734de --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_syswm.h @@ -0,0 +1,386 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_syswm.h + * + * Include file for SDL custom system window manager hooks. + */ + +#ifndef SDL_syswm_h_ +#define SDL_syswm_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_video.h" +#include "SDL_version.h" + +/** + * \brief SDL_syswm.h + * + * Your application has access to a special type of event ::SDL_SYSWMEVENT, + * which contains window-manager specific information and arrives whenever + * an unhandled window event occurs. This event is ignored by default, but + * you can enable it with SDL_EventState(). + */ +struct SDL_SysWMinfo; + +#if !defined(SDL_PROTOTYPES_ONLY) + +#if defined(SDL_VIDEO_DRIVER_WINDOWS) +#ifndef WIN32_LEAN_AND_MEAN +#define WIN32_LEAN_AND_MEAN +#endif +#ifndef NOMINMAX /* don't define min() and max(). */ +#define NOMINMAX +#endif +#include +#endif + +#if defined(SDL_VIDEO_DRIVER_WINRT) +#include +#endif + +/* This is the structure for custom window manager events */ +#if defined(SDL_VIDEO_DRIVER_X11) +#if defined(__APPLE__) && defined(__MACH__) +/* conflicts with Quickdraw.h */ +#define Cursor X11Cursor +#endif + +#include +#include + +#if defined(__APPLE__) && defined(__MACH__) +/* matches the re-define above */ +#undef Cursor +#endif + +#endif /* defined(SDL_VIDEO_DRIVER_X11) */ + +#if defined(SDL_VIDEO_DRIVER_DIRECTFB) +#include +#endif + +#if defined(SDL_VIDEO_DRIVER_COCOA) +#ifdef __OBJC__ +@class NSWindow; +#else +typedef struct _NSWindow NSWindow; +#endif +#endif + +#if defined(SDL_VIDEO_DRIVER_UIKIT) +#ifdef __OBJC__ +#include +#else +typedef struct _UIWindow UIWindow; +typedef struct _UIViewController UIViewController; +#endif +typedef Uint32 GLuint; +#endif + +#if defined(SDL_VIDEO_VULKAN) || defined(SDL_VIDEO_METAL) +#define SDL_METALVIEW_TAG 255 +#endif + +#if defined(SDL_VIDEO_DRIVER_ANDROID) +typedef struct ANativeWindow ANativeWindow; +typedef void *EGLSurface; +#endif + +#if defined(SDL_VIDEO_DRIVER_VIVANTE) +#include "SDL_egl.h" +#endif + +#if defined(SDL_VIDEO_DRIVER_OS2) +#define INCL_WIN +#include +#endif +#endif /* SDL_PROTOTYPES_ONLY */ + +#if defined(SDL_VIDEO_DRIVER_KMSDRM) +struct gbm_device; +#endif + + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +#if !defined(SDL_PROTOTYPES_ONLY) +/** + * These are the various supported windowing subsystems + */ +typedef enum +{ + SDL_SYSWM_UNKNOWN, + SDL_SYSWM_WINDOWS, + SDL_SYSWM_X11, + SDL_SYSWM_DIRECTFB, + SDL_SYSWM_COCOA, + SDL_SYSWM_UIKIT, + SDL_SYSWM_WAYLAND, + SDL_SYSWM_MIR, /* no longer available, left for API/ABI compatibility. Remove in 2.1! */ + SDL_SYSWM_WINRT, + SDL_SYSWM_ANDROID, + SDL_SYSWM_VIVANTE, + SDL_SYSWM_OS2, + SDL_SYSWM_HAIKU, + SDL_SYSWM_KMSDRM, + SDL_SYSWM_RISCOS +} SDL_SYSWM_TYPE; + +/** + * The custom event structure. + */ +struct SDL_SysWMmsg +{ + SDL_version version; + SDL_SYSWM_TYPE subsystem; + union + { +#if defined(SDL_VIDEO_DRIVER_WINDOWS) + struct { + HWND hwnd; /**< The window for the message */ + UINT msg; /**< The type of message */ + WPARAM wParam; /**< WORD message parameter */ + LPARAM lParam; /**< LONG message parameter */ + } win; +#endif +#if defined(SDL_VIDEO_DRIVER_X11) + struct { + XEvent event; + } x11; +#endif +#if defined(SDL_VIDEO_DRIVER_DIRECTFB) + struct { + DFBEvent event; + } dfb; +#endif +#if defined(SDL_VIDEO_DRIVER_COCOA) + struct + { + /* Latest version of Xcode clang complains about empty structs in C v. C++: + error: empty struct has size 0 in C, size 1 in C++ + */ + int dummy; + /* No Cocoa window events yet */ + } cocoa; +#endif +#if defined(SDL_VIDEO_DRIVER_UIKIT) + struct + { + int dummy; + /* No UIKit window events yet */ + } uikit; +#endif +#if defined(SDL_VIDEO_DRIVER_VIVANTE) + struct + { + int dummy; + /* No Vivante window events yet */ + } vivante; +#endif +#if defined(SDL_VIDEO_DRIVER_OS2) + struct + { + BOOL fFrame; /**< TRUE if hwnd is a frame window */ + HWND hwnd; /**< The window receiving the message */ + ULONG msg; /**< The message identifier */ + MPARAM mp1; /**< The first first message parameter */ + MPARAM mp2; /**< The second first message parameter */ + } os2; +#endif + /* Can't have an empty union */ + int dummy; + } msg; +}; + +/** + * The custom window manager information structure. + * + * When this structure is returned, it holds information about which + * low level system it is using, and will be one of SDL_SYSWM_TYPE. + */ +struct SDL_SysWMinfo +{ + SDL_version version; + SDL_SYSWM_TYPE subsystem; + union + { +#if defined(SDL_VIDEO_DRIVER_WINDOWS) + struct + { + HWND window; /**< The window handle */ + HDC hdc; /**< The window device context */ + HINSTANCE hinstance; /**< The instance handle */ + } win; +#endif +#if defined(SDL_VIDEO_DRIVER_WINRT) + struct + { + IInspectable * window; /**< The WinRT CoreWindow */ + } winrt; +#endif +#if defined(SDL_VIDEO_DRIVER_X11) + struct + { + Display *display; /**< The X11 display */ + Window window; /**< The X11 window */ + } x11; +#endif +#if defined(SDL_VIDEO_DRIVER_DIRECTFB) + struct + { + IDirectFB *dfb; /**< The directfb main interface */ + IDirectFBWindow *window; /**< The directfb window handle */ + IDirectFBSurface *surface; /**< The directfb client surface */ + } dfb; +#endif +#if defined(SDL_VIDEO_DRIVER_COCOA) + struct + { +#if defined(__OBJC__) && defined(__has_feature) + #if __has_feature(objc_arc) + NSWindow __unsafe_unretained *window; /**< The Cocoa window */ + #else + NSWindow *window; /**< The Cocoa window */ + #endif +#else + NSWindow *window; /**< The Cocoa window */ +#endif + } cocoa; +#endif +#if defined(SDL_VIDEO_DRIVER_UIKIT) + struct + { +#if defined(__OBJC__) && defined(__has_feature) + #if __has_feature(objc_arc) + UIWindow __unsafe_unretained *window; /**< The UIKit window */ + #else + UIWindow *window; /**< The UIKit window */ + #endif +#else + UIWindow *window; /**< The UIKit window */ +#endif + GLuint framebuffer; /**< The GL view's Framebuffer Object. It must be bound when rendering to the screen using GL. */ + GLuint colorbuffer; /**< The GL view's color Renderbuffer Object. It must be bound when SDL_GL_SwapWindow is called. */ + GLuint resolveFramebuffer; /**< The Framebuffer Object which holds the resolve color Renderbuffer, when MSAA is used. */ + } uikit; +#endif +#if defined(SDL_VIDEO_DRIVER_WAYLAND) + struct + { + struct wl_display *display; /**< Wayland display */ + struct wl_surface *surface; /**< Wayland surface */ + void *shell_surface; /**< DEPRECATED Wayland shell_surface (window manager handle) */ + struct wl_egl_window *egl_window; /**< Wayland EGL window (native window) */ + struct xdg_surface *xdg_surface; /**< Wayland xdg surface (window manager handle) */ + struct xdg_toplevel *xdg_toplevel; /**< Wayland xdg toplevel role */ + struct xdg_popup *xdg_popup; /**< Wayland xdg popup role */ + struct xdg_positioner *xdg_positioner; /**< Wayland xdg positioner, for popup */ + } wl; +#endif +#if defined(SDL_VIDEO_DRIVER_MIR) /* no longer available, left for API/ABI compatibility. Remove in 2.1! */ + struct + { + void *connection; /**< Mir display server connection */ + void *surface; /**< Mir surface */ + } mir; +#endif + +#if defined(SDL_VIDEO_DRIVER_ANDROID) + struct + { + ANativeWindow *window; + EGLSurface surface; + } android; +#endif + +#if defined(SDL_VIDEO_DRIVER_OS2) + struct + { + HWND hwnd; /**< The window handle */ + HWND hwndFrame; /**< The frame window handle */ + } os2; +#endif + +#if defined(SDL_VIDEO_DRIVER_VIVANTE) + struct + { + EGLNativeDisplayType display; + EGLNativeWindowType window; + } vivante; +#endif + +#if defined(SDL_VIDEO_DRIVER_KMSDRM) + struct + { + int dev_index; /**< Device index (ex: the X in /dev/dri/cardX) */ + int drm_fd; /**< DRM FD (unavailable on Vulkan windows) */ + struct gbm_device *gbm_dev; /**< GBM device (unavailable on Vulkan windows) */ + } kmsdrm; +#endif + + /* Make sure this union is always 64 bytes (8 64-bit pointers). */ + /* Be careful not to overflow this if you add a new target! */ + Uint8 dummy[64]; + } info; +}; + +#endif /* SDL_PROTOTYPES_ONLY */ + +typedef struct SDL_SysWMinfo SDL_SysWMinfo; + + +/** + * Get driver-specific information about a window. + * + * You must include SDL_syswm.h for the declaration of SDL_SysWMinfo. + * + * The caller must initialize the `info` structure's version by using + * `SDL_VERSION(&info.version)`, and then this function will fill in the rest + * of the structure with information about the given window. + * + * \param window the window about which information is being requested + * \param info an SDL_SysWMinfo structure filled in with window information + * \returns SDL_TRUE if the function is implemented and the `version` member + * of the `info` struct is valid, or SDL_FALSE if the information + * could not be retrieved; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_GetWindowWMInfo(SDL_Window * window, + SDL_SysWMinfo * info); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_syswm_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test.h new file mode 100644 index 00000000..80daaafb --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test.h @@ -0,0 +1,69 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_test.h + * + * Include file for SDL test framework. + * + * This code is a part of the SDL2_test library, not the main SDL library. + */ + +#ifndef SDL_test_h_ +#define SDL_test_h_ + +#include "SDL.h" +#include "SDL_test_assert.h" +#include "SDL_test_common.h" +#include "SDL_test_compare.h" +#include "SDL_test_crc32.h" +#include "SDL_test_font.h" +#include "SDL_test_fuzzer.h" +#include "SDL_test_harness.h" +#include "SDL_test_images.h" +#include "SDL_test_log.h" +#include "SDL_test_md5.h" +#include "SDL_test_memory.h" +#include "SDL_test_random.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* Global definitions */ + +/* + * Note: Maximum size of SDLTest log message is less than SDL's limit + * to ensure we can fit additional information such as the timestamp. + */ +#define SDLTEST_MAX_LOGMESSAGE_LENGTH 3584 + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_test_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_assert.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_assert.h new file mode 100644 index 00000000..341e490f --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_assert.h @@ -0,0 +1,105 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_test_assert.h + * + * Include file for SDL test framework. + * + * This code is a part of the SDL2_test library, not the main SDL library. + */ + +/* + * + * Assert API for test code and test cases + * + */ + +#ifndef SDL_test_assert_h_ +#define SDL_test_assert_h_ + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \brief Fails the assert. + */ +#define ASSERT_FAIL 0 + +/** + * \brief Passes the assert. + */ +#define ASSERT_PASS 1 + +/** + * \brief Assert that logs and break execution flow on failures. + * + * \param assertCondition Evaluated condition or variable to assert; fail (==0) or pass (!=0). + * \param assertDescription Message to log with the assert describing it. + */ +void SDLTest_Assert(int assertCondition, SDL_PRINTF_FORMAT_STRING const char *assertDescription, ...) SDL_PRINTF_VARARG_FUNC(2); + +/** + * \brief Assert for test cases that logs but does not break execution flow on failures. Updates assertion counters. + * + * \param assertCondition Evaluated condition or variable to assert; fail (==0) or pass (!=0). + * \param assertDescription Message to log with the assert describing it. + * + * \returns the assertCondition so it can be used to externally to break execution flow if desired. + */ +int SDLTest_AssertCheck(int assertCondition, SDL_PRINTF_FORMAT_STRING const char *assertDescription, ...) SDL_PRINTF_VARARG_FUNC(2); + +/** + * \brief Explicitly pass without checking an assertion condition. Updates assertion counter. + * + * \param assertDescription Message to log with the assert describing it. + */ +void SDLTest_AssertPass(SDL_PRINTF_FORMAT_STRING const char *assertDescription, ...) SDL_PRINTF_VARARG_FUNC(1); + +/** + * \brief Resets the assert summary counters to zero. + */ +void SDLTest_ResetAssertSummary(void); + +/** + * \brief Logs summary of all assertions (total, pass, fail) since last reset as INFO or ERROR. + */ +void SDLTest_LogAssertSummary(void); + + +/** + * \brief Converts the current assert summary state to a test result. + * + * \returns TEST_RESULT_PASSED, TEST_RESULT_FAILED, or TEST_RESULT_NO_ASSERT + */ +int SDLTest_AssertSummaryToTestResult(void); + +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_test_assert_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_common.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_common.h new file mode 100644 index 00000000..6de63cad --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_common.h @@ -0,0 +1,236 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_test_common.h + * + * Include file for SDL test framework. + * + * This code is a part of the SDL2_test library, not the main SDL library. + */ + +/* Ported from original test\common.h file. */ + +#ifndef SDL_test_common_h_ +#define SDL_test_common_h_ + +#include "SDL.h" + +#if defined(__PSP__) +#define DEFAULT_WINDOW_WIDTH 480 +#define DEFAULT_WINDOW_HEIGHT 272 +#elif defined(__VITA__) +#define DEFAULT_WINDOW_WIDTH 960 +#define DEFAULT_WINDOW_HEIGHT 544 +#else +#define DEFAULT_WINDOW_WIDTH 640 +#define DEFAULT_WINDOW_HEIGHT 480 +#endif + +#define VERBOSE_VIDEO 0x00000001 +#define VERBOSE_MODES 0x00000002 +#define VERBOSE_RENDER 0x00000004 +#define VERBOSE_EVENT 0x00000008 +#define VERBOSE_AUDIO 0x00000010 +#define VERBOSE_MOTION 0x00000020 + +typedef struct +{ + /* SDL init flags */ + char **argv; + Uint32 flags; + Uint32 verbose; + + /* Video info */ + const char *videodriver; + int display; + const char *window_title; + const char *window_icon; + Uint32 window_flags; + SDL_bool flash_on_focus_loss; + int window_x; + int window_y; + int window_w; + int window_h; + int window_minW; + int window_minH; + int window_maxW; + int window_maxH; + int logical_w; + int logical_h; + float scale; + int depth; + int refresh_rate; + int num_windows; + SDL_Window **windows; + + /* Renderer info */ + const char *renderdriver; + Uint32 render_flags; + SDL_bool skip_renderer; + SDL_Renderer **renderers; + SDL_Texture **targets; + + /* Audio info */ + const char *audiodriver; + SDL_AudioSpec audiospec; + + /* GL settings */ + int gl_red_size; + int gl_green_size; + int gl_blue_size; + int gl_alpha_size; + int gl_buffer_size; + int gl_depth_size; + int gl_stencil_size; + int gl_double_buffer; + int gl_accum_red_size; + int gl_accum_green_size; + int gl_accum_blue_size; + int gl_accum_alpha_size; + int gl_stereo; + int gl_multisamplebuffers; + int gl_multisamplesamples; + int gl_retained_backing; + int gl_accelerated; + int gl_major_version; + int gl_minor_version; + int gl_debug; + int gl_profile_mask; + + /* Additional fields added in 2.0.18 */ + SDL_Rect confine; + +} SDLTest_CommonState; + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* Function prototypes */ + +/** + * \brief Parse command line parameters and create common state. + * + * \param argv Array of command line parameters + * \param flags Flags indicating which subsystem to initialize (i.e. SDL_INIT_VIDEO | SDL_INIT_AUDIO) + * + * \returns a newly allocated common state object. + */ +SDLTest_CommonState *SDLTest_CommonCreateState(char **argv, Uint32 flags); + +/** + * \brief Process one common argument. + * + * \param state The common state describing the test window to create. + * \param index The index of the argument to process in argv[]. + * + * \returns the number of arguments processed (i.e. 1 for --fullscreen, 2 for --video [videodriver], or -1 on error. + */ +int SDLTest_CommonArg(SDLTest_CommonState * state, int index); + + +/** + * \brief Logs command line usage info. + * + * This logs the appropriate command line options for the subsystems in use + * plus other common options, and then any application-specific options. + * This uses the SDL_Log() function and splits up output to be friendly to + * 80-character-wide terminals. + * + * \param state The common state describing the test window for the app. + * \param argv0 argv[0], as passed to main/SDL_main. + * \param options an array of strings for application specific options. The last element of the array should be NULL. + */ +void SDLTest_CommonLogUsage(SDLTest_CommonState * state, const char *argv0, const char **options); + +/** + * \brief Returns common usage information + * + * You should (probably) be using SDLTest_CommonLogUsage() instead, but this + * function remains for binary compatibility. Strings returned from this + * function are valid until SDLTest_CommonQuit() is called, in which case + * those strings' memory is freed and can no longer be used. + * + * \param state The common state describing the test window to create. + * \returns a string with usage information + */ +const char *SDLTest_CommonUsage(SDLTest_CommonState * state); + +/** + * \brief Open test window. + * + * \param state The common state describing the test window to create. + * + * \returns SDL_TRUE if initialization succeeded, false otherwise + */ +SDL_bool SDLTest_CommonInit(SDLTest_CommonState * state); + +/** + * \brief Easy argument handling when test app doesn't need any custom args. + * + * \param state The common state describing the test window to create. + * \param argc argc, as supplied to SDL_main + * \param argv argv, as supplied to SDL_main + * + * \returns SDL_FALSE if app should quit, true otherwise. + */ +SDL_bool SDLTest_CommonDefaultArgs(SDLTest_CommonState * state, const int argc, char **argv); + +/** + * \brief Common event handler for test windows. + * + * \param state The common state used to create test window. + * \param event The event to handle. + * \param done Flag indicating we are done. + * + */ +void SDLTest_CommonEvent(SDLTest_CommonState * state, SDL_Event * event, int *done); + +/** + * \brief Close test window. + * + * \param state The common state used to create test window. + * + */ +void SDLTest_CommonQuit(SDLTest_CommonState * state); + +/** + * \brief Draws various window information (position, size, etc.) to the renderer. + * + * \param renderer The renderer to draw to. + * \param window The window whose information should be displayed. + * \param usedHeight Returns the height used, so the caller can draw more below. + * + */ +void SDLTest_CommonDrawWindowInfo(SDL_Renderer * renderer, SDL_Window * window, int * usedHeight); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_test_common_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_compare.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_compare.h new file mode 100644 index 00000000..5fce25ca --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_compare.h @@ -0,0 +1,69 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_test_compare.h + * + * Include file for SDL test framework. + * + * This code is a part of the SDL2_test library, not the main SDL library. + */ + +/* + + Defines comparison functions (i.e. for surfaces). + +*/ + +#ifndef SDL_test_compare_h_ +#define SDL_test_compare_h_ + +#include "SDL.h" + +#include "SDL_test_images.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \brief Compares a surface and with reference image data for equality + * + * \param surface Surface used in comparison + * \param referenceSurface Test Surface used in comparison + * \param allowable_error Allowable difference (=sum of squared difference for each RGB component) in blending accuracy. + * + * \returns 0 if comparison succeeded, >0 (=number of pixels for which the comparison failed) if comparison failed, -1 if any of the surfaces were NULL, -2 if the surface sizes differ. + */ +int SDLTest_CompareSurfaces(SDL_Surface *surface, SDL_Surface *referenceSurface, int allowable_error); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_test_compare_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_crc32.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_crc32.h new file mode 100644 index 00000000..bf347821 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_crc32.h @@ -0,0 +1,124 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_test_crc32.h + * + * Include file for SDL test framework. + * + * This code is a part of the SDL2_test library, not the main SDL library. + */ + +/* + + Implements CRC32 calculations (default output is Perl String::CRC32 compatible). + +*/ + +#ifndef SDL_test_crc32_h_ +#define SDL_test_crc32_h_ + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + + +/* ------------ Definitions --------- */ + +/* Definition shared by all CRC routines */ + +#ifndef CrcUint32 + #define CrcUint32 unsigned int +#endif +#ifndef CrcUint8 + #define CrcUint8 unsigned char +#endif + +#ifdef ORIGINAL_METHOD + #define CRC32_POLY 0x04c11db7 /* AUTODIN II, Ethernet, & FDDI */ +#else + #define CRC32_POLY 0xEDB88320 /* Perl String::CRC32 compatible */ +#endif + +/** + * Data structure for CRC32 (checksum) computation + */ + typedef struct { + CrcUint32 crc32_table[256]; /* CRC table */ + } SDLTest_Crc32Context; + +/* ---------- Function Prototypes ------------- */ + +/** + * \brief Initialize the CRC context + * + * Note: The function initializes the crc table required for all crc calculations. + * + * \param crcContext pointer to context variable + * + * \returns 0 for OK, -1 on error + * + */ + int SDLTest_Crc32Init(SDLTest_Crc32Context * crcContext); + + +/** + * \brief calculate a crc32 from a data block + * + * \param crcContext pointer to context variable + * \param inBuf input buffer to checksum + * \param inLen length of input buffer + * \param crc32 pointer to Uint32 to store the final CRC into + * + * \returns 0 for OK, -1 on error + * + */ +int SDLTest_Crc32Calc(SDLTest_Crc32Context * crcContext, CrcUint8 *inBuf, CrcUint32 inLen, CrcUint32 *crc32); + +/* Same routine broken down into three steps */ +int SDLTest_Crc32CalcStart(SDLTest_Crc32Context * crcContext, CrcUint32 *crc32); +int SDLTest_Crc32CalcEnd(SDLTest_Crc32Context * crcContext, CrcUint32 *crc32); +int SDLTest_Crc32CalcBuffer(SDLTest_Crc32Context * crcContext, CrcUint8 *inBuf, CrcUint32 inLen, CrcUint32 *crc32); + + +/** + * \brief clean up CRC context + * + * \param crcContext pointer to context variable + * + * \returns 0 for OK, -1 on error + * +*/ + +int SDLTest_Crc32Done(SDLTest_Crc32Context * crcContext); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_test_crc32_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_font.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_font.h new file mode 100644 index 00000000..18a82ffc --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_font.h @@ -0,0 +1,168 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_test_font.h + * + * Include file for SDL test framework. + * + * This code is a part of the SDL2_test library, not the main SDL library. + */ + +#ifndef SDL_test_font_h_ +#define SDL_test_font_h_ + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* Function prototypes */ + +#define FONT_CHARACTER_SIZE 8 +#define FONT_LINE_HEIGHT (FONT_CHARACTER_SIZE + 2) + +/** + * \brief Draw a string in the currently set font. + * + * \param renderer The renderer to draw on. + * \param x The X coordinate of the upper left corner of the character. + * \param y The Y coordinate of the upper left corner of the character. + * \param c The character to draw. + * + * \returns 0 on success, -1 on failure. + */ +int SDLTest_DrawCharacter(SDL_Renderer *renderer, int x, int y, Uint32 c); + +/** + * \brief Draw a UTF-8 string in the currently set font. + * + * The font currently only supports characters in the Basic Latin and Latin-1 Supplement sets. + * + * \param renderer The renderer to draw on. + * \param x The X coordinate of the upper left corner of the string. + * \param y The Y coordinate of the upper left corner of the string. + * \param s The string to draw. + * + * \returns 0 on success, -1 on failure. + */ +int SDLTest_DrawString(SDL_Renderer *renderer, int x, int y, const char *s); + +/** + * \brief Data used for multi-line text output + */ +typedef struct SDLTest_TextWindow +{ + SDL_Rect rect; + int current; + int numlines; + char **lines; +} SDLTest_TextWindow; + +/** + * \brief Create a multi-line text output window + * + * \param x The X coordinate of the upper left corner of the window. + * \param y The Y coordinate of the upper left corner of the window. + * \param w The width of the window (currently ignored) + * \param h The height of the window (currently ignored) + * + * \returns the new window, or NULL on failure. + * + * \since This function is available since SDL 2.24.0 + */ +SDLTest_TextWindow *SDLTest_TextWindowCreate(int x, int y, int w, int h); + +/** + * \brief Display a multi-line text output window + * + * This function should be called every frame to display the text + * + * \param textwin The text output window + * \param renderer The renderer to use for display + * + * \since This function is available since SDL 2.24.0 + */ +void SDLTest_TextWindowDisplay(SDLTest_TextWindow *textwin, SDL_Renderer *renderer); + +/** + * \brief Add text to a multi-line text output window + * + * Adds UTF-8 text to the end of the current text. The newline character starts a + * new line of text. The backspace character deletes the last character or, if the + * line is empty, deletes the line and goes to the end of the previous line. + * + * \param textwin The text output window + * \param fmt A printf() style format string + * \param ... additional parameters matching % tokens in the `fmt` string, if any + * + * \since This function is available since SDL 2.24.0 + */ +void SDLTest_TextWindowAddText(SDLTest_TextWindow *textwin, SDL_PRINTF_FORMAT_STRING const char *fmt, ...) SDL_PRINTF_VARARG_FUNC(2); + +/** + * \brief Add text to a multi-line text output window + * + * Adds UTF-8 text to the end of the current text. The newline character starts a + * new line of text. The backspace character deletes the last character or, if the + * line is empty, deletes the line and goes to the end of the previous line. + * + * \param textwin The text output window + * \param text The text to add to the window + * \param len The length, in bytes, of the text to add to the window + * + * \since This function is available since SDL 2.24.0 + */ +void SDLTest_TextWindowAddTextWithLength(SDLTest_TextWindow *textwin, const char *text, size_t len); + +/** + * \brief Clear the text in a multi-line text output window + * + * \param textwin The text output window + * + * \since This function is available since SDL 2.24.0 + */ +void SDLTest_TextWindowClear(SDLTest_TextWindow *textwin); + +/** + * \brief Free the storage associated with a multi-line text output window + * + * \param textwin The text output window + * + * \since This function is available since SDL 2.24.0 + */ +void SDLTest_TextWindowDestroy(SDLTest_TextWindow *textwin); + +/** + * \brief Cleanup textures used by font drawing functions. + */ +void SDLTest_CleanupTextDrawing(void); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_test_font_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_fuzzer.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_fuzzer.h new file mode 100644 index 00000000..cfe6a14f --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_fuzzer.h @@ -0,0 +1,386 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_test_fuzzer.h + * + * Include file for SDL test framework. + * + * This code is a part of the SDL2_test library, not the main SDL library. + */ + +/* + + Data generators for fuzzing test data in a reproducible way. + +*/ + +#ifndef SDL_test_fuzzer_h_ +#define SDL_test_fuzzer_h_ + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + + +/* + Based on GSOC code by Markus Kauppila +*/ + + +/** + * \file + * Note: The fuzzer implementation uses a static instance of random context + * internally which makes it thread-UNsafe. + */ + +/** + * Initializes the fuzzer for a test + * + * \param execKey Execution "Key" that initializes the random number generator uniquely for the test. + * + */ +void SDLTest_FuzzerInit(Uint64 execKey); + + +/** + * Returns a random Uint8 + * + * \returns a generated integer + */ +Uint8 SDLTest_RandomUint8(void); + +/** + * Returns a random Sint8 + * + * \returns a generated signed integer + */ +Sint8 SDLTest_RandomSint8(void); + + +/** + * Returns a random Uint16 + * + * \returns a generated integer + */ +Uint16 SDLTest_RandomUint16(void); + +/** + * Returns a random Sint16 + * + * \returns a generated signed integer + */ +Sint16 SDLTest_RandomSint16(void); + + +/** + * Returns a random integer + * + * \returns a generated integer + */ +Sint32 SDLTest_RandomSint32(void); + + +/** + * Returns a random positive integer + * + * \returns a generated integer + */ +Uint32 SDLTest_RandomUint32(void); + +/** + * Returns random Uint64. + * + * \returns a generated integer + */ +Uint64 SDLTest_RandomUint64(void); + + +/** + * Returns random Sint64. + * + * \returns a generated signed integer + */ +Sint64 SDLTest_RandomSint64(void); + +/** + * \returns a random float in range [0.0 - 1.0] + */ +float SDLTest_RandomUnitFloat(void); + +/** + * \returns a random double in range [0.0 - 1.0] + */ +double SDLTest_RandomUnitDouble(void); + +/** + * \returns a random float. + * + */ +float SDLTest_RandomFloat(void); + +/** + * \returns a random double. + * + */ +double SDLTest_RandomDouble(void); + +/** + * Returns a random boundary value for Uint8 within the given boundaries. + * Boundaries are inclusive, see the usage examples below. If validDomain + * is true, the function will only return valid boundaries, otherwise non-valid + * boundaries are also possible. + * If boundary1 > boundary2, the values are swapped + * + * Usage examples: + * RandomUint8BoundaryValue(10, 20, SDL_TRUE) returns 10, 11, 19 or 20 + * RandomUint8BoundaryValue(1, 20, SDL_FALSE) returns 0 or 21 + * RandomUint8BoundaryValue(0, 99, SDL_FALSE) returns 100 + * RandomUint8BoundaryValue(0, 255, SDL_FALSE) returns 0 (error set) + * + * \param boundary1 Lower boundary limit + * \param boundary2 Upper boundary limit + * \param validDomain Should the generated boundary be valid (=within the bounds) or not? + * + * \returns a random boundary value for the given range and domain or 0 with error set + */ +Uint8 SDLTest_RandomUint8BoundaryValue(Uint8 boundary1, Uint8 boundary2, SDL_bool validDomain); + +/** + * Returns a random boundary value for Uint16 within the given boundaries. + * Boundaries are inclusive, see the usage examples below. If validDomain + * is true, the function will only return valid boundaries, otherwise non-valid + * boundaries are also possible. + * If boundary1 > boundary2, the values are swapped + * + * Usage examples: + * RandomUint16BoundaryValue(10, 20, SDL_TRUE) returns 10, 11, 19 or 20 + * RandomUint16BoundaryValue(1, 20, SDL_FALSE) returns 0 or 21 + * RandomUint16BoundaryValue(0, 99, SDL_FALSE) returns 100 + * RandomUint16BoundaryValue(0, 0xFFFF, SDL_FALSE) returns 0 (error set) + * + * \param boundary1 Lower boundary limit + * \param boundary2 Upper boundary limit + * \param validDomain Should the generated boundary be valid (=within the bounds) or not? + * + * \returns a random boundary value for the given range and domain or 0 with error set + */ +Uint16 SDLTest_RandomUint16BoundaryValue(Uint16 boundary1, Uint16 boundary2, SDL_bool validDomain); + +/** + * Returns a random boundary value for Uint32 within the given boundaries. + * Boundaries are inclusive, see the usage examples below. If validDomain + * is true, the function will only return valid boundaries, otherwise non-valid + * boundaries are also possible. + * If boundary1 > boundary2, the values are swapped + * + * Usage examples: + * RandomUint32BoundaryValue(10, 20, SDL_TRUE) returns 10, 11, 19 or 20 + * RandomUint32BoundaryValue(1, 20, SDL_FALSE) returns 0 or 21 + * RandomUint32BoundaryValue(0, 99, SDL_FALSE) returns 100 + * RandomUint32BoundaryValue(0, 0xFFFFFFFF, SDL_FALSE) returns 0 (with error set) + * + * \param boundary1 Lower boundary limit + * \param boundary2 Upper boundary limit + * \param validDomain Should the generated boundary be valid (=within the bounds) or not? + * + * \returns a random boundary value for the given range and domain or 0 with error set + */ +Uint32 SDLTest_RandomUint32BoundaryValue(Uint32 boundary1, Uint32 boundary2, SDL_bool validDomain); + +/** + * Returns a random boundary value for Uint64 within the given boundaries. + * Boundaries are inclusive, see the usage examples below. If validDomain + * is true, the function will only return valid boundaries, otherwise non-valid + * boundaries are also possible. + * If boundary1 > boundary2, the values are swapped + * + * Usage examples: + * RandomUint64BoundaryValue(10, 20, SDL_TRUE) returns 10, 11, 19 or 20 + * RandomUint64BoundaryValue(1, 20, SDL_FALSE) returns 0 or 21 + * RandomUint64BoundaryValue(0, 99, SDL_FALSE) returns 100 + * RandomUint64BoundaryValue(0, 0xFFFFFFFFFFFFFFFF, SDL_FALSE) returns 0 (with error set) + * + * \param boundary1 Lower boundary limit + * \param boundary2 Upper boundary limit + * \param validDomain Should the generated boundary be valid (=within the bounds) or not? + * + * \returns a random boundary value for the given range and domain or 0 with error set + */ +Uint64 SDLTest_RandomUint64BoundaryValue(Uint64 boundary1, Uint64 boundary2, SDL_bool validDomain); + +/** + * Returns a random boundary value for Sint8 within the given boundaries. + * Boundaries are inclusive, see the usage examples below. If validDomain + * is true, the function will only return valid boundaries, otherwise non-valid + * boundaries are also possible. + * If boundary1 > boundary2, the values are swapped + * + * Usage examples: + * RandomSint8BoundaryValue(-10, 20, SDL_TRUE) returns -11, -10, 19 or 20 + * RandomSint8BoundaryValue(-100, -10, SDL_FALSE) returns -101 or -9 + * RandomSint8BoundaryValue(SINT8_MIN, 99, SDL_FALSE) returns 100 + * RandomSint8BoundaryValue(SINT8_MIN, SINT8_MAX, SDL_FALSE) returns SINT8_MIN (== error value) with error set + * + * \param boundary1 Lower boundary limit + * \param boundary2 Upper boundary limit + * \param validDomain Should the generated boundary be valid (=within the bounds) or not? + * + * \returns a random boundary value for the given range and domain or SINT8_MIN with error set + */ +Sint8 SDLTest_RandomSint8BoundaryValue(Sint8 boundary1, Sint8 boundary2, SDL_bool validDomain); + + +/** + * Returns a random boundary value for Sint16 within the given boundaries. + * Boundaries are inclusive, see the usage examples below. If validDomain + * is true, the function will only return valid boundaries, otherwise non-valid + * boundaries are also possible. + * If boundary1 > boundary2, the values are swapped + * + * Usage examples: + * RandomSint16BoundaryValue(-10, 20, SDL_TRUE) returns -11, -10, 19 or 20 + * RandomSint16BoundaryValue(-100, -10, SDL_FALSE) returns -101 or -9 + * RandomSint16BoundaryValue(SINT16_MIN, 99, SDL_FALSE) returns 100 + * RandomSint16BoundaryValue(SINT16_MIN, SINT16_MAX, SDL_FALSE) returns SINT16_MIN (== error value) with error set + * + * \param boundary1 Lower boundary limit + * \param boundary2 Upper boundary limit + * \param validDomain Should the generated boundary be valid (=within the bounds) or not? + * + * \returns a random boundary value for the given range and domain or SINT16_MIN with error set + */ +Sint16 SDLTest_RandomSint16BoundaryValue(Sint16 boundary1, Sint16 boundary2, SDL_bool validDomain); + +/** + * Returns a random boundary value for Sint32 within the given boundaries. + * Boundaries are inclusive, see the usage examples below. If validDomain + * is true, the function will only return valid boundaries, otherwise non-valid + * boundaries are also possible. + * If boundary1 > boundary2, the values are swapped + * + * Usage examples: + * RandomSint32BoundaryValue(-10, 20, SDL_TRUE) returns -11, -10, 19 or 20 + * RandomSint32BoundaryValue(-100, -10, SDL_FALSE) returns -101 or -9 + * RandomSint32BoundaryValue(SINT32_MIN, 99, SDL_FALSE) returns 100 + * RandomSint32BoundaryValue(SINT32_MIN, SINT32_MAX, SDL_FALSE) returns SINT32_MIN (== error value) + * + * \param boundary1 Lower boundary limit + * \param boundary2 Upper boundary limit + * \param validDomain Should the generated boundary be valid (=within the bounds) or not? + * + * \returns a random boundary value for the given range and domain or SINT32_MIN with error set + */ +Sint32 SDLTest_RandomSint32BoundaryValue(Sint32 boundary1, Sint32 boundary2, SDL_bool validDomain); + +/** + * Returns a random boundary value for Sint64 within the given boundaries. + * Boundaries are inclusive, see the usage examples below. If validDomain + * is true, the function will only return valid boundaries, otherwise non-valid + * boundaries are also possible. + * If boundary1 > boundary2, the values are swapped + * + * Usage examples: + * RandomSint64BoundaryValue(-10, 20, SDL_TRUE) returns -11, -10, 19 or 20 + * RandomSint64BoundaryValue(-100, -10, SDL_FALSE) returns -101 or -9 + * RandomSint64BoundaryValue(SINT64_MIN, 99, SDL_FALSE) returns 100 + * RandomSint64BoundaryValue(SINT64_MIN, SINT64_MAX, SDL_FALSE) returns SINT64_MIN (== error value) and error set + * + * \param boundary1 Lower boundary limit + * \param boundary2 Upper boundary limit + * \param validDomain Should the generated boundary be valid (=within the bounds) or not? + * + * \returns a random boundary value for the given range and domain or SINT64_MIN with error set + */ +Sint64 SDLTest_RandomSint64BoundaryValue(Sint64 boundary1, Sint64 boundary2, SDL_bool validDomain); + + +/** + * Returns integer in range [min, max] (inclusive). + * Min and max values can be negative values. + * If Max in smaller than min, then the values are swapped. + * Min and max are the same value, that value will be returned. + * + * \param min Minimum inclusive value of returned random number + * \param max Maximum inclusive value of returned random number + * + * \returns a generated random integer in range + */ +Sint32 SDLTest_RandomIntegerInRange(Sint32 min, Sint32 max); + + +/** + * Generates random null-terminated string. The minimum length for + * the string is 1 character, maximum length for the string is 255 + * characters and it can contain ASCII characters from 32 to 126. + * + * Note: Returned string needs to be deallocated. + * + * \returns a newly allocated random string; or NULL if length was invalid or string could not be allocated. + */ +char * SDLTest_RandomAsciiString(void); + + +/** + * Generates random null-terminated string. The maximum length for + * the string is defined by the maxLength parameter. + * String can contain ASCII characters from 32 to 126. + * + * Note: Returned string needs to be deallocated. + * + * \param maxLength The maximum length of the generated string. + * + * \returns a newly allocated random string; or NULL if maxLength was invalid or string could not be allocated. + */ +char * SDLTest_RandomAsciiStringWithMaximumLength(int maxLength); + + +/** + * Generates random null-terminated string. The length for + * the string is defined by the size parameter. + * String can contain ASCII characters from 32 to 126. + * + * Note: Returned string needs to be deallocated. + * + * \param size The length of the generated string + * + * \returns a newly allocated random string; or NULL if size was invalid or string could not be allocated. + */ +char * SDLTest_RandomAsciiStringOfSize(int size); + +/** + * Get the invocation count for the fuzzer since last ...FuzzerInit. + * + * \returns the invocation count. + */ +int SDLTest_GetFuzzerInvocationCount(void); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_test_fuzzer_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_harness.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_harness.h new file mode 100644 index 00000000..26231dcd --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_harness.h @@ -0,0 +1,134 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_test_harness.h + * + * Include file for SDL test framework. + * + * This code is a part of the SDL2_test library, not the main SDL library. + */ + +/* + Defines types for test case definitions and the test execution harness API. + + Based on original GSOC code by Markus Kauppila +*/ + +#ifndef SDL_test_h_arness_h +#define SDL_test_h_arness_h + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + + +/* ! Definitions for test case structures */ +#define TEST_ENABLED 1 +#define TEST_DISABLED 0 + +/* ! Definition of all the possible test return values of the test case method */ +#define TEST_ABORTED -1 +#define TEST_STARTED 0 +#define TEST_COMPLETED 1 +#define TEST_SKIPPED 2 + +/* ! Definition of all the possible test results for the harness */ +#define TEST_RESULT_PASSED 0 +#define TEST_RESULT_FAILED 1 +#define TEST_RESULT_NO_ASSERT 2 +#define TEST_RESULT_SKIPPED 3 +#define TEST_RESULT_SETUP_FAILURE 4 + +/* !< Function pointer to a test case setup function (run before every test) */ +typedef void (*SDLTest_TestCaseSetUpFp)(void *arg); + +/* !< Function pointer to a test case function */ +typedef int (*SDLTest_TestCaseFp)(void *arg); + +/* !< Function pointer to a test case teardown function (run after every test) */ +typedef void (*SDLTest_TestCaseTearDownFp)(void *arg); + +/** + * Holds information about a single test case. + */ +typedef struct SDLTest_TestCaseReference { + /* !< Func2Stress */ + SDLTest_TestCaseFp testCase; + /* !< Short name (or function name) "Func2Stress" */ + const char *name; + /* !< Long name or full description "This test pushes func2() to the limit." */ + const char *description; + /* !< Set to TEST_ENABLED or TEST_DISABLED (test won't be run) */ + int enabled; +} SDLTest_TestCaseReference; + +/** + * Holds information about a test suite (multiple test cases). + */ +typedef struct SDLTest_TestSuiteReference { + /* !< "PlatformSuite" */ + const char *name; + /* !< The function that is run before each test. NULL skips. */ + SDLTest_TestCaseSetUpFp testSetUp; + /* !< The test cases that are run as part of the suite. Last item should be NULL. */ + const SDLTest_TestCaseReference **testCases; + /* !< The function that is run after each test. NULL skips. */ + SDLTest_TestCaseTearDownFp testTearDown; +} SDLTest_TestSuiteReference; + + +/** + * \brief Generates a random run seed string for the harness. The generated seed will contain alphanumeric characters (0-9A-Z). + * + * Note: The returned string needs to be deallocated by the caller. + * + * \param length The length of the seed string to generate + * + * \returns the generated seed string + */ +char *SDLTest_GenerateRunSeed(const int length); + +/** + * \brief Execute a test suite using the given run seed and execution key. + * + * \param testSuites Suites containing the test case. + * \param userRunSeed Custom run seed provided by user, or NULL to autogenerate one. + * \param userExecKey Custom execution key provided by user, or 0 to autogenerate one. + * \param filter Filter specification. NULL disables. Case sensitive. + * \param testIterations Number of iterations to run each test case. + * + * \returns the test run result: 0 when all tests passed, 1 if any tests failed. + */ +int SDLTest_RunSuites(SDLTest_TestSuiteReference *testSuites[], const char *userRunSeed, Uint64 userExecKey, const char *filter, int testIterations); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_test_h_arness_h */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_images.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_images.h new file mode 100644 index 00000000..12113717 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_images.h @@ -0,0 +1,78 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_test_images.h + * + * Include file for SDL test framework. + * + * This code is a part of the SDL2_test library, not the main SDL library. + */ + +/* + + Defines some images for tests. + +*/ + +#ifndef SDL_test_images_h_ +#define SDL_test_images_h_ + +#include "SDL.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + *Type for test images. + */ +typedef struct SDLTest_SurfaceImage_s { + int width; + int height; + unsigned int bytes_per_pixel; /* 3:RGB, 4:RGBA */ + const char *pixel_data; +} SDLTest_SurfaceImage_t; + +/* Test images */ +SDL_Surface *SDLTest_ImageBlit(void); +SDL_Surface *SDLTest_ImageBlitColor(void); +SDL_Surface *SDLTest_ImageBlitAlpha(void); +SDL_Surface *SDLTest_ImageBlitBlendAdd(void); +SDL_Surface *SDLTest_ImageBlitBlend(void); +SDL_Surface *SDLTest_ImageBlitBlendMod(void); +SDL_Surface *SDLTest_ImageBlitBlendNone(void); +SDL_Surface *SDLTest_ImageBlitBlendAll(void); +SDL_Surface *SDLTest_ImageFace(void); +SDL_Surface *SDLTest_ImagePrimitives(void); +SDL_Surface *SDLTest_ImagePrimitivesBlend(void); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_test_images_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_log.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_log.h new file mode 100644 index 00000000..a27ffc20 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_log.h @@ -0,0 +1,67 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_test_log.h + * + * Include file for SDL test framework. + * + * This code is a part of the SDL2_test library, not the main SDL library. + */ + +/* + * + * Wrapper to log in the TEST category + * + */ + +#ifndef SDL_test_log_h_ +#define SDL_test_log_h_ + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \brief Prints given message with a timestamp in the TEST category and INFO priority. + * + * \param fmt Message to be logged + */ +void SDLTest_Log(SDL_PRINTF_FORMAT_STRING const char *fmt, ...) SDL_PRINTF_VARARG_FUNC(1); + +/** + * \brief Prints given message with a timestamp in the TEST category and the ERROR priority. + * + * \param fmt Message to be logged + */ +void SDLTest_LogError(SDL_PRINTF_FORMAT_STRING const char *fmt, ...) SDL_PRINTF_VARARG_FUNC(1); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_test_log_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_md5.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_md5.h new file mode 100644 index 00000000..538c7ae3 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_md5.h @@ -0,0 +1,129 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_test_md5.h + * + * Include file for SDL test framework. + * + * This code is a part of the SDL2_test library, not the main SDL library. + */ + +/* + *********************************************************************** + ** Header file for implementation of MD5 ** + ** RSA Data Security, Inc. MD5 Message-Digest Algorithm ** + ** Created: 2/17/90 RLR ** + ** Revised: 12/27/90 SRD,AJ,BSK,JT Reference C version ** + ** Revised (for MD5): RLR 4/27/91 ** + ** -- G modified to have y&~z instead of y&z ** + ** -- FF, GG, HH modified to add in last register done ** + ** -- Access pattern: round 2 works mod 5, round 3 works mod 3 ** + ** -- distinct additive constant for each step ** + ** -- round 4 added, working mod 7 ** + *********************************************************************** +*/ + +/* + *********************************************************************** + ** Message-digest routines: ** + ** To form the message digest for a message M ** + ** (1) Initialize a context buffer mdContext using MD5Init ** + ** (2) Call MD5Update on mdContext and M ** + ** (3) Call MD5Final on mdContext ** + ** The message digest is now in mdContext->digest[0...15] ** + *********************************************************************** +*/ + +#ifndef SDL_test_md5_h_ +#define SDL_test_md5_h_ + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* ------------ Definitions --------- */ + +/* typedef a 32-bit type */ + typedef unsigned long int MD5UINT4; + +/* Data structure for MD5 (Message-Digest) computation */ + typedef struct { + MD5UINT4 i[2]; /* number of _bits_ handled mod 2^64 */ + MD5UINT4 buf[4]; /* scratch buffer */ + unsigned char in[64]; /* input buffer */ + unsigned char digest[16]; /* actual digest after Md5Final call */ + } SDLTest_Md5Context; + +/* ---------- Function Prototypes ------------- */ + +/** + * \brief initialize the context + * + * \param mdContext pointer to context variable + * + * Note: The function initializes the message-digest context + * mdContext. Call before each new use of the context - + * all fields are set to zero. + */ + void SDLTest_Md5Init(SDLTest_Md5Context * mdContext); + + +/** + * \brief update digest from variable length data + * + * \param mdContext pointer to context variable + * \param inBuf pointer to data array/string + * \param inLen length of data array/string + * + * Note: The function updates the message-digest context to account + * for the presence of each of the characters inBuf[0..inLen-1] + * in the message whose digest is being computed. +*/ + + void SDLTest_Md5Update(SDLTest_Md5Context * mdContext, unsigned char *inBuf, + unsigned int inLen); + + +/** + * \brief complete digest computation + * + * \param mdContext pointer to context variable + * + * Note: The function terminates the message-digest computation and + * ends with the desired message digest in mdContext.digest[0..15]. + * Always call before using the digest[] variable. +*/ + + void SDLTest_Md5Final(SDLTest_Md5Context * mdContext); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_test_md5_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_memory.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_memory.h new file mode 100644 index 00000000..f959177d --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_memory.h @@ -0,0 +1,63 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_test_memory.h + * + * Include file for SDL test framework. + * + * This code is a part of the SDL2_test library, not the main SDL library. + */ + +#ifndef SDL_test_memory_h_ +#define SDL_test_memory_h_ + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * \brief Start tracking SDL memory allocations + * + * \note This should be called before any other SDL functions for complete tracking coverage + */ +int SDLTest_TrackAllocations(void); + +/** + * \brief Print a log of any outstanding allocations + * + * \note This can be called after SDL_Quit() + */ +void SDLTest_LogAllocations(void); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_test_memory_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_random.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_random.h new file mode 100644 index 00000000..0035a803 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_test_random.h @@ -0,0 +1,115 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_test_random.h + * + * Include file for SDL test framework. + * + * This code is a part of the SDL2_test library, not the main SDL library. + */ + +/* + + A "32-bit Multiply with carry random number generator. Very fast. + Includes a list of recommended multipliers. + + multiply-with-carry generator: x(n) = a*x(n-1) + carry mod 2^32. + period: (a*2^31)-1 + +*/ + +#ifndef SDL_test_random_h_ +#define SDL_test_random_h_ + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* --- Definitions */ + +/* + * Macros that return a random number in a specific format. + */ +#define SDLTest_RandomInt(c) ((int)SDLTest_Random(c)) + +/* + * Context structure for the random number generator state. + */ + typedef struct { + unsigned int a; + unsigned int x; + unsigned int c; + unsigned int ah; + unsigned int al; + } SDLTest_RandomContext; + + +/* --- Function prototypes */ + +/** + * \brief Initialize random number generator with two integers. + * + * Note: The random sequence of numbers returned by ...Random() is the + * same for the same two integers and has a period of 2^31. + * + * \param rndContext pointer to context structure + * \param xi integer that defines the random sequence + * \param ci integer that defines the random sequence + * + */ + void SDLTest_RandomInit(SDLTest_RandomContext * rndContext, unsigned int xi, + unsigned int ci); + +/** + * \brief Initialize random number generator based on current system time. + * + * \param rndContext pointer to context structure + * + */ + void SDLTest_RandomInitTime(SDLTest_RandomContext *rndContext); + + +/** + * \brief Initialize random number generator based on current system time. + * + * Note: ...RandomInit() or ...RandomInitTime() must have been called + * before using this function. + * + * \param rndContext pointer to context structure + * + * \returns a random number (32bit unsigned integer) + * + */ + unsigned int SDLTest_Random(SDLTest_RandomContext *rndContext); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_test_random_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_thread.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_thread.h new file mode 100644 index 00000000..b829bbad --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_thread.h @@ -0,0 +1,464 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +#ifndef SDL_thread_h_ +#define SDL_thread_h_ + +/** + * \file SDL_thread.h + * + * Header for the SDL thread management routines. + */ + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +/* Thread synchronization primitives */ +#include "SDL_atomic.h" +#include "SDL_mutex.h" + +#if (defined(__WIN32__) || defined(__GDK__)) && !defined(__WINRT__) +#include /* _beginthreadex() and _endthreadex() */ +#endif +#if defined(__OS2__) /* for _beginthread() and _endthread() */ +#ifndef __EMX__ +#include +#else +#include +#endif +#endif + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* The SDL thread structure, defined in SDL_thread.c */ +struct SDL_Thread; +typedef struct SDL_Thread SDL_Thread; + +/* The SDL thread ID */ +typedef unsigned long SDL_threadID; + +/* Thread local storage ID, 0 is the invalid ID */ +typedef unsigned int SDL_TLSID; + +/** + * The SDL thread priority. + * + * SDL will make system changes as necessary in order to apply the thread priority. + * Code which attempts to control thread state related to priority should be aware + * that calling SDL_SetThreadPriority may alter such state. + * SDL_HINT_THREAD_PRIORITY_POLICY can be used to control aspects of this behavior. + * + * \note On many systems you require special privileges to set high or time critical priority. + */ +typedef enum { + SDL_THREAD_PRIORITY_LOW, + SDL_THREAD_PRIORITY_NORMAL, + SDL_THREAD_PRIORITY_HIGH, + SDL_THREAD_PRIORITY_TIME_CRITICAL +} SDL_ThreadPriority; + +/** + * The function passed to SDL_CreateThread(). + * + * \param data what was passed as `data` to SDL_CreateThread() + * \returns a value that can be reported through SDL_WaitThread(). + */ +typedef int (SDLCALL * SDL_ThreadFunction) (void *data); + + +#if (defined(__WIN32__) || defined(__GDK__)) && !defined(__WINRT__) +/** + * \file SDL_thread.h + * + * We compile SDL into a DLL. This means, that it's the DLL which + * creates a new thread for the calling process with the SDL_CreateThread() + * API. There is a problem with this, that only the RTL of the SDL2.DLL will + * be initialized for those threads, and not the RTL of the calling + * application! + * + * To solve this, we make a little hack here. + * + * We'll always use the caller's _beginthread() and _endthread() APIs to + * start a new thread. This way, if it's the SDL2.DLL which uses this API, + * then the RTL of SDL2.DLL will be used to create the new thread, and if it's + * the application, then the RTL of the application will be used. + * + * So, in short: + * Always use the _beginthread() and _endthread() of the calling runtime + * library! + */ +#define SDL_PASSED_BEGINTHREAD_ENDTHREAD + +typedef uintptr_t (__cdecl * pfnSDL_CurrentBeginThread) + (void *, unsigned, unsigned (__stdcall *func)(void *), + void * /*arg*/, unsigned, unsigned * /* threadID */); +typedef void (__cdecl * pfnSDL_CurrentEndThread) (unsigned code); + +#ifndef SDL_beginthread +#define SDL_beginthread _beginthreadex +#endif +#ifndef SDL_endthread +#define SDL_endthread _endthreadex +#endif + +extern DECLSPEC SDL_Thread *SDLCALL +SDL_CreateThread(SDL_ThreadFunction fn, const char *name, void *data, + pfnSDL_CurrentBeginThread pfnBeginThread, + pfnSDL_CurrentEndThread pfnEndThread); + +extern DECLSPEC SDL_Thread *SDLCALL +SDL_CreateThreadWithStackSize(SDL_ThreadFunction fn, + const char *name, const size_t stacksize, void *data, + pfnSDL_CurrentBeginThread pfnBeginThread, + pfnSDL_CurrentEndThread pfnEndThread); + + +#if defined(SDL_CreateThread) && SDL_DYNAMIC_API +#undef SDL_CreateThread +#define SDL_CreateThread(fn, name, data) SDL_CreateThread_REAL(fn, name, data, (pfnSDL_CurrentBeginThread)SDL_beginthread, (pfnSDL_CurrentEndThread)SDL_endthread) +#undef SDL_CreateThreadWithStackSize +#define SDL_CreateThreadWithStackSize(fn, name, stacksize, data) SDL_CreateThreadWithStackSize_REAL(fn, name, stacksize, data, (pfnSDL_CurrentBeginThread)SDL_beginthread, (pfnSDL_CurrentEndThread)SDL_endthread) +#else +#define SDL_CreateThread(fn, name, data) SDL_CreateThread(fn, name, data, (pfnSDL_CurrentBeginThread)SDL_beginthread, (pfnSDL_CurrentEndThread)SDL_endthread) +#define SDL_CreateThreadWithStackSize(fn, name, stacksize, data) SDL_CreateThreadWithStackSize(fn, name, stacksize, data, (pfnSDL_CurrentBeginThread)SDL_beginthread, (pfnSDL_CurrentEndThread)SDL_endthread) +#endif + +#elif defined(__OS2__) +/* + * just like the windows case above: We compile SDL2 + * into a dll with Watcom's runtime statically linked. + */ +#define SDL_PASSED_BEGINTHREAD_ENDTHREAD + +typedef int (*pfnSDL_CurrentBeginThread)(void (*func)(void *), void *, unsigned, void * /*arg*/); +typedef void (*pfnSDL_CurrentEndThread)(void); + +#ifndef SDL_beginthread +#define SDL_beginthread _beginthread +#endif +#ifndef SDL_endthread +#define SDL_endthread _endthread +#endif + +extern DECLSPEC SDL_Thread *SDLCALL +SDL_CreateThread(SDL_ThreadFunction fn, const char *name, void *data, + pfnSDL_CurrentBeginThread pfnBeginThread, + pfnSDL_CurrentEndThread pfnEndThread); +extern DECLSPEC SDL_Thread *SDLCALL +SDL_CreateThreadWithStackSize(SDL_ThreadFunction fn, const char *name, const size_t stacksize, void *data, + pfnSDL_CurrentBeginThread pfnBeginThread, + pfnSDL_CurrentEndThread pfnEndThread); + +#if defined(SDL_CreateThread) && SDL_DYNAMIC_API +#undef SDL_CreateThread +#define SDL_CreateThread(fn, name, data) SDL_CreateThread_REAL(fn, name, data, (pfnSDL_CurrentBeginThread)SDL_beginthread, (pfnSDL_CurrentEndThread)SDL_endthread) +#undef SDL_CreateThreadWithStackSize +#define SDL_CreateThreadWithStackSize(fn, name, stacksize, data) SDL_CreateThreadWithStackSize_REAL(fn, name, stacksize, data, (pfnSDL_CurrentBeginThread)SDL_beginthread, (pfnSDL_CurrentEndThread)SDL_endthread) +#else +#define SDL_CreateThread(fn, name, data) SDL_CreateThread(fn, name, data, (pfnSDL_CurrentBeginThread)SDL_beginthread, (pfnSDL_CurrentEndThread)SDL_endthread) +#define SDL_CreateThreadWithStackSize(fn, name, stacksize, data) SDL_CreateThreadWithStackSize(fn, name, stacksize, data, (pfnSDL_CurrentBeginThread)SDL_beginthread, (pfnSDL_CurrentEndThread)SDL_endthread) +#endif + +#else + +/** + * Create a new thread with a default stack size. + * + * This is equivalent to calling: + * + * ```c + * SDL_CreateThreadWithStackSize(fn, name, 0, data); + * ``` + * + * \param fn the SDL_ThreadFunction function to call in the new thread + * \param name the name of the thread + * \param data a pointer that is passed to `fn` + * \returns an opaque pointer to the new thread object on success, NULL if the + * new thread could not be created; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateThreadWithStackSize + * \sa SDL_WaitThread + */ +extern DECLSPEC SDL_Thread *SDLCALL +SDL_CreateThread(SDL_ThreadFunction fn, const char *name, void *data); + +/** + * Create a new thread with a specific stack size. + * + * SDL makes an attempt to report `name` to the system, so that debuggers can + * display it. Not all platforms support this. + * + * Thread naming is a little complicated: Most systems have very small limits + * for the string length (Haiku has 32 bytes, Linux currently has 16, Visual + * C++ 6.0 has _nine_!), and possibly other arbitrary rules. You'll have to + * see what happens with your system's debugger. The name should be UTF-8 (but + * using the naming limits of C identifiers is a better bet). There are no + * requirements for thread naming conventions, so long as the string is + * null-terminated UTF-8, but these guidelines are helpful in choosing a name: + * + * https://stackoverflow.com/questions/149932/naming-conventions-for-threads + * + * If a system imposes requirements, SDL will try to munge the string for it + * (truncate, etc), but the original string contents will be available from + * SDL_GetThreadName(). + * + * The size (in bytes) of the new stack can be specified. Zero means "use the + * system default" which might be wildly different between platforms. x86 + * Linux generally defaults to eight megabytes, an embedded device might be a + * few kilobytes instead. You generally need to specify a stack that is a + * multiple of the system's page size (in many cases, this is 4 kilobytes, but + * check your system documentation). + * + * In SDL 2.1, stack size will be folded into the original SDL_CreateThread + * function, but for backwards compatibility, this is currently a separate + * function. + * + * \param fn the SDL_ThreadFunction function to call in the new thread + * \param name the name of the thread + * \param stacksize the size, in bytes, to allocate for the new thread stack. + * \param data a pointer that is passed to `fn` + * \returns an opaque pointer to the new thread object on success, NULL if the + * new thread could not be created; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.9. + * + * \sa SDL_WaitThread + */ +extern DECLSPEC SDL_Thread *SDLCALL +SDL_CreateThreadWithStackSize(SDL_ThreadFunction fn, const char *name, const size_t stacksize, void *data); + +#endif + +/** + * Get the thread name as it was specified in SDL_CreateThread(). + * + * This is internal memory, not to be freed by the caller, and remains valid + * until the specified thread is cleaned up by SDL_WaitThread(). + * + * \param thread the thread to query + * \returns a pointer to a UTF-8 string that names the specified thread, or + * NULL if it doesn't have a name. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateThread + */ +extern DECLSPEC const char *SDLCALL SDL_GetThreadName(SDL_Thread *thread); + +/** + * Get the thread identifier for the current thread. + * + * This thread identifier is as reported by the underlying operating system. + * If SDL is running on a platform that does not support threads the return + * value will always be zero. + * + * This function also returns a valid thread ID when called from the main + * thread. + * + * \returns the ID of the current thread. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetThreadID + */ +extern DECLSPEC SDL_threadID SDLCALL SDL_ThreadID(void); + +/** + * Get the thread identifier for the specified thread. + * + * This thread identifier is as reported by the underlying operating system. + * If SDL is running on a platform that does not support threads the return + * value will always be zero. + * + * \param thread the thread to query + * \returns the ID of the specified thread, or the ID of the current thread if + * `thread` is NULL. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ThreadID + */ +extern DECLSPEC SDL_threadID SDLCALL SDL_GetThreadID(SDL_Thread * thread); + +/** + * Set the priority for the current thread. + * + * Note that some platforms will not let you alter the priority (or at least, + * promote the thread to a higher priority) at all, and some require you to be + * an administrator account. Be prepared for this to fail. + * + * \param priority the SDL_ThreadPriority to set + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC int SDLCALL SDL_SetThreadPriority(SDL_ThreadPriority priority); + +/** + * Wait for a thread to finish. + * + * Threads that haven't been detached will remain (as a "zombie") until this + * function cleans them up. Not doing so is a resource leak. + * + * Once a thread has been cleaned up through this function, the SDL_Thread + * that references it becomes invalid and should not be referenced again. As + * such, only one thread may call SDL_WaitThread() on another. + * + * The return code for the thread function is placed in the area pointed to by + * `status`, if `status` is not NULL. + * + * You may not wait on a thread that has been used in a call to + * SDL_DetachThread(). Use either that function or this one, but not both, or + * behavior is undefined. + * + * It is safe to pass a NULL thread to this function; it is a no-op. + * + * Note that the thread pointer is freed by this function and is not valid + * afterward. + * + * \param thread the SDL_Thread pointer that was returned from the + * SDL_CreateThread() call that started this thread + * \param status pointer to an integer that will receive the value returned + * from the thread function by its 'return', or NULL to not + * receive such value back. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateThread + * \sa SDL_DetachThread + */ +extern DECLSPEC void SDLCALL SDL_WaitThread(SDL_Thread * thread, int *status); + +/** + * Let a thread clean up on exit without intervention. + * + * A thread may be "detached" to signify that it should not remain until + * another thread has called SDL_WaitThread() on it. Detaching a thread is + * useful for long-running threads that nothing needs to synchronize with or + * further manage. When a detached thread is done, it simply goes away. + * + * There is no way to recover the return code of a detached thread. If you + * need this, don't detach the thread and instead use SDL_WaitThread(). + * + * Once a thread is detached, you should usually assume the SDL_Thread isn't + * safe to reference again, as it will become invalid immediately upon the + * detached thread's exit, instead of remaining until someone has called + * SDL_WaitThread() to finally clean it up. As such, don't detach the same + * thread more than once. + * + * If a thread has already exited when passed to SDL_DetachThread(), it will + * stop waiting for a call to SDL_WaitThread() and clean up immediately. It is + * not safe to detach a thread that might be used with SDL_WaitThread(). + * + * You may not call SDL_WaitThread() on a thread that has been detached. Use + * either that function or this one, but not both, or behavior is undefined. + * + * It is safe to pass NULL to this function; it is a no-op. + * + * \param thread the SDL_Thread pointer that was returned from the + * SDL_CreateThread() call that started this thread + * + * \since This function is available since SDL 2.0.2. + * + * \sa SDL_CreateThread + * \sa SDL_WaitThread + */ +extern DECLSPEC void SDLCALL SDL_DetachThread(SDL_Thread * thread); + +/** + * Create a piece of thread-local storage. + * + * This creates an identifier that is globally visible to all threads but + * refers to data that is thread-specific. + * + * \returns the newly created thread local storage identifier or 0 on error. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_TLSGet + * \sa SDL_TLSSet + */ +extern DECLSPEC SDL_TLSID SDLCALL SDL_TLSCreate(void); + +/** + * Get the current thread's value associated with a thread local storage ID. + * + * \param id the thread local storage ID + * \returns the value associated with the ID for the current thread or NULL if + * no value has been set; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_TLSCreate + * \sa SDL_TLSSet + */ +extern DECLSPEC void * SDLCALL SDL_TLSGet(SDL_TLSID id); + +/** + * Set the current thread's value associated with a thread local storage ID. + * + * The function prototype for `destructor` is: + * + * ```c + * void destructor(void *value) + * ``` + * + * where its parameter `value` is what was passed as `value` to SDL_TLSSet(). + * + * \param id the thread local storage ID + * \param value the value to associate with the ID for the current thread + * \param destructor a function called when the thread exits, to free the + * value + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_TLSCreate + * \sa SDL_TLSGet + */ +extern DECLSPEC int SDLCALL SDL_TLSSet(SDL_TLSID id, const void *value, void (SDLCALL *destructor)(void*)); + +/** + * Cleanup all TLS data for this thread. + * + * \since This function is available since SDL 2.0.16. + */ +extern DECLSPEC void SDLCALL SDL_TLSCleanup(void); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_thread_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_timer.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_timer.h new file mode 100644 index 00000000..98f9ad16 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_timer.h @@ -0,0 +1,222 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +#ifndef SDL_timer_h_ +#define SDL_timer_h_ + +/** + * \file SDL_timer.h + * + * Header for the SDL time management routines. + */ + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * Get the number of milliseconds since SDL library initialization. + * + * This value wraps if the program runs for more than ~49 days. + * + * This function is not recommended as of SDL 2.0.18; use SDL_GetTicks64() + * instead, where the value doesn't wrap every ~49 days. There are places in + * SDL where we provide a 32-bit timestamp that can not change without + * breaking binary compatibility, though, so this function isn't officially + * deprecated. + * + * \returns an unsigned 32-bit value representing the number of milliseconds + * since the SDL library initialized. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_TICKS_PASSED + */ +extern DECLSPEC Uint32 SDLCALL SDL_GetTicks(void); + +/** + * Get the number of milliseconds since SDL library initialization. + * + * Note that you should not use the SDL_TICKS_PASSED macro with values + * returned by this function, as that macro does clever math to compensate for + * the 32-bit overflow every ~49 days that SDL_GetTicks() suffers from. 64-bit + * values from this function can be safely compared directly. + * + * For example, if you want to wait 100 ms, you could do this: + * + * ```c + * const Uint64 timeout = SDL_GetTicks64() + 100; + * while (SDL_GetTicks64() < timeout) { + * // ... do work until timeout has elapsed + * } + * ``` + * + * \returns an unsigned 64-bit value representing the number of milliseconds + * since the SDL library initialized. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC Uint64 SDLCALL SDL_GetTicks64(void); + +/** + * Compare 32-bit SDL ticks values, and return true if `A` has passed `B`. + * + * This should be used with results from SDL_GetTicks(), as this macro + * attempts to deal with the 32-bit counter wrapping back to zero every ~49 + * days, but should _not_ be used with SDL_GetTicks64(), which does not have + * that problem. + * + * For example, with SDL_GetTicks(), if you want to wait 100 ms, you could + * do this: + * + * ```c + * const Uint32 timeout = SDL_GetTicks() + 100; + * while (!SDL_TICKS_PASSED(SDL_GetTicks(), timeout)) { + * // ... do work until timeout has elapsed + * } + * ``` + * + * Note that this does not handle tick differences greater + * than 2^31 so take care when using the above kind of code + * with large timeout delays (tens of days). + */ +#define SDL_TICKS_PASSED(A, B) ((Sint32)((B) - (A)) <= 0) + +/** + * Get the current value of the high resolution counter. + * + * This function is typically used for profiling. + * + * The counter values are only meaningful relative to each other. Differences + * between values can be converted to times by using + * SDL_GetPerformanceFrequency(). + * + * \returns the current counter value. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetPerformanceFrequency + */ +extern DECLSPEC Uint64 SDLCALL SDL_GetPerformanceCounter(void); + +/** + * Get the count per second of the high resolution counter. + * + * \returns a platform-specific count per second. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetPerformanceCounter + */ +extern DECLSPEC Uint64 SDLCALL SDL_GetPerformanceFrequency(void); + +/** + * Wait a specified number of milliseconds before returning. + * + * This function waits a specified number of milliseconds before returning. It + * waits at least the specified time, but possibly longer due to OS + * scheduling. + * + * \param ms the number of milliseconds to delay + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC void SDLCALL SDL_Delay(Uint32 ms); + +/** + * Function prototype for the timer callback function. + * + * The callback function is passed the current timer interval and returns + * the next timer interval. If the returned value is the same as the one + * passed in, the periodic alarm continues, otherwise a new alarm is + * scheduled. If the callback returns 0, the periodic alarm is cancelled. + */ +typedef Uint32 (SDLCALL * SDL_TimerCallback) (Uint32 interval, void *param); + +/** + * Definition of the timer ID type. + */ +typedef int SDL_TimerID; + +/** + * Call a callback function at a future time. + * + * If you use this function, you must pass `SDL_INIT_TIMER` to SDL_Init(). + * + * The callback function is passed the current timer interval and the user + * supplied parameter from the SDL_AddTimer() call and should return the next + * timer interval. If the value returned from the callback is 0, the timer is + * canceled. + * + * The callback is run on a separate thread. + * + * Timers take into account the amount of time it took to execute the + * callback. For example, if the callback took 250 ms to execute and returned + * 1000 (ms), the timer would only wait another 750 ms before its next + * iteration. + * + * Timing may be inexact due to OS scheduling. Be sure to note the current + * time with SDL_GetTicks() or SDL_GetPerformanceCounter() in case your + * callback needs to adjust for variances. + * + * \param interval the timer delay, in milliseconds, passed to `callback` + * \param callback the SDL_TimerCallback function to call when the specified + * `interval` elapses + * \param param a pointer that is passed to `callback` + * \returns a timer ID or 0 if an error occurs; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RemoveTimer + */ +extern DECLSPEC SDL_TimerID SDLCALL SDL_AddTimer(Uint32 interval, + SDL_TimerCallback callback, + void *param); + +/** + * Remove a timer created with SDL_AddTimer(). + * + * \param id the ID of the timer to remove + * \returns SDL_TRUE if the timer is removed or SDL_FALSE if the timer wasn't + * found. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_AddTimer + */ +extern DECLSPEC SDL_bool SDLCALL SDL_RemoveTimer(SDL_TimerID id); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_timer_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_touch.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_touch.h new file mode 100644 index 00000000..c12d4a1c --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_touch.h @@ -0,0 +1,150 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_touch.h + * + * Include file for SDL touch event handling. + */ + +#ifndef SDL_touch_h_ +#define SDL_touch_h_ + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_video.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +typedef Sint64 SDL_TouchID; +typedef Sint64 SDL_FingerID; + +typedef enum +{ + SDL_TOUCH_DEVICE_INVALID = -1, + SDL_TOUCH_DEVICE_DIRECT, /* touch screen with window-relative coordinates */ + SDL_TOUCH_DEVICE_INDIRECT_ABSOLUTE, /* trackpad with absolute device coordinates */ + SDL_TOUCH_DEVICE_INDIRECT_RELATIVE /* trackpad with screen cursor-relative coordinates */ +} SDL_TouchDeviceType; + +typedef struct SDL_Finger +{ + SDL_FingerID id; + float x; + float y; + float pressure; +} SDL_Finger; + +/* Used as the device ID for mouse events simulated with touch input */ +#define SDL_TOUCH_MOUSEID ((Uint32)-1) + +/* Used as the SDL_TouchID for touch events simulated with mouse input */ +#define SDL_MOUSE_TOUCHID ((Sint64)-1) + + +/** + * Get the number of registered touch devices. + * + * On some platforms SDL first sees the touch device if it was actually used. + * Therefore SDL_GetNumTouchDevices() may return 0 although devices are + * available. After using all devices at least once the number will be + * correct. + * + * This was fixed for Android in SDL 2.0.1. + * + * \returns the number of registered touch devices. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetTouchDevice + */ +extern DECLSPEC int SDLCALL SDL_GetNumTouchDevices(void); + +/** + * Get the touch ID with the given index. + * + * \param index the touch device index + * \returns the touch ID with the given index on success or 0 if the index is + * invalid; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetNumTouchDevices + */ +extern DECLSPEC SDL_TouchID SDLCALL SDL_GetTouchDevice(int index); + +/** + * Get the touch device name as reported from the driver or NULL if the index + * is invalid. + * + * \since This function is available since SDL 2.0.22. + */ +extern DECLSPEC const char* SDLCALL SDL_GetTouchName(int index); + +/** + * Get the type of the given touch device. + * + * \since This function is available since SDL 2.0.10. + */ +extern DECLSPEC SDL_TouchDeviceType SDLCALL SDL_GetTouchDeviceType(SDL_TouchID touchID); + +/** + * Get the number of active fingers for a given touch device. + * + * \param touchID the ID of a touch device + * \returns the number of active fingers for a given touch device on success + * or 0 on failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetTouchFinger + */ +extern DECLSPEC int SDLCALL SDL_GetNumTouchFingers(SDL_TouchID touchID); + +/** + * Get the finger object for specified touch device ID and finger index. + * + * The returned resource is owned by SDL and should not be deallocated. + * + * \param touchID the ID of the requested touch device + * \param index the index of the requested finger + * \returns a pointer to the SDL_Finger object or NULL if no object at the + * given ID and index could be found. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_RecordGesture + */ +extern DECLSPEC SDL_Finger * SDLCALL SDL_GetTouchFinger(SDL_TouchID touchID, int index); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_touch_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_types.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_types.h new file mode 100644 index 00000000..b5d7192f --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_types.h @@ -0,0 +1,29 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_types.h + * + * \deprecated + */ + +/* DEPRECATED */ +#include "SDL_stdinc.h" diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_version.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_version.h new file mode 100644 index 00000000..7585eece --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_version.h @@ -0,0 +1,204 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_version.h + * + * This header defines the current SDL version. + */ + +#ifndef SDL_version_h_ +#define SDL_version_h_ + +#include "SDL_stdinc.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * Information about the version of SDL in use. + * + * Represents the library's version as three levels: major revision + * (increments with massive changes, additions, and enhancements), + * minor revision (increments with backwards-compatible changes to the + * major revision), and patchlevel (increments with fixes to the minor + * revision). + * + * \sa SDL_VERSION + * \sa SDL_GetVersion + */ +typedef struct SDL_version +{ + Uint8 major; /**< major version */ + Uint8 minor; /**< minor version */ + Uint8 patch; /**< update version */ +} SDL_version; + +/* Printable format: "%d.%d.%d", MAJOR, MINOR, PATCHLEVEL +*/ +#define SDL_MAJOR_VERSION 2 +#define SDL_MINOR_VERSION 28 +#define SDL_PATCHLEVEL 5 + +/** + * Macro to determine SDL version program was compiled against. + * + * This macro fills in a SDL_version structure with the version of the + * library you compiled against. This is determined by what header the + * compiler uses. Note that if you dynamically linked the library, you might + * have a slightly newer or older version at runtime. That version can be + * determined with SDL_GetVersion(), which, unlike SDL_VERSION(), + * is not a macro. + * + * \param x A pointer to a SDL_version struct to initialize. + * + * \sa SDL_version + * \sa SDL_GetVersion + */ +#define SDL_VERSION(x) \ +{ \ + (x)->major = SDL_MAJOR_VERSION; \ + (x)->minor = SDL_MINOR_VERSION; \ + (x)->patch = SDL_PATCHLEVEL; \ +} + +/* TODO: Remove this whole block in SDL 3 */ +#if SDL_MAJOR_VERSION < 3 +/** + * This macro turns the version numbers into a numeric value: + * \verbatim + (1,2,3) -> (1203) + \endverbatim + * + * This assumes that there will never be more than 100 patchlevels. + * + * In versions higher than 2.9.0, the minor version overflows into + * the thousands digit: for example, 2.23.0 is encoded as 4300, + * and 2.255.99 would be encoded as 25799. + * This macro will not be available in SDL 3.x. + */ +#define SDL_VERSIONNUM(X, Y, Z) \ + ((X)*1000 + (Y)*100 + (Z)) + +/** + * This is the version number macro for the current SDL version. + * + * In versions higher than 2.9.0, the minor version overflows into + * the thousands digit: for example, 2.23.0 is encoded as 4300. + * This macro will not be available in SDL 3.x. + * + * Deprecated, use SDL_VERSION_ATLEAST or SDL_VERSION instead. + */ +#define SDL_COMPILEDVERSION \ + SDL_VERSIONNUM(SDL_MAJOR_VERSION, SDL_MINOR_VERSION, SDL_PATCHLEVEL) +#endif /* SDL_MAJOR_VERSION < 3 */ + +/** + * This macro will evaluate to true if compiled with SDL at least X.Y.Z. + */ +#define SDL_VERSION_ATLEAST(X, Y, Z) \ + ((SDL_MAJOR_VERSION >= X) && \ + (SDL_MAJOR_VERSION > X || SDL_MINOR_VERSION >= Y) && \ + (SDL_MAJOR_VERSION > X || SDL_MINOR_VERSION > Y || SDL_PATCHLEVEL >= Z)) + +/** + * Get the version of SDL that is linked against your program. + * + * If you are linking to SDL dynamically, then it is possible that the current + * version will be different than the version you compiled against. This + * function returns the current version, while SDL_VERSION() is a macro that + * tells you what version you compiled with. + * + * This function may be called safely at any time, even before SDL_Init(). + * + * \param ver the SDL_version structure that contains the version information + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetRevision + */ +extern DECLSPEC void SDLCALL SDL_GetVersion(SDL_version * ver); + +/** + * Get the code revision of SDL that is linked against your program. + * + * This value is the revision of the code you are linked with and may be + * different from the code you are compiling with, which is found in the + * constant SDL_REVISION. + * + * The revision is arbitrary string (a hash value) uniquely identifying the + * exact revision of the SDL library in use, and is only useful in comparing + * against other revisions. It is NOT an incrementing number. + * + * If SDL wasn't built from a git repository with the appropriate tools, this + * will return an empty string. + * + * Prior to SDL 2.0.16, before development moved to GitHub, this returned a + * hash for a Mercurial repository. + * + * You shouldn't use this function for anything but logging it for debugging + * purposes. The string is not intended to be reliable in any way. + * + * \returns an arbitrary string, uniquely identifying the exact revision of + * the SDL library in use. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetVersion + */ +extern DECLSPEC const char *SDLCALL SDL_GetRevision(void); + +/** + * Obsolete function, do not use. + * + * When SDL was hosted in a Mercurial repository, and was built carefully, + * this would return the revision number that the build was created from. This + * number was not reliable for several reasons, but more importantly, SDL is + * now hosted in a git repository, which does not offer numbers at all, only + * hashes. This function only ever returns zero now. Don't use it. + * + * Before SDL 2.0.16, this might have returned an unreliable, but non-zero + * number. + * + * \deprecated Use SDL_GetRevision() instead; if SDL was carefully built, it + * will return a git hash. + * + * \returns zero, always, in modern SDL releases. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetRevision + */ +extern SDL_DEPRECATED DECLSPEC int SDLCALL SDL_GetRevisionNumber(void); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_version_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_video.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_video.h new file mode 100644 index 00000000..c8b2d7a0 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_video.h @@ -0,0 +1,2178 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_video.h + * + * Header file for SDL video functions. + */ + +#ifndef SDL_video_h_ +#define SDL_video_h_ + +#include "SDL_stdinc.h" +#include "SDL_pixels.h" +#include "SDL_rect.h" +#include "SDL_surface.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/** + * \brief The structure that defines a display mode + * + * \sa SDL_GetNumDisplayModes() + * \sa SDL_GetDisplayMode() + * \sa SDL_GetDesktopDisplayMode() + * \sa SDL_GetCurrentDisplayMode() + * \sa SDL_GetClosestDisplayMode() + * \sa SDL_SetWindowDisplayMode() + * \sa SDL_GetWindowDisplayMode() + */ +typedef struct +{ + Uint32 format; /**< pixel format */ + int w; /**< width, in screen coordinates */ + int h; /**< height, in screen coordinates */ + int refresh_rate; /**< refresh rate (or zero for unspecified) */ + void *driverdata; /**< driver-specific data, initialize to 0 */ +} SDL_DisplayMode; + +/** + * \brief The type used to identify a window + * + * \sa SDL_CreateWindow() + * \sa SDL_CreateWindowFrom() + * \sa SDL_DestroyWindow() + * \sa SDL_FlashWindow() + * \sa SDL_GetWindowData() + * \sa SDL_GetWindowFlags() + * \sa SDL_GetWindowGrab() + * \sa SDL_GetWindowKeyboardGrab() + * \sa SDL_GetWindowMouseGrab() + * \sa SDL_GetWindowPosition() + * \sa SDL_GetWindowSize() + * \sa SDL_GetWindowTitle() + * \sa SDL_HideWindow() + * \sa SDL_MaximizeWindow() + * \sa SDL_MinimizeWindow() + * \sa SDL_RaiseWindow() + * \sa SDL_RestoreWindow() + * \sa SDL_SetWindowData() + * \sa SDL_SetWindowFullscreen() + * \sa SDL_SetWindowGrab() + * \sa SDL_SetWindowKeyboardGrab() + * \sa SDL_SetWindowMouseGrab() + * \sa SDL_SetWindowIcon() + * \sa SDL_SetWindowPosition() + * \sa SDL_SetWindowSize() + * \sa SDL_SetWindowBordered() + * \sa SDL_SetWindowResizable() + * \sa SDL_SetWindowTitle() + * \sa SDL_ShowWindow() + */ +typedef struct SDL_Window SDL_Window; + +/** + * \brief The flags on a window + * + * \sa SDL_GetWindowFlags() + */ +typedef enum +{ + SDL_WINDOW_FULLSCREEN = 0x00000001, /**< fullscreen window */ + SDL_WINDOW_OPENGL = 0x00000002, /**< window usable with OpenGL context */ + SDL_WINDOW_SHOWN = 0x00000004, /**< window is visible */ + SDL_WINDOW_HIDDEN = 0x00000008, /**< window is not visible */ + SDL_WINDOW_BORDERLESS = 0x00000010, /**< no window decoration */ + SDL_WINDOW_RESIZABLE = 0x00000020, /**< window can be resized */ + SDL_WINDOW_MINIMIZED = 0x00000040, /**< window is minimized */ + SDL_WINDOW_MAXIMIZED = 0x00000080, /**< window is maximized */ + SDL_WINDOW_MOUSE_GRABBED = 0x00000100, /**< window has grabbed mouse input */ + SDL_WINDOW_INPUT_FOCUS = 0x00000200, /**< window has input focus */ + SDL_WINDOW_MOUSE_FOCUS = 0x00000400, /**< window has mouse focus */ + SDL_WINDOW_FULLSCREEN_DESKTOP = ( SDL_WINDOW_FULLSCREEN | 0x00001000 ), + SDL_WINDOW_FOREIGN = 0x00000800, /**< window not created by SDL */ + SDL_WINDOW_ALLOW_HIGHDPI = 0x00002000, /**< window should be created in high-DPI mode if supported. + On macOS NSHighResolutionCapable must be set true in the + application's Info.plist for this to have any effect. */ + SDL_WINDOW_MOUSE_CAPTURE = 0x00004000, /**< window has mouse captured (unrelated to MOUSE_GRABBED) */ + SDL_WINDOW_ALWAYS_ON_TOP = 0x00008000, /**< window should always be above others */ + SDL_WINDOW_SKIP_TASKBAR = 0x00010000, /**< window should not be added to the taskbar */ + SDL_WINDOW_UTILITY = 0x00020000, /**< window should be treated as a utility window */ + SDL_WINDOW_TOOLTIP = 0x00040000, /**< window should be treated as a tooltip */ + SDL_WINDOW_POPUP_MENU = 0x00080000, /**< window should be treated as a popup menu */ + SDL_WINDOW_KEYBOARD_GRABBED = 0x00100000, /**< window has grabbed keyboard input */ + SDL_WINDOW_VULKAN = 0x10000000, /**< window usable for Vulkan surface */ + SDL_WINDOW_METAL = 0x20000000, /**< window usable for Metal view */ + + SDL_WINDOW_INPUT_GRABBED = SDL_WINDOW_MOUSE_GRABBED /**< equivalent to SDL_WINDOW_MOUSE_GRABBED for compatibility */ +} SDL_WindowFlags; + +/** + * \brief Used to indicate that you don't care what the window position is. + */ +#define SDL_WINDOWPOS_UNDEFINED_MASK 0x1FFF0000u +#define SDL_WINDOWPOS_UNDEFINED_DISPLAY(X) (SDL_WINDOWPOS_UNDEFINED_MASK|(X)) +#define SDL_WINDOWPOS_UNDEFINED SDL_WINDOWPOS_UNDEFINED_DISPLAY(0) +#define SDL_WINDOWPOS_ISUNDEFINED(X) \ + (((X)&0xFFFF0000) == SDL_WINDOWPOS_UNDEFINED_MASK) + +/** + * \brief Used to indicate that the window position should be centered. + */ +#define SDL_WINDOWPOS_CENTERED_MASK 0x2FFF0000u +#define SDL_WINDOWPOS_CENTERED_DISPLAY(X) (SDL_WINDOWPOS_CENTERED_MASK|(X)) +#define SDL_WINDOWPOS_CENTERED SDL_WINDOWPOS_CENTERED_DISPLAY(0) +#define SDL_WINDOWPOS_ISCENTERED(X) \ + (((X)&0xFFFF0000) == SDL_WINDOWPOS_CENTERED_MASK) + +/** + * \brief Event subtype for window events + */ +typedef enum +{ + SDL_WINDOWEVENT_NONE, /**< Never used */ + SDL_WINDOWEVENT_SHOWN, /**< Window has been shown */ + SDL_WINDOWEVENT_HIDDEN, /**< Window has been hidden */ + SDL_WINDOWEVENT_EXPOSED, /**< Window has been exposed and should be + redrawn */ + SDL_WINDOWEVENT_MOVED, /**< Window has been moved to data1, data2 + */ + SDL_WINDOWEVENT_RESIZED, /**< Window has been resized to data1xdata2 */ + SDL_WINDOWEVENT_SIZE_CHANGED, /**< The window size has changed, either as + a result of an API call or through the + system or user changing the window size. */ + SDL_WINDOWEVENT_MINIMIZED, /**< Window has been minimized */ + SDL_WINDOWEVENT_MAXIMIZED, /**< Window has been maximized */ + SDL_WINDOWEVENT_RESTORED, /**< Window has been restored to normal size + and position */ + SDL_WINDOWEVENT_ENTER, /**< Window has gained mouse focus */ + SDL_WINDOWEVENT_LEAVE, /**< Window has lost mouse focus */ + SDL_WINDOWEVENT_FOCUS_GAINED, /**< Window has gained keyboard focus */ + SDL_WINDOWEVENT_FOCUS_LOST, /**< Window has lost keyboard focus */ + SDL_WINDOWEVENT_CLOSE, /**< The window manager requests that the window be closed */ + SDL_WINDOWEVENT_TAKE_FOCUS, /**< Window is being offered a focus (should SetWindowInputFocus() on itself or a subwindow, or ignore) */ + SDL_WINDOWEVENT_HIT_TEST, /**< Window had a hit test that wasn't SDL_HITTEST_NORMAL. */ + SDL_WINDOWEVENT_ICCPROF_CHANGED,/**< The ICC profile of the window's display has changed. */ + SDL_WINDOWEVENT_DISPLAY_CHANGED /**< Window has been moved to display data1. */ +} SDL_WindowEventID; + +/** + * \brief Event subtype for display events + */ +typedef enum +{ + SDL_DISPLAYEVENT_NONE, /**< Never used */ + SDL_DISPLAYEVENT_ORIENTATION, /**< Display orientation has changed to data1 */ + SDL_DISPLAYEVENT_CONNECTED, /**< Display has been added to the system */ + SDL_DISPLAYEVENT_DISCONNECTED, /**< Display has been removed from the system */ + SDL_DISPLAYEVENT_MOVED /**< Display has changed position */ +} SDL_DisplayEventID; + +/** + * \brief Display orientation + */ +typedef enum +{ + SDL_ORIENTATION_UNKNOWN, /**< The display orientation can't be determined */ + SDL_ORIENTATION_LANDSCAPE, /**< The display is in landscape mode, with the right side up, relative to portrait mode */ + SDL_ORIENTATION_LANDSCAPE_FLIPPED, /**< The display is in landscape mode, with the left side up, relative to portrait mode */ + SDL_ORIENTATION_PORTRAIT, /**< The display is in portrait mode */ + SDL_ORIENTATION_PORTRAIT_FLIPPED /**< The display is in portrait mode, upside down */ +} SDL_DisplayOrientation; + +/** + * \brief Window flash operation + */ +typedef enum +{ + SDL_FLASH_CANCEL, /**< Cancel any window flash state */ + SDL_FLASH_BRIEFLY, /**< Flash the window briefly to get attention */ + SDL_FLASH_UNTIL_FOCUSED /**< Flash the window until it gets focus */ +} SDL_FlashOperation; + +/** + * \brief An opaque handle to an OpenGL context. + */ +typedef void *SDL_GLContext; + +/** + * \brief OpenGL configuration attributes + */ +typedef enum +{ + SDL_GL_RED_SIZE, + SDL_GL_GREEN_SIZE, + SDL_GL_BLUE_SIZE, + SDL_GL_ALPHA_SIZE, + SDL_GL_BUFFER_SIZE, + SDL_GL_DOUBLEBUFFER, + SDL_GL_DEPTH_SIZE, + SDL_GL_STENCIL_SIZE, + SDL_GL_ACCUM_RED_SIZE, + SDL_GL_ACCUM_GREEN_SIZE, + SDL_GL_ACCUM_BLUE_SIZE, + SDL_GL_ACCUM_ALPHA_SIZE, + SDL_GL_STEREO, + SDL_GL_MULTISAMPLEBUFFERS, + SDL_GL_MULTISAMPLESAMPLES, + SDL_GL_ACCELERATED_VISUAL, + SDL_GL_RETAINED_BACKING, + SDL_GL_CONTEXT_MAJOR_VERSION, + SDL_GL_CONTEXT_MINOR_VERSION, + SDL_GL_CONTEXT_EGL, + SDL_GL_CONTEXT_FLAGS, + SDL_GL_CONTEXT_PROFILE_MASK, + SDL_GL_SHARE_WITH_CURRENT_CONTEXT, + SDL_GL_FRAMEBUFFER_SRGB_CAPABLE, + SDL_GL_CONTEXT_RELEASE_BEHAVIOR, + SDL_GL_CONTEXT_RESET_NOTIFICATION, + SDL_GL_CONTEXT_NO_ERROR, + SDL_GL_FLOATBUFFERS +} SDL_GLattr; + +typedef enum +{ + SDL_GL_CONTEXT_PROFILE_CORE = 0x0001, + SDL_GL_CONTEXT_PROFILE_COMPATIBILITY = 0x0002, + SDL_GL_CONTEXT_PROFILE_ES = 0x0004 /**< GLX_CONTEXT_ES2_PROFILE_BIT_EXT */ +} SDL_GLprofile; + +typedef enum +{ + SDL_GL_CONTEXT_DEBUG_FLAG = 0x0001, + SDL_GL_CONTEXT_FORWARD_COMPATIBLE_FLAG = 0x0002, + SDL_GL_CONTEXT_ROBUST_ACCESS_FLAG = 0x0004, + SDL_GL_CONTEXT_RESET_ISOLATION_FLAG = 0x0008 +} SDL_GLcontextFlag; + +typedef enum +{ + SDL_GL_CONTEXT_RELEASE_BEHAVIOR_NONE = 0x0000, + SDL_GL_CONTEXT_RELEASE_BEHAVIOR_FLUSH = 0x0001 +} SDL_GLcontextReleaseFlag; + +typedef enum +{ + SDL_GL_CONTEXT_RESET_NO_NOTIFICATION = 0x0000, + SDL_GL_CONTEXT_RESET_LOSE_CONTEXT = 0x0001 +} SDL_GLContextResetNotification; + +/* Function prototypes */ + +/** + * Get the number of video drivers compiled into SDL. + * + * \returns a number >= 1 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetVideoDriver + */ +extern DECLSPEC int SDLCALL SDL_GetNumVideoDrivers(void); + +/** + * Get the name of a built in video driver. + * + * The video drivers are presented in the order in which they are normally + * checked during initialization. + * + * \param index the index of a video driver + * \returns the name of the video driver with the given **index**. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetNumVideoDrivers + */ +extern DECLSPEC const char *SDLCALL SDL_GetVideoDriver(int index); + +/** + * Initialize the video subsystem, optionally specifying a video driver. + * + * This function initializes the video subsystem, setting up a connection to + * the window manager, etc, and determines the available display modes and + * pixel formats, but does not initialize a window or graphics mode. + * + * If you use this function and you haven't used the SDL_INIT_VIDEO flag with + * either SDL_Init() or SDL_InitSubSystem(), you should call SDL_VideoQuit() + * before calling SDL_Quit(). + * + * It is safe to call this function multiple times. SDL_VideoInit() will call + * SDL_VideoQuit() itself if the video subsystem has already been initialized. + * + * You can use SDL_GetNumVideoDrivers() and SDL_GetVideoDriver() to find a + * specific `driver_name`. + * + * \param driver_name the name of a video driver to initialize, or NULL for + * the default driver + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetNumVideoDrivers + * \sa SDL_GetVideoDriver + * \sa SDL_InitSubSystem + * \sa SDL_VideoQuit + */ +extern DECLSPEC int SDLCALL SDL_VideoInit(const char *driver_name); + +/** + * Shut down the video subsystem, if initialized with SDL_VideoInit(). + * + * This function closes all windows, and restores the original video mode. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_VideoInit + */ +extern DECLSPEC void SDLCALL SDL_VideoQuit(void); + +/** + * Get the name of the currently initialized video driver. + * + * \returns the name of the current video driver or NULL if no driver has been + * initialized. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetNumVideoDrivers + * \sa SDL_GetVideoDriver + */ +extern DECLSPEC const char *SDLCALL SDL_GetCurrentVideoDriver(void); + +/** + * Get the number of available video displays. + * + * \returns a number >= 1 or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetDisplayBounds + */ +extern DECLSPEC int SDLCALL SDL_GetNumVideoDisplays(void); + +/** + * Get the name of a display in UTF-8 encoding. + * + * \param displayIndex the index of display from which the name should be + * queried + * \returns the name of a display or NULL for an invalid display index or + * failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetNumVideoDisplays + */ +extern DECLSPEC const char * SDLCALL SDL_GetDisplayName(int displayIndex); + +/** + * Get the desktop area represented by a display. + * + * The primary display (`displayIndex` zero) is always located at 0,0. + * + * \param displayIndex the index of the display to query + * \param rect the SDL_Rect structure filled in with the display bounds + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetNumVideoDisplays + */ +extern DECLSPEC int SDLCALL SDL_GetDisplayBounds(int displayIndex, SDL_Rect * rect); + +/** + * Get the usable desktop area represented by a display. + * + * The primary display (`displayIndex` zero) is always located at 0,0. + * + * This is the same area as SDL_GetDisplayBounds() reports, but with portions + * reserved by the system removed. For example, on Apple's macOS, this + * subtracts the area occupied by the menu bar and dock. + * + * Setting a window to be fullscreen generally bypasses these unusable areas, + * so these are good guidelines for the maximum space available to a + * non-fullscreen window. + * + * The parameter `rect` is ignored if it is NULL. + * + * This function also returns -1 if the parameter `displayIndex` is out of + * range. + * + * \param displayIndex the index of the display to query the usable bounds + * from + * \param rect the SDL_Rect structure filled in with the display bounds + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.5. + * + * \sa SDL_GetDisplayBounds + * \sa SDL_GetNumVideoDisplays + */ +extern DECLSPEC int SDLCALL SDL_GetDisplayUsableBounds(int displayIndex, SDL_Rect * rect); + +/** + * Get the dots/pixels-per-inch for a display. + * + * Diagonal, horizontal and vertical DPI can all be optionally returned if the + * appropriate parameter is non-NULL. + * + * A failure of this function usually means that either no DPI information is + * available or the `displayIndex` is out of range. + * + * **WARNING**: This reports the DPI that the hardware reports, and it is not + * always reliable! It is almost always better to use SDL_GetWindowSize() to + * find the window size, which might be in logical points instead of pixels, + * and then SDL_GL_GetDrawableSize(), SDL_Vulkan_GetDrawableSize(), + * SDL_Metal_GetDrawableSize(), or SDL_GetRendererOutputSize(), and compare + * the two values to get an actual scaling value between the two. We will be + * rethinking how high-dpi details should be managed in SDL3 to make things + * more consistent, reliable, and clear. + * + * \param displayIndex the index of the display from which DPI information + * should be queried + * \param ddpi a pointer filled in with the diagonal DPI of the display; may + * be NULL + * \param hdpi a pointer filled in with the horizontal DPI of the display; may + * be NULL + * \param vdpi a pointer filled in with the vertical DPI of the display; may + * be NULL + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.4. + * + * \sa SDL_GetNumVideoDisplays + */ +extern DECLSPEC int SDLCALL SDL_GetDisplayDPI(int displayIndex, float * ddpi, float * hdpi, float * vdpi); + +/** + * Get the orientation of a display. + * + * \param displayIndex the index of the display to query + * \returns The SDL_DisplayOrientation enum value of the display, or + * `SDL_ORIENTATION_UNKNOWN` if it isn't available. + * + * \since This function is available since SDL 2.0.9. + * + * \sa SDL_GetNumVideoDisplays + */ +extern DECLSPEC SDL_DisplayOrientation SDLCALL SDL_GetDisplayOrientation(int displayIndex); + +/** + * Get the number of available display modes. + * + * The `displayIndex` needs to be in the range from 0 to + * SDL_GetNumVideoDisplays() - 1. + * + * \param displayIndex the index of the display to query + * \returns a number >= 1 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetDisplayMode + * \sa SDL_GetNumVideoDisplays + */ +extern DECLSPEC int SDLCALL SDL_GetNumDisplayModes(int displayIndex); + +/** + * Get information about a specific display mode. + * + * The display modes are sorted in this priority: + * + * - width -> largest to smallest + * - height -> largest to smallest + * - bits per pixel -> more colors to fewer colors + * - packed pixel layout -> largest to smallest + * - refresh rate -> highest to lowest + * + * \param displayIndex the index of the display to query + * \param modeIndex the index of the display mode to query + * \param mode an SDL_DisplayMode structure filled in with the mode at + * `modeIndex` + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetNumDisplayModes + */ +extern DECLSPEC int SDLCALL SDL_GetDisplayMode(int displayIndex, int modeIndex, + SDL_DisplayMode * mode); + +/** + * Get information about the desktop's display mode. + * + * There's a difference between this function and SDL_GetCurrentDisplayMode() + * when SDL runs fullscreen and has changed the resolution. In that case this + * function will return the previous native display mode, and not the current + * display mode. + * + * \param displayIndex the index of the display to query + * \param mode an SDL_DisplayMode structure filled in with the current display + * mode + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetCurrentDisplayMode + * \sa SDL_GetDisplayMode + * \sa SDL_SetWindowDisplayMode + */ +extern DECLSPEC int SDLCALL SDL_GetDesktopDisplayMode(int displayIndex, SDL_DisplayMode * mode); + +/** + * Get information about the current display mode. + * + * There's a difference between this function and SDL_GetDesktopDisplayMode() + * when SDL runs fullscreen and has changed the resolution. In that case this + * function will return the current display mode, and not the previous native + * display mode. + * + * \param displayIndex the index of the display to query + * \param mode an SDL_DisplayMode structure filled in with the current display + * mode + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetDesktopDisplayMode + * \sa SDL_GetDisplayMode + * \sa SDL_GetNumVideoDisplays + * \sa SDL_SetWindowDisplayMode + */ +extern DECLSPEC int SDLCALL SDL_GetCurrentDisplayMode(int displayIndex, SDL_DisplayMode * mode); + + +/** + * Get the closest match to the requested display mode. + * + * The available display modes are scanned and `closest` is filled in with the + * closest mode matching the requested mode and returned. The mode format and + * refresh rate default to the desktop mode if they are set to 0. The modes + * are scanned with size being first priority, format being second priority, + * and finally checking the refresh rate. If all the available modes are too + * small, then NULL is returned. + * + * \param displayIndex the index of the display to query + * \param mode an SDL_DisplayMode structure containing the desired display + * mode + * \param closest an SDL_DisplayMode structure filled in with the closest + * match of the available display modes + * \returns the passed in value `closest` or NULL if no matching video mode + * was available; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetDisplayMode + * \sa SDL_GetNumDisplayModes + */ +extern DECLSPEC SDL_DisplayMode * SDLCALL SDL_GetClosestDisplayMode(int displayIndex, const SDL_DisplayMode * mode, SDL_DisplayMode * closest); + +/** + * Get the index of the display containing a point + * + * \param point the point to query + * \returns the index of the display containing the point or a negative error + * code on failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.24.0. + * + * \sa SDL_GetDisplayBounds + * \sa SDL_GetNumVideoDisplays + */ +extern DECLSPEC int SDLCALL SDL_GetPointDisplayIndex(const SDL_Point * point); + +/** + * Get the index of the display primarily containing a rect + * + * \param rect the rect to query + * \returns the index of the display entirely containing the rect or closest + * to the center of the rect on success or a negative error code on + * failure; call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.24.0. + * + * \sa SDL_GetDisplayBounds + * \sa SDL_GetNumVideoDisplays + */ +extern DECLSPEC int SDLCALL SDL_GetRectDisplayIndex(const SDL_Rect * rect); + +/** + * Get the index of the display associated with a window. + * + * \param window the window to query + * \returns the index of the display containing the center of the window on + * success or a negative error code on failure; call SDL_GetError() + * for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetDisplayBounds + * \sa SDL_GetNumVideoDisplays + */ +extern DECLSPEC int SDLCALL SDL_GetWindowDisplayIndex(SDL_Window * window); + +/** + * Set the display mode to use when a window is visible at fullscreen. + * + * This only affects the display mode used when the window is fullscreen. To + * change the window size when the window is not fullscreen, use + * SDL_SetWindowSize(). + * + * \param window the window to affect + * \param mode the SDL_DisplayMode structure representing the mode to use, or + * NULL to use the window's dimensions and the desktop's format + * and refresh rate + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowDisplayMode + * \sa SDL_SetWindowFullscreen + */ +extern DECLSPEC int SDLCALL SDL_SetWindowDisplayMode(SDL_Window * window, + const SDL_DisplayMode * mode); + +/** + * Query the display mode to use when a window is visible at fullscreen. + * + * \param window the window to query + * \param mode an SDL_DisplayMode structure filled in with the fullscreen + * display mode + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetWindowDisplayMode + * \sa SDL_SetWindowFullscreen + */ +extern DECLSPEC int SDLCALL SDL_GetWindowDisplayMode(SDL_Window * window, + SDL_DisplayMode * mode); + +/** + * Get the raw ICC profile data for the screen the window is currently on. + * + * Data returned should be freed with SDL_free. + * + * \param window the window to query + * \param size the size of the ICC profile + * \returns the raw ICC profile data on success or NULL on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.18. + */ +extern DECLSPEC void* SDLCALL SDL_GetWindowICCProfile(SDL_Window * window, size_t* size); + +/** + * Get the pixel format associated with the window. + * + * \param window the window to query + * \returns the pixel format of the window on success or + * SDL_PIXELFORMAT_UNKNOWN on failure; call SDL_GetError() for more + * information. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC Uint32 SDLCALL SDL_GetWindowPixelFormat(SDL_Window * window); + +/** + * Create a window with the specified position, dimensions, and flags. + * + * `flags` may be any of the following OR'd together: + * + * - `SDL_WINDOW_FULLSCREEN`: fullscreen window + * - `SDL_WINDOW_FULLSCREEN_DESKTOP`: fullscreen window at desktop resolution + * - `SDL_WINDOW_OPENGL`: window usable with an OpenGL context + * - `SDL_WINDOW_VULKAN`: window usable with a Vulkan instance + * - `SDL_WINDOW_METAL`: window usable with a Metal instance + * - `SDL_WINDOW_HIDDEN`: window is not visible + * - `SDL_WINDOW_BORDERLESS`: no window decoration + * - `SDL_WINDOW_RESIZABLE`: window can be resized + * - `SDL_WINDOW_MINIMIZED`: window is minimized + * - `SDL_WINDOW_MAXIMIZED`: window is maximized + * - `SDL_WINDOW_INPUT_GRABBED`: window has grabbed input focus + * - `SDL_WINDOW_ALLOW_HIGHDPI`: window should be created in high-DPI mode if + * supported (>= SDL 2.0.1) + * + * `SDL_WINDOW_SHOWN` is ignored by SDL_CreateWindow(). The SDL_Window is + * implicitly shown if SDL_WINDOW_HIDDEN is not set. `SDL_WINDOW_SHOWN` may be + * queried later using SDL_GetWindowFlags(). + * + * On Apple's macOS, you **must** set the NSHighResolutionCapable Info.plist + * property to YES, otherwise you will not receive a High-DPI OpenGL canvas. + * + * If the window is created with the `SDL_WINDOW_ALLOW_HIGHDPI` flag, its size + * in pixels may differ from its size in screen coordinates on platforms with + * high-DPI support (e.g. iOS and macOS). Use SDL_GetWindowSize() to query the + * client area's size in screen coordinates, and SDL_GL_GetDrawableSize() or + * SDL_GetRendererOutputSize() to query the drawable size in pixels. Note that + * when this flag is set, the drawable size can vary after the window is + * created and should be queried after major window events such as when the + * window is resized or moved between displays. + * + * If the window is set fullscreen, the width and height parameters `w` and + * `h` will not be used. However, invalid size parameters (e.g. too large) may + * still fail. Window size is actually limited to 16384 x 16384 for all + * platforms at window creation. + * + * If the window is created with any of the SDL_WINDOW_OPENGL or + * SDL_WINDOW_VULKAN flags, then the corresponding LoadLibrary function + * (SDL_GL_LoadLibrary or SDL_Vulkan_LoadLibrary) is called and the + * corresponding UnloadLibrary function is called by SDL_DestroyWindow(). + * + * If SDL_WINDOW_VULKAN is specified and there isn't a working Vulkan driver, + * SDL_CreateWindow() will fail because SDL_Vulkan_LoadLibrary() will fail. + * + * If SDL_WINDOW_METAL is specified on an OS that does not support Metal, + * SDL_CreateWindow() will fail. + * + * On non-Apple devices, SDL requires you to either not link to the Vulkan + * loader or link to a dynamic library version. This limitation may be removed + * in a future version of SDL. + * + * \param title the title of the window, in UTF-8 encoding + * \param x the x position of the window, `SDL_WINDOWPOS_CENTERED`, or + * `SDL_WINDOWPOS_UNDEFINED` + * \param y the y position of the window, `SDL_WINDOWPOS_CENTERED`, or + * `SDL_WINDOWPOS_UNDEFINED` + * \param w the width of the window, in screen coordinates + * \param h the height of the window, in screen coordinates + * \param flags 0, or one or more SDL_WindowFlags OR'd together + * \returns the window that was created or NULL on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateWindowFrom + * \sa SDL_DestroyWindow + */ +extern DECLSPEC SDL_Window * SDLCALL SDL_CreateWindow(const char *title, + int x, int y, int w, + int h, Uint32 flags); + +/** + * Create an SDL window from an existing native window. + * + * In some cases (e.g. OpenGL) and on some platforms (e.g. Microsoft Windows) + * the hint `SDL_HINT_VIDEO_WINDOW_SHARE_PIXEL_FORMAT` needs to be configured + * before using SDL_CreateWindowFrom(). + * + * \param data a pointer to driver-dependent window creation data, typically + * your native window cast to a void* + * \returns the window that was created or NULL on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateWindow + * \sa SDL_DestroyWindow + */ +extern DECLSPEC SDL_Window * SDLCALL SDL_CreateWindowFrom(const void *data); + +/** + * Get the numeric ID of a window. + * + * The numeric ID is what SDL_WindowEvent references, and is necessary to map + * these events to specific SDL_Window objects. + * + * \param window the window to query + * \returns the ID of the window on success or 0 on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowFromID + */ +extern DECLSPEC Uint32 SDLCALL SDL_GetWindowID(SDL_Window * window); + +/** + * Get a window from a stored ID. + * + * The numeric ID is what SDL_WindowEvent references, and is necessary to map + * these events to specific SDL_Window objects. + * + * \param id the ID of the window + * \returns the window associated with `id` or NULL if it doesn't exist; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowID + */ +extern DECLSPEC SDL_Window * SDLCALL SDL_GetWindowFromID(Uint32 id); + +/** + * Get the window flags. + * + * \param window the window to query + * \returns a mask of the SDL_WindowFlags associated with `window` + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateWindow + * \sa SDL_HideWindow + * \sa SDL_MaximizeWindow + * \sa SDL_MinimizeWindow + * \sa SDL_SetWindowFullscreen + * \sa SDL_SetWindowGrab + * \sa SDL_ShowWindow + */ +extern DECLSPEC Uint32 SDLCALL SDL_GetWindowFlags(SDL_Window * window); + +/** + * Set the title of a window. + * + * This string is expected to be in UTF-8 encoding. + * + * \param window the window to change + * \param title the desired window title in UTF-8 format + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowTitle + */ +extern DECLSPEC void SDLCALL SDL_SetWindowTitle(SDL_Window * window, + const char *title); + +/** + * Get the title of a window. + * + * \param window the window to query + * \returns the title of the window in UTF-8 format or "" if there is no + * title. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetWindowTitle + */ +extern DECLSPEC const char *SDLCALL SDL_GetWindowTitle(SDL_Window * window); + +/** + * Set the icon for a window. + * + * \param window the window to change + * \param icon an SDL_Surface structure containing the icon for the window + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC void SDLCALL SDL_SetWindowIcon(SDL_Window * window, + SDL_Surface * icon); + +/** + * Associate an arbitrary named pointer with a window. + * + * `name` is case-sensitive. + * + * \param window the window to associate with the pointer + * \param name the name of the pointer + * \param userdata the associated pointer + * \returns the previous value associated with `name`. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowData + */ +extern DECLSPEC void* SDLCALL SDL_SetWindowData(SDL_Window * window, + const char *name, + void *userdata); + +/** + * Retrieve the data pointer associated with a window. + * + * \param window the window to query + * \param name the name of the pointer + * \returns the value associated with `name`. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetWindowData + */ +extern DECLSPEC void *SDLCALL SDL_GetWindowData(SDL_Window * window, + const char *name); + +/** + * Set the position of a window. + * + * The window coordinate origin is the upper left of the display. + * + * \param window the window to reposition + * \param x the x coordinate of the window in screen coordinates, or + * `SDL_WINDOWPOS_CENTERED` or `SDL_WINDOWPOS_UNDEFINED` + * \param y the y coordinate of the window in screen coordinates, or + * `SDL_WINDOWPOS_CENTERED` or `SDL_WINDOWPOS_UNDEFINED` + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowPosition + */ +extern DECLSPEC void SDLCALL SDL_SetWindowPosition(SDL_Window * window, + int x, int y); + +/** + * Get the position of a window. + * + * If you do not need the value for one of the positions a NULL may be passed + * in the `x` or `y` parameter. + * + * \param window the window to query + * \param x a pointer filled in with the x position of the window, in screen + * coordinates, may be NULL + * \param y a pointer filled in with the y position of the window, in screen + * coordinates, may be NULL + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetWindowPosition + */ +extern DECLSPEC void SDLCALL SDL_GetWindowPosition(SDL_Window * window, + int *x, int *y); + +/** + * Set the size of a window's client area. + * + * The window size in screen coordinates may differ from the size in pixels, + * if the window was created with `SDL_WINDOW_ALLOW_HIGHDPI` on a platform + * with high-dpi support (e.g. iOS or macOS). Use SDL_GL_GetDrawableSize() or + * SDL_GetRendererOutputSize() to get the real client area size in pixels. + * + * Fullscreen windows automatically match the size of the display mode, and + * you should use SDL_SetWindowDisplayMode() to change their size. + * + * \param window the window to change + * \param w the width of the window in pixels, in screen coordinates, must be + * > 0 + * \param h the height of the window in pixels, in screen coordinates, must be + * > 0 + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowSize + * \sa SDL_SetWindowDisplayMode + */ +extern DECLSPEC void SDLCALL SDL_SetWindowSize(SDL_Window * window, int w, + int h); + +/** + * Get the size of a window's client area. + * + * NULL can safely be passed as the `w` or `h` parameter if the width or + * height value is not desired. + * + * The window size in screen coordinates may differ from the size in pixels, + * if the window was created with `SDL_WINDOW_ALLOW_HIGHDPI` on a platform + * with high-dpi support (e.g. iOS or macOS). Use SDL_GL_GetDrawableSize(), + * SDL_Vulkan_GetDrawableSize(), or SDL_GetRendererOutputSize() to get the + * real client area size in pixels. + * + * \param window the window to query the width and height from + * \param w a pointer filled in with the width of the window, in screen + * coordinates, may be NULL + * \param h a pointer filled in with the height of the window, in screen + * coordinates, may be NULL + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GL_GetDrawableSize + * \sa SDL_Vulkan_GetDrawableSize + * \sa SDL_SetWindowSize + */ +extern DECLSPEC void SDLCALL SDL_GetWindowSize(SDL_Window * window, int *w, + int *h); + +/** + * Get the size of a window's borders (decorations) around the client area. + * + * Note: If this function fails (returns -1), the size values will be + * initialized to 0, 0, 0, 0 (if a non-NULL pointer is provided), as if the + * window in question was borderless. + * + * Note: This function may fail on systems where the window has not yet been + * decorated by the display server (for example, immediately after calling + * SDL_CreateWindow). It is recommended that you wait at least until the + * window has been presented and composited, so that the window system has a + * chance to decorate the window and provide the border dimensions to SDL. + * + * This function also returns -1 if getting the information is not supported. + * + * \param window the window to query the size values of the border + * (decorations) from + * \param top pointer to variable for storing the size of the top border; NULL + * is permitted + * \param left pointer to variable for storing the size of the left border; + * NULL is permitted + * \param bottom pointer to variable for storing the size of the bottom + * border; NULL is permitted + * \param right pointer to variable for storing the size of the right border; + * NULL is permitted + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.5. + * + * \sa SDL_GetWindowSize + */ +extern DECLSPEC int SDLCALL SDL_GetWindowBordersSize(SDL_Window * window, + int *top, int *left, + int *bottom, int *right); + +/** + * Get the size of a window in pixels. + * + * This may differ from SDL_GetWindowSize() if we're rendering to a high-DPI + * drawable, i.e. the window was created with `SDL_WINDOW_ALLOW_HIGHDPI` on a + * platform with high-DPI support (Apple calls this "Retina"), and not + * disabled by the `SDL_HINT_VIDEO_HIGHDPI_DISABLED` hint. + * + * \param window the window from which the drawable size should be queried + * \param w a pointer to variable for storing the width in pixels, may be NULL + * \param h a pointer to variable for storing the height in pixels, may be + * NULL + * + * \since This function is available since SDL 2.26.0. + * + * \sa SDL_CreateWindow + * \sa SDL_GetWindowSize + */ +extern DECLSPEC void SDLCALL SDL_GetWindowSizeInPixels(SDL_Window * window, + int *w, int *h); + +/** + * Set the minimum size of a window's client area. + * + * \param window the window to change + * \param min_w the minimum width of the window in pixels + * \param min_h the minimum height of the window in pixels + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowMinimumSize + * \sa SDL_SetWindowMaximumSize + */ +extern DECLSPEC void SDLCALL SDL_SetWindowMinimumSize(SDL_Window * window, + int min_w, int min_h); + +/** + * Get the minimum size of a window's client area. + * + * \param window the window to query + * \param w a pointer filled in with the minimum width of the window, may be + * NULL + * \param h a pointer filled in with the minimum height of the window, may be + * NULL + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowMaximumSize + * \sa SDL_SetWindowMinimumSize + */ +extern DECLSPEC void SDLCALL SDL_GetWindowMinimumSize(SDL_Window * window, + int *w, int *h); + +/** + * Set the maximum size of a window's client area. + * + * \param window the window to change + * \param max_w the maximum width of the window in pixels + * \param max_h the maximum height of the window in pixels + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowMaximumSize + * \sa SDL_SetWindowMinimumSize + */ +extern DECLSPEC void SDLCALL SDL_SetWindowMaximumSize(SDL_Window * window, + int max_w, int max_h); + +/** + * Get the maximum size of a window's client area. + * + * \param window the window to query + * \param w a pointer filled in with the maximum width of the window, may be + * NULL + * \param h a pointer filled in with the maximum height of the window, may be + * NULL + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowMinimumSize + * \sa SDL_SetWindowMaximumSize + */ +extern DECLSPEC void SDLCALL SDL_GetWindowMaximumSize(SDL_Window * window, + int *w, int *h); + +/** + * Set the border state of a window. + * + * This will add or remove the window's `SDL_WINDOW_BORDERLESS` flag and add + * or remove the border from the actual window. This is a no-op if the + * window's border already matches the requested state. + * + * You can't change the border state of a fullscreen window. + * + * \param window the window of which to change the border state + * \param bordered SDL_FALSE to remove border, SDL_TRUE to add border + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowFlags + */ +extern DECLSPEC void SDLCALL SDL_SetWindowBordered(SDL_Window * window, + SDL_bool bordered); + +/** + * Set the user-resizable state of a window. + * + * This will add or remove the window's `SDL_WINDOW_RESIZABLE` flag and + * allow/disallow user resizing of the window. This is a no-op if the window's + * resizable state already matches the requested state. + * + * You can't change the resizable state of a fullscreen window. + * + * \param window the window of which to change the resizable state + * \param resizable SDL_TRUE to allow resizing, SDL_FALSE to disallow + * + * \since This function is available since SDL 2.0.5. + * + * \sa SDL_GetWindowFlags + */ +extern DECLSPEC void SDLCALL SDL_SetWindowResizable(SDL_Window * window, + SDL_bool resizable); + +/** + * Set the window to always be above the others. + * + * This will add or remove the window's `SDL_WINDOW_ALWAYS_ON_TOP` flag. This + * will bring the window to the front and keep the window above the rest. + * + * \param window The window of which to change the always on top state + * \param on_top SDL_TRUE to set the window always on top, SDL_FALSE to + * disable + * + * \since This function is available since SDL 2.0.16. + * + * \sa SDL_GetWindowFlags + */ +extern DECLSPEC void SDLCALL SDL_SetWindowAlwaysOnTop(SDL_Window * window, + SDL_bool on_top); + +/** + * Show a window. + * + * \param window the window to show + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_HideWindow + * \sa SDL_RaiseWindow + */ +extern DECLSPEC void SDLCALL SDL_ShowWindow(SDL_Window * window); + +/** + * Hide a window. + * + * \param window the window to hide + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_ShowWindow + */ +extern DECLSPEC void SDLCALL SDL_HideWindow(SDL_Window * window); + +/** + * Raise a window above other windows and set the input focus. + * + * \param window the window to raise + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC void SDLCALL SDL_RaiseWindow(SDL_Window * window); + +/** + * Make a window as large as possible. + * + * \param window the window to maximize + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_MinimizeWindow + * \sa SDL_RestoreWindow + */ +extern DECLSPEC void SDLCALL SDL_MaximizeWindow(SDL_Window * window); + +/** + * Minimize a window to an iconic representation. + * + * \param window the window to minimize + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_MaximizeWindow + * \sa SDL_RestoreWindow + */ +extern DECLSPEC void SDLCALL SDL_MinimizeWindow(SDL_Window * window); + +/** + * Restore the size and position of a minimized or maximized window. + * + * \param window the window to restore + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_MaximizeWindow + * \sa SDL_MinimizeWindow + */ +extern DECLSPEC void SDLCALL SDL_RestoreWindow(SDL_Window * window); + +/** + * Set a window's fullscreen state. + * + * `flags` may be `SDL_WINDOW_FULLSCREEN`, for "real" fullscreen with a + * videomode change; `SDL_WINDOW_FULLSCREEN_DESKTOP` for "fake" fullscreen + * that takes the size of the desktop; and 0 for windowed mode. + * + * \param window the window to change + * \param flags `SDL_WINDOW_FULLSCREEN`, `SDL_WINDOW_FULLSCREEN_DESKTOP` or 0 + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowDisplayMode + * \sa SDL_SetWindowDisplayMode + */ +extern DECLSPEC int SDLCALL SDL_SetWindowFullscreen(SDL_Window * window, + Uint32 flags); + +/** + * Return whether the window has a surface associated with it. + * + * \returns SDL_TRUE if there is a surface associated with the window, or SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.28.0. + * + * \sa SDL_GetWindowSurface + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasWindowSurface(SDL_Window *window); + +/** + * Get the SDL surface associated with the window. + * + * A new surface will be created with the optimal format for the window, if + * necessary. This surface will be freed when the window is destroyed. Do not + * free this surface. + * + * This surface will be invalidated if the window is resized. After resizing a + * window this function must be called again to return a valid surface. + * + * You may not combine this with 3D or the rendering API on this window. + * + * This function is affected by `SDL_HINT_FRAMEBUFFER_ACCELERATION`. + * + * \param window the window to query + * \returns the surface associated with the window, or NULL on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_DestroyWindowSurface + * \sa SDL_HasWindowSurface + * \sa SDL_UpdateWindowSurface + * \sa SDL_UpdateWindowSurfaceRects + */ +extern DECLSPEC SDL_Surface * SDLCALL SDL_GetWindowSurface(SDL_Window * window); + +/** + * Copy the window surface to the screen. + * + * This is the function you use to reflect any changes to the surface on the + * screen. + * + * This function is equivalent to the SDL 1.2 API SDL_Flip(). + * + * \param window the window to update + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowSurface + * \sa SDL_UpdateWindowSurfaceRects + */ +extern DECLSPEC int SDLCALL SDL_UpdateWindowSurface(SDL_Window * window); + +/** + * Copy areas of the window surface to the screen. + * + * This is the function you use to reflect changes to portions of the surface + * on the screen. + * + * This function is equivalent to the SDL 1.2 API SDL_UpdateRects(). + * + * \param window the window to update + * \param rects an array of SDL_Rect structures representing areas of the + * surface to copy, in pixels + * \param numrects the number of rectangles + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowSurface + * \sa SDL_UpdateWindowSurface + */ +extern DECLSPEC int SDLCALL SDL_UpdateWindowSurfaceRects(SDL_Window * window, + const SDL_Rect * rects, + int numrects); + +/** + * Destroy the surface associated with the window. + * + * \param window the window to update + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.28.0. + * + * \sa SDL_GetWindowSurface + * \sa SDL_HasWindowSurface + */ +extern DECLSPEC int SDLCALL SDL_DestroyWindowSurface(SDL_Window *window); + +/** + * Set a window's input grab mode. + * + * When input is grabbed, the mouse is confined to the window. This function + * will also grab the keyboard if `SDL_HINT_GRAB_KEYBOARD` is set. To grab the + * keyboard without also grabbing the mouse, use SDL_SetWindowKeyboardGrab(). + * + * If the caller enables a grab while another window is currently grabbed, the + * other window loses its grab in favor of the caller's window. + * + * \param window the window for which the input grab mode should be set + * \param grabbed SDL_TRUE to grab input or SDL_FALSE to release input + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetGrabbedWindow + * \sa SDL_GetWindowGrab + */ +extern DECLSPEC void SDLCALL SDL_SetWindowGrab(SDL_Window * window, + SDL_bool grabbed); + +/** + * Set a window's keyboard grab mode. + * + * Keyboard grab enables capture of system keyboard shortcuts like Alt+Tab or + * the Meta/Super key. Note that not all system keyboard shortcuts can be + * captured by applications (one example is Ctrl+Alt+Del on Windows). + * + * This is primarily intended for specialized applications such as VNC clients + * or VM frontends. Normal games should not use keyboard grab. + * + * When keyboard grab is enabled, SDL will continue to handle Alt+Tab when the + * window is full-screen to ensure the user is not trapped in your + * application. If you have a custom keyboard shortcut to exit fullscreen + * mode, you may suppress this behavior with + * `SDL_HINT_ALLOW_ALT_TAB_WHILE_GRABBED`. + * + * If the caller enables a grab while another window is currently grabbed, the + * other window loses its grab in favor of the caller's window. + * + * \param window The window for which the keyboard grab mode should be set. + * \param grabbed This is SDL_TRUE to grab keyboard, and SDL_FALSE to release. + * + * \since This function is available since SDL 2.0.16. + * + * \sa SDL_GetWindowKeyboardGrab + * \sa SDL_SetWindowMouseGrab + * \sa SDL_SetWindowGrab + */ +extern DECLSPEC void SDLCALL SDL_SetWindowKeyboardGrab(SDL_Window * window, + SDL_bool grabbed); + +/** + * Set a window's mouse grab mode. + * + * Mouse grab confines the mouse cursor to the window. + * + * \param window The window for which the mouse grab mode should be set. + * \param grabbed This is SDL_TRUE to grab mouse, and SDL_FALSE to release. + * + * \since This function is available since SDL 2.0.16. + * + * \sa SDL_GetWindowMouseGrab + * \sa SDL_SetWindowKeyboardGrab + * \sa SDL_SetWindowGrab + */ +extern DECLSPEC void SDLCALL SDL_SetWindowMouseGrab(SDL_Window * window, + SDL_bool grabbed); + +/** + * Get a window's input grab mode. + * + * \param window the window to query + * \returns SDL_TRUE if input is grabbed, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetWindowGrab + */ +extern DECLSPEC SDL_bool SDLCALL SDL_GetWindowGrab(SDL_Window * window); + +/** + * Get a window's keyboard grab mode. + * + * \param window the window to query + * \returns SDL_TRUE if keyboard is grabbed, and SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.16. + * + * \sa SDL_SetWindowKeyboardGrab + * \sa SDL_GetWindowGrab + */ +extern DECLSPEC SDL_bool SDLCALL SDL_GetWindowKeyboardGrab(SDL_Window * window); + +/** + * Get a window's mouse grab mode. + * + * \param window the window to query + * \returns SDL_TRUE if mouse is grabbed, and SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.16. + * + * \sa SDL_SetWindowKeyboardGrab + * \sa SDL_GetWindowGrab + */ +extern DECLSPEC SDL_bool SDLCALL SDL_GetWindowMouseGrab(SDL_Window * window); + +/** + * Get the window that currently has an input grab enabled. + * + * \returns the window if input is grabbed or NULL otherwise. + * + * \since This function is available since SDL 2.0.4. + * + * \sa SDL_GetWindowGrab + * \sa SDL_SetWindowGrab + */ +extern DECLSPEC SDL_Window * SDLCALL SDL_GetGrabbedWindow(void); + +/** + * Confines the cursor to the specified area of a window. + * + * Note that this does NOT grab the cursor, it only defines the area a cursor + * is restricted to when the window has mouse focus. + * + * \param window The window that will be associated with the barrier. + * \param rect A rectangle area in window-relative coordinates. If NULL the + * barrier for the specified window will be destroyed. + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_GetWindowMouseRect + * \sa SDL_SetWindowMouseGrab + */ +extern DECLSPEC int SDLCALL SDL_SetWindowMouseRect(SDL_Window * window, const SDL_Rect * rect); + +/** + * Get the mouse confinement rectangle of a window. + * + * \param window The window to query + * \returns A pointer to the mouse confinement rectangle of a window, or NULL + * if there isn't one. + * + * \since This function is available since SDL 2.0.18. + * + * \sa SDL_SetWindowMouseRect + */ +extern DECLSPEC const SDL_Rect * SDLCALL SDL_GetWindowMouseRect(SDL_Window * window); + +/** + * Set the brightness (gamma multiplier) for a given window's display. + * + * Despite the name and signature, this method sets the brightness of the + * entire display, not an individual window. A window is considered to be + * owned by the display that contains the window's center pixel. (The index of + * this display can be retrieved using SDL_GetWindowDisplayIndex().) The + * brightness set will not follow the window if it is moved to another + * display. + * + * Many platforms will refuse to set the display brightness in modern times. + * You are better off using a shader to adjust gamma during rendering, or + * something similar. + * + * \param window the window used to select the display whose brightness will + * be changed + * \param brightness the brightness (gamma multiplier) value to set where 0.0 + * is completely dark and 1.0 is normal brightness + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowBrightness + * \sa SDL_SetWindowGammaRamp + */ +extern DECLSPEC int SDLCALL SDL_SetWindowBrightness(SDL_Window * window, float brightness); + +/** + * Get the brightness (gamma multiplier) for a given window's display. + * + * Despite the name and signature, this method retrieves the brightness of the + * entire display, not an individual window. A window is considered to be + * owned by the display that contains the window's center pixel. (The index of + * this display can be retrieved using SDL_GetWindowDisplayIndex().) + * + * \param window the window used to select the display whose brightness will + * be queried + * \returns the brightness for the display where 0.0 is completely dark and + * 1.0 is normal brightness. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetWindowBrightness + */ +extern DECLSPEC float SDLCALL SDL_GetWindowBrightness(SDL_Window * window); + +/** + * Set the opacity for a window. + * + * The parameter `opacity` will be clamped internally between 0.0f + * (transparent) and 1.0f (opaque). + * + * This function also returns -1 if setting the opacity isn't supported. + * + * \param window the window which will be made transparent or opaque + * \param opacity the opacity value (0.0f - transparent, 1.0f - opaque) + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.5. + * + * \sa SDL_GetWindowOpacity + */ +extern DECLSPEC int SDLCALL SDL_SetWindowOpacity(SDL_Window * window, float opacity); + +/** + * Get the opacity of a window. + * + * If transparency isn't supported on this platform, opacity will be reported + * as 1.0f without error. + * + * The parameter `opacity` is ignored if it is NULL. + * + * This function also returns -1 if an invalid window was provided. + * + * \param window the window to get the current opacity value from + * \param out_opacity the float filled in (0.0f - transparent, 1.0f - opaque) + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.5. + * + * \sa SDL_SetWindowOpacity + */ +extern DECLSPEC int SDLCALL SDL_GetWindowOpacity(SDL_Window * window, float * out_opacity); + +/** + * Set the window as a modal for another window. + * + * \param modal_window the window that should be set modal + * \param parent_window the parent window for the modal window + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.5. + */ +extern DECLSPEC int SDLCALL SDL_SetWindowModalFor(SDL_Window * modal_window, SDL_Window * parent_window); + +/** + * Explicitly set input focus to the window. + * + * You almost certainly want SDL_RaiseWindow() instead of this function. Use + * this with caution, as you might give focus to a window that is completely + * obscured by other windows. + * + * \param window the window that should get the input focus + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.5. + * + * \sa SDL_RaiseWindow + */ +extern DECLSPEC int SDLCALL SDL_SetWindowInputFocus(SDL_Window * window); + +/** + * Set the gamma ramp for the display that owns a given window. + * + * Set the gamma translation table for the red, green, and blue channels of + * the video hardware. Each table is an array of 256 16-bit quantities, + * representing a mapping between the input and output for that channel. The + * input is the index into the array, and the output is the 16-bit gamma value + * at that index, scaled to the output color precision. + * + * Despite the name and signature, this method sets the gamma ramp of the + * entire display, not an individual window. A window is considered to be + * owned by the display that contains the window's center pixel. (The index of + * this display can be retrieved using SDL_GetWindowDisplayIndex().) The gamma + * ramp set will not follow the window if it is moved to another display. + * + * \param window the window used to select the display whose gamma ramp will + * be changed + * \param red a 256 element array of 16-bit quantities representing the + * translation table for the red channel, or NULL + * \param green a 256 element array of 16-bit quantities representing the + * translation table for the green channel, or NULL + * \param blue a 256 element array of 16-bit quantities representing the + * translation table for the blue channel, or NULL + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GetWindowGammaRamp + */ +extern DECLSPEC int SDLCALL SDL_SetWindowGammaRamp(SDL_Window * window, + const Uint16 * red, + const Uint16 * green, + const Uint16 * blue); + +/** + * Get the gamma ramp for a given window's display. + * + * Despite the name and signature, this method retrieves the gamma ramp of the + * entire display, not an individual window. A window is considered to be + * owned by the display that contains the window's center pixel. (The index of + * this display can be retrieved using SDL_GetWindowDisplayIndex().) + * + * \param window the window used to select the display whose gamma ramp will + * be queried + * \param red a 256 element array of 16-bit quantities filled in with the + * translation table for the red channel, or NULL + * \param green a 256 element array of 16-bit quantities filled in with the + * translation table for the green channel, or NULL + * \param blue a 256 element array of 16-bit quantities filled in with the + * translation table for the blue channel, or NULL + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_SetWindowGammaRamp + */ +extern DECLSPEC int SDLCALL SDL_GetWindowGammaRamp(SDL_Window * window, + Uint16 * red, + Uint16 * green, + Uint16 * blue); + +/** + * Possible return values from the SDL_HitTest callback. + * + * \sa SDL_HitTest + */ +typedef enum +{ + SDL_HITTEST_NORMAL, /**< Region is normal. No special properties. */ + SDL_HITTEST_DRAGGABLE, /**< Region can drag entire window. */ + SDL_HITTEST_RESIZE_TOPLEFT, + SDL_HITTEST_RESIZE_TOP, + SDL_HITTEST_RESIZE_TOPRIGHT, + SDL_HITTEST_RESIZE_RIGHT, + SDL_HITTEST_RESIZE_BOTTOMRIGHT, + SDL_HITTEST_RESIZE_BOTTOM, + SDL_HITTEST_RESIZE_BOTTOMLEFT, + SDL_HITTEST_RESIZE_LEFT +} SDL_HitTestResult; + +/** + * Callback used for hit-testing. + * + * \param win the SDL_Window where hit-testing was set on + * \param area an SDL_Point which should be hit-tested + * \param data what was passed as `callback_data` to SDL_SetWindowHitTest() + * \return an SDL_HitTestResult value. + * + * \sa SDL_SetWindowHitTest + */ +typedef SDL_HitTestResult (SDLCALL *SDL_HitTest)(SDL_Window *win, + const SDL_Point *area, + void *data); + +/** + * Provide a callback that decides if a window region has special properties. + * + * Normally windows are dragged and resized by decorations provided by the + * system window manager (a title bar, borders, etc), but for some apps, it + * makes sense to drag them from somewhere else inside the window itself; for + * example, one might have a borderless window that wants to be draggable from + * any part, or simulate its own title bar, etc. + * + * This function lets the app provide a callback that designates pieces of a + * given window as special. This callback is run during event processing if we + * need to tell the OS to treat a region of the window specially; the use of + * this callback is known as "hit testing." + * + * Mouse input may not be delivered to your application if it is within a + * special area; the OS will often apply that input to moving the window or + * resizing the window and not deliver it to the application. + * + * Specifying NULL for a callback disables hit-testing. Hit-testing is + * disabled by default. + * + * Platforms that don't support this functionality will return -1 + * unconditionally, even if you're attempting to disable hit-testing. + * + * Your callback may fire at any time, and its firing does not indicate any + * specific behavior (for example, on Windows, this certainly might fire when + * the OS is deciding whether to drag your window, but it fires for lots of + * other reasons, too, some unrelated to anything you probably care about _and + * when the mouse isn't actually at the location it is testing_). Since this + * can fire at any time, you should try to keep your callback efficient, + * devoid of allocations, etc. + * + * \param window the window to set hit-testing on + * \param callback the function to call when doing a hit-test + * \param callback_data an app-defined void pointer passed to **callback** + * \returns 0 on success or -1 on error (including unsupported); call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.4. + */ +extern DECLSPEC int SDLCALL SDL_SetWindowHitTest(SDL_Window * window, + SDL_HitTest callback, + void *callback_data); + +/** + * Request a window to demand attention from the user. + * + * \param window the window to be flashed + * \param operation the flash operation + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.16. + */ +extern DECLSPEC int SDLCALL SDL_FlashWindow(SDL_Window * window, SDL_FlashOperation operation); + +/** + * Destroy a window. + * + * If `window` is NULL, this function will return immediately after setting + * the SDL error message to "Invalid window". See SDL_GetError(). + * + * \param window the window to destroy + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_CreateWindow + * \sa SDL_CreateWindowFrom + */ +extern DECLSPEC void SDLCALL SDL_DestroyWindow(SDL_Window * window); + + +/** + * Check whether the screensaver is currently enabled. + * + * The screensaver is disabled by default since SDL 2.0.2. Before SDL 2.0.2 + * the screensaver was enabled by default. + * + * The default can also be changed using `SDL_HINT_VIDEO_ALLOW_SCREENSAVER`. + * + * \returns SDL_TRUE if the screensaver is enabled, SDL_FALSE if it is + * disabled. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_DisableScreenSaver + * \sa SDL_EnableScreenSaver + */ +extern DECLSPEC SDL_bool SDLCALL SDL_IsScreenSaverEnabled(void); + +/** + * Allow the screen to be blanked by a screen saver. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_DisableScreenSaver + * \sa SDL_IsScreenSaverEnabled + */ +extern DECLSPEC void SDLCALL SDL_EnableScreenSaver(void); + +/** + * Prevent the screen from being blanked by a screen saver. + * + * If you disable the screensaver, it is automatically re-enabled when SDL + * quits. + * + * The screensaver is disabled by default since SDL 2.0.2. Before SDL 2.0.2 + * the screensaver was enabled by default. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_EnableScreenSaver + * \sa SDL_IsScreenSaverEnabled + */ +extern DECLSPEC void SDLCALL SDL_DisableScreenSaver(void); + + +/** + * \name OpenGL support functions + */ +/* @{ */ + +/** + * Dynamically load an OpenGL library. + * + * This should be done after initializing the video driver, but before + * creating any OpenGL windows. If no OpenGL library is loaded, the default + * library will be loaded upon creation of the first OpenGL window. + * + * If you do this, you need to retrieve all of the GL functions used in your + * program from the dynamic library using SDL_GL_GetProcAddress(). + * + * \param path the platform dependent OpenGL library name, or NULL to open the + * default OpenGL library + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GL_GetProcAddress + * \sa SDL_GL_UnloadLibrary + */ +extern DECLSPEC int SDLCALL SDL_GL_LoadLibrary(const char *path); + +/** + * Get an OpenGL function by name. + * + * If the GL library is loaded at runtime with SDL_GL_LoadLibrary(), then all + * GL functions must be retrieved this way. Usually this is used to retrieve + * function pointers to OpenGL extensions. + * + * There are some quirks to looking up OpenGL functions that require some + * extra care from the application. If you code carefully, you can handle + * these quirks without any platform-specific code, though: + * + * - On Windows, function pointers are specific to the current GL context; + * this means you need to have created a GL context and made it current + * before calling SDL_GL_GetProcAddress(). If you recreate your context or + * create a second context, you should assume that any existing function + * pointers aren't valid to use with it. This is (currently) a + * Windows-specific limitation, and in practice lots of drivers don't suffer + * this limitation, but it is still the way the wgl API is documented to + * work and you should expect crashes if you don't respect it. Store a copy + * of the function pointers that comes and goes with context lifespan. + * - On X11, function pointers returned by this function are valid for any + * context, and can even be looked up before a context is created at all. + * This means that, for at least some common OpenGL implementations, if you + * look up a function that doesn't exist, you'll get a non-NULL result that + * is _NOT_ safe to call. You must always make sure the function is actually + * available for a given GL context before calling it, by checking for the + * existence of the appropriate extension with SDL_GL_ExtensionSupported(), + * or verifying that the version of OpenGL you're using offers the function + * as core functionality. + * - Some OpenGL drivers, on all platforms, *will* return NULL if a function + * isn't supported, but you can't count on this behavior. Check for + * extensions you use, and if you get a NULL anyway, act as if that + * extension wasn't available. This is probably a bug in the driver, but you + * can code defensively for this scenario anyhow. + * - Just because you're on Linux/Unix, don't assume you'll be using X11. + * Next-gen display servers are waiting to replace it, and may or may not + * make the same promises about function pointers. + * - OpenGL function pointers must be declared `APIENTRY` as in the example + * code. This will ensure the proper calling convention is followed on + * platforms where this matters (Win32) thereby avoiding stack corruption. + * + * \param proc the name of an OpenGL function + * \returns a pointer to the named OpenGL function. The returned pointer + * should be cast to the appropriate function signature. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GL_ExtensionSupported + * \sa SDL_GL_LoadLibrary + * \sa SDL_GL_UnloadLibrary + */ +extern DECLSPEC void *SDLCALL SDL_GL_GetProcAddress(const char *proc); + +/** + * Unload the OpenGL library previously loaded by SDL_GL_LoadLibrary(). + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GL_LoadLibrary + */ +extern DECLSPEC void SDLCALL SDL_GL_UnloadLibrary(void); + +/** + * Check if an OpenGL extension is supported for the current context. + * + * This function operates on the current GL context; you must have created a + * context and it must be current before calling this function. Do not assume + * that all contexts you create will have the same set of extensions + * available, or that recreating an existing context will offer the same + * extensions again. + * + * While it's probably not a massive overhead, this function is not an O(1) + * operation. Check the extensions you care about after creating the GL + * context and save that information somewhere instead of calling the function + * every time you need to know. + * + * \param extension the name of the extension to check + * \returns SDL_TRUE if the extension is supported, SDL_FALSE otherwise. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_GL_ExtensionSupported(const char + *extension); + +/** + * Reset all previously set OpenGL context attributes to their default values. + * + * \since This function is available since SDL 2.0.2. + * + * \sa SDL_GL_GetAttribute + * \sa SDL_GL_SetAttribute + */ +extern DECLSPEC void SDLCALL SDL_GL_ResetAttributes(void); + +/** + * Set an OpenGL window attribute before window creation. + * + * This function sets the OpenGL attribute `attr` to `value`. The requested + * attributes should be set before creating an OpenGL window. You should use + * SDL_GL_GetAttribute() to check the values after creating the OpenGL + * context, since the values obtained can differ from the requested ones. + * + * \param attr an SDL_GLattr enum value specifying the OpenGL attribute to set + * \param value the desired value for the attribute + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GL_GetAttribute + * \sa SDL_GL_ResetAttributes + */ +extern DECLSPEC int SDLCALL SDL_GL_SetAttribute(SDL_GLattr attr, int value); + +/** + * Get the actual value for an attribute from the current context. + * + * \param attr an SDL_GLattr enum value specifying the OpenGL attribute to get + * \param value a pointer filled in with the current value of `attr` + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GL_ResetAttributes + * \sa SDL_GL_SetAttribute + */ +extern DECLSPEC int SDLCALL SDL_GL_GetAttribute(SDL_GLattr attr, int *value); + +/** + * Create an OpenGL context for an OpenGL window, and make it current. + * + * Windows users new to OpenGL should note that, for historical reasons, GL + * functions added after OpenGL version 1.1 are not available by default. + * Those functions must be loaded at run-time, either with an OpenGL + * extension-handling library or with SDL_GL_GetProcAddress() and its related + * functions. + * + * SDL_GLContext is an alias for `void *`. It's opaque to the application. + * + * \param window the window to associate with the context + * \returns the OpenGL context associated with `window` or NULL on error; call + * SDL_GetError() for more details. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GL_DeleteContext + * \sa SDL_GL_MakeCurrent + */ +extern DECLSPEC SDL_GLContext SDLCALL SDL_GL_CreateContext(SDL_Window * + window); + +/** + * Set up an OpenGL context for rendering into an OpenGL window. + * + * The context must have been created with a compatible window. + * + * \param window the window to associate with the context + * \param context the OpenGL context to associate with the window + * \returns 0 on success or a negative error code on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GL_CreateContext + */ +extern DECLSPEC int SDLCALL SDL_GL_MakeCurrent(SDL_Window * window, + SDL_GLContext context); + +/** + * Get the currently active OpenGL window. + * + * \returns the currently active OpenGL window on success or NULL on failure; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC SDL_Window* SDLCALL SDL_GL_GetCurrentWindow(void); + +/** + * Get the currently active OpenGL context. + * + * \returns the currently active OpenGL context or NULL on failure; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GL_MakeCurrent + */ +extern DECLSPEC SDL_GLContext SDLCALL SDL_GL_GetCurrentContext(void); + +/** + * Get the size of a window's underlying drawable in pixels. + * + * This returns info useful for calling glViewport(). + * + * This may differ from SDL_GetWindowSize() if we're rendering to a high-DPI + * drawable, i.e. the window was created with `SDL_WINDOW_ALLOW_HIGHDPI` on a + * platform with high-DPI support (Apple calls this "Retina"), and not + * disabled by the `SDL_HINT_VIDEO_HIGHDPI_DISABLED` hint. + * + * \param window the window from which the drawable size should be queried + * \param w a pointer to variable for storing the width in pixels, may be NULL + * \param h a pointer to variable for storing the height in pixels, may be + * NULL + * + * \since This function is available since SDL 2.0.1. + * + * \sa SDL_CreateWindow + * \sa SDL_GetWindowSize + */ +extern DECLSPEC void SDLCALL SDL_GL_GetDrawableSize(SDL_Window * window, int *w, + int *h); + +/** + * Set the swap interval for the current OpenGL context. + * + * Some systems allow specifying -1 for the interval, to enable adaptive + * vsync. Adaptive vsync works the same as vsync, but if you've already missed + * the vertical retrace for a given frame, it swaps buffers immediately, which + * might be less jarring for the user during occasional framerate drops. If an + * application requests adaptive vsync and the system does not support it, + * this function will fail and return -1. In such a case, you should probably + * retry the call with 1 for the interval. + * + * Adaptive vsync is implemented for some glX drivers with + * GLX_EXT_swap_control_tear, and for some Windows drivers with + * WGL_EXT_swap_control_tear. + * + * Read more on the Khronos wiki: + * https://www.khronos.org/opengl/wiki/Swap_Interval#Adaptive_Vsync + * + * \param interval 0 for immediate updates, 1 for updates synchronized with + * the vertical retrace, -1 for adaptive vsync + * \returns 0 on success or -1 if setting the swap interval is not supported; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GL_GetSwapInterval + */ +extern DECLSPEC int SDLCALL SDL_GL_SetSwapInterval(int interval); + +/** + * Get the swap interval for the current OpenGL context. + * + * If the system can't determine the swap interval, or there isn't a valid + * current context, this function will return 0 as a safe default. + * + * \returns 0 if there is no vertical retrace synchronization, 1 if the buffer + * swap is synchronized with the vertical retrace, and -1 if late + * swaps happen immediately instead of waiting for the next retrace; + * call SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GL_SetSwapInterval + */ +extern DECLSPEC int SDLCALL SDL_GL_GetSwapInterval(void); + +/** + * Update a window with OpenGL rendering. + * + * This is used with double-buffered OpenGL contexts, which are the default. + * + * On macOS, make sure you bind 0 to the draw framebuffer before swapping the + * window, otherwise nothing will happen. If you aren't using + * glBindFramebuffer(), this is the default and you won't have to do anything + * extra. + * + * \param window the window to change + * + * \since This function is available since SDL 2.0.0. + */ +extern DECLSPEC void SDLCALL SDL_GL_SwapWindow(SDL_Window * window); + +/** + * Delete an OpenGL context. + * + * \param context the OpenGL context to be deleted + * + * \since This function is available since SDL 2.0.0. + * + * \sa SDL_GL_CreateContext + */ +extern DECLSPEC void SDLCALL SDL_GL_DeleteContext(SDL_GLContext context); + +/* @} *//* OpenGL support functions */ + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_video_h_ */ + +/* vi: set ts=4 sw=4 expandtab: */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/SDL_vulkan.h b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_vulkan.h new file mode 100644 index 00000000..ab86a0b8 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/SDL_vulkan.h @@ -0,0 +1,215 @@ +/* + Simple DirectMedia Layer + Copyright (C) 2017, Mark Callow + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file SDL_vulkan.h + * + * Header file for functions to creating Vulkan surfaces on SDL windows. + */ + +#ifndef SDL_vulkan_h_ +#define SDL_vulkan_h_ + +#include "SDL_video.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* Avoid including vulkan.h, don't define VkInstance if it's already included */ +#ifdef VULKAN_H_ +#define NO_SDL_VULKAN_TYPEDEFS +#endif +#ifndef NO_SDL_VULKAN_TYPEDEFS +#define VK_DEFINE_HANDLE(object) typedef struct object##_T* object; + +#if defined(__LP64__) || defined(_WIN64) || defined(__x86_64__) || defined(_M_X64) || defined(__ia64) || defined (_M_IA64) || defined(__aarch64__) || defined(__powerpc64__) +#define VK_DEFINE_NON_DISPATCHABLE_HANDLE(object) typedef struct object##_T *object; +#else +#define VK_DEFINE_NON_DISPATCHABLE_HANDLE(object) typedef uint64_t object; +#endif + +VK_DEFINE_HANDLE(VkInstance) +VK_DEFINE_NON_DISPATCHABLE_HANDLE(VkSurfaceKHR) + +#endif /* !NO_SDL_VULKAN_TYPEDEFS */ + +typedef VkInstance SDL_vulkanInstance; +typedef VkSurfaceKHR SDL_vulkanSurface; /* for compatibility with Tizen */ + +/** + * \name Vulkan support functions + * + * \note SDL_Vulkan_GetInstanceExtensions & SDL_Vulkan_CreateSurface API + * is compatable with Tizen's implementation of Vulkan in SDL. + */ +/* @{ */ + +/** + * Dynamically load the Vulkan loader library. + * + * This should be called after initializing the video driver, but before + * creating any Vulkan windows. If no Vulkan loader library is loaded, the + * default library will be loaded upon creation of the first Vulkan window. + * + * It is fairly common for Vulkan applications to link with libvulkan instead + * of explicitly loading it at run time. This will work with SDL provided the + * application links to a dynamic library and both it and SDL use the same + * search path. + * + * If you specify a non-NULL `path`, an application should retrieve all of the + * Vulkan functions it uses from the dynamic library using + * SDL_Vulkan_GetVkGetInstanceProcAddr unless you can guarantee `path` points + * to the same vulkan loader library the application linked to. + * + * On Apple devices, if `path` is NULL, SDL will attempt to find the + * `vkGetInstanceProcAddr` address within all the Mach-O images of the current + * process. This is because it is fairly common for Vulkan applications to + * link with libvulkan (and historically MoltenVK was provided as a static + * library). If it is not found, on macOS, SDL will attempt to load + * `vulkan.framework/vulkan`, `libvulkan.1.dylib`, + * `MoltenVK.framework/MoltenVK`, and `libMoltenVK.dylib`, in that order. On + * iOS, SDL will attempt to load `libMoltenVK.dylib`. Applications using a + * dynamic framework or .dylib must ensure it is included in its application + * bundle. + * + * On non-Apple devices, application linking with a static libvulkan is not + * supported. Either do not link to the Vulkan loader or link to a dynamic + * library version. + * + * \param path The platform dependent Vulkan loader library name or NULL + * \returns 0 on success or -1 if the library couldn't be loaded; call + * SDL_GetError() for more information. + * + * \since This function is available since SDL 2.0.6. + * + * \sa SDL_Vulkan_GetVkInstanceProcAddr + * \sa SDL_Vulkan_UnloadLibrary + */ +extern DECLSPEC int SDLCALL SDL_Vulkan_LoadLibrary(const char *path); + +/** + * Get the address of the `vkGetInstanceProcAddr` function. + * + * This should be called after either calling SDL_Vulkan_LoadLibrary() or + * creating an SDL_Window with the `SDL_WINDOW_VULKAN` flag. + * + * \returns the function pointer for `vkGetInstanceProcAddr` or NULL on error. + * + * \since This function is available since SDL 2.0.6. + */ +extern DECLSPEC void *SDLCALL SDL_Vulkan_GetVkGetInstanceProcAddr(void); + +/** + * Unload the Vulkan library previously loaded by SDL_Vulkan_LoadLibrary() + * + * \since This function is available since SDL 2.0.6. + * + * \sa SDL_Vulkan_LoadLibrary + */ +extern DECLSPEC void SDLCALL SDL_Vulkan_UnloadLibrary(void); + +/** + * Get the names of the Vulkan instance extensions needed to create a surface + * with SDL_Vulkan_CreateSurface. + * + * If `pNames` is NULL, then the number of required Vulkan instance extensions + * is returned in `pCount`. Otherwise, `pCount` must point to a variable set + * to the number of elements in the `pNames` array, and on return the variable + * is overwritten with the number of names actually written to `pNames`. If + * `pCount` is less than the number of required extensions, at most `pCount` + * structures will be written. If `pCount` is smaller than the number of + * required extensions, SDL_FALSE will be returned instead of SDL_TRUE, to + * indicate that not all the required extensions were returned. + * + * The `window` parameter is currently needed to be valid as of SDL 2.0.8, + * however, this parameter will likely be removed in future releases + * + * \param window A window for which the required Vulkan instance extensions + * should be retrieved (will be deprecated in a future release) + * \param pCount A pointer to an unsigned int corresponding to the number of + * extensions to be returned + * \param pNames NULL or a pointer to an array to be filled with required + * Vulkan instance extensions + * \returns SDL_TRUE on success, SDL_FALSE on error. + * + * \since This function is available since SDL 2.0.6. + * + * \sa SDL_Vulkan_CreateSurface + */ +extern DECLSPEC SDL_bool SDLCALL SDL_Vulkan_GetInstanceExtensions(SDL_Window *window, + unsigned int *pCount, + const char **pNames); + +/** + * Create a Vulkan rendering surface for a window. + * + * The `window` must have been created with the `SDL_WINDOW_VULKAN` flag and + * `instance` must have been created with extensions returned by + * SDL_Vulkan_GetInstanceExtensions() enabled. + * + * \param window The window to which to attach the Vulkan surface + * \param instance The Vulkan instance handle + * \param surface A pointer to a VkSurfaceKHR handle to output the newly + * created surface + * \returns SDL_TRUE on success, SDL_FALSE on error. + * + * \since This function is available since SDL 2.0.6. + * + * \sa SDL_Vulkan_GetInstanceExtensions + * \sa SDL_Vulkan_GetDrawableSize + */ +extern DECLSPEC SDL_bool SDLCALL SDL_Vulkan_CreateSurface(SDL_Window *window, + VkInstance instance, + VkSurfaceKHR* surface); + +/** + * Get the size of the window's underlying drawable dimensions in pixels. + * + * This may differ from SDL_GetWindowSize() if we're rendering to a high-DPI + * drawable, i.e. the window was created with `SDL_WINDOW_ALLOW_HIGHDPI` on a + * platform with high-DPI support (Apple calls this "Retina"), and not + * disabled by the `SDL_HINT_VIDEO_HIGHDPI_DISABLED` hint. + * + * \param window an SDL_Window for which the size is to be queried + * \param w Pointer to the variable to write the width to or NULL + * \param h Pointer to the variable to write the height to or NULL + * + * \since This function is available since SDL 2.0.6. + * + * \sa SDL_GetWindowSize + * \sa SDL_CreateWindow + * \sa SDL_Vulkan_CreateSurface + */ +extern DECLSPEC void SDLCALL SDL_Vulkan_GetDrawableSize(SDL_Window * window, + int *w, int *h); + +/* @} *//* Vulkan support functions */ + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* SDL_vulkan_h_ */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/begin_code.h b/desktop/cores/sdl/SDL2.macos.arm/include/begin_code.h new file mode 100644 index 00000000..4142ffeb --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/begin_code.h @@ -0,0 +1,187 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file begin_code.h + * + * This file sets things up for C dynamic library function definitions, + * static inlined functions, and structures aligned at 4-byte alignment. + * If you don't like ugly C preprocessor code, don't look at this file. :) + */ + +/* This shouldn't be nested -- included it around code only. */ +#ifdef SDL_begin_code_h +#error Nested inclusion of begin_code.h +#endif +#define SDL_begin_code_h + +#ifndef SDL_DEPRECATED +# if defined(__GNUC__) && (__GNUC__ >= 4) /* technically, this arrived in gcc 3.1, but oh well. */ +# define SDL_DEPRECATED __attribute__((deprecated)) +# else +# define SDL_DEPRECATED +# endif +#endif + +#ifndef SDL_UNUSED +# ifdef __GNUC__ +# define SDL_UNUSED __attribute__((unused)) +# else +# define SDL_UNUSED +# endif +#endif + +/* Some compilers use a special export keyword */ +#ifndef DECLSPEC +# if defined(__WIN32__) || defined(__WINRT__) || defined(__CYGWIN__) || defined(__GDK__) +# ifdef DLL_EXPORT +# define DECLSPEC __declspec(dllexport) +# else +# define DECLSPEC +# endif +# elif defined(__OS2__) +# ifdef BUILD_SDL +# define DECLSPEC __declspec(dllexport) +# else +# define DECLSPEC +# endif +# else +# if defined(__GNUC__) && __GNUC__ >= 4 +# define DECLSPEC __attribute__ ((visibility("default"))) +# else +# define DECLSPEC +# endif +# endif +#endif + +/* By default SDL uses the C calling convention */ +#ifndef SDLCALL +#if (defined(__WIN32__) || defined(__WINRT__) || defined(__GDK__)) && !defined(__GNUC__) +#define SDLCALL __cdecl +#elif defined(__OS2__) || defined(__EMX__) +#define SDLCALL _System +# if defined (__GNUC__) && !defined(_System) +# define _System /* for old EMX/GCC compat. */ +# endif +#else +#define SDLCALL +#endif +#endif /* SDLCALL */ + +/* Removed DECLSPEC on Symbian OS because SDL cannot be a DLL in EPOC */ +#ifdef __SYMBIAN32__ +#undef DECLSPEC +#define DECLSPEC +#endif /* __SYMBIAN32__ */ + +/* Force structure packing at 4 byte alignment. + This is necessary if the header is included in code which has structure + packing set to an alternate value, say for loading structures from disk. + The packing is reset to the previous value in close_code.h + */ +#if defined(_MSC_VER) || defined(__MWERKS__) || defined(__BORLANDC__) +#ifdef _MSC_VER +#pragma warning(disable: 4103) +#endif +#ifdef __clang__ +#pragma clang diagnostic ignored "-Wpragma-pack" +#endif +#ifdef __BORLANDC__ +#pragma nopackwarning +#endif +#ifdef _WIN64 +/* Use 8-byte alignment on 64-bit architectures, so pointers are aligned */ +#pragma pack(push,8) +#else +#pragma pack(push,4) +#endif +#endif /* Compiler needs structure packing set */ + +#ifndef SDL_INLINE +#if defined(__GNUC__) +#define SDL_INLINE __inline__ +#elif defined(_MSC_VER) || defined(__BORLANDC__) || \ + defined(__DMC__) || defined(__SC__) || \ + defined(__WATCOMC__) || defined(__LCC__) || \ + defined(__DECC) || defined(__CC_ARM) +#define SDL_INLINE __inline +#ifndef __inline__ +#define __inline__ __inline +#endif +#else +#define SDL_INLINE inline +#ifndef __inline__ +#define __inline__ inline +#endif +#endif +#endif /* SDL_INLINE not defined */ + +#ifndef SDL_FORCE_INLINE +#if defined(_MSC_VER) +#define SDL_FORCE_INLINE __forceinline +#elif ( (defined(__GNUC__) && (__GNUC__ >= 4)) || defined(__clang__) ) +#define SDL_FORCE_INLINE __attribute__((always_inline)) static __inline__ +#else +#define SDL_FORCE_INLINE static SDL_INLINE +#endif +#endif /* SDL_FORCE_INLINE not defined */ + +#ifndef SDL_NORETURN +#if defined(__GNUC__) +#define SDL_NORETURN __attribute__((noreturn)) +#elif defined(_MSC_VER) +#define SDL_NORETURN __declspec(noreturn) +#else +#define SDL_NORETURN +#endif +#endif /* SDL_NORETURN not defined */ + +/* Apparently this is needed by several Windows compilers */ +#if !defined(__MACH__) +#ifndef NULL +#ifdef __cplusplus +#define NULL 0 +#else +#define NULL ((void *)0) +#endif +#endif /* NULL */ +#endif /* ! Mac OS X - breaks precompiled headers */ + +#ifndef SDL_FALLTHROUGH +#if (defined(__cplusplus) && __cplusplus >= 201703L) || \ + (defined(__STDC_VERSION__) && __STDC_VERSION__ >= 202000L) +#define SDL_FALLTHROUGH [[fallthrough]] +#else +#if defined(__has_attribute) +#define SDL_HAS_FALLTHROUGH __has_attribute(__fallthrough__) +#else +#define SDL_HAS_FALLTHROUGH 0 +#endif /* __has_attribute */ +#if SDL_HAS_FALLTHROUGH && \ + ((defined(__GNUC__) && __GNUC__ >= 7) || \ + (defined(__clang_major__) && __clang_major__ >= 10)) +#define SDL_FALLTHROUGH __attribute__((__fallthrough__)) +#else +#define SDL_FALLTHROUGH do {} while (0) /* fallthrough */ +#endif /* SDL_HAS_FALLTHROUGH */ +#undef SDL_HAS_FALLTHROUGH +#endif /* C++17 or C2x */ +#endif /* SDL_FALLTHROUGH not defined */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/include/close_code.h b/desktop/cores/sdl/SDL2.macos.arm/include/close_code.h new file mode 100644 index 00000000..b5ff3e20 --- /dev/null +++ b/desktop/cores/sdl/SDL2.macos.arm/include/close_code.h @@ -0,0 +1,40 @@ +/* + Simple DirectMedia Layer + Copyright (C) 1997-2023 Sam Lantinga + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. +*/ + +/** + * \file close_code.h + * + * This file reverses the effects of begin_code.h and should be included + * after you finish any function and structure declarations in your headers + */ + +#ifndef SDL_begin_code_h +#error close_code.h included without matching begin_code.h +#endif +#undef SDL_begin_code_h + +/* Reset structure packing at previous byte alignment */ +#if defined(_MSC_VER) || defined(__MWERKS__) || defined(__BORLANDC__) +#ifdef __BORLANDC__ +#pragma nopackwarning +#endif +#pragma pack(pop) +#endif /* Compiler needs structure packing set */ diff --git a/desktop/cores/sdl/SDL2.macos.arm/lib/libSDL2-2.0.0.dylib b/desktop/cores/sdl/SDL2.macos.arm/lib/libSDL2-2.0.0.dylib new file mode 100644 index 0000000000000000000000000000000000000000..c76a277f080bb046277eb3716712a65b73e7eb70 GIT binary patch literal 1540848 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 z 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 z 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

FZ>uenqIh*D=S#E6-Za91s0wU&I^_{r+o? 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⋙dCQm-u%%~7txV7c&CGO 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

@@m}oYK<U;U={X2 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

ofk->b*s^z_dmfvYJ#VsEoI6;1EDApA*sF~8 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 M&C_Mcvv2$V21-JJ(*OVf literal 0 HcmV?d00001 diff --git a/desktop/cores/sdl/SDL2.macos.arm/lib/libSDL2.dylib b/desktop/cores/sdl/SDL2.macos.arm/lib/libSDL2.dylib new file mode 100644 index 0000000000000000000000000000000000000000..c76a277f080bb046277eb3716712a65b73e7eb70 GIT binary patch literal 1540848 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 z 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 z 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

FZ>uenqIh*D=S#E6-Za91s0wU&I^_{r+o? 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⋙dCQm-u%%~7txV7c&CGO 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

@@m}oYK<U;U={X2 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

Y z{7c_-%E-;SKLPv6tVbk0b7qe9{|5f;a`=(9KQ&eIO}SA2;=bh;Qx0z$|KumI{dVGC z@_&-{BkmXBe*^7lgOT0TyuClhEIoyvGZf<*-ib3e96c!5?`SE&*NU&fm5Gu%F}KarhexTU!eO# zxPK#{{-B_H$p2WIr?g>-DpkD`F&!g{0+60bxkJd=MZ@z1%3wHr!8trGA#I z_u`(i%F_sQSwI+4FLIU_yylpW8@X?HEqu~j!=*DgOT3LU#aPZ3XL80ki*w)Gd9GyQ ziC^32?;-v^fb+b`Q$3x}ro^tOr)=XU&o0klEfD95E~h{E-bjvL5vOLB50aj^kEfI` zqP!-%BDSuc$=ZPa%RFzN-#^D1VM`AFnR|7Gp%* zWKa2Xj7^g#c($&-$tsUzoW5Q-^%KXgk1i43HHJcJI)^HH>~lSW{4bXuS4W)bV}~9aS}N&w&EC42{pA|QF)8Pk8PMe}cQJ3^ zf6O}YuW*0exQOT3xwVI$ySw(#^Td0JJ>Gi(f49&gX-}R|Zc#sRuE=s1WnGV-$rG%t ztJhNA3xe_yP*#$Uz`p(m64nHFxre;RO?H>J2wn5aa~aE>I;)NqFur*CZS#J@0*oJ#5h5N9>buXnP6tOQ)Z* zMxoz4`_*S(_MEJpxMs1OqsbZC(P^oRQ(zw7x`m!s>)<=n!w;OF5_Z7<8SCSB`8yaV zeUH-roNL}KFmk;y;F`WUqURauk?WLz>(yM3lsRoYo;B8W^D1rUmfB9c9n%+_{A@T> zH1r1P39JwD8x!0ocs_q54UDrR*LQOL73%$u3Eye^A0D$#(*8&Cd+Vog!&n;=+%6`C z(1nTa8=s==zZ7=h{PV5@{wdbSPFc$w@+*)Vl^pb{v`uSD}U7P4q2=^h+1JC_sirq&p(MR6>tFeD8^yi<_U8A_WoBqamn2aaE zXV%BDciQChj61>hb=W&?TgJfWB)?y?PQDwEk0)^xTn+~%-k|)JhtL&=(D5kmNdFJ4 zGkXmw@duRmt`PorgwU~ou+#JG!jy#I@n+*E*rk4go!th4@|YAN?qwl#_lD3VgwV|k zp_AvAg3ESR2%W4|g5Ak^O|VY(_Q5(?tBCGd=GVYFpiei}%Ua^&&y){xt{>-$DCZd< zdoJo`%#U+TEq5W)Ru08an03pF3Dzz5Pl&zcS?1)`&$!C}iMf2?=UrQ~3s_^YXL0(} z^Qj?4Pax}##T>-bmWtl1R_@R#4&knoGY2%}%QPX3bi?2J`|wCgjB zb(3ZN!h9aa8ljeTN~m>j@*|e-<0l_sE%q>LvYo8WvRI?-;C}CRp0Aj6f-qxbt@R9h zSHk#K{aoj7`K%N1b9mNlg7vLGlX9kC#VzN#0^*8$faeLAV{0jUDQotv9Yvg9bH-i1 zkv*ZDeaP8u!uwDCB&w)tD9(y{Pv-ANo@-d`ig{1&+1C?h;R?=1qq+ZP!pY$|pip_3 zv(4hqisjz%C+9<+>-rXR{=!$Rt=X~;V*l>cMR48m?A=JbEj$w>_w&wEmlsT34rJ1P zT&z`CXS<$qO&{1YDu*GvJ&3mF5~{+2q9oD#X^L$Wnek860(tC;Y^33IwIMxIMp3(Aoj<$Wz zp~8_htK>iOgzM%HX=kgcch5@8xB3A0YuLA}envk};>0I$jkNs)Pr{Tr$qDb>#hU3J zi(7g6KD_i3ms{VnaG_(bdEx>(E3!7@9dn*3=X3GYGtYN)hH3}rQ$M*a%d;ni4|%Ry zw7FXIY>~)uMTtZ9*>s?ub8lNO_W{sL8A_WK-Ffyiy`(FWy5Mi1_rJ*5pp#e6Z?BEq z6MBz(ZgT%@Htn5p%>KK88Bm`2x!&1lk@Sm0hlgGWoBST{X7Wk8Zb$z6308-+Kj%Kg z`6(L{EXpFB`(_v8S~4d1U�m&fg}XleBX0uHz#qckXqB@%Ojk(VnXjv9iv(o#$Ef zIk|_nCif4|n|%h)KJj-5vtFKS!mz@IR!CTK*6!yy7W#^;-HfcC<8$tZXgB&Ej)Y;C z7k$2&`x-BNKmNUqSFicLM7gd0wN*anSJ!-zzq&Tk9$VhOHm*HN`*QL8vh=GkC~aB7 zJW3exmq-1EY6mYSTE2J4i_?yrHYRQ9Mq@?)m#P5YA@+BkUWNp+jTGZR-z;9a=2l*`rgdQJru$mxxS5YVC0@9dtMof zi)dpBhpL9&h~R#hw0Ev$&bn}yp11UU8+6hiN#DSNjR|s&J@2IFiQ|k@@7}c`Ve(DZ z%`zS)e7k;D!c$?>HzLc}n2m1#x1XD}AKj(TMkPctHg5R(bGTv5TredeFl@PZ>BvqS zm9)xv&kNs+fA5F%r|tf0iIN7(f2~6pY28v0`P8&b%Vo(kfc9SB8Pk_Bw`wf2cIOW_`MF(|$4Cd!Yb^3WdWjoNP>srdvf1R>S<_!Br z{h{qgqiOQ|Xf&^$X*D79`l6g$Q$JG2vfghyqtl`1G46Hqw**{|%(-c}ssG_<9PS^6 zZa3#Suf9=QBYEb+nM?e=qiJ=ASs~M!EN9v#t=XjGZ{_bC%slV^!Ki=!)?r1cdQIk> zMEuj=GPdB~%z7i^cwFLqhPH`ob4CBsldKOZ^4*tx$32 zvL6-qIo#_GY0vy8_BSfCP=24`9MH+vjr{-3tD|-2d3ChR;<@(_=W;*hxpK`HDvn&v z93_r9xVuptG7bfw9n0F_+7Ws7L9fwRJL`EcT<+uQ=K~~s@!QSwcusyqcAjNBKgBar z<~l8U{trHf4SsG@+A@=QLoLr>N_et9X))KbM(H-!FYrw0YkcXqS94@;z-m(R1#y1@4xz8CpQ zeDsfoJl3pIZ_<}|rq$MqoP29-Mhf3Gd>!}VN7hWv+Qh#`KgT2bcer-KNiokRiTr8e zJ!+l}yk5NLzc$*oa#$z4&H7R9kq_La&v$Q>p6{9Gv~J^Xv(8WF86cj`k+>w?-8|cW z>8>n$T)0s<&NH;XS;%utJQL`IF|c4e&mHaN_&mQ|=L_c*@VqTovF}&j0-O{ezb>j_`b?xR112?dxY) zcL%za^FQ2_=ivTGSaf-dO?P>+%wYjxGN8m9NwKv3QL8+e{07=TguTpJc6_A$NR+31 z4&@r?CRx%I5T=~}1cJf{_C&+|ZIWag?;Dpm$8a{>CI10w@w_XbiMC-@0#zzudJH`$6ayr!7SI!=ksJd z(fx%r*mD0PnF&MtxRr$c)o||1$jQM&^(Uf9QEt?$e!r-a0@Te*F8SHI!Q? zp15f453`2izRE&pPxPFew=v#1_lO<;cH0lAcWL{M9eziy ziD_$}W}V@*>7Ur`D)bu1&zyDsE*+n&ojz&Kl5{zLcQrO`dhYp&%%}g2;a%dn-y+tj zy|l@A-h&rveXpMhnSSGELe^aGZwj6Dq&)v!n-P2l7|Xj03-9ioMcuEs#@~f-2F-pV zX5{a+f(4FaSqS*UhnVx_zUH77QM_j`rO0%*PJ0a`=s9Ovkr5O zc#W;~wYXXu2kUW*c4B`KM?SPS`-8P5-)nsS>5TV5zTdSXq9#Q}MSaHltG|Bb^xuCM z+uvG~U7vAr&$jo+{no|z-v7<8^^Lze{i^j3Z~hauD_oy*d0gvVn_b&n54*Bm`K}^Y ziL268!vXqH*B4#f<9My-4c>#pzyGv;XGIcWz3b0iZ(DC$zi?f(`L-1u_O`Wvuj2c* zm5#ZGZ-p(rZ9PfiKXiTM;<=73k7Pdb$Rj(fNl`aN+i914F&9{rizWGT^Y!qp@leeBu?pinWotEi-` zx(e%j^LFjUiw_j=M#RPP7Sqzlu_!I${oVoc!Rq4uxh13ad#j6!EAuJ}3M7(IVKM31 zUsYO$l2n#HzPPYxUtxY(5wZ@tlB=qsDqfCFW_bG&$m8Yhb~ejOD|y|o)|Hm*Eh;Hk zoL5~@$s2qZFRrf4<&D~l_sBc1C5(LqrKGsx5SIsvO2&w}HGv`pP?X0je-|&V1QKn2 zK^ZSDmjd2fQBWyY2P+Si&wc*(#IwB zg+=-Kb|KhBx_EDCUbUp}KxuVl!HBfCIJdHJacPNGP$|6DmONDC?g9G?4(%!BEoF{e zA#We1P)jQg2X3APC6PYhECNia1m|Id(P%PnA<{l_3rd=$S zycHG{QyZHzQ%TqXN;;$9U{NK7T$w6Oigz>@Slcp_*RQo-n(5}lyE7jdZIHXI&86-= z)kVcsv^}@o>fHGiypy@2GRjH`sb9%!WF(aNxZo}9#rY)*s@#-eRZ(s+Wj7`ei8adF zE=}G9t2LXe59}$ZaF_0Nn-*G$K0nGz)$NsQ?aqE*w{2I$D?D?nuvjGSczHjS=hn?d zypFX_P2YfWZ_WI@QI^}irn-VIROOc5WZZh@=chff2!HN9MOBrH%L*zMmlf3%6uaZ+ zS0;!b`D7I2yEo-l?&ngum(t+4S}#3ATLxRD@V0j2E!fI_X?L(M%0KrYFT%4k zG}_ZlG1?=5#JX;zLk8Nc)CrPmCNsZJ<~}idZuci-(Cz+&EV|v-%j8HS)5BSaRyNjx ztZk{Tk|x5qSz2*uP82PUOpdlWh9|m`TMsiJQlh4 zl}h{Jl6IRbjh43;x~t3F0flZ_qtjFZTR7uYJ^{<1ZPM;8VKB)j-f>CN&I&4YqMY_O zqKz8U>Fg0ZL~)GKuN;jdpsNKs(Qzfq@IR)jjdW2N^4+!%lPr5&=55*T(h{c|Qa({u zWHjLp}V3qZ-3>SsGX(N?)@dDkGqSi=8(R!B1%Kv5H23_ zii^tZVNa?px2&pK-pft7GHg;}ROT9M8-sFXbzUCt!RHme@-j%U#_?wo>4@&4N_WA5 zvZ_N-R&q+_7VkFSTKA?c+cFv0oV3labpFvR9`UAZ^ZB-H-pwri$R_97{Yb{>W!6&X zT7DPBy{Vwm{_PNQwVpW&0O)B)_D8q$JlDm+r|e4q?xXA2_$~mL))i)BN;&Z|lrT)V`LrsGC*ZVcX|n zdnP)pJsq}Jq1rTSQQHfPG24f|B0a2Eq=)Sx#{^=$+iNM?N4}Zx>8B(qaFOahOurbj zOtm#*7OJ+VfZ^!9co{US&7;jdzEWn>24OW-V7U;v!(UW0ox4TlF}JLh zWXNU$)TyXpc9f~L#Igsr`@ESc=4z93l{lhH9ML6N8S7J)@v}J~F)`6fS)U<5TE2X_ zNP=W-naD(HTXTWtKHCb7eIY(pXdeJ+sT~+5*GWl95|%)GLOb4M`$7V>FQCzg_xqN} zKWpdq`<7Z8c5d5l?cBC=skLqA_MPZskZYO z^{E`@;c{$}N8_u?-7jP2bt5S(VwXUsLnmRwnl@bYrx9SGt+AOY&sLP0L=Al*Qq|{K`e*Ay^rhkg*|GFel_n z(EqI=+k3Bqd`@E6YR#QrId@F(p;Ke~v1j}n`Z$nV7OFPvM!1rlP++gK_ety*ubTqn zYfp7m6-%%A3{|GEokJciwkPRcWxCwgHSlO`;554zJfy! z3N$d20oE(stn-T{99`^t4sk7m-L}meH*eXo*;>Plu{OinmTuWol|wD1iVNG2*>U8; z5($?9qr#Xd3+=KDsOT{R$QY;Ns`a{oj=FTJEYRti(h`0+lG3mEuQSb*l3U)%;?%@hqjW(bqLf_L7HeGhYk&TC)2QD^{!m<8*@M4DJ6aNIT;? zJK4^7aPMO{mz8pUK^03AinG|x*L7S~6y%jwNUs^|Y7;*YTQm}s&ar*|8<-cp zizK3hMR^7E6!{gBfJSSq`LR_N*p=h$lgrAOdn%Z**<|x$aRFY%s-n82Ldr{s;2XIZv01%(wLOky+2&3?;ALs-w0w+) z>(^$;K<8ZBqrC3?kPAesO%|>hewhgJ=dGRG@`0P+JeH+l)4)vw@&rI90&6~;? ztM!hQnx34R%^+%15457OW3+WTzL8T|bBJB3e|MPv5P0Uy0XK_J`!<}m(bp`RtK^`< zdOTp)P?>wMz_@knwV!p{yxz<;8Ee;Wwzj0N-Mrpwk#r`2v<<1L=u)<1kPGkH%uLJP zZOMMb+O%QwhE43eQmswNS@AyRuV-a$+>oB_OU~Sw zlFS_D+u+^cqwcnB@%lETQ|0N~(z7?M-Mr1(xOQjCmgJ1okyLHp=G~aQ+1j+$mrOR< zEb0?p4s8?U7VZE0{J#?T|DXge69>;q$fE>fKBAY;*{IO=Jv_`PpJBc{TR+Z!HxKj3 zr(Q!IR~Yk2yGe^(JQu-(8{_;(@WTP}DdQU|eO1POnDmA6U&e!*@=51Q6yw6@z4rDe zmBfegA2CMUXJMQx}l@I42HIYu8jwuowG!DDJADxGRJik{(Uvg+IW46%=>z zP~6RiTX7ewsaS1TZ(x7fquupGan}daad$>j^+5hkd+})~;k80>*95oVu2EARhf*J9 zP~7D}-u>a-rKz%@*sp?ix}cpdxJlBbsiwkz#sBbX%Q_DSp}6mbQolXWi~DX()d6|0 zh__u+wZeyxTQpT8+<@Gmsp?=Fa;>JSg6omXG*v!ahn%CSe318(c+)jiDojQ8XsSfG z26>65ng>&m-I^)}CL>2{st73U%hFVrAG9nF_Jf+r4_71iYpOo@Ao3Ya)eRMLm!@il ztB{*C)nWKKWXalM<@P6c8P1OijAU9~L8n_&} zN>llu)Jq~fO1_q8szopnck!Ak7G9tpqoLG$1YCx_rKyHj>3SZ3V&4y!Vt-Ckoq+@R zKMkclv_iWb!uxR7sHy7Ve_~$)rM>Nf;@%6F;4V#5r5gKqV;=)0yh!+2+(l@rq0d>? z75ra>5>6i!cRlc4+;wZJQ}EZ=H$ib%2gO|tT#UOaO_c*b#6BI0yH!x!ErE-0w@6dX zhW``$NGR^EtkmuX;61pzsHx7te(bxUwaNO`0 z@G^gSE%6O%?)F28?<^E|y^z0OUEHMYiyJj}*TH|sz6OfBVmJYJtF(P_lIHGscoF;A zP}~jk(iw?wNOSif6n#IGdO8Q^Q7&gSRS&!ixm#0p!MVt%HB|@vTl}4ZbI`YGs#fSm zZqZZ?Q1Vp|e+>CGS!)PJzz^V+rFy*VhLVmh7$@;)st))u`gV9H`cs;!6^4;-eva18 zSFz^qd?@i{LCK#N-T`N8`{G#5-B<3@{w{0o_QTH*UKgA#@oTCMcsp{trfP;#j!l}o z8=%Bf3;Dg}#f4DDf1lAWg3|BjLCIe%e4Q2djg;Jg^@D|)RYpNXhHu@|mNFhSQL1G*vOY3As>H?SfO0 zvow_#W-yMVX{uB>1$&RCS_S(^*K8PtK2}rp-lNN<2Tn%ct*K7INaQw6)dVLYH)^Un zI1#y4Q*gO?4Q4gj}Pk3gK1cd`;zpL&)ix$^(Ccyh>9g z!v8{EqN(EHhsg6ZRSf(!ak67rCyx(El5 z{hF#5{wH#erfP@(fqY6+wL;#%=WWqc$KgffdQDXY-$5?ZR5|c1+GjfaUvL$?fPIpt zS_01_FVa-=pdZ<-siNV}k*8{^p#_A4d__}r!T*VTT2r;be&kk7)d+uv+@PuI;J>oO zErb7tzF1T7d$qRS3(uiX(^MY#Hu5S>l?Z=|yhKyE;lCiy)>KpBS>#Afb@6V_2ax@m zsu%tkxkpoV!MBi4YpQnGhkQy?wZeZ!ZqZat@JGmvn(8?GC**ofRSVA`AJ$Z>;17|L zG*u#e6M2cIiid9?&(l;x^Vzo}U(r;4_>ai_nyLf-0J&XLHNbyBKCY?CU=MP!rb>n1 zNA_r{82CNpXiXIfUq_D6RA=Wg{v!8jsy6r#onW- z+^`FIwx+V+-y;v-WmEORze7HwsZPUA^fPCm;`M zssZ>4@NtRuaG^O$_-^+ zv!JZ+`{w9$pV8di10~(v@LTxn(o}6m-wJo2Z_!kBa6A2^7S>C=nyMI{LN3%)yWlsG zvow_#zKoovsaC<0$Vr-N5o|+_*Hp9NUm(Y7s;TfB$dQ_A*lk%~M;_8tgRm8OKvVU@ zuOXk)RA=Bz$i14X3%-baT2q~ZEy!(}su{k3+@z_F!#_u^*Hnk$pCQ+1s$$rTT&Ssb z!9PXL(o~D!SCQj2)o`3;{RuKJ#={4#QqrkV;H zks~!#|DBfgCFFCOsu}(fa+9Xo1y3MnX{y<<0XbGv_1|Gxe<=PnRU`Z#$PJpx3;zH) zO;cI$IP&mpo2m&)JE@1yqp#CctDt>O0G~s@NK?6?w8Pm@+FPuqiiQ$S1bh~~rKtvQ z*ZO`a?eLtYYKLD$KBcMZq4>{-UqGLusdmB7BWGzUexSvwLr&LJUU&>SO;e@9qsSgj zwFFAI@$eb+^E8zkO8#PCE&6Cp6#*rG7CeG}c$Q7Ydn@g694b?;Q(O7LLQRze z4S-F9}xTE>Tm(!;jEM!>e#0mhl?4!daAGBm5b14g4ughf~n= zE@uA9m3 z$6+`b{lE;?<**0-9sb(k9mp+E+D{Gq5&AS(NqSN>l?PTJuhLZf*ds53WZu?Pi(naY zyrvqymHijt4I1_v=D=9==}^+=g{AmQ(^MWj0_pzGFGWy|LEGrlNkfyo_#l0WqpzqgI zz3@%+9Z=d&8M;2seEu3a=NBk z1tq*B_z3z$O%)Giza0&~2Q9@(~!>@aLIY&L8#tTU`JEH=zB^ckiZt}7Le`K3v7_b&KX=o8^d*guW)Fw)s&c*?L5 zZY8`1O~vnn+wQ82zRc+PeS2G<4l{7?)l{pDevx4el<=Z8RfN$G-(=rI_YP^QK`7&0 zAC&pC)v(^M4yF-Kt)}uAd6kiq;AYBk5!{Hocuh4AK7`Eg;t17jxCJ>@Q^i2>7Xihe zrKyIe>T~-cO*IH59DXmu4(B5D^53tinxV9-2DkzJaZOch^t+5c%jkVj;!D?5iBSB{ z)AnAsrkV{U-dIf)3B_H6w)a|^ir<5_%jb%wx(vnLAY70C0Zml}dC|$@U4|Yg^Xw|P z4tGhKYLU^$!?oz=X{uPGkAbP^qcv58(OYm0dU<{Vs>@KqAA~9B2Q*c`(Vv6K=;e73 zsCtdQ2YS$VYpM>TZ-=YVpVCx~Q1UY{S+mcu$FRe&*|5RzFr1FRd?RNW^4>~2yj6xv z4CfgRMrwP%VT@s6_X_{)4(I-Jkf1;*}H~M+-e)Mil6=U?#Q1UfZQ&~no9HG-Qq^Vlr z?UY*!OvHV&rfPsvp8V#TwG4f|raEl&HE=2VDos^r^!bn~^5$qNpV6no`_OwemB;8; z!6oREG!?(4W5*j0KZ|~zriwNC7{&sCvXvZ-g6{W{7$E> zKL_tYe^yiV8hsC3h`w7>br^j+OhA81Q?(dg z?5ofZX{tdeLVGW2s&htv7T$@zPgC_6eK%z5>+RB1?M8nJvQ_o+{&b;gHu@&W z7SP+Msp^fsPKdr%Q&ky#8DvZ2E!I>yM!yTPCGloyDzDL}LADs)R85s+^oei=`X!ob zp3%D@OL*^WO%-kQQz1)fZ=|N;o|9cZ+?$gnsh4L7go-_lt><2%EbY90O~u;5)^iU+ zmQ-HWltR^Q^xQj=7y5X)XC+jpjJ^%BwD7iSswSgvgiQI~22E9G^tF(w)_Yh}l^K08 zWJ>cEYN}mEp9Ps(ygp5pX7s6$DZuN|REb8v1TtiM7ilWD(a(kq)!tZ5bPX2D}{2=_xZ@_r?btv`M3Z>k>2Bq9yf>LfTLMgWv zIE4Qf;AQydZ~*=p^uuO&4*n_ZgI|Tc@K0bjY=WoZAH#O|71##93|n9$Y=U2c4e*a( zJ^Vvh3*V=y*BF+;6X*+J1I#h{Ecge==|)b2{{z_rpNC2CI9vjsgYocL=!Or&Shy2L z!z>sHcR*mr@24F2Ke5}8GO^`a!yH48q1(_h^b-!36T1yt3~LQ@3_XT!Lty)x6gDqD zEUd@X>vDyia`m{vFx$Cqb)Dk6)zyYsRWm89DXckcQrM}mcFgXu9?Y|0=P(Dt1~G@j zYQramHH0@}wuZN1o(}KAJQLoB=?}k%c_n-Zvp(W@#H6t1h!)KDhz`u2h+fQd5&f8h z5tlJ*C)7=t6xKMQ3A1g&Da@`3-I#q7&SGAiFn~EUVHorH#DzQ#)euD*qFOOKqE2J>MxDXzkMd*IPdPpX z_fuLh+oyD3_Dt!;JU68ub8yOK%-X4SQ%T#c?YEM)TYE6i-g*vm;MPIR;ah8GkhU3( zn5{F~Fi+3u!mN$0izQXJ4c&&z+XgXPVq4KP#WrJ}i|xnki#>~Zd}hN;+{_%r^v}GA zSv#w47U`ck%yrwWQ(U*qYQ=1t)r{FSs~fZS_PX0~bNeu6!|cY{l*a5n%*(T{U^d** zct=EdQ(SXgOB^G6;jYE(nbV8eH|H$o(41k+ zj(Ml&k*;|cG0)BG$LyPT7IWb4!MjP(-OZRo^M~hCqVxTjSLO|2dwN0F0?K1S8)n0T zM$EniXE6sCT*kZ{esFnbpDE<&@Y8?${;2WH!% zQ9d% zh2&4XDCUZbE6CG|e$2iVXEA$MoWbl~(SzB#q7AcQMI+|$^4b*<;TM+=EFWA>OSr%N ze*E3vk9jJoJ&96CYR0Ths>7^%p#A}TJW#um@K&C|Z1`N`=WzeIe$1v-&8xUx)sNYv znibcoAG7|!;}2454|ZV=JUEEiw7PjUA*?=wIkb8hv(0nLgL_Xu=747qbI3D{S)Y77 z8TZM}m~F|YFi$6UVfH4U!917Tk2#P$h&hxzj9H&@JO%eD&6sT|r!Y^abYb?UoWVSo z(vLZiGKe{xQoDw>x26@dYfbl>i10IO`qrF9>Q3!R#XhxmEg`LK#%x&Ih}pfi2eWVO zSO zZ)n5p+i(`M?xFgJh~c3|%jeg848;3Aky=`7X_SSA9 zZ{9Ah+r1r_^_z~PY24I=*|Mn>^VFtxakHrlvu9H;rhn5#%=*p8H9`7&n1mc}i_wWSU7^p-BnD_e#zkEb`J6MlLt=Be~{%+u*zn3vP9VAgLv zz7^Z8t(d2_wqy2e?ZxzOy@)xqbr|z_MneW}GFmZvGtOXM&bWfvklC0?sxn(J+cP^b zyE3~m&t{&(9LOBR9L}uu;nLTL+3IVBhkV1BjoX^G;bvP0 z=9z7MnEq`SF|TYJ!mQtZd^<62Z^3Ne-htV(y%+P`_I}L4?UylYchv15oE=S=Z97h3 zcJJuHY|3iRA{RSbcM@0DFxLZFgP7;C`Z0U6&R}+D^<+ha_w4N5c?MVY42&w3D z(Ul^~`mwW*MTFPwuit-se?<7X{r&s>`>C1Y#^Q+Z_TrA>)3EM9{ej~LD1igL2hJRz z6ib>)NMT7wNkn*eNl!^H^p{*L8Gy~DEv2oc5#hCEb!GKsxUFugCXH3+swkzZmMZ#J z+Yb`f6CF>G?k7%R);>{(dG1jEA$%Wd!tAX%Q$zo%smJVoyytOB z`rr`0t~@b>%Z4W#pTy0RXE2AJ9L8)t+;*6p9X^XWba)uE>8a+YDCeiTG0!~JhuQy> zA9L`j%b2xK*F8-*Pd8$=Jl%@f{&WXs_tQO?eNUgo^gn$O^YYVIFl&$09l`yPM$DEY zt(fgcIxxGB^k6pCx|(ZU#Pf`+?-|(mjH^l5U+eOVzUio|`6#@2)HNV%Kj!K<1{>;J zjdifA&ebivSmzoL4%fMAKM#+8-qj#%`n;=I*!p=_o3Q=!t`1?>=Uv^l{TE%$UqtTx zg6qr|;PB^NwO?>iLtk`V{Gw~XwyJlXsmH3j-qj;)f7aFUtjN#0dRTaFWpg`?1)=k) zWC68jjW_aK568y zvl$h+)5zyJ1Q7WRBmas`zQ{i{a*Rpedq&=C()U{3hJ)e`C_O!N~t;(&sbsB@P)R{5&In&ZMux$bVqcci71P z$>D&wuQPJ8N#AiJ|EWpeSB(5?Hud8E&yD=BNne|hpEv32Fmkg=-)lzxu1VjUMvgY= z`>~PtnDqV3$XzCVmyA5ar0+LIE;H%7Nj6%1-ZbfpG4f|k`sNuq$E5EbBg;9vHU()&;3UJcaxu$M!w&K z_n?tanEFUH@_Qyf4;gu_sgKP@e!kS;qZWOn5m)e#n%^Q6qQK_9VQBxjOuB znDl;byO!nfRMP*Tkyn`hdYfqGv(VUw8~bBM{HD&ge>YXf zch-b=*~Gus$gQUQUN&;x6z%@1k^jKN-E>Go3c&Mnc(h#e)+7Ls4NFYFhqG*ZLwm32^R64cR*4Wx9 zP}^~)IF;6pv^pKyI#abw2RrU;ESUdiowd#*Cjl;?CHyBRJ9n+MpX<9{ z=bXLw+1unPRz9zr{3|Qp;;S`%S?40>##oTYnSA8d&yG%-m{`Z>uW7Z#>Z0_e+ z{yb*xXPZ6~%)QC_qe7Ga&fKpw|Nmw2{lw&vmj7Qi`4<-cMU%5Fzw1rjZ}oq_$vIZO zUod&DmG7CBewpc8VeVhgSA1`o{BI_I%jD-R{;3xJ2NvJm=KfL3zn)vq^AUT*pK zeUm?9_2WNHE;YPoS$elh{_(t=Ywbzy6Qb`Blb^TnJu6ke=S6?qCz<;w!~4}a>i*YK zp12p7dru4h{g0{pEm9u1k2Ux2iT_;2nyc0QVauO4KB4m2gVg^R^S|HfQ~CAkey-FH z;=9P)@38zBevP`HWA)=h7T;UuKEm+)O5!8@C=0*JZ{OEx_)D$+?6&Z~xBPp>rCEncpqf(lw8BDet+NOCFXyp$v?69dd~5@J6Y!?_2FETZ?^m>H~G_&-{^az>H7@# z2Fbr|b2a?M7XL)UcemA-A6s}?&m-~QX7X06FRz-sg}CH?AFF!>xSzy4M}w@Up0 z&q9-bVfpuv$!A-8e%|D-57qqIWBJu+{{PqN({C)lhnfCA6aU2D*LOEO<4k_h^gsRM znm4-lZ=R+1eXEb(Emil|B|Y5#&fIeh|F12+O7s5*lV_Pe z=iaE{msx(!H2E3J&l^oX#malB$xAFh8%+NADT-%{#s3}i|5KCO$Xn?@N+xQ03oXB1 zHU0m^>fZsA$6EfJZ{Y`6eH?A_4vRl!@@Fjlq{)4TsXl{EF1GxbVRC;f|9`dg)>!_` zHuoP|d1QWA^&MpTo@{cR>3g2Z-?s9;(BvmAe?MmOe_Q_EX0oimmiqQJlRss6o-lc* zrwUacy(@g#u_im{VPg;Gbko-gbuE{_5+ozS^P)l!~z0baG z`SsuCKFsvL%-m(2qv(H+$+Jx!Z1UNLf2)Om!rIRWb3e=Uf7skVY2i1R{IdD~uE~q7 zK7Q1~Z#DfsVe+M>-_Onem(BlnlXv^|$Nax*_3J{D8!SJTn*8r4E51*ee6i)v8y4R~ z=D*eCedfQ1`IqlGB!3<^_p7YF6j}K+n)`Joue1DGVDk42&u2{TY4zo+CSPOqWvj{m zVEVmcav!TNxwEu>RQFYVCYpS!;HN(PlhubWS$rS1`_G#e-?b)RVe#E$@;5EMdXt~B z@+-Ia&oaDoOm4FGReztSs}28c=KeQU-k-Pp`=vZ@@V~lR^h z<=+Dq-)hUhZ=3uZs}E0^EbITJe12i_XAJLqCO7v}_o9zz`WG19i%ec`coHfs-aOqTCpBtL3RE;jvcGx=`Q@3SWV+R9_S$wdP+{8vr>Yt!%VO@77N#}<<> zGX0)0`5UI+c9XZ8ey^DPs1;eBK8 zS@grN$o}=+ES2RsQm%(+OXBYDd&f5rp6u_lH#Jjbf8W14(h~pAP%q@_Y3@%>RQWn{ z--=#xaVDj#dkQXD9~+e~kI82eU)23~2vPYO@I>YE-tFa6!4q|7kBoNm&1e#J|2_AX zs9Y0^|DR(1Yh(UjjJZ#b>GNmWW;FgyvG7;M^!t3w{a?t>X!w7Qx$lj+pB-}_Mma^p zH^=<{O$_gavGTew7T+gg@Ch5{G3MC z4*!p0_2btuc}1+e-ipQdVoblU#LEBMvHaZ^)9?IPd3-OX->#TEEf&9g9}z9j>X^QH zvH0JP`KSA87hd9u#`juGelI4=cMwtkUx?x3+pc!*xv~8Cbj*EMO#ky@{#l#bF237i z=`D`M_ph<|hQ#7KD;A%8hZDs!A?98k!{0aN|CU&N7#+)>J7WIV#q?PoE1#yAf7bf8 z)9<4(|38iSzb}UWJ26?->qqgP7sGc_EdH`s_&a0%e-@LAWAVQo(|=|xz87QZ$@fZ8 z{XQGZud8G6-x!lGh{ZoURvwqeGhbR8JTQyy^noT@WnNzXq7ep;M?8-Y~teu@5 zo*T8B?w!jSwJFo%ZK76!$Ug?5oZaOEjXSXovTbG!2WHiqUp>bQjQ)9tUKHY9F}-fl zqPoQw%Yn&bK+)iFojSYPih2$jsai0mK@MOYLxIL)P|du9HDyD46}{se12GaLu&W3; zFYINUkQ)gbh>@^?T}4g(+?x5me@-B%=Y#{VvMFfGyfV&Nh+2~ul6#?bka6DmD51Sq zw<;}c{o*B@&Dk)o%C{gI&gazwEA@E3dQAlf{}`J~yrX?QmDIZ`rUzVx-igNp--*4( zuSX4g(R&$DE(c}PZtBoRPB9K$Xl8Ny3Qe?(UGiti>AjY&$6w+duYB?1<%`Pe^AOkp z*_xXiWf}T4OFeip@NKqwp0T;rR*a#50xuVP$9sEO9bqY2Kg%m`@%n}!fSouR$@svj zGrNullm^}-Wu$g9XkBz0nsT-3=#`t7U+t_bv>Rw|7gWfpwYOB>T7T8z^13A&Gx1h& zqJalfcuq(1d(2h)oy(z-bH!umHk zT~$+}?)teXXI^*W;9Is!&eOwHgl96BTYLdN|6tT08LcWEtgZYbXMSpkAfn5tmuf|> z)N&HJmqUSbsDh&o*K?-O!iuT|3oGO_UC(P+I@7#EX{0TL4i~SelpBZjtJ5oMYvm*p zO{21*t|s)UjfU2uj%a2gBQ~Lm% zoT#W)l#^y)J1f6#h3DbMBJl25%NK=#f~c>Q`&QIN>@p)+K0y>X|3>VW4FRi{rEjT=LdUdpoPhlU31B@xUKEk+r8q+Sig}#Ku89O?Y&jv_u(|}bs52Mq20L@=)ULw;z(ML| zJexJpop@!}F7<=BT!QYi^#B-+#JbK)rv>7=@l#?r?GC?ix)sSdvVAnLN?@*<3q0@n+VyE zOO6Y@hHN5aLoPWs^cu2>kPW$HGV~gG5c_v*F?yMTyjF-UL3NCkPW%y_|R*}CPFsklH)?J zA)5%$AsgX}L$4v52-%QJCPJ?vn~1Q1dy*~SeO*H~5wbW_O5^=f zg7Hf!U=x!9Hsp#Y29}AC4Y}lmz`ZzR6CoRN$?>7rkWGYa$R)>xUPCq!vLTmbGkPCO z$R=n z5wandOom=VHW9KBt~m4>vWbulxnv^r8nTH98@MNn0=7706CsN;r8L$rB^bYy0yZ%z zU_-8$^8?IlB4k4@IU#T_4%tM=hFo%d=rv>$AsceZaiQ0cO@wU7B{`zq#}cxMkPW$H zGV~g2phO3ivqSdWD_BaGo_StwbN%^R#XAi2Rs_pCME@}$E8u+L4e|k0h1jt5wand93OfO*+j^OTyk9K zHDnVZ8*<69q1TX2glxzqlcCp;O@wTOD-OMeY$9ZZ(|$1#_)LauamXejZ0NNpV2eXG z0c+QpiSvsm7KB$($QFz%N+K0|RsN#T3!!-)=uZoHr>+5|1T2V1$YWdXA!q>x3 zUtCN@&r=){lRFG8$?&vZas7O~`dh#f$*9N`ja#f`vWU#CYN%Mg_!j1cDk@56M7+BK zQ>?%&uLY+-e=M9|ZWm=SZY;I4xp5YOIr;bYPLzG-+!KV8C0G(j{Ov(zDV>|e{CNR7 z%W|z*{L3Z0{0kIWrp1)?5npXjXFtEcZqUYe=by5%wPNtbx33wpaqm^9ZalDZ=*9zg zWv^q|%-YIardTyR8Z)?lg*f!YhBaO1t z_+OA+vhS3OOD1iA4i+kD;P;ALS3cA=zA%jcQ~Ae!-^QVhTW(raa^K9;dwnG%W9YpL zu&cqY@buna88UV9y=_msEzMb3>t4zltm!{pvgKjoeW}f@d->09-7D|7b+5i9M-I64 zzTaJ0LpHsXbxPpBQvClB|F7f!cldu<{kOE0Oxpd&l1aH4y{^c;Bloo3e=eEy+Uq5g zMrQQBqVV)yr|o_R`F9cdw32<2HzPA#IZbISd#3iM+T@P~^X{xuH?eoYzTH`;ZEE>e z#>PoGZrz3fjawefI_2K$vIgHfB`f#!dt2MeCEr%&xc;lMdM#Y|PFp$47bX0F4O!Q< zQdQQDq`c&RN=EMM8NFP2b5{PQ11qniY(GXBUrSkELz#bc<@uGVq@f z#NqMn!VV0=wqDa--_32*IO1d)i!QaMzHV_r-MoS&-V}2Is|6Mm$oTp#vS6T~T4ra( z`RtkknRKh3D_ld(qU!oF?vk2i^$i7cs~ec8$DyvOYWcEd)r+gD3o085mekeP$Q(ER z&m)M&TTs80(S7`uT{>;_+?sjw7}OX4UIJsBYUEEA)Sq26c5a1CiRcVicqY7eE3nA} z^;K4FQF)$xr-FHm!uxvb9mInxhsn<^sH`tI=j{4(3Mv=R^%(M`-IYtHHp1H~vYoZ- zfjxC^^nMDIK4^3+lrF1uuTpQ{Qu?*gJCRwcI{G;%OPfb;g;M9eeMzAyTfbBK81~}7 z$^5T}(nXi9Rhn7@mGHMi>3U07C`~Pa(hqw3kcu7)kWsC4u2OIRQL&$ijPdp#g{H>x zBjL_7TBy|9f0QN3qlY02-GL|8+|_PZdP516CBLI(9U4o8N6S8qBJU-8M7|wL&pw*= z?a{NK623sG_mv$%vU_9|oK@H(YkSs?teshG@6)rjC)?V1+xD&PxgNf?$NnB{jMJlq zU)l1H@Nx|yuzcU-U!?8rKT~2JCW}l4NV&L zW5xX?lY2>c^2Sf^U32%-dtGF7DKLG`M2@aso{k(*U(SANC*L2Fm&fEMWAdkCvcw&Y ze|}89Iwnge7WJ<)SS-i!)@8{mBYEPowluwN`C?sB9P8~#-HX9Mna5qTxgvC(JODBsz0Ebq(q*c*M0T~?JW)~DF`^Q%-F_EB}NfPXG|er@r% z&VTk;;rY+FzD!@_J-<>_y(``4*P=*#epoMp4$rS%S9;LrLqDFan!J00$fOC$zSH!3 zamxH55<$%Zy;PQRQBPPn{M=ZQ2fZh#4CP|=9-h~Hs|t8)obqBvR>c+(Xvq) zjb*QAyGHD#IspIo+qFKgmT|_b9dH5*VVAS6mbt|%5a9n45-)yv9vV4H!Zo9VA6E-;H4l8~xqVV*YL7bu+S;%(w@+{FfV1dQq>)PZUch*=|NL(#hg`=W={qzKnq^}q?mjNwT zyRsAuvzIdzslN{K%l%0%?tOA~aBspAFWHSo=NK(9 zT3{67`THz1df&NIWMcK}`C?@0=Y8Dgne5}Z)?{CAQBV9gLFKxbf02E?rQM4hT(Yu2 zdtK=~rdQThvBYLsv^N&}V0mSGuhW*xQ&!kBPZz&4>K5ywk@kKsUaTuD+S~g5p>=~t z+z*Z)KViJS@<;l?d37va)tC8rKghF>TOg%YpgADa;10GIyNc2~ypZ(oze`(h8q z0(x#dFg;`4-mI)mZ^`?<+NX)Vlhn|Dod z>t4E$_v{J0XD8)dP{#`PlJ-`@?3z+C>7@%xCcQkNWRk>{@A5}^_DPj_XBhQk{JnG+ zaZC@xH)21^>uS-1`&uUdSv?xddNe;(_UeeHvh3!E%6>OO-!Yrf!yDT`=X;iYXg1Yh z?d2J6-DMfQ?(N;LWM7{lC0p{iPmazgne?5rpB?%X_r>3HU%K+v^PX74z3`#a-O|&! zhh3i0=iZUX^4>fGyGrhlf@hbFgKP!6KPIvboeNvr9+!L3ft44N_cM4`pT3c@@ZQh& z*ZqKZ_6*+LGqG=9Hg8cjd$Te(@h-gSZSQ?KYm?;7oW5@JJIFiQ9x8kBw8pX@Kx;Bf zCQW%WYboEPloy^lV!#Oa(=%PU*gc%($`@oamN97ffDsv)OIrp%G?YFkf6A_`rR+yv zE|hP%)KC8p1%C4J^9|yb`)V)#|D5ffzevLUIBRK^#0}jA_2NEhSP+hS;N6Qy5$4gk?enC;fpP_F0o96I13^5Dndt(iMC`EpjT zO|?1hd7gdV+&$r;q2J&(w~(@3mE+$0b5`CaDO;hGU)jt|H(@0I!Y^1cVnEI9eP&Nt z)GgP!k;}_6Mz%b=rfl1@k1kvN z$-=MnIxyI_XX${i)R5 zXFPr0zUD{RWuOcC&`xvnb!_ik%2&n~rLCVr_(45Nw%l`a$-Xmhy}9*i`jtB~OL!I> zNIvN};%dgheoc8jKFam~E&23q%HY7tX~aF1^q%(8+I!VMlMZRo7qm_9_xwx5@AqT5 zjFHLzhP#HhZos{Tv8m=W-MVkUXEa)0fGqfU_uaG$-p9A$u1|CrYkQcoB#m_kR!&8S zpLpp@*mvju5bPP~l!-n0zmk5HF{{0%%S&LJ*~2}5&rJ&^P8sQzw&pxEbWp~p?OFO3X_w=3cn-#24gPX(t!eE|eI7~J-}ZKI zzKm|apk0>Mygl*ltUj9_%E(AuK_68^9WKnsy?4Z$BUl^e^6!!Uf%uwf6V$;Mq#UHp z`0+a8eKB*;y{9uaDLOEQ?)5Q@RNuRc`s+&Z>(0dWqRn<$Y;+_6+VNdarzo_VLB{+8%$1`%An% zkXCTz!pVR4r8zKU@??%$9pYFh=UwB+ui|gK%eyFRc+m*%ELF)Nd2CuUZ2Yjr7bX|G z>}*;n*O0vI?CiWDj{hO>e8)ezZSp)E^75wS*~=HayqZrD|K5eeoFaT@Ww?u~>Q~my zzsTjCoX0iA4Nn#g%T6jSN)C4uCy!6ADq7*DWQ`~id02MVh$M7Mk_d=rID*LhTvRlK zJE9~S{%1=nqMw&M8o*1+k`71i<&pl%l`FG`yF4P8!cvtf^X6SNZ{8F)d0tjl*1XAX z#Q1sh=8YfWR$X$+@KY{Xr8K**=UzYWcE|D;f+gK>Q9p0ETTz6X=re!b_~GN{&3Apt zo=fJfa#gpiTD7pM%2my;TUiesHomH=ip@eLy=>wm`Q(MeCwj7k8=lNLd7|JPt?}Z_ zHAT`R`aDPIg{y8MJ|ZD5u}6jBZdFxcNGa(JC*ADq;ZDF8%_(6YJh_IuT=5)UZFyea zh{f=tizwHe>}>E~C~`83^d}Z!jlIZI$fsfBvwcNd`bEKJrWAlYHWoI>`Yd^x|8Aym+1$zgO{GZr(+CdFM&_Nifkv zf=R^!|0=1hWPk?=d-S24PtE(QDQ;q3UQS-#L(7mdG2>euio(xUzRy#1%c#*ZIH+q0M^Py6zxKmFxtlP8ZKFH-;< zRy!{_?^%zV60QQ^tiowu{pMG{Ic@SMJtxj!BNkyv7ZpC~hS4l#3?Et~4stH0UbMrD zva{a%umAe*9Xoc275?oSQZzg_FV~C5kI#|%Jd}I7D*2-~-hL|!x)O^$S zr;)awE%iG`TH>lI&Sq9RY&=~8GWEoxRi1T~bSX4|q`RH&D4DF+)CZc6s>Z0eiQ zya6hYH#x`a8E8jF^9BeroHGoi$Dn=BtE-d#hL$~H%7sbhOomLJH+cy9D4je{si#lX zMeb=tad+c~(UdvscRc;)_zw||{Gq>EVw67ElapS%8SdFj58qdsx3st6$q~sBS-dJp zyZ@x<=h06-J$cUU^Y(*ZW%Tsy(bc1zu3hwA@;x$scouhtVTAYk1;Trj@bf;IJz4O1 za*p*2!;D^3HH<1G?%H3RcJgV&OFuBe>lkFvnvMZkB7{=GJPLlbH`%K!QUm5pzWI}j zCub9m4uD!UQ7Pe6_Nmx={R0_r5w|0;_hhX@oc-mvqw%=c{t|n+Ci6f==hDIbIlT>f zW6}(zba09$x;s3rwX7{sZ#PnXi5qXliiFgbgR)O zqm4%A7%eecU=-qo%gh{B!1*GX>oa>~ZRq`I@5g#`PE@a!UT?^t7*{okMaQu#MCQMl za*}IdOqM&Y%zyghju(nM8RL%|eoW+u`|Ozel_FC{{^z{$9_U+*3v{^&S&(%k)a zH{0C(cJjX_`|aX~Cj0H+qZZz8m((dQe!m?^IY@eQz#!L;;4#7VxtJ{VBI^EVOkP9C zsJrk{`MsF`|B1=xlRr`acgNgMiMdk;+xb5u=Dt1VerYVe=9oM0SMB1{iB|ovb$ZRk4W||&PP(x|XHHY+&J};5)AR+E%fyDSey3MfEvPQ7@wdQ` z`P=HneE2C}U{99~Lc&m<6(L!s7MNfAu1+=+c9T#uP7-KG4~wRu;%^GbVc&_UvpC?|4L|B}RTuz4lvu(&$R+gLpQywVCL zLg zqk)utZiU^$!-(8Nx!OfpjD=uPy%`sf)ks`p_?J9;j+^}xPzfadR&(<%5{e*adq?M$ z{Lnl{xI+Fd>*3PB;^SK~CYHZ{&aY(d=y7k|-=Iwyj8}dt)5cyI$I>xH9oHhwwcNKD z%WFTzEA}p<#9I$=VC7Sctum&JT`%kyZ|u){r7dVFk0;)kC7Hz>%a6VBzcl?1JR<*- zk1+npE{OjSdNXfvP)zypyp=rNz<4=(xAot^noX+6+IlaK!}&DNo%|7Zf9$X&D}Pk( z?Q^~-YcAsX{Fg{0I%YcX_LBPw`5)mj7XA(6*LxY?d<(s(^Z9#aO$cMOhscxljOUNK zCv)I0yg9g&e)d4_oxM(T83X%7$LIa>f0?-u@z4CgsAlE?cM(7JWqq`+v`x==K01Dn z|4lT>V7=`|<`Jg2%-0KwKJ~!J#!o%4inT1e(2=>G*LlV-A90d-)V=53dTuLwob?}(k$n9%Fv_|b+FO1TW4*nZ7kDkl?Mo7dcAC$;N|~(d{R#QO z`@lMhr|@IR)_T?v(=F$(8t#@&CCx_sdHOSkP1)z?_F(Sf8*U?Eo|m=7_u+o1ykjI` zK+SMBpm(pI4W0Qg>zGML#@Y`y@8OTHlSkfsrKH_y-ZD=ZPcrVNAj6)(fi=7K6**6TALyX%3=w%+x? zLzLme{B-hF>L2}TzRYirX1?8iP{tJ*V_$!rd6H7fn7WfM>#nC{WW4_KcI7#?Rn~>m|K>kLc}lu|c}iI_ud}Y@ z+hJL*PopgFiIrvYAZ7Vu^w7BpoiFLEJlo}S+vmTqtL^hz*CPE`r)6?i4Dq3&(VgLxj_#N6HvEEr9}&NzyR0uc*jfhBX%Tae zG6(E>c*|G3Iq9YzZnMlq{)zh&V-9O4R^DCq#D~7UY+s+oG9F{rGHJf9MecXBu@>)s zrgQ1Jw?FZf+yird)yw5xk=ygKKDl>f&(6K`)YEeBWM1&5@e^hK_^HfOUO!#t@NOD6 zF&YPXdr(~A{C2oj!=HEe*KCjv&24Re{8yuw&f+a?5g!RG)8+Ox^#wJHZ?3GZnd=r+ zuCSM^df5vrc3q3iJum(^>y3AbS9=_9_v|P3bLEGtNJ)qA>(&0C?#~>?eeYr1{dIhu zJ)26<5__ONnEUnSE;u^F&;8%K4(5KA<;W6z;k6@vb3ad`>EOQA++}@ZXW{qA!#CH# z!Vj|;3)NeP_-B|6^;fPA<#(;QA1wa6&3*Dg^xJALW(N!ZhPfY1zkJ3c<&t*;t`7R0 zXYP~Ly#qe_4ey$95cfyT{b1>DH}@5UXPUG_{G~QtH})X$H=6ru+&e43+*}RV0_kjA zc^mHMGH&?{lzVmQR;8&Ypp2`QZdRIl6iVilK4NqUw2*KMOrB%(dMJ97UaK@UQ|(h@ zp_I{R8Bdh--hfJaJXZ)!$+#q5N@YDJG_?cTAAg&nh0w>K67Qo*QyZY7Pm|FHm8Nco zO1N66gj)cWaC4QW=0GK!j9UvWSDGp^xzOlYW}gR@cv>kZiD$3U)Mn@z*xwJuOX+&0 zskKl{=^CXe8UGf4rBI1yCRE~?p)@7q;-beyqhpn(WSpGJFgn-hPU^A9PeOUO9eqFa ze8$1oE4_i|E-(HiP>H`*X=(vf;-6#mdZnon=qTcOkVqw-`<33%0F`vg`Jw5Lo@g}B z=o?r{`p-fo+%}~*Y=8>BN1&2^lhV|KP)UEi(X~obOQ4ee4jd((?MiQu@1t-UT?_Tf z4=VA=cTUjM94MyrTBGGkQ$3*Ke?OkYeXr3spkn_D6jR!)G_?~d{` z`?W^py9=>zgknl>SDIP@m3U`)T*${}rhCZ-$Ehok~+XpyKa2 zqt7Z$ZH9{f$Drc>QKhL3>R#Gp^g*SmyP@L05i0&~SDKQ3P2!jPlh9hFsq0O?*5nx` zmzX>mI+FP4Gh{vX=wVQacZkx|94ZNNc8Spfqf&XqzL~^?ZZ+Cuw9)7sqa{WQj5?#u z1opxkZ8F+ubdJ#yqXkBt(Pq^%d#lkVqm4%A7%eecU=$)zWanfK8J3xO?e#gCx61+Z zk7hi^@6L>7e%CPVlaskQ^9g>R%iPZI{>(Vy?&!Ibuj`s}9?9vE zwJqn_oadnT_j<5bQ?H!NH+t`3x@di$`};iDhp+DPw&i7Ju0QGilQJ^boV5O=On%p% z#J74UwetJQ$t@?7e&5IX=47_?eS_bfEQZd>+|h4mKTenL|4e^$>Hh?Mu6JeAPekXP zz~apt`QwglCO?y>{^h}1{BI@S<&r@xk^TGrA1F_mxAMnRr5}pQJVSeOYpxgEF)rrK zTMgx=BG+&Emw7CIobpw3Kg-;|Co=!Oe%~>7-~URH=_~zlN&2dG^H|J%da~4Gx#pp} z%sZ8GQ;_RUll}Yf1tvF|yPOjt{=X_80pYy8GN8J7A(q887KL`EN z-2La$FPgjm9P}k~-%1>EU1je1mi`6izFwW(Gx(GA{O7c*%>5a2mw7I6_n#BDn0pC! zaxJs?{pZ9x&HZ|Fm+w*~y#Jhdwz<#M;O^_Vi97E;-X-6ciR?dDRbeagd8Ypmcua6j zhl|RRZ&CSwV)Ch^A9cSb<}Uqk)LoXxMP*m9Y%zNVFK%cz?ZxEQCElFHWs7gN?Z9K- zSo+SkQMxz3QpU&1Y%CgVwjS9$J>q^f?w8coRYqpnG+@l~((1*0ztT{B_2T-P`HQRP zdOLJiF1z)km9@)lZ*MPF-{0)ceCZ}F!TeEV6K`YZbp|s!X{Wj|4{*>Op4G?H`l_0m z+1^$cGCybAAIGQZI`|D|@H#kMH80paIhuWJjy|znUc{rAeOMn?<0CQtu#p_bi^MX1 z>|tX$j2DR|S$x=74&y~C)r`93>uiSJ^9(*~YPPH_g;5(H+UWCp$WhSO1ak=$HauTwPxuo%rnRd%A7XAr(lfI`1jZln1*C z7`&6K#r*iP+@9p0?#;W>o%hU>8zPs)CtY?flQi;d%1cEfzrl65c||{-f8^dP*Iovm zqnF(d0;|K#D{hq$S*|(WFW(vGE4BqJ95IaTqAbRrJ!`^A;;I1eKHj`yBjIuhA%Vo- zY6kT_Cj8l~dEt2@IVFmhEMX(Ra_i{ht^=tol{%or6#7^<&pRcO>60e8?@$P2Mwf z0po>_%Qpa-gI>SW8$TR7QO4|vcP;;3{4!3+?YEtsW4vC{)^Yas& zxcqeeHtvc~m$d$ZZv+ZPYJ};k=j42Zz6DFHi^L+|)nq!ls z@fP2~NcmOrjfCKmeQv~m)75V5g5K;KL^`+d?Zgnio!G_xJr76xln}q?r)EX#5bDBU z(wDMKu3X-F-+{Krd3N4MTtPXGBfYDsH!=q8kLgpW-njDy(qAq4N*G^<4Wut+Fq8BL zlYVYb%_o2C-*d0IrS%=Y;V2|+8HXOX^5)jZNjsWub5g$-;gP&u9b(>SKxJloyK?gS zdhBE$rM6q#mbM9QUm~tnw{bUNqUlTd$u~s1iBIz2 zKT>Y9)akY=(zwR$8;o1qQ>0bzw(MrlpD7u9AIJB0d?PlBI=7`|f_w|=-fZEUrrlFy zU!pUH@4lsE3*Q{|ZuzX+(wlh_)*`OuU1Hrn_K12IeWvh@eJ*yn%qQe$-1&ffzsLOM zI>Jl1QH2?9LNjSeS#M>!@Iuu91qKS+VZ}==em7ye^Lg+D5JN|cl(B& zuX`V~%?H=V-M%5lqa>y;~d&35}lua?i2Y+23T0}oTy&EOTiN|0M- z6E}0NEtj@$BPsN2ArGht+e-@s&#J(l-)O&L?PUGL*%bAdTdv~hmzhtm}b0%|HUjMVuZQO-h zyL7U8cyTmCcaeWDSy+)J?q-lIllxv@Sj@moOt2yMdNJ@?hJt=Ub>GRT|L$(u~-nYc+A zH)pwVBl<|aeDTl-9 zD458He9kpZY$dM2nQmM(9FHjZQTb4LOPwPsx4xpUTlx-l!tcv?@h$(&|F(^5`8fK> zoa8m+tI%)t-pI3Ke#?BbuQK~{@-|A|OvxI!@W(H=l`{k+@do(F3$N}_MqYpP&Q&FJ zm)dV$Z~K_#xih~ey6{B(wZ>Bm_~7E);eBR~8b0U8XAGax$9dmTH|Mwmw6}J7H|Nxm z!`;?1n3r}3{4^yGqO!#M))giDDu|~UY!x}o1M_V4n;veQ)H~K#t&{%lPXhP{{&xW`RB;Lc;1NL*C^MS8AJN)rX1yaX!c`g^xA?%p3=+lWWjcN}dT``DZBt@BIvjaymY`o(`~dz{yVe7Qe+_iFCh za_<&hdPV9>fmdG&hi7mMWT56U^@z=HB3vJrV@_Yn@=G z50_&z*N%ej<{q#ceDQv2EM?@iOX3XLF#dKQBEEC|_{ht6e6*Fax2PAZvfa`?taFH# z!&_McN4-tG8H7%<_Hf&0N?LpMq)gDQR-f-G}Q68t_qH&`>AoO>tK*_I6i9^wv)^1XoZcYu`PMY@bCh z1bZyjvT}mtFOZOnMUT40wYL`3b5>)uZd-UTFA{+*xd39^zq-t=nOm^9uAyL2C5J?^ z6XLC&qU!s6Ii+rl^RRirFDN*>-W7P4#O;X}uc&D#sNxK&3kr%|A!C`cuw$f4$)(vc z+e`WEnhP|=`h_)1mPn=?!i%485iirxg{>sVbc~0i5UZ-aVx;8f*3?VR&NUb{vbm&} z5A(cyEV!kzo&sucqvcZlI>xNYD{$85S1qeqB0E_s0JXkMHvX~gEiWjjUQu1e@lge5 z*VZm~R0G)`(>BfYb_nv$dL+BXB<7KNm7B_8d>kI9=~JxYU!2FOb9r4uWo-vX3wzN; z7f}peUP=M<_VjVHD(mYx9_tcsi%zL;Q<0Z7)H{t!%OH}HMMgxkmoHkxkS}#uQcayb zw?N|(PNNfUww4;K^Ww62!74fL>u<4QE4kn|1~*g3s95$R4NTfV^^#?EUK8V^kNOK- z!RXOmGew=zqphTmO&z1h;Hbq)HMz}~B(a$s8r7pVa49EeS!i84V8^jkii zf?uHXb%310ALGuh_x{L$>W1Y@eC;prs+Z<30VOG~LM`(*B&{#_@YSWI&N8r|kQ2ka z8ah%lvcQYY=~%xkjPj2LrEp0~v%R2j*0Q?!tfHcs%ISTzw~q7y6neGWdYbd-z1Wb| z+VnE?T(-EN{NmZ=71OV}a#rcZ8xQrg)#`;_98>$yc zC3D_)D6*L=O^p82{Afqzc_=sBzD#BE$O>FZ7D$+GxM+QFgzRb zxYGU-d%1@3P&gu|qx&A4FO+#;t`6aI8Fyq3ZnS)iSOVRTe?FTnZB?3j1Nu?ycR(+K zZikM;{W+znyP?eW)bhcw&>L1j#a{z-EdG`#O)Y?4h5fbWUII16qWJg~K-=sA4AXLKL50&`WD^1CF91{QCN>g*8qQ@L4bD*WyD@{#? zUV;5s=;hEnsQAl;iXJ_bruOpiBsljdO}zmXe=SfCm(tWu=xF@yP@38X75$!pN_<h(P&qaPzY3dQE z=+y+3a1Sa?@c`?ETdOp6H*^&4jY?B%r81umO|?L0;@%7uTsxtEh5HVrskKn?zeZ`Q5z5%s=oL`$-vAXn zvR(|DS^$-Nn5#4;>%|x{_tuL+Q>DmfAa@xKD)-LkY>q(%0&Rj$hOU5$ zJ`13t|6C~Z7^QQRrmlrDE>l{rG$rfT_#V1+rqYzGSCf9Q2s#y72%Qcc0-Xl!2b~Sg zgYy1SnyWM=>kwvPza4rtbemCG*Cz5~MmInuynH8~PxvOKskKlB2}Tz|MgQ}lyhD~2 zDoyn>`#dNw3#E)l2u;1g9YgB*D^RKD&p{>pvryhgOSdUaJqqQ$ymW)o)LLjc{+2*b z!oF5%YAjU37eV`Cf1c8mtaFfb@}ZJW9#s70LQlqD52Y#jK3(G70hM^SD@{EEmG-n% zY3f?2gfEBo!G4y~)L7_D?9YQr`4vKY<9?RXlB}nO7r74~v+zk9rgkB6Sgi3uaf#NT_z$nC% z`365@J&Up)$=Hz5BTJTMt!F{j-C3(yinXK1PBt~y4bJ!U*bDz$&+RPAYU%j~zx#VO zvdZg`><#=rm%W|etvS!+5mhT=SmgcYEFrew*`N;rGdtwz7z8`$;?aZ9eH0e%GITKMT56 zpS*_O{U4bU+LGv@0$K=Sq!$L|4x2e`|s!X*#XZDV0qYpC-~htpqbwX2R02Po`LK6 zZ5((vzuN{r%kPr|xAObwz{mLAJFu1CmVs~ZyJO%^;q%|fCw~S#GYCI}9^xLnZ^1Elq-XT4*8c)6Z)YYd_VyABB zxAoNhbhdI`O688Odpn8BBJ0%rdF+eC9eDzcU9RV-pu1!|n^_{!=go^h$K>Zs{v(~W zT#HRU-Q4G3C*`q+&Qz`{i*F$}1GyHNyu$n!oBSPtpgj z^Z$6EhcX? zd9ukrH#x)PT_%q(`Bkg0a(|QjS!4Op&(inX*NfuLzrW673Vvl>pnts!Bl32$xDOg} zc0Y^BOJlO!r=tFOW@{&(5p(}}O#Xi{|L?@yzZjEyqd!)`C45v~81rxI#RLDcK0NB5 zIqP=v^jLh~jJaPIlb?+FkMBovZA{+@G5>#v$-86m-x`zeiOEmIWLY%%D zgmE5zt!KL=%j+bXvg-QfwM0_Q2&)%W#Wj`n!OBlBq0Ux_UO9JM;7Qa5X{ZRJa^6r? zy8}A}P`GN-i#R;KLq^L(#~5~R!D_YbvQk#PsP5)!82~a1*-9mp7_O^g47Y005(R_I zfPF+YZ){!N1iDS2|y<=w*(-8dfrsfk+Ky%NXs%!m+^_^pP{}1H16p`oJo3G<$e3dccJz zs0UV&Q^LbDx&tme%sa4(99SJ57aefnxzd4Ec+7I-6y@LymGBJYL|_*(tfSP^QzA$E z26o}Wyx|GFfn#_OZSEcVsOZ4WL0Kh7oJNRtkg6`gVQU+gEGT& zFcX1mu#i)OMGkL_oPy}9%W;WZM|%F@*s){B6??NRk#(LmiKaep^W>e&yV70fxx@TRe00T*xm};K9-D385@~`;*Hhw%$%L^LU zR*CEo*WuQAHf_|vb1aV4=q1>Sco8&=?NTVlZ3zFGE%NZH53pn4**7MK zbtLTb-eY(pd-^w)>7M>gWw^5!#Z>mv_rE{yTJfX_zuQ}P4s_>0cMf#tKz9yw=fGb$ z2in>kMz5~Jt0vLkW6zc#HlpfMe4tE6=BQBzQJp|wRp+CC90hw6!H-LjKxddEx`Kcj zO2w|m2TBG^R6{J&zXE{KsJav%DASPvrdp3g&w*m;Gab=&v6=4x&j|47BH%OAa-OrqU&fAgz5?yqdEkNrO$Lk*QreqVKlj}zz2$@!*mL* zQ%j9aow^zy&Gs1T083Qo!1B;cG)PzD)7SjaK4Y$34Q5oIn5u!8j-SADM@ED5%hlER zn6^>%VopaYYE&Ak>+k`gK= zYo~JBBdsdUfnEhsVmR4!v+$P5Cce=B*N@KsLb7Niu)FvMRQKbBG zDN+RsOf_QfAyu%zR3r8tQU$9>6)Z5-h@U1i!h1*+tRhvgz*HlC4uUjlk2mYhggb)k zL_Mn;b!{ncjg+0G5aPrv&rW-yP1Cbow?0n%+F3Y({H~s_qa}t`V3H8$TW!J#iJtS zS9lMpf(51;vG4!v6RzPItI3u4Xpi;wfPRCS8bug7(dz7`t7%Nmcy9+ zTvzM2-`W)pWA1oYn;gdMH@jND{nl=C7_;B$YW?opnR*@=LV5$-S9#RDhOf_QfAyu%zR3r9HW`y^UDp*CTV1cPd{2T;n)E;j^t%g7B zbwYzx;9yK2GI+V7CF@z706!f|IZdTwVAtwKeO}e*oUYiRY#`wu@v)_I<&vF>E zpX)aJ?RWDEhcWv#ZnNKhH#a$q*>856{lqqJa~QMV={EcAck>?CsZpN+Y73bLQM`Cm zq?*1WRj`Uw!2(l_`1g=1SYWCVdk?9C1*RIYZ!#mihg88TQUwc4HR9(WNTc?63u-m| zVXqS!tO5sP`jElP6)joM;sp5VSjuTC9Rs_=IgDOA@*T$bLbtj}JN)*$ zW3I!P{R+3kZ@)X%IE>jhxgAYDzuE23_CuoE#F(F*ZinA~ckFSU8ub~VwvcHM#fwKp zs$dnVf(51;@#7&?u)tI!_8w9N3rsa)?;%yNz*Hml9#RFXNEIwF)rg;iAdTANEvVJ- zhrLc{unHWE=|k4q6fN1qW>FXM(6N-$R4fK|zjGM9_UAi{@r7=`-+uQ`b{Mmt<@WpS zcmG_6G5Zy6{|cX9bj%$zA^`*txGT_on^m&FtniTT4H#1s^WdFLH5C!Z{4|NhYI+2aCpOu#w> zR-_76kt*1E^PHMXE^&YJH)gKB+I{CcWsuWWfIFS7Lw+*jHZ_17yHzxd0ik z3J}PERUp^^RsmuY0~^=}sDo05ImRmrHkov4{_3yXRacoAH_pwOW9EVjLcX}zU2~0@ zbI!3EP9;C>G&gCI8QR6jk!Esp#Ys(8mh0Qs4EpEfgeEi71x9PQ{=P4e1F*yi(n~KD zL*oMA>eXV(%iY~~yYGEZOm(&U!WZ1X{hRyw&)xj_?sK1$qWsVQbX8UEQ=f9p&F;6q zl~P{1R7&}kSENyY;uG$**Tj^RxXUkhfBd7EM8b_9@AmE$L*u2O_w5s-xDFg}7%gIL z1YzYGZd+wR>4ghEGit(+jGGAjgazX2R&^f!Qv7%f>)t9TT61F)nU zfTbEMSnBFscNtiU?7QEUV*T-t-R-vvz?WWf|M4HvQUCXUivoy?771j!**D&B*I)16 zddvO!&jOQTrR>ljC%}Rl2sj1p7bj3e!Nwrtfg}jXFHT1d8N8G%O;zzyvh*|DL8z-# zMNm_L@3_N#^rLR&N-19|E%so)=_Ysc&2oRC+5?UHeBOC-8@l2OY4RWXkfWK?gfG0% z_3tmYGnz2J?@RmDtWd~DTEq%g@sjg&wtxa7DAulZPdwo^Y;Y@9$S;lbXFqed-RA!8 z@7%6kQnJgIN%iJtx^t)G&!;~v4VETMBsl%g|HRwx)T!@J`wglAkdO2V@QxTEWucug zu=K`MPLO{2%hFEoyHCK zF8`;>NUmxuB-U zra)_JrEYVJsjPH2+~DTUJr3;`$VYl9Sj7tvAaG0NUZYi)hhEm-fA+KPpZ-aTk4n30 zl~ifk=C6M(jg+jQ%~Hy=-_LyJ#Av^ORlL)uyH9>n4E-%N`N=26zz>fn|{ zkAwj@?U&mtM%ynn4b%18FJO5r)b@+!+=D2rZ+t_ZjlcLsDM7mBXP*_+JTTK#DOp~K zo_kK(FU^@uIN{naFFxEW>6~f5jg8V6wf*u-4ru$m_g<;z+J3o1@o1s#_r@Df$o30f zdQ@${cit(FRkTyuFJ(%vrR|qKncN84FOQKYbo=G$NM8_WzuaZ?g`M`>)Z}*T5JT1d z`@ffm4b1;<+AnpLJJC14DbQ%M4?ZZ*%)Agi{IL7M53K#tuDjdsU;g>|Snj{w?YF!A zcDG-ipS$nBUHttwcz*7_|90PhyPuy=-1pz^_WOUK{qmm6yDj5&j8-v%@xTKzUdNje z^CmoR@Pf~{D`PwQzQCaL?|*Oa`-EbEPRHwb!(wP%$Lp9`U}#;(+Zpm@i4Q*E52#ZJU^HeM{a48E?b@oUz$=-xb5~ z4n9>4W)>LBQuTCXQhD$)v!FbKmVi|rylH|}{|Z@o211qL_vpr-_5RBQ7Lx@!e#IkG zMTPr^e-QK6e=Xx4Pdz1u(e>H0rO+{7`HBoq{_uxl$Pgwbbi8ZRCU^SjGJcC;lwi;x z84kic{j|%=vx#p;bhEQ%{sqIUbVi1I^G!1+o#X-oG&Phd(9|$*cr17sd{(@}h6xtl zd>LkBvTV&7smZ)DFxkiC0%KW>(OrA3Z5l!y#xwI}AYy`#LS!a_+D@fq5`moAx>YJI z1xgJ8ECs}5iq7(*EolK0u<)Q$u<$SnHfpakP}pmo=NE5yP;;d9S;yOf%=``z6f)D( z)K!LL8NXu6j5mEsmJtB(QsR^#zxZURmKjn|D_%x^fDGP${%6693Gf0rzzgaCFOUPg zidymNuiEfSDJxzr;$z1PYVa~L$;<-gBOeh&k31rVadl=qbq1eyO=;?)3kI)fk`yS0 zF<4r(wq@#rc1Pqt^D;rx3mPx>N7Q(=9D{fej|DIDgIdH)Is;RuL7C%Y(E^Y$*w8Q; zam6rL_nY6?+%BchLIo-thAB35P`u1F&@-!&lsm4&0uQX@00xPc-7+&*=Un);>h>4AT4WD z7@?&VGqpso%X}z35_p-%M|}Qysn6VYsq@r)UIv*N1Uqvs+3k@7Q4h!%#jE9b6!n>S)KJ#oRj@~am--J_H40f#D`Z6tWL>Pn=sHELW6?BfYCZrl zbh%V%#YmN=VraWYF(50(V|m^PkRChO0IxRuLiWj>DP_%b0^JuggY;cYEm5)m@gHUA zhSgI%RP!)N>9dAMhu`Va>E5{+@o)uLrp|e~pu}0&aLOq%_XJq(Q`}RSdm<~CuHt!v zg#ly*4<|ft=u?x{UVb%;_yr(Nl%v*Ro>=tK@-^YeiI%U112RUh$&(#MFQ%3-EWqf| zLpm`GsiQXI^BDM^J>@9{)2EO4;h_tIf1Zfw?l5=&nm(pxWl5||q++n->IeaQb;6bS z-!HI|m|v@vKNw`5H!ynXolAM%CNO&OfQ`|Mjvb>HPa_z;=;Sea>53-hqmVyPu$o-Z zkP#KK7BodnCC5-$6md|vFw8Dt$a#u*-8wN8E=A0ADhB@)aZtE0)N>4lODXddjR6-$ zOn;8y(yB}C)sod>)sofR;P?H#6!rgw^|9}7zIDgeodexD(47OvmIE_2c#n82W-r^= z%hkbst=b>deH-q*oEyCX%A-W-BT7?^W)OMH#2kYHj_`Cq&^m!eBDx&8%*~7SBs3| zO&ir+q8<^6|4Nhnd^pWyKi`tb5%11b%$`wNUR~c%aoM8E`PI{EYZ_c_#hChrWu3T8 zudA(7`-FAL(MEE6`+x z6rNob73CLSTkcCCU10G@d{hfJ@k1G|uQ>N`jY9mZqN1v9aeWwJJU&03i%-=EXZecw z5yYs(w`fUaRYS${#kbTfo?B5-IwR`kv7s7IlUd$BEW^Y082U+`af!OYRSais1Wwiq z>zd&)aQ*n-LAU~_1QLI($l_zi@C?@s?Zf{p{^iQGe@W*g{#6s0Be>e`a?c-FnX&P} zbhj?I`Khv&EGJLZ(k#|x^lmKsR8C`AR#szKkKv7Fua0Oe%Wi(C>~|xY%5YzQVCB?} zgkQHctK!$)@f{n7cMf#tKz9yw=RkK3`~`DB2j=uwM^U@ZZ&bHq&yr|t zQH*VE?zo$%K2bf78g(?1XynHoX`nzGlfdE1}<*S~LI;}1eP*J3b zw~Od}jA{Tmor`LJj0ts-;OV#DE~4-;s*z^-cDLVS%;)13Jk9d$ZokK?fQ~WUG|RWU z{T^dJAFtqPmT!0aJzfQLjOnIXzTNHj81wmf1y8elyW8*aDxhObH_h_xZokKv&&Mlx zn&sQwevelH9b>v_mT!0aJ;r=KUcu8W-~Y4im;5#H>{#a zXFOJYp3sm32VfhAN#_diDsyy+9LRr|jrpN?*w zKsuaNP-=LL@rq6@j|<;XrD!G`4%wmH1Eeqtt?T?obvqP>Zny3n=+1%e9O%x0?i}dO zf$kjW&VlY6=+1%e9C%-HKwm<1Q@18>do~X5!W>^-Q-6P7kb>9%tMT&-(tr6EGiHzZZJF=e4#^!wNk<1pV7*2fR93=MI{0s0Ts_C+s zrE0<{TVMN{wJ9J|6$9nduh<$y!)l!&D)ze8tpowcUb$?OiQ$Ydwp3(~Sj=C5XM<19 z^TePD+e@oHRBsMz)g!skJb;`g-T*8y1Yp^~TO+54NlwACSu!x$^q0M3$#zcE{6Kjk zCpjQfvZ@1+=`>Ib^?;1Fs`C*+)+`FbQnK_#>=#S1axw$^d9yt#zj~xIr*5+KCdWQ= zNF(M0<%yi+Fr}rY18T5eFN&caMoU@CNFk?F#Q4-8SI3 zVUuR%Q4huV3|^F1$m!r!u!@&BDNyp0l5J>^w!>!3>_i^4Q#NuA_z$}Ml9sk# z?JL+o8f7@{fK9lmS15_99A=_;6>>UwX}@^J(DrFE`cT0(?d)H!_gPM&0Ge*;%`J&G ztGu?~4~{1-($ymZfXv4Est?^J$DZg$%M>wPjq0Cfe?|MnKQ%<_Gfy()8(?WBoC`sl z4cadoJhONEU%vgaO*rkBK7@l@*h-v}mrxjG&`{ei%3}kNbS}J6r2)lr8vOuaxiL|h zyLkI$t95O^`Vevy?H9cCNZNkch5CeOzdR&>SL-v7)48JUmmH(XU}(Se4BUM%U7`JQ zz6b4B_0jgL$MR4*$KHNv$bhApbe;A~uchsm13ftUfQC%h-Q9lEsn5avx4Zpzx8LJ+ z|LtzS-S^+_`){Y+b;sxD?)z^SeE;okzuoP(yZw>_9p8T$%;9~3r*U3{dB4{8eMTD? zMbYPt`|gu*GYmE#9IxYbq{5J$j@QBShK^#q0RHlq@5B2quUA08FlfUFq#nPH=cxKHuBBOxo{Z`o_4_aH3yinxxDW$G^vQYz-^PuO zlSMg$5yPQvmt5k${cSP$x%lD_kS8r(!#R!^!&!Np3#a-}5R9NwG#G|1gYmi}8o#23 zF!~B)Muf;Wz<&SxG7`vD^$(CIdr6u@4te2au!Zv; zNt)qAfOAS7V3|nL(I*|>NPqlFA2}GG1g}B{FUO97oufpVU?MI2a5S0XMZ_rH4~_>N zNpr#>hH)EGM+ekHF-`(x774Hzo{qqa6DFNYeew%l+6BE6hS6*Wc>lll&IK;2GX3N4 zT;w9Ccn#5Tykux17cZ$n;oTBZCp0q^2vA8B1vGCJrd!6?2F1UutgNtrw8+J>v?Mh% zx789Wx66j*rkmB;YNe(-|KD?-a}I|y!!TU4{mlD$=e*bRp6B~M&-teEkJh(JGZW*eaFlqis6bJxu z$Emy+?>Gy#0bbY%K6u67qGIn3`o##sjEy)p^dJLn5SS!!C&HzhE{^*VZVebv7_h(u zFI35jz?=*rF;g)n>&ty!MuvX*L?DBg2L;;1C1Bx*pjUAh#=aSfcw4?n%G?wqgBO=T%kbbuXCes?{o>VqBTZlY9M0>!(*KT74tYk=Qnj6oypf_ zSdgl)%w7*zm6r!Q*wGZg@lW*_FSsILu@1q#3GC>E5hLi1fDXq-cT>l430xgxhes2H zfo=0+3=mRXV6eX2nXJ@0+EZM|@t3*yV^6Nr3~^k79iEkfX9v=Tz;Yw5F}V)<7QXr8 zjSj0t1~?q!t8t7Jc=R@Mnm^8(Wwo+qwc@ex^*(Y_+&E8z7|!s&Z=A2LVj|*ko*wY@ zfTssMJ>cm9PY-x{z|#Ys9`N*lrw2Se;OPNR4|saO(*vF!@brME2RuFC=>bmcm9PY-x{z|#Ys9`N*lrw2Se;OPNR z4|saO(*vF!@brME2RuFC=>bmcm9PY-x{z|#Ys9`N*lrw2Se;OPNR4|saO(*vF!@brME2RuFC=>bm< zczVFo1D+o6^gx4qpr$888r<+@*8r2+?ZT;7N-sOH>n9L!oMP%SrE2xN!ew-po3PCO zM@Oy`>>nMu92!ClbGg*<@M<-S-f-O^Qt9QQJ9p%$6{pq;?p)>aspA?) zt81t6Kb(Hi^YuplBf@HVAwN5(s+2mWS_LF7A49E^B$DKGw{^#HZ~Sp#xo6x&{@;Mr zAv;wxlX6j=jOTK$%=WCJ=^{$$o*YjPczVFo z1D+o6^nj-aF25cy3jxp0R|t4^t%ZPRr_J{l0-jwPA>i4y6#|~!cp>1~#R~z?Zk7=6 z?B)mo&+d>APQ6mzI3bfVEE6#-lQL{OA>cWmNg0+&8J0;Iwu2Dx49lbp%cKm;qzt=A z2zaNkYG$L*x(^FpIQL<}3+Fy;t-Jtlpwq(o2*Dn0wo<&C?Q1Q;ol|ZSg1tMHQAbsA zLy!$MT&$*iR7a6o%5gq&zj1rsLcp`@Ed+d!UcKi)O^{os7o}Ir@sJ^vPz{~mFg}>t4ga~*!{~F=1_<*4To?Y;R zLiHhK0$$E97QxOum#Wlq?K%ZD@CKU%Ji{_6!>X8wFe$_OqcXe%%cKmeVj{w%4C{Zb zfM-}HWmu<}YUa4p0v?W$>jWxScE-!F%v%RoiPwEt=5-$y)Xdu;WM{li`_0cFRPVHc zF4MQWP`$3mX|nwSp?aqkkg4*j$TiihG2b8z%BbWz9fe?jg%Ey{M>%dU+FNT!lYp1& zOL5LI83er1AiP@$ZxeaB&Op&|V6`}s@Bvqn_iZQIj}cu5lV=xuNI3Oe8hJ>tmuuHC z(DJIVplgAfg)~ni&WlGh+|8Rj!!jcuyLk!tR%YQVVfyR)??zp!5_yyC z-K?*wSVea8L~XS{8$_}o50@xK5XYS;*&xycd3LEVW+lGDAmC>U^2-fXzK>bJvzzQK z;MuL{E#PknBEP(MmA`SMfM+*tmVjsXYO!$Yxm2Z=Yu7Pg-b7|pcv}iFPvS*mn3+`Y zmyH6RVYQ_B20>n$tbm;COCIj!YsfRKmh{;{0{(gF{V$MLtRxTT6jRMCJ1yYh7@1;q zbby6VbOsAvI7e9U>eMR3O7(#Z&st_!9Cd1CSRCPi49{9txoV0qZ$pp)`+#K6WWbW2 zCj={(q`AzT%Bv#RRY1V9Ju@$0u{#)`v~t)WRR66(z_UAS6skXC6!3C>cX3$p=R*X% zoPUittoZXo1w6Zt9~7#;SSH}*{9^I3GtZ?ewOqR{ftD9mI$!c=;A0bchGkNQRWT7^ zQieT@%J32_lQOJ|i3pQ2?BQz#Ji{_6!#c%OGmoAY@NkS=7f`t_k6&0aR`)ts=B)#) z#OppR^STcUYUXVSGV@~kWnLFO_Lf1Y-bFRFsIa?Gy_)GFVp||oUsE-eS4FO=W=;9v zEs*Xz;AO{_jN`vd;%5VStd9R`6g?%+b@J=bPCb{Z)N<{*1Xvv~FQ25swlRtxMge)R6h!*9nLHeG zF{~O##6F1p#jC{Lu7vs9Nb*!);$m1ej)?tb*}u4mFfV;g9?mJIE)&%lnNFyq!>tFr z>{Qs}EeIAqr!KJI1?+E;m$R07fuD##+|}~7tRHp(;}S0qnJ*vivuzbUhpz3j+JKdtFJg~B4dVA zO;;ON&6r{TJ67hUI&PRTYqox{+yf+8-JQydQ!#AWR~W^2-r|H&bds!H%d5RYSSP_A znb!u3!5zuNX?eBvd4j?bZt8g1Bd%&tTUpISMC;UiZKj)XY*MMbObX><&l#}Hi*_7g z|0B&^ObX@pYD=*2zdBeli*ar^g$#?C3KKHJUXT`4s{AU~u{Wr`WBxZMo##`odk(zt zY|pB4Yvtv(RpqL@ZYo#h1vQ)_7q?&6d_K$dk$G{1QwQ%sY4Gy+MZQ+QY8(-0mOW-F z3|=0;RKHmUFZTw?T}P)PL%)m+-hcYa^A#FX8MX00IxkpMUf6RlFY>ic&TP`G-`m1! zzwPo6#u|g~xwYPQS#?^&*2TF(U5~6{b-sdr|F?P3KS1uF+>F?s-O=wKar~;rz{pI^ z$Y1|2t~XR(L}1e31vuOlEd+S~s#$MH*>u2U#SY^aDn@d5pg2F6#xLF+Q3Yv=mMcj~ zj${1d{)^J*ZxrVXXxWT=6P#|c)eTZLsWzyqujbby!lRf^#NvwAS4i?K?bsksS)8w~ zuc&@?Wc5UzvN*}B4vxEz;C@Z@qOK#CGjgNDa#t`ie?G=qc4r}tH~Vm`WjEJas9)?Drf-??*`Wix8TSU&vu}j+~!}5c_ViGdPSMBIgGpM3b@f!~sI7K0>cVPJ6}3 zp+yc6P(3eiV~0R* zINK$szr%-h*O>7$@RdI z$}O@yx5{ZnuhDW^(W4L2a#h6O(i8e;WoL?nn@7bajLb|;%1*V89dYyG%-KoFsbexT z=1V2VlNv>FG5U#bqo=t*LB6MoLXZHRTUaNs*=OKTwGb z5b4?T7e*>cM39V7l87J~PIx^d!Xk%Qs$6(g)G{c108)`sRz!IC;DG}zmVrYDL~@nH zmeeKLM4m?EW6&Y}GpJ2+hc5i^U#xDVuA~FKaoJB13tn)Y@gkeA$t?`vl9O^BB~8y& z&yUor>mla8v#%B(U0Pu%zBE$g z`&90-o-_#>qIr1b8_YRYBVH!F%mF#pe|F2UdR0DSJ<)BW73Bq&Rtzu3`T6Chgir9b zbo$8|&&$&X9vk=r*Z{^a{l-#_ucjKy$~9~2qN;k0H>XgWwOvpSDDDX%X4HI1>F+O93XuGiLIU+HIV z$Mv{yR<}OacO`@N8a2h0KB@1umAQ(~XXcu&rO$7yYde#H+SXrBNh{Y?>L}+oM(FGs z+UqIts(S9twRM$EX}_khuI+j1H;tbDDE$gMs^;W<7TThy=7u1Jy^2tI4@BM1q)OPo zkoGGnJ{&J+`;yB3&2?0cO%f$d-`JpiMP)yx2hy}}s5l=)G|4*e3)1eGvM0w|al8YE z{kH*wHG6Kg?Y~6~iX6lnaQgkXNttQ$)3Vc+q-Is^!Zmy!u9P|)PT7Z3(v7tbw|-c| z_Th%h@-=iHE>CX9BGFQd3Tm`$WM zUo&kG`?a8BD=(Uf+ThFe_-QI1I=1&Eb#`n?1K|okI(C-6hu)u$Y)!{*ZSVxG z=pI1I?%9@(SGB{YxVXA~D`58NK*wu3w#G5EQyUz^uApN?U;vH-I=97fP!Juj?a~g% zp!jVs?Io0=-nBZ6R!@! z)!`V~$4r?15?X)+)uAOp1xg@YB?LRP(4Fop2o2i!493H|_OL3JHQ3@qN4EIIZ86IL_!FjALTB1;>=g?l{gJ z*aOG(!M$)?IOHlEvxoJ@acNWtjw`S4gJVuiUmVwr=!fIlQK2}l8`B@hym8?;7Tgqp zP>j>+Vw|S78n%608O}x#3+@!oU!MfmWYUHNlbph*|my!32ls}Mn4O=TONYTO{ zg_lKjZ8=b)3&*5?2xL{rhVx$jT2^5pYNPO8&d6HWM&!Lbk#(?*%X_&YJHj?P@8yH+ z6t*dNF9+nBV4H~d_eFLA+jP9Y7qV;ECgp8@bvAa3&31G#y4>jYaT-nt?;18-en>&jav_VH?it0=aQ{8wIvecxys# zMBc`NZ7g1=$c@F@h_H>w>xkSayp0RnxV$>#hVwQ$Y@_pPksHX{6tGRft0Fgqw~1hz zh?kM;%?ntUn>`IG#QR^@g4MeeV!JrECZU@Y78h+gUckCQcFNn-uuaYD0@;zb8DN`% zwu+7Pt0Mm{u)hm0 zBjeK5+&E6v`+K}&alZC@yz~V&?e}(v>(bKqdhv}t<@=oUbw2I)c)KG_d8(ry_N)5& zzN#iT`&!Tg=yn!nF0{hpxwrQU}AuAcJ!UC;M< zrLX3yX3&uD^G-u6m+SX=OH*4t{9{@<;SVcgsafcP-TyJuVl`+%r>OR3;aV zI5^s@%NVS z=m+-bjcQ)%7VaF!qwaE-NwVLCD|UQ4=+nvlL2a{Ho;@9r^Go08ZGJbr(=qE()U5|Y zr+paO{cnCVFAjyD_*n$}q%R`bpSPdY`u|)yt@TwsEI|?~X*Evue)Cy8xqbN;m z_3Yg}=ilYK_#D zcj2xWZy(AL)00Myblua5BeC{ja#5LF)Ul!+UsOH%ACtwfoU%~MCn8|OsV{c<@gFzr zn%$>#Qk&AW)`;0cu)@tXiZOFE#`9FV8ue^0V9(<}V zsP8Gc2GomRivakq_9Fx?u>&0b%d-`Z__(yT2h@I0*|ibZ@;jE++96!Q@kHc!2{Wc9 zjMlhk{h?D}zBRqAee*4)liLJ#@;>)X+^V&$SK2~Bol_EV?_Xr_mc7jke)F?^pmcaN z#q1p>v}e_i+!Uf}pD!Jl9?0PL@Awl(ezO07BCvaBo4E(VN_%i*mOVy3>$yOVr0h%K zhSmIvI-Y8RCcD0QKSr`ZM*~9m81N0Ry}k{EheCVDMx1k67ugW4)REx$@^A z29STSf5WQxpY6>`zrKiOeyxzb9`D7;tUXzX0QtZ63)~NuSGsXz#=#ldQ^4-G)th^u z@26LBB*Q+L>-73gZV(Yo_=@ADKBR;ia-fk}`8Z1RHZ`4t!?OJM#Zg;>{gI6U~!4W}bUGZz` z1hsvR;1&$tw%)$Dx55-NO2VNp=>swVD|uDAmc|&Ub~?SEBcj(u46@a?d8kV zi|z|YYoLB;PYf&^K74vOsNttQ8qPBtUk>UHNxzDfKK|Koozlbes2#ZMoeLTNH~UYV z)7Iw8JrK1sOlSPz`{c8x9-YdS)noXva5jMJzq=k0LMj z4Xow;w*1D1EQoCYd)HtFmw%HOg4s zDmZdfr-CDgHLlaUS`!g1zMAhQew2v9Eg0+DxWlwQ#C%KNinc zdKXjY|1y!)9d&;c7u~ge3A3NK{|h;w4qWMnBjPLft>ilDG3u=-W=}XYlOt+T$5KD; zf$RS<6cIHa5$2^SOX5Z2s)|*sh6Z(7#te33bf9e8ilUX-o7!h;smJJQuyC!BBS)uh zt)0^e8f!QR;18C>P1e8r?$#n1~fc_4#Di+pfh0%kD#{*^$MY}>9V z+Ro<977l74yP!td-eOsAm60>i(FY8*VTZ$^bDrDqw$&pOG;DKQ*-YNV0 ztm|I!tvToET~8wd@L%o!Km@&QS)^O$A(Skz#rsvAkjz+NeoLv?;bgtJtoBciLUOX_&h2{;Aq;6k`GDyvqFxKm{0 z1KH8GB}MDa+bK0&MY(?=&W^!3?v$E~RCf8a+6)%<=mF@OZ3?BbdJX6oycB4`xhw=O35d(Poku+7~ z$N@!<%>g*V-+toMqcq6|FIE zn!t6sMr%yXy4D&wirletRD)ABu49*!NENW;z>!y3;N@}6vlgoumxd&$gSi?~g*+6vmZ%FdaIA`ubrp8w@MPgzJ} zmd5ew4((XQZ|^TgQtIu4P%$J-9Gu9=4=;{lh}&Op$KmL{GqJt%txzsDp=Qv&m}T-!M*~n%tug9C>n($lB*A| zW>bI1eh-%>9wjcg%;T9H!BJH=Od3n9Dqdyb!4=RHnpIWI$m&2=wVX30Ja3lW+Z^`T zl-pjIY2PwA!v5ScA5+-A2{B&guobiH^QSI>_i7O~cZMB)#*{TJTb^#wB5dk3J3QjC z)_ltG3X=woEGAPU!PHW>s{|Bbbu&1U3|_5SUGCYt~Y9Pa-O&~i`^O$0A)X_Nx=S-%?QCEkP z;Qjg18N@MVrMF9G-hY4lZwzp)iqk{W`QEhVfoPZ?>Ti8MmXSyI?g-rhXW7joSj%r^ zuY#6=d}FunyQbTr=x1)7%!+>6_982~UuZfjIwj*yrhd4XbdAR2SI~YPknh}B?JXXV zr#wCuMW9}^eFh?0mwqUnsh1yJ!ub{Ud}dGoE?upuPSLJS8uhg*W@Oc{p)y8x)bbW< z93!i?ty8qtykXKT7~1Dc=yZ+JT8$iaT-do&UU`bCvtq*{yTKkOOMH) zJl=D1CBV6w3;m*kZWJTOsZ~Qxt%`vRt*x4xrk%7goOQO1I*iOCniY*p@U#GA9kpu< z*G_0Ua3Ny96q`j*QlTl zZNwp`j(Uw)gM9G*aP}0pcI%hdF>hF>&OC(YZHZFv)Uaw;#hSru&<!A&_s)Hw>P_TK;Ng7H(CrZIdw&mEo5yxeL_1EMa-(EE@}I zXgT+}xtzmCP|sOUW()!Tv~!WR2=Tewr?NU8KK`&HwW?p#)K1aNOQ$=`TF7K%L>QUZ z40Y17Q`?53dHt!SRu!!^Z(N#9i)DZqh-h6+x&|+*;axC}T3#e|vovy8#fYf9h~rvV zC$NfP)jSN<>AI$|>^Pj+w$UIjQ+3EHH7cl1HE-lV4A<0rmX z+9v}?@V@i*7C!O$9aGqBCWdweBar7z`J7c8)rS_1K#uM{f(?6k5gBM8`T1F|a!Pv0XH3{BJ^4m!? z?x<+Wl-99Kjil;o)wNo)CSFuHB-ELIiGM(J0+Sz2Sxp;6>5S(!wz}+rfKXPdg7} zlYuN8V|m4_qfjy8udLqAhV5m3m<@YUObYUWe6EmI+R$=Z|1coqjbz!qcdDBP(#0`# z(_2L#Km8)@Gq3|P{DyrScpiKCXw}nDtm_TOHlQXf{E2TKQBI;z)i2^YUdLG)bySl^ zoK+0T8JVrxddn75iz~f+3#P}Q4{gKcfy|E zXKElravLAx|?W4;>1_KY}BS&s$Legf?IKb8N{skmq` zodp%|eRC@!aIcgnffUI7gNne5_#HPd2QTCwbAj3cM_%cSu*U_h<(%K+Zh>#%~p*}Mn*(u*vL_} zY`58Ro%vGQcaRz1R6!t=Avs-a)ZU9Pp9JwF)8@Qlp1EpgGj zjfgsLOPTT{=2X#q!(v9hXK~z-7x`Lbl^6L=N3qHa{Q?=h1f{WRPC8E4$k&Z!PyiW6 zMn*)3>}cDLQ#a0nc{81snsWyD;29Ot?}Z;PAi`5L^OpB2$CQQ4PY)gB%6W&TBLWqd z-%ySSCh7yPZwDrj$MmFC2YAh<3`QQ`=QbY5&#(C}YkXwy(O`s@6JpX4!Hhj`(s@QU zx5$8D+wHrzzP*)^?WcD`%MgCX@J>8~9X_xdNg&&Q*w2c#&g_I7AaDOKAoX6+^#BYE2r+@X?zYnJ2Dcv3Gz)Yt=|<=PpK8*C9x%THf;1 zWFW(9wXB=L`0fl$AmdomB&J=0LP*uQ=0!E_T?t98Nh7IUZXt=~2V{8olZBtz%Mh8m zfsFIf19N!U!>b@nCVU5BML;iXlPo!qaf%jMO41C52oxoW~hA>*>55xz^ zfgAS~eTb~iSBS@45yJNTYvr#)GEbh2JZW?wih%re`MZcfa%xjT{@}SMS<8E$*@z-&*1{`E812_w_9sagjWYk9Rp7}d1ANa{=)IXV?j ze32C4sA6QQTILmCJ+zEQ7+HIFMyAfTp=b!k?mjK->r;g8W%u*GASAnAn*TRKO@`i|B`avUog5e->N>@`!>$y zae>Rh9sTa84%wY43n!|xG~z&3XK5sLDnx^_y0yy?0CxhTh9ZW%-TwQsMYU`@P5bv@2`KQIUH{1N6T>p^6}}%r40KE z579?;s)`Z6>qa^YTK;L*Pl#w$d_0fNf|h@L=SQ8EkH2)BwS4NpDb%4SC=}%5`{*nn zSH8VprG}Q__e{Nx_es90{1|n#PN!cK>85Dp=w@l;Xh$)(rkkkip<=Y-reUL|s%6CM zWZR9%i%U1qs+$N|{eq+%Y3TQPW6_TG`@HmBTg8kwfKwd6h3*CnRL2F3sk; zbWBKyA3ZI;Dy_u!%h#gzsHvrBpgR?sMGgguOGrpY_ogkFBWLK_Q`({b|CF2}h@6zg zlv53=uO9Nz8IveMTX{Z)nDg{1T7BiYv&kaA5-)?uu^P)) zS#h0$qLCv1q@OrX`OhPMY}YF5*QWL*RE8cn%%Lc#yjc`O#=H?iAHJ!f~{!IFl-42*h55dChPnNzQnLB)P7fl)J`CWGh+~Ls>nhH(c#x#|i%Ayg zey7Zny(taw7`cJta`~q?hup__(f#pDIekV>e<-Jakkd+g|Dd#{y+f2n$5+brrz5Ss ze_}jJX|3JJRZ$}K3H6KW?v?YDjB&d3PFE=_W_I+=dFyN&=69(AkU7A^OhoR*Y zz9FKxQ;f*JqFm%>d5OunonnUgwh(QLMugZ}z9WX-U@$Ct$4|sp8e?Kp3Vp)%dj9^n&s{fc-2UON!qhA<#_Tim{` zJK`dO%i^rTug9I}`exjku5ZL;1-}-zsq6N*4syG%yZoI@?~O9xe1WeLm8)~2j$v(9Y3d}c6@9VhIcf4O$vR^nRJ3}A z!#+xMs+Y6!~> zFtoO{GPbgP_}!z{(}8bgZkEmuus!&-7y^&EdA!qWBeD>)!@%+lil7S?_=4?aY50yrlHE zko!=|#FB4uokPN$J6810EfoQ_0He2U?7Qo&Cr%e-wlVU#ry+M6>%@{1bguBqEf81d zhR2R9HX4n#5u$@Fd-qyv2d{0J69v`pVEY6GF_2-#5g2(3a(LR z>|p!&*C(w1cg?GrT(_g`NZ}n7_w^l9@^_p!$P}1+!8kN`VBnbIM~$6qpBRH|(|g@( z4a(e-`LVI1Z5z_BeZ8#WkgG=2AYDTbkL5BifK1YD+x4R zVGDxXpF{4UqO)x%>RxAFWqopRapq9erSz~D@2I%-np;bTngVUZP`3-({=`5scd^~J zd5!h+J#S}{{FMIeyA>5vGp3e&0+~NE211S@xmOrT<}S7_b04*K@!6h9@>BZ4XLnZI zl5k7OuaeB2A;ZwzY3Li0_qv;G*3bW1SS9a!nHAHA{SEqCCd+%<&+Dz94}Pmk z-dEBqZXHlv-q)Z@l5dCKz1H8}ebc49ze;`F#kL~%Uh9+NHoBB|h$8QY8?4uL*x^#% zHw@grU6ws#U2$QjOL>2h`}gymk6W*cDs(CD>vI2oej&&D+ichUJ4CjJ-%3|mcRca7 zOL-rc`*+7v_gZgzsmP_gpD6undCIyXYMV=WA2Z1Ia8H5tx2JZw?BAubyw}|ZwjVaT zmKQch_Ig|X1J>I%y;_63fd_6gU2#CMhucE#x87#o1$i+>eq33xB#Ui#x~bEF$KUHq`KPRNXY!qw^o@Fde$0B| z)7NX3PttXe?X#l~S(mrpRHJ-%%KbOpzQ#H%{;eA2W19@Jy%}(y^}B>R~-&wPK(ilg-JoAus`q}4el<#qA ze51V&a;;DN-}Y+q@!WZB3E4q1`Y%}o*&ZFY#ya>$z}VtI1lo?keeULmAGbbo?A1(44}g4& zrcW-3KpO*$SCO0tPN3}*qP^`e4?S%?b7@y5#m|&4s+cooY{?1H)^%d5zqpcvxL3bxh=82l2UZJ`uZ8XS`wv;oKW~A2W?Kr$ zH<$XawJk+lU9v{BJ`jPnBd~rje6hj0>fK$C&tU5g`68yVeBF)xOS%iw0b08RplsFc zYpvZA%Q7i$K>vl0np6^CY-=;1|GEpvhqezI(8j=ft%i4sA)mqa4&>`E$roUI2l90n z0SC5Vjkg75!#{k;n)KbpJ;b z*;u|-2i9R7w+?MAyyXe&>FJv?DV_Juq6+^pu_f!UuFErolAH&AMB4{Z7MA&h_0Yn% zAYU8XA;|Y*Xk5ul20vRa)|xk=jvwBWJYev(g&2Koz7_-4uA{Jq9f`H<2&`#o|MOjG z|I@z5h`mo=ydP?beOjaL{ZR0^#1`~ENR*t#D`)T?tg-Kp@)5m*eehn$w@r?ciNA|`q>A2wg19;Joj+I_P%K$=L*`1e4p;v7k$U@6Ta^n zpFIM5yV82SgS1|!`e_FZ`MGz@J=gc6ZbK-%>C(80zQ>gKq>8>jDe-9)eMgHSQhTe! zPf7o^#}}=h<*>$oR$AlV1b(dT_ZZc+y`kr5+VT<)y!5w!Q~2L6-ds=NlK4pJ?Q0Gt z8o^n(4Trb1a=sXd66{N8w*sUy;^i}rx5U2JB*(YJL!>v;LYzZw^tr?N{qS}i@1;Zi z5n>-Q-G{@OCkluX&-bghtaVJss1<)XHJ3~)J_OO%J0SD%m9ScP86}m8;t=S3`U;-qtVCTY#I_3 zV>T6#ewkBg4@01ut8M>pEp-k#S&(vkfJX97!#I(f zuGVBE$^zBuG?;im^+$}x{(2<$Bczk4ULy8o`ysSoz!f1=yZ=Y!^rE`-yV*1*>+0;Q z#YdM`;Pb|jxW~wQdTUK$8`_~EFWrw27Hkw^V5Lc%*X2uffG*v;AcypsBKY%?tIlKW`y(s>FWwL^Q8IcO*8?ha2@nmU#cBZbOmj=$W(pL05Ei_X=yMQzh(mgayd7bGMbe8!CPx&OgP=SzFPqU0FreJ%H;t9IrZg{{4>wjF#%jdo)ESq^T8KerBo zoSpFsgsjXV^6~6J-Y58sx$Cm!=QenWvbEkZ=g;*N#U!TzGAni{=~R|ox@HTPcEN(Za>DJFWSAq(D77`mx%X?7=3=N z-M)?c6LvoV&q~17Hp=Vsj+0guNzW;?lx-c?miiHL9FyZ6jXBmp{~YW2GuPzfHp$O7 zpq%P;GmAy1jQETK>93Qe=P_s+E;!jH=KS0{2A!k)V>s`d)gu4iE9GklF(?u5OHa0s zIe#8@Yp~?tISpRRlYIoDKgmXpU9`uaT)yPo+{hN^&nzE&u4wgb4tg;_FX&r5pD_yk zJreys0{uT6y5aXJB#X1>G%8_3D3j*qV?9OwH|;Ur<^G0TL-{!nkss2^$0pF9>BHIVw2TE=A$^8f;^(GjrdpElx+=x8Ff(;Y zTE^lmOHx&(ewLK9tfbr1@%WNuc4lg7NFU3Rr1Uhb;LwDaAg!9T^@p_-qn9pBP0qGN zYO`@}#(c}2Y1wnr7Fg&r;IXOM_*{C#;@Kg6T8b*6AO@cCNU_|$EIT!;rSJ`3dR2P* zQhMy8rHIeSPD)3#f8=b-_}eXL2o*J=w#1Vm@YnD;(1h#?I8>HfN;W;fBe@tHF}FID zAK_v_>t@`49m+B8C6`!3d5&B@5arI=FO$nRK(V;jIkZ1X&Y|Z_uw-#4KaMv`Cuet=wl8bVX~ zafDR<1wtx6im)y6k8l{h9bo{{WgJGABBbkW<}jM+UDeNyy9I0GS; zxUthXjHZdA2l59Zr1~KUslEjv)$f21Z>eMHeKW%7vrq=rJI!JANrdpR$2p8XijeyI z2!}K2`6n-wujeqD>=SP#Lgyl+_vbU@bQD5rm!5Y*80{mMpN74fk^eo1(d7u~yds3C z5?jb&bPhs{i_nz_@m42xDTmS72r-w&F61ye9UA5C^(J4sdEktZ0htV?-c11hW z5R%;S97azxJMmfxp!$diZkwc3dBA|A>j7I+eZ)0FrZ)5mCZ{yOH z-o}O5^z4wPx(Z1nLq?)Ba8i#*?TzGl-|pL0GJeT>GFzTYFSv}IXKv#Hdt%x}9NTJYcCkH^CN za{P_Pv;OC(E5viKKka_W0>IIJiSqsweJ!Q)pj2|F@S^UhNZartz9Lzkbh&)I{C;bkcak*%4EJb>B^}sZF?cQ`U7?8f9le2>C(^X(wIZ5wLeBz{vTaAKlX+P+|rX5E)(?dbi$m} z?Bt9E(&OU^SqaIR$pa$saJ8JFtiwi9wz{8|AV0WA&)Cz`a_Lx|amzQSX7l=L%;E*f z(u$4qv2=3C&q~Oq)m>^zf@VRaR*zdeziNq%7p79lxw}7lR<2L&Vd@f-&;ZsAbW_ivKym8xk#s|c(?1`1 zjXZQraKErSF^C=X4a*;k|@*SUG@SFEM_7(f!*+^e7pf5Z4h@5tnCZ7^{ z-$lG~yAaF25NfD&R5(Yf>5Q`Ed@&bs=+2?5*YkN|28RykuR%Wa6r00gbRt5mbEW6= zVaL+?6@48_x~2SBgp@xHA+4W#Aw;9G77ms5s~4pqkF+YG&}g*uG8*ZbeH6gmi5IP7 z6t{@dXh<0cf6}GzlGCz*ig}cV;EJ77KhU~D;iaZ2t++9ATIsKjNNb8ksxCcCmyXe; zt7cfMG)Yz6T5+eGl$uE&QY9@&P92%C7BJ R|4RB8Dk33VRDB#J{vT;d3e*4q literal 0 HcmV?d00001 diff --git a/desktop/cores/sdl/SDL2.macos.arm/lib/libSDL2main.a b/desktop/cores/sdl/SDL2.macos.arm/lib/libSDL2main.a new file mode 100644 index 0000000000000000000000000000000000000000..1faff76ca806884df085ff19e0366fb84760bd89 GIT binary patch literal 736 zcmY$iNi0gvu;WrT)HgCvKmcU6sev(20wQf_U|?oqqM%>~SCGIJAFmf2>FeU^rV#8O z6yoZ_00bN$0tgy_SOSRSgI#>$a}zW3;)_d4lj4gjbCdFOfa0XuYieQvbz}k;$jFq^ z+}ujA!Fu^X;~38Wdi#%ufq@ak1`#YkoB?7e05K3q0`Utl3p=3!7486=z`y|GvjfFJ zmUa%YsIA~zXfclg3a|;raOX5rO$}{s) z;^Te5;RsUzGe-d`Fac;DGmyOjNJE{9V!RPh4ixSXIX)l@gduJKiSK~we*vUHe&hz? z03ZedkU<~}(hIVKJkStyfI0peOfS^^C~kzw0qtRv1##p`Pfq{< literal 0 HcmV?d00001 diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_assert.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_assert.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_assert.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_assert.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_atomic.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_atomic.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_atomic.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_atomic.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_audio.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_audio.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_audio.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_audio.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_bits.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_bits.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_bits.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_bits.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_blendmode.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_blendmode.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_blendmode.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_blendmode.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_clipboard.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_clipboard.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_clipboard.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_clipboard.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_config.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_config.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_config.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_config.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_cpuinfo.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_cpuinfo.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_cpuinfo.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_cpuinfo.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_egl.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_egl.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_egl.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_egl.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_endian.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_endian.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_endian.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_endian.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_error.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_error.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_error.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_error.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_events.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_events.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_events.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_events.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_filesystem.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_filesystem.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_filesystem.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_filesystem.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_gamecontroller.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_gamecontroller.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_gamecontroller.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_gamecontroller.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_gesture.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_gesture.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_gesture.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_gesture.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_haptic.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_haptic.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_haptic.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_haptic.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_hints.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_hints.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_hints.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_hints.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_joystick.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_joystick.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_joystick.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_joystick.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_keyboard.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_keyboard.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_keyboard.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_keyboard.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_keycode.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_keycode.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_keycode.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_keycode.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_loadso.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_loadso.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_loadso.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_loadso.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_log.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_log.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_log.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_log.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_main.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_main.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_main.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_main.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_messagebox.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_messagebox.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_messagebox.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_messagebox.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_metal.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_metal.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_metal.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_metal.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_mouse.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_mouse.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_mouse.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_mouse.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_mutex.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_mutex.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_mutex.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_mutex.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_name.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_name.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_name.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_name.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_opengl.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengl.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_opengl.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengl.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_opengl_glext.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengl_glext.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_opengl_glext.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengl_glext.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_opengles.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengles.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_opengles.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengles.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_opengles2.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengles2.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_opengles2.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengles2.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_opengles2_gl2.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengles2_gl2.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_opengles2_gl2.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengles2_gl2.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_opengles2_gl2ext.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengles2_gl2ext.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_opengles2_gl2ext.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengles2_gl2ext.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_opengles2_gl2platform.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengles2_gl2platform.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_opengles2_gl2platform.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengles2_gl2platform.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_opengles2_khrplatform.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengles2_khrplatform.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_opengles2_khrplatform.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_opengles2_khrplatform.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_pixels.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_pixels.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_pixels.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_pixels.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_platform.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_platform.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_platform.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_platform.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_power.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_power.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_power.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_power.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_quit.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_quit.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_quit.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_quit.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_rect.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_rect.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_rect.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_rect.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_render.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_render.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_render.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_render.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_revision.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_revision.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_revision.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_revision.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_rwops.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_rwops.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_rwops.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_rwops.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_scancode.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_scancode.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_scancode.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_scancode.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_sensor.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_sensor.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_sensor.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_sensor.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_shape.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_shape.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_shape.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_shape.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_stdinc.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_stdinc.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_stdinc.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_stdinc.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_surface.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_surface.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_surface.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_surface.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_system.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_system.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_system.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_system.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_syswm.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_syswm.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_syswm.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_syswm.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_test.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_test.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_test.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_test.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_test_assert.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_assert.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_test_assert.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_assert.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_test_common.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_common.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_test_common.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_common.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_test_compare.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_compare.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_test_compare.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_compare.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_test_crc32.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_crc32.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_test_crc32.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_crc32.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_test_font.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_font.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_test_font.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_font.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_test_fuzzer.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_fuzzer.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_test_fuzzer.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_fuzzer.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_test_harness.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_harness.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_test_harness.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_harness.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_test_images.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_images.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_test_images.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_images.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_test_log.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_log.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_test_log.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_log.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_test_md5.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_md5.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_test_md5.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_md5.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_test_memory.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_memory.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_test_memory.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_memory.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_test_random.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_random.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_test_random.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_test_random.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_thread.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_thread.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_thread.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_thread.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_timer.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_timer.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_timer.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_timer.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_touch.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_touch.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_touch.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_touch.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_types.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_types.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_types.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_types.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_version.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_version.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_version.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_version.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_video.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_video.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_video.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_video.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/SDL_vulkan.h b/desktop/cores/sdl/SDL2.macos.intel/include/SDL_vulkan.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/SDL_vulkan.h rename to desktop/cores/sdl/SDL2.macos.intel/include/SDL_vulkan.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/begin_code.h b/desktop/cores/sdl/SDL2.macos.intel/include/begin_code.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/begin_code.h rename to desktop/cores/sdl/SDL2.macos.intel/include/begin_code.h diff --git a/desktop/cores/sdl/SDL2.macosx/include/close_code.h b/desktop/cores/sdl/SDL2.macos.intel/include/close_code.h similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/include/close_code.h rename to desktop/cores/sdl/SDL2.macos.intel/include/close_code.h diff --git a/desktop/cores/sdl/SDL2.macosx/lib/libSDL2-2.0.0.dylib b/desktop/cores/sdl/SDL2.macos.intel/lib/libSDL2-2.0.0.dylib similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/lib/libSDL2-2.0.0.dylib rename to desktop/cores/sdl/SDL2.macos.intel/lib/libSDL2-2.0.0.dylib diff --git a/desktop/cores/sdl/SDL2.macosx/lib/libSDL2.a b/desktop/cores/sdl/SDL2.macos.intel/lib/libSDL2.a similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/lib/libSDL2.a rename to desktop/cores/sdl/SDL2.macos.intel/lib/libSDL2.a diff --git a/desktop/cores/sdl/SDL2.macosx/lib/libSDL2.dylib b/desktop/cores/sdl/SDL2.macos.intel/lib/libSDL2.dylib similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/lib/libSDL2.dylib rename to desktop/cores/sdl/SDL2.macos.intel/lib/libSDL2.dylib diff --git a/desktop/cores/sdl/SDL2.macosx/lib/libSDL2_test.a b/desktop/cores/sdl/SDL2.macos.intel/lib/libSDL2_test.a similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/lib/libSDL2_test.a rename to desktop/cores/sdl/SDL2.macos.intel/lib/libSDL2_test.a diff --git a/desktop/cores/sdl/SDL2.macosx/lib/libSDL2main.a b/desktop/cores/sdl/SDL2.macos.intel/lib/libSDL2main.a similarity index 100% rename from desktop/cores/sdl/SDL2.macosx/lib/libSDL2main.a rename to desktop/cores/sdl/SDL2.macos.intel/lib/libSDL2main.a diff --git a/desktop/libraries/KlangWellen/examples/01.Wavetable/01.Wavetable.ino b/desktop/libraries/KlangWellen/examples/01.Wavetable/01.Wavetable.ino index 72de4e3b..1a21d3d8 100644 --- a/desktop/libraries/KlangWellen/examples/01.Wavetable/01.Wavetable.ino +++ b/desktop/libraries/KlangWellen/examples/01.Wavetable/01.Wavetable.ino @@ -7,6 +7,11 @@ using namespace klangwellen; Wavetable fWavetable; void setup() { + Serial.begin(115200); + Serial.println("------------"); + Serial.println("01.Wavetable"); + Serial.println("------------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SAWTOOTH, 16); fWavetable.set_frequency(55); fWavetable.set_amplitude(0.5); @@ -19,4 +24,4 @@ void audioblock(float** input_signal, float** output_signal) { output_signal[LEFT][i] = fWavetable.process(); } KlangWellen::copy(output_signal[LEFT], output_signal[RIGHT]); -} \ No newline at end of file +} diff --git a/desktop/libraries/KlangWellen/examples/02.ADSR/02.ADSR.ino b/desktop/libraries/KlangWellen/examples/02.ADSR/02.ADSR.ino index adfe4cc7..67a16c88 100644 --- a/desktop/libraries/KlangWellen/examples/02.ADSR/02.ADSR.ino +++ b/desktop/libraries/KlangWellen/examples/02.ADSR/02.ADSR.ino @@ -9,6 +9,11 @@ Wavetable fWavetable; ADSR fADSR; void setup() { + Serial.begin(115200); + Serial.println("-------"); + Serial.println("02.ADSR"); + Serial.println("-------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SINE); } diff --git a/desktop/libraries/KlangWellen/examples/03.LowPassFilter/03.LowPassFilter.ino b/desktop/libraries/KlangWellen/examples/03.LowPassFilter/03.LowPassFilter.ino index 9cc9ac65..dc806717 100644 --- a/desktop/libraries/KlangWellen/examples/03.LowPassFilter/03.LowPassFilter.ino +++ b/desktop/libraries/KlangWellen/examples/03.LowPassFilter/03.LowPassFilter.ino @@ -8,6 +8,11 @@ Wavetable fWavetable; FilterLowPassMoogLadder fFilter; void setup() { + Serial.begin(115200); + Serial.println("----------------"); + Serial.println("03.LowPassFilter"); + Serial.println("----------------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SQUARE); fWavetable.set_frequency(55); fFilter.set_resonance(0.85f); diff --git a/desktop/libraries/KlangWellen/examples/04.Reverb/04.Reverb.ino b/desktop/libraries/KlangWellen/examples/04.Reverb/04.Reverb.ino index a9ee38c3..6561e9da 100644 --- a/desktop/libraries/KlangWellen/examples/04.Reverb/04.Reverb.ino +++ b/desktop/libraries/KlangWellen/examples/04.Reverb/04.Reverb.ino @@ -11,6 +11,11 @@ ADSR fADSR; Reverb fReverb; void setup() { + Serial.begin(115200); + Serial.println("---------"); + Serial.println("04.Reverb"); + Serial.println("---------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SINE); fReverb.set_roomsize(0.9); } diff --git a/desktop/libraries/KlangWellen/examples/05.FormantFilter/05.FormantFilter.ino b/desktop/libraries/KlangWellen/examples/05.FormantFilter/05.FormantFilter.ino index 0af62dd1..5e58a2df 100644 --- a/desktop/libraries/KlangWellen/examples/05.FormantFilter/05.FormantFilter.ino +++ b/desktop/libraries/KlangWellen/examples/05.FormantFilter/05.FormantFilter.ino @@ -8,6 +8,11 @@ Wavetable fWavetable; FilterVowelFormant fFilter; void setup() { + Serial.begin(115200); + Serial.println("----------------"); + Serial.println("05.FormantFilter"); + Serial.println("----------------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SAWTOOTH); fWavetable.set_frequency(35); fWavetable.set_amplitude(0.125f); diff --git a/desktop/libraries/KlangWellen/examples/06.SAM/06.SAM.ino b/desktop/libraries/KlangWellen/examples/06.SAM/06.SAM.ino index 0505aa7f..3c96c193 100644 --- a/desktop/libraries/KlangWellen/examples/06.SAM/06.SAM.ino +++ b/desktop/libraries/KlangWellen/examples/06.SAM/06.SAM.ino @@ -10,6 +10,11 @@ SAM fSAM_left(fBuffer, 48000); SAM fSAM_right(48000); void setup() { + Serial.begin(115200); + Serial.println("------"); + Serial.println("06.SAM"); + Serial.println("------"); + fSAM_right.set_speed(120); fSAM_right.set_throat(100); beats_per_minute(60); diff --git a/desktop/libraries/KlangWellen/examples/07.Sampler/07.Sampler.ino b/desktop/libraries/KlangWellen/examples/07.Sampler/07.Sampler.ino index 90ddc012..0dca2a05 100644 --- a/desktop/libraries/KlangWellen/examples/07.Sampler/07.Sampler.ino +++ b/desktop/libraries/KlangWellen/examples/07.Sampler/07.Sampler.ino @@ -6,8 +6,13 @@ using namespace klangwellen; Sampler fSampler{KlangWellen::DEFAULT_SAMPLING_RATE / 8}; void setup() { - for (size_t i = 0; i < fSampler.get_buffer_length(); i++) { - float ratio = 1.0 - (float)i / fSampler.get_buffer_length(); + Serial.begin(115200); + Serial.println("----------"); + Serial.println("07.Sampler"); + Serial.println("----------"); + + for (int32_t i = 0; i < fSampler.get_buffer_length(); i++) { + float ratio = 1.0 - (float)i / fSampler.get_buffer_length(); // fSampler.get_buffer()[i] = KlangWellen::random() * 0.2f * ratio * 127; // for SamplerI8 fSampler.get_buffer()[i] = KlangWellen::random() * 0.2f * ratio; } diff --git a/desktop/libraries/KlangWellen/examples/08.Delay/08.Delay.ino b/desktop/libraries/KlangWellen/examples/08.Delay/08.Delay.ino index 49a3abc4..e92e9746 100644 --- a/desktop/libraries/KlangWellen/examples/08.Delay/08.Delay.ino +++ b/desktop/libraries/KlangWellen/examples/08.Delay/08.Delay.ino @@ -1,3 +1,5 @@ +// @TODO Delay is broken, fix it! + #include "ADSR.h" #include "Delay.h" #include "KlangWellen.h" @@ -12,6 +14,11 @@ ADSR fADSR; Delay fDelay; void setup() { + Serial.begin(115200); + Serial.println("--------"); + Serial.println("08.Delay"); + Serial.println("--------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_TRIANGLE); fWavetable.set_amplitude(0.4); fDelay.set_echo_length(0.05); @@ -34,7 +41,7 @@ void beat(uint32_t beat_counter) { } void audioblock(float** input_signal, float** output_signal) { - // /* process as float ( mono, single sample ) ... */ + /* process as float ( mono, single sample ) ... */ // for (uint16_t i = 0; i < KLANG_SAMPLES_PER_AUDIO_BLOCK; i++) { // output_signal[LEFT][i] = fDelay.process(fADSR.process(fWavetable.process())); // output_signal[RIGHT][i] = output_signal[LEFT][i]; diff --git a/desktop/libraries/KlangWellen/examples/09.LFO/09.LFO.ino b/desktop/libraries/KlangWellen/examples/09.LFO/09.LFO.ino index 91599add..72d111c5 100644 --- a/desktop/libraries/KlangWellen/examples/09.LFO/09.LFO.ino +++ b/desktop/libraries/KlangWellen/examples/09.LFO/09.LFO.ino @@ -9,6 +9,11 @@ Wavetable fWavetable; FilterLowPassMoogLadder fFilter; void setup() { + Serial.begin(115200); + Serial.println("------"); + Serial.println("09.LFO"); + Serial.println("------"); + fLFO.set_waveform(KlangWellen::WAVEFORM_SINE); fLFO.set_oscillation_range(55, 1000); fLFO.set_frequency(0.5); diff --git a/desktop/libraries/KlangWellen/examples/10.BeatDSP/10.BeatDSP.ino b/desktop/libraries/KlangWellen/examples/10.BeatDSP/10.BeatDSP.ino index 8dff8030..4072cc9c 100644 --- a/desktop/libraries/KlangWellen/examples/10.BeatDSP/10.BeatDSP.ino +++ b/desktop/libraries/KlangWellen/examples/10.BeatDSP/10.BeatDSP.ino @@ -13,6 +13,11 @@ ADSR fADSR_DSP; BeatDSP fBeat; void setup() { + Serial.begin(115200); + Serial.println("----------"); + Serial.println("10.BeatDSP"); + Serial.println("----------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SINE); fWavetable.set_frequency(220.0); fWavetable_DSP.set_waveform(KlangWellen::WAVEFORM_TRIANGLE); diff --git a/desktop/libraries/KlangWellen/examples/11.Trigger/11.Trigger.ino b/desktop/libraries/KlangWellen/examples/11.Trigger/11.Trigger.ino index 2840b9c6..4cf5a49c 100644 --- a/desktop/libraries/KlangWellen/examples/11.Trigger/11.Trigger.ino +++ b/desktop/libraries/KlangWellen/examples/11.Trigger/11.Trigger.ino @@ -14,6 +14,11 @@ ADSR fADSR_DSP; Trigger fTrigger; void setup() { + Serial.begin(115200); + Serial.println("----------"); + Serial.println("11.Trigger"); + Serial.println("----------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SINE); fWavetable.set_frequency(220.0); fWavetable_DSP.set_waveform(KlangWellen::WAVEFORM_TRIANGLE); diff --git a/desktop/libraries/KlangWellen/examples/12.Clamp/12.Clamp.ino b/desktop/libraries/KlangWellen/examples/12.Clamp/12.Clamp.ino index 60de9f8f..a6953541 100644 --- a/desktop/libraries/KlangWellen/examples/12.Clamp/12.Clamp.ino +++ b/desktop/libraries/KlangWellen/examples/12.Clamp/12.Clamp.ino @@ -11,6 +11,11 @@ ADSR fADSR; Clamp fClamp; void setup() { + Serial.begin(115200); + Serial.println("--------"); + Serial.println("12.Clamp"); + Serial.println("--------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SINE); fClamp.set_min(-0.1); fClamp.set_max(0.1); diff --git a/desktop/libraries/KlangWellen/examples/13.Vocoder/13.Vocoder.ino b/desktop/libraries/KlangWellen/examples/13.Vocoder/13.Vocoder.ino index ac45a65c..446c7e6f 100644 --- a/desktop/libraries/KlangWellen/examples/13.Vocoder/13.Vocoder.ino +++ b/desktop/libraries/KlangWellen/examples/13.Vocoder/13.Vocoder.ino @@ -9,9 +9,14 @@ using namespace klangwellen; SAM fSAM(48000); Wavetable fWavetable; -Vocoder fVocoder{13, 3}; // this still works on KLST_SHEEP but only with `fastest` compiler option +Vocoder fVocoder{13, 3}; // this still works on KLST_SHEEP but only with `fastest` compiler option void setup() { + Serial.begin(115200); + Serial.println("----------"); + Serial.println("13.Vocoder"); + Serial.println("----------"); + fSAM.speak("hello world"); fWavetable.set_waveform(KlangWellen::WAVEFORM_SAWTOOTH); diff --git a/desktop/libraries/KlangWellen/examples/14.Filters/14.Filters.ino b/desktop/libraries/KlangWellen/examples/14.Filters/14.Filters.ino index 5401c11d..d2ffe693 100644 --- a/desktop/libraries/KlangWellen/examples/14.Filters/14.Filters.ino +++ b/desktop/libraries/KlangWellen/examples/14.Filters/14.Filters.ino @@ -8,6 +8,11 @@ Wavetable fWavetable; Filter fFilter; void setup() { + Serial.begin(115200); + Serial.println("----------"); + Serial.println("14.Filters"); + Serial.println("----------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SQUARE); fWavetable.set_frequency(55); klangstrom::beats_per_minute(120 * 4); diff --git a/desktop/libraries/KlangWellen/examples/15.Ramp/15.Ramp.ino b/desktop/libraries/KlangWellen/examples/15.Ramp/15.Ramp.ino index db8df4e4..a5fb8ace 100644 --- a/desktop/libraries/KlangWellen/examples/15.Ramp/15.Ramp.ino +++ b/desktop/libraries/KlangWellen/examples/15.Ramp/15.Ramp.ino @@ -12,6 +12,11 @@ Ramp fRampFilterFrequency; Ramp fRampFrequency; void setup() { + Serial.begin(115200); + Serial.println("-------"); + Serial.println("15.Ramp"); + Serial.println("-------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SAWTOOTH); fWavetable.set_frequency(27.5); fFilter.set_resonance(0.3); diff --git a/desktop/libraries/KlangWellen/examples/16.Envelope/16.Envelope.ino b/desktop/libraries/KlangWellen/examples/16.Envelope/16.Envelope.ino index a8d396c8..e904bd24 100644 --- a/desktop/libraries/KlangWellen/examples/16.Envelope/16.Envelope.ino +++ b/desktop/libraries/KlangWellen/examples/16.Envelope/16.Envelope.ino @@ -10,6 +10,11 @@ Envelope fFrequencyEnvelope; Envelope fAmplitudeEnvelope; void setup() { + Serial.begin(115200); + Serial.println("-----------"); + Serial.println("16.Envelope"); + Serial.println("-----------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SAWTOOTH); fWavetable.set_frequency(55.0); fWavetable.set_amplitude(0.0); diff --git a/desktop/libraries/KlangWellen/examples/17.Waveshaper/17.Waveshaper.ino b/desktop/libraries/KlangWellen/examples/17.Waveshaper/17.Waveshaper.ino index 59036be8..4b2e2693 100644 --- a/desktop/libraries/KlangWellen/examples/17.Waveshaper/17.Waveshaper.ino +++ b/desktop/libraries/KlangWellen/examples/17.Waveshaper/17.Waveshaper.ino @@ -9,6 +9,11 @@ Waveshaper fWaveshaper; uint8_t fWaveshaperType = Waveshaper::SIN; void setup() { + Serial.begin(115200); + Serial.println("-------------"); + Serial.println("17.Waveshaper"); + Serial.println("-------------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SINE); fWavetable.set_amplitude(4.0); fWavetable.set_frequency(55); diff --git a/desktop/libraries/KlangWellen/src/ADSR.h b/desktop/libraries/KlangWellen/src/ADSR.h index 532a6565..f755b56f 100644 --- a/desktop/libraries/KlangWellen/src/ADSR.h +++ b/desktop/libraries/KlangWellen/src/ADSR.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/desktop/libraries/KlangWellen/src/BeatDSP.h b/desktop/libraries/KlangWellen/src/BeatDSP.h index 2b70c730..1240e7a3 100644 --- a/desktop/libraries/KlangWellen/src/BeatDSP.h +++ b/desktop/libraries/KlangWellen/src/BeatDSP.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/desktop/libraries/KlangWellen/src/Clamp.h b/desktop/libraries/KlangWellen/src/Clamp.h index e7ee8d90..874cf422 100644 --- a/desktop/libraries/KlangWellen/src/Clamp.h +++ b/desktop/libraries/KlangWellen/src/Clamp.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/desktop/libraries/KlangWellen/src/Delay.h b/desktop/libraries/KlangWellen/src/Delay.h index b0eaab96..d2b1c822 100644 --- a/desktop/libraries/KlangWellen/src/Delay.h +++ b/desktop/libraries/KlangWellen/src/Delay.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -27,6 +27,8 @@ * - [ ] void process(float*, float*, uint32_t) */ +// @TODO Delay is broken, fix it + #pragma once #include @@ -46,8 +48,7 @@ namespace klangwellen { * @param sample_rate the sample rate in Hz. * @param decay_rate the decay of the echo, a value between 0 and 1. 1 meaning no decay, 0 means immediate decay */ - Delay(float echo_length = 0.5, float decay_rate = 0.75, float wet = 0.8, uint32_t sample_rate = KlangWellen::DEFAULT_SAMPLING_RATE) { - fSampleRate = sample_rate; + Delay(float echo_length = 0.5, float decay_rate = 0.75, float wet = 0.8, uint32_t sample_rate = KlangWellen::DEFAULT_SAMPLING_RATE) : fSampleRate(sample_rate) { set_decay_rate(decay_rate); set_echo_length(echo_length); set_wet(wet); @@ -105,14 +106,14 @@ namespace klangwellen { } private: - int32_t fBufferPosition; - float fDecayRate; - float fWet; - float* fBuffer; - int32_t fBufferLength; - float fNewEchoLength; + int32_t fBufferPosition = 0; + float fDecayRate = 0; + float fWet = 0; + float* fBuffer = nullptr; + bool fAllocatedBuffer = false; + int32_t fBufferLength = 0; + float fNewEchoLength = 0; uint32_t fSampleRate; - bool fAllocatedBuffer; void adaptEchoLength() { if (fNewEchoLength > 0) { diff --git a/desktop/libraries/KlangWellen/src/Envelope.h b/desktop/libraries/KlangWellen/src/Envelope.h index c98ef281..49c4c3a0 100644 --- a/desktop/libraries/KlangWellen/src/Envelope.h +++ b/desktop/libraries/KlangWellen/src/Envelope.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -130,14 +130,15 @@ namespace klangwellen { */ float process() { if (!mEnvelopeDone) { - if (mEnvStage < mEnvelopeStages.size()) { + const int mNumberOfStages = mEnvelopeStages.size(); + if (mEnvStage < mNumberOfStages) { mValue += mTimeScale * mDelta; mStageDuration += mTimeScale * 1.0f / mSamplingRate; if (mStageDuration > mEnvelopeStages[mEnvStage].duration) { const float mRemainder = mStageDuration - mEnvelopeStages[mEnvStage].duration; finished_stage(mEnvStage); mEnvStage++; - if (mEnvStage < mEnvelopeStages.size() - 1) { + if (mEnvStage < mNumberOfStages - 1) { prepareNextStage(mEnvStage, mRemainder); } else { stop(); diff --git a/desktop/libraries/KlangWellen/src/Filter.h b/desktop/libraries/KlangWellen/src/Filter.h index ee76426b..c6625f1a 100644 --- a/desktop/libraries/KlangWellen/src/Filter.h +++ b/desktop/libraries/KlangWellen/src/Filter.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://githubiquad_com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/desktop/libraries/KlangWellen/src/FilterLowPassMoogLadder.h b/desktop/libraries/KlangWellen/src/FilterLowPassMoogLadder.h index 0cbe3724..b7c177c7 100644 --- a/desktop/libraries/KlangWellen/src/FilterLowPassMoogLadder.h +++ b/desktop/libraries/KlangWellen/src/FilterLowPassMoogLadder.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/desktop/libraries/KlangWellen/src/FilterVowelFormant.h b/desktop/libraries/KlangWellen/src/FilterVowelFormant.h index 976b5651..001391f4 100644 --- a/desktop/libraries/KlangWellen/src/FilterVowelFormant.h +++ b/desktop/libraries/KlangWellen/src/FilterVowelFormant.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/desktop/libraries/KlangWellen/src/Gain.h b/desktop/libraries/KlangWellen/src/Gain.h index 4d5f55ba..95eb0d16 100644 --- a/desktop/libraries/KlangWellen/src/Gain.h +++ b/desktop/libraries/KlangWellen/src/Gain.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/desktop/libraries/KlangWellen/src/KlangWellen.cpp b/desktop/libraries/KlangWellen/src/KlangWellen.cpp deleted file mode 100644 index 1e4c4ec4..00000000 --- a/desktop/libraries/KlangWellen/src/KlangWellen.cpp +++ /dev/null @@ -1,31 +0,0 @@ -#include "KlangWellen.h" - -/* xorshift32 ( ref: https://en.wikipedia.org/wiki/Xorshift ) */ -uint32_t klangwellen::KlangWellen::x32Seed = 23; -uint32_t klangwellen::KlangWellen::xorshift32() { - x32Seed ^= x32Seed << 13; - x32Seed ^= x32Seed >> 17; - x32Seed ^= x32Seed << 5; - return x32Seed; -} - -/** - * returns a random number between 0 ... 1 - */ -float klangwellen::KlangWellen::random_normalized() { - // TODO replace with mapping, without division - return ((float)xorshift32() / UINT32_MAX); -} - -float klangwellen::KlangWellen::random() { - // TODO replace with mapping, without division - return ((float)xorshift32() / UINT32_MAX) * 2 - 1; -} - -uint32_t klangwellen::KlangWellen::millis_to_samples(float pMillis, float pSamplingRate) { - return (uint32_t)(pMillis / 1000.0f * pSamplingRate); -} - -uint32_t klangwellen::KlangWellen::millis_to_samples(float pMillis) { - return (uint32_t)(pMillis / 1000.0f * (float)klangwellen::KlangWellen::DEFAULT_SAMPLING_RATE); -} diff --git a/desktop/libraries/KlangWellen/src/KlangWellen.h b/desktop/libraries/KlangWellen/src/KlangWellen.h index 41ace193..9d384dae 100755 --- a/desktop/libraries/KlangWellen/src/KlangWellen.h +++ b/desktop/libraries/KlangWellen/src/KlangWellen.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -30,6 +30,11 @@ #define KLANG_SAMPLES_PER_AUDIO_BLOCK 512 #endif +#ifndef KLANG_SAMPLING_RATE +#warning KLANG_SAMPLING_RATE not defined - using default value of 48000 +#define KLANG_SAMPLING_RATE 48000 +#endif + #ifndef PI #define PI M_PI #endif @@ -45,238 +50,331 @@ namespace klangwellen { class KlangWellen { public: - static const uint8_t BITS_PER_SAMPLE_16 = 16; - static const uint8_t BITS_PER_SAMPLE_24 = 24; - static const uint8_t BITS_PER_SAMPLE_32 = 32; - static const uint8_t BITS_PER_SAMPLE_8 = 8; - static constexpr float DEFAULT_ATTACK = 0.005f; - static const int DEFAULT_AUDIOBLOCK_SIZE = 1024; - static const int DEFAULT_AUDIO_DEVICE = -1; - static const uint8_t DEFAULT_BITS_PER_SAMPLE = BITS_PER_SAMPLE_16; - static constexpr float DEFAULT_DECAY = 0.01f; - static constexpr float DEFAULT_FILTER_BANDWIDTH = 100.0f; - static constexpr float DEFAULT_FILTER_FREQUENCY = 1000.0f; - static constexpr float DEFAULT_RELEASE = 0.075f; - static constexpr uint32_t DEFAULT_SAMPLING_RATE = 48000; - static constexpr uint32_t DEFAULT_INTERPOLATE_AMP_FREQ_DURATION = 5.f / 1000.f * (float)DEFAULT_SAMPLING_RATE; // KlangWellen::millis_to_samples(5); - static constexpr float DEFAULT_SUSTAIN = 0.5f; - static const int DEFAULT_WAVETABLE_SIZE = 512; - static const uint8_t DISTORTION_HARD_CLIPPING = 0; - static const uint8_t DISTORTION_FOLDBACK = 1; - static const uint8_t DISTORTION_FOLDBACK_SINGLE = 2; - static const uint8_t DISTORTION_FULL_WAVE_RECTIFICATION = 3; - static const uint8_t DISTORTION_HALF_WAVE_RECTIFICATION = 4; - static const uint8_t DISTORTION_INFINITE_CLIPPING = 5; - static const uint8_t DISTORTION_SOFT_CLIPPING_CUBIC = 6; - static const uint8_t DISTORTION_SOFT_CLIPPING_ARC_TANGENT = 7; - static const uint8_t DISTORTION_BIT_CRUSHING = 8; - static const int EVENT_UNDEFINED = -1; - static const int EVENT_CHANNEL = 0; - static const int EVENT_NOTE_ON = 0; - static const int EVENT_NOTE = 1; - static const int EVENT_NOTE_OFF = 1; - static const int EVENT_CONTROLCHANGE = 2; - static const int EVENT_PITCHBEND = 3; - static const int EVENT_PROGRAMCHANGE = 4; - static const int EVENT_VELOCITY = 2; - static const uint8_t FILTER_MODE_LOW_PASS = 0; - static const uint8_t FILTER_MODE_HIGH_PASS = 1; - static const uint8_t FILTER_MODE_BAND_PASS = 2; - static const uint8_t FILTER_MODE_NOTCH = 3; - static const uint8_t FILTER_MODE_PEAK = 4; - static const uint8_t FILTER_MODE_LOWSHELF = 5; - static const uint8_t FILTER_MODE_HIGHSHELF = 6; - static const uint8_t FILTER_MODE_BAND_REJECT = 7; - static const int LOOP_INFINITE = std::numeric_limits::max(); - static const uint8_t MONO = 1; - static const uint8_t NOISE_WHITE = 0; - static const uint8_t NOISE_GAUSSIAN_WHITE = 1; - static const uint8_t NOISE_GAUSSIAN_WHITE2 = 2; - static const uint8_t NOISE_PINK = 3; - static const uint8_t NOISE_PINK2 = 4; - static const uint8_t NOISE_PINK3 = 5; - static const uint8_t NOISE_SIMPLEX = 6; - static constexpr float NOTE_WHOLE = 0.25f; - static constexpr float NOTE_HALF = 0.5f; - static const uint8_t NOTE_QUARTER = 1; - static const uint8_t NOTE_EIGHTH = 2; - static const uint8_t NOTE_SIXTEENTH = 4; - static const uint8_t NOTE_THIRTYSECOND = 8; - static const int NO_AUDIO_DEVICE = -2; - static const int NO_CHANNELS = 0; - static const int NO_EVENT = -1; - static const int NO_INPOINT = 0; - static const int NO_LOOP = -2; - static const int NO_LOOP_COUNT = -1; - static const int NO_OUTPOINT = -1; - static const int NO_POSITION = -1; - static const int NO_VALUE = -1; - static const uint8_t PAN_LINEAR = 0; - static const uint8_t PAN_SINE_LAW = 2; - static const uint8_t PAN_SQUARE_LAW = 1; - static const uint8_t SIGNAL_LEFT = 0; - static constexpr float SIGNAL_MAX = 1.0f; - static constexpr float SIGNAL_MIN = -1.0f; - static const int SIGNAL_MONO = 1; - static const int SIGNAL_PROCESSING_IGNORE_IN_OUTPOINTS = -3; - static const int SIGNAL_RIGHT = 1; - static const int SIGNAL_STEREO = 2; - static const uint8_t SIG_INT16_BIG_ENDIAN = 2; - static const uint8_t SIG_INT16_LITTLE_ENDIAN = 3; - static const uint8_t SIG_INT24_3_BIG_ENDIAN = 4; - static const uint8_t SIG_INT24_3_LITTLE_ENDIAN = 5; - static const uint8_t SIG_INT24_4_BIG_ENDIAN = 6; - static const uint8_t SIG_INT24_4_LITTLE_ENDIAN = 7; - static const uint8_t SIG_INT32_BIG_ENDIAN = 8; - static const uint8_t SIG_INT32_LITTLE_ENDIAN = 9; - static const uint8_t SIG_INT8 = 0; - static const uint8_t SIG_UINT8 = 1; - static const uint8_t STEREO = 2; - static const uint8_t VERSION_MAJOR = 0; - static const uint8_t VERSION_MINOR = 8; - static const uint8_t WAVEFORM_SINE = 0; - static const uint8_t WAVEFORM_TRIANGLE = 1; - static const uint8_t WAVEFORM_SAWTOOTH = 2; - static const uint8_t WAVEFORM_SQUARE = 3; - static const uint8_t WAVEFORM_NOISE = 4; - static const uint8_t WAVESHAPE_INTERPOLATE_NONE = 0; - static const uint8_t WAVESHAPE_INTERPOLATE_LINEAR = 1; - static const uint8_t WAVESHAPE_INTERPOLATE_CUBIC = 2; - static const uint8_t WAV_FORMAT_PCM = 1; - static const uint8_t WAV_FORMAT_IEEE_FLOAT_32BIT = 3; + static const uint8_t BITS_PER_SAMPLE_16 = 16; + static const uint8_t BITS_PER_SAMPLE_24 = 24; + static const uint8_t BITS_PER_SAMPLE_32 = 32; + static const uint8_t BITS_PER_SAMPLE_8 = 8; + static constexpr float DEFAULT_ATTACK = 0.005f; + static const int DEFAULT_AUDIOBLOCK_SIZE = KLANG_SAMPLES_PER_AUDIO_BLOCK; // TODO decide for either one + static const int DEFAULT_AUDIO_DEVICE = -1; + static const uint8_t DEFAULT_BITS_PER_SAMPLE = BITS_PER_SAMPLE_16; + static constexpr float DEFAULT_DECAY = 0.01f; + static constexpr float DEFAULT_FILTER_BANDWIDTH = 100.0f; + static constexpr float DEFAULT_FILTER_FREQUENCY = 1000.0f; + static constexpr float DEFAULT_RELEASE = 0.075f; + static constexpr uint32_t DEFAULT_SAMPLING_RATE = KLANG_SAMPLING_RATE; // TODO decide for either one + static constexpr uint32_t DEFAULT_INTERPOLATE_AMP_FREQ_DURATION = 5.f / 1000.f * (float) DEFAULT_SAMPLING_RATE; + // KlangWellen::millis_to_samples(5); + static constexpr float DEFAULT_SUSTAIN = 0.5f; + static const int DEFAULT_WAVETABLE_SIZE = 512; + static const uint8_t DISTORTION_HARD_CLIPPING = 0; + static const uint8_t DISTORTION_FOLDBACK = 1; + static const uint8_t DISTORTION_FOLDBACK_SINGLE = 2; + static const uint8_t DISTORTION_FULL_WAVE_RECTIFICATION = 3; + static const uint8_t DISTORTION_HALF_WAVE_RECTIFICATION = 4; + static const uint8_t DISTORTION_INFINITE_CLIPPING = 5; + static const uint8_t DISTORTION_SOFT_CLIPPING_CUBIC = 6; + static const uint8_t DISTORTION_SOFT_CLIPPING_ARC_TANGENT = 7; + static const uint8_t DISTORTION_BIT_CRUSHING = 8; + static const int EVENT_UNDEFINED = -1; + static const int EVENT_CHANNEL = 0; + static const int EVENT_NOTE_ON = 0; + static const int EVENT_NOTE = 1; + static const int EVENT_NOTE_OFF = 1; + static const int EVENT_CONTROLCHANGE = 2; + static const int EVENT_PITCHBEND = 3; + static const int EVENT_PROGRAMCHANGE = 4; + static const int EVENT_VELOCITY = 2; + static const uint8_t FILTER_MODE_LOW_PASS = 0; + static const uint8_t FILTER_MODE_HIGH_PASS = 1; + static const uint8_t FILTER_MODE_BAND_PASS = 2; + static const uint8_t FILTER_MODE_NOTCH = 3; + static const uint8_t FILTER_MODE_PEAK = 4; + static const uint8_t FILTER_MODE_LOWSHELF = 5; + static const uint8_t FILTER_MODE_HIGHSHELF = 6; + static const uint8_t FILTER_MODE_BAND_REJECT = 7; + static const int LOOP_INFINITE = std::numeric_limits::max(); + static const uint8_t MONO = 1; + static const uint8_t NOISE_WHITE = 0; + static const uint8_t NOISE_GAUSSIAN_WHITE = 1; + static const uint8_t NOISE_GAUSSIAN_WHITE2 = 2; + static const uint8_t NOISE_PINK = 3; + static const uint8_t NOISE_PINK2 = 4; + static const uint8_t NOISE_PINK3 = 5; + static const uint8_t NOISE_SIMPLEX = 6; + static constexpr float NOTE_WHOLE = 0.25f; + static constexpr float NOTE_HALF = 0.5f; + static const uint8_t NOTE_QUARTER = 1; + static const uint8_t NOTE_EIGHTH = 2; + static const uint8_t NOTE_SIXTEENTH = 4; + static const uint8_t NOTE_THIRTYSECOND = 8; + static const int NO_AUDIO_DEVICE = -2; + static const int NO_CHANNELS = 0; + static const int NO_EVENT = -1; + static const int NO_INPOINT = 0; + static const int NO_LOOP = -2; + static const int NO_LOOP_COUNT = -1; + static const int NO_OUTPOINT = -1; + static const int NO_POSITION = -1; + static const int NO_VALUE = -1; + static const uint8_t PAN_LINEAR = 0; + static const uint8_t PAN_SINE_LAW = 2; + static const uint8_t PAN_SQUARE_LAW = 1; + static const uint8_t SIGNAL_LEFT = 0; + static constexpr float SIGNAL_MAX = 1.0f; + static constexpr float SIGNAL_MIN = -1.0f; + static const int SIGNAL_MONO = 1; + static const int SIGNAL_PROCESSING_IGNORE_IN_OUTPOINTS = -3; + static const int SIGNAL_RIGHT = 1; + static const int SIGNAL_STEREO = 2; + static const uint8_t SIG_INT16_BIG_ENDIAN = 2; + static const uint8_t SIG_INT16_LITTLE_ENDIAN = 3; + static const uint8_t SIG_INT24_3_BIG_ENDIAN = 4; + static const uint8_t SIG_INT24_3_LITTLE_ENDIAN = 5; + static const uint8_t SIG_INT24_4_BIG_ENDIAN = 6; + static const uint8_t SIG_INT24_4_LITTLE_ENDIAN = 7; + static const uint8_t SIG_INT32_BIG_ENDIAN = 8; + static const uint8_t SIG_INT32_LITTLE_ENDIAN = 9; + static const uint8_t SIG_INT8 = 0; + static const uint8_t SIG_UINT8 = 1; + static const uint8_t STEREO = 2; + static const uint8_t VERSION_MAJOR = 0; + static const uint8_t VERSION_MINOR = 8; + static const uint8_t WAVEFORM_SINE = 0; + static const uint8_t WAVEFORM_TRIANGLE = 1; + static const uint8_t WAVEFORM_SAWTOOTH = 2; + static const uint8_t WAVEFORM_SQUARE = 3; + static const uint8_t WAVEFORM_NOISE = 4; + static const uint8_t WAVESHAPE_INTERPOLATE_NONE = 0; + static const uint8_t WAVESHAPE_INTERPOLATE_LINEAR = 1; + static const uint8_t WAVESHAPE_INTERPOLATE_CUBIC = 2; + static const uint8_t WAV_FORMAT_PCM = 1; + static const uint8_t WAV_FORMAT_IEEE_FLOAT_32BIT = 3; /* --- sound --- */ static constexpr float MIDI_NOTE_CONVERSION_BASE_FREQUENCY = 440.0; static const uint8_t NOTE_OFFSET = 69; - static float note_to_frequency(uint8_t pMidiNote) { + + static float note_to_frequency(uint8_t pMidiNote) { return MIDI_NOTE_CONVERSION_BASE_FREQUENCY * pow(2, ((pMidiNote - NOTE_OFFSET) / 12.0)); } + static void normalize(float *buffer, const uint32_t numSamples) { + if (buffer == nullptr || numSamples == 0) { + return; + } + + float peakValue = 0.0f; + + // find the peak value in the buffer + for (uint32_t i = 0; i < numSamples; i++) { + const float sample = std::abs(buffer[i]); + if (sample > peakValue) { + peakValue = sample; + } + } + + // avoid division by zero + if (peakValue == 0.0f) { + return; + } + + const float normalizationFactor = 1.0f / peakValue; + + // normalize the buffer + for (uint32_t i = 0; i < numSamples; i++) { + buffer[i] *= normalizationFactor; + } + } + + static void peak(float *buffer, const uint32_t length, float &min, float &max) { + if (buffer == nullptr || length == 0) { + return; + } + + min = 0.0f; + max = 0.0f; + + for (uint32_t i = 0; i < length; i++) { + const float sample = buffer[i]; + if (sample < min) { + min = sample; + } + if (sample > max) { + max = sample; + } + } + } + /* --- math --- */ - static uint32_t millis_to_samples(float pMillis, float pSamplingRate); - static uint32_t millis_to_samples(float pMillis); - static float random_normalized(); - static uint32_t x32Seed; - static uint32_t xorshift32(); - static float random(); + // static uint32_t millis_to_samples(float pMillis, float pSamplingRate); + // static uint32_t millis_to_samples(float pMillis); + // static float random_normalized(); + // static uint32_t x32Seed; + // static uint32_t xorshift32(); + // static float random(); + inline static uint32_t x32Seed = 23; + + static uint32_t millis_to_samples(const float pMillis, const float pSamplingRate) { + return static_cast(pMillis / 1000.0f * pSamplingRate); + } + + static uint32_t millis_to_samples(const float pMillis) { + return static_cast(pMillis / 1000.0f * static_cast(DEFAULT_SAMPLING_RATE)); + } + + /* xorshift32 ( ref: https://en.wikipedia.org/wiki/Xorshift ) */ + // static uint32_t x32Seed = 23; + static uint32_t xorshift32() { + x32Seed ^= x32Seed << 13; + x32Seed ^= x32Seed >> 17; + x32Seed ^= x32Seed << 5; + return x32Seed; + } + + /** + * returns a random number between 0 ... 1 + */ + static float random_normalized() { + // TODO replace with mapping, without division + return ((float) xorshift32() / UINT32_MAX); + } + + static float random() { + // TODO replace with mapping, without division + return ((float) xorshift32() / UINT32_MAX) * 2 - 1; + } - inline static float clamp(float value, - float minimum, - float maximum) { + static float clamp(const float value, + const float minimum, + const float maximum) { return value > maximum ? maximum : (value < minimum ? minimum : value); } - inline static float clamp(float value) { + static float clip(const float value) { + return clamp(value); + } + + static float clamp(const float value) { return value > SIGNAL_MAX ? SIGNAL_MAX : (value < SIGNAL_MIN ? SIGNAL_MIN : value); } - inline static uint8_t clamp127(uint8_t value) { + static uint8_t clamp127(const uint8_t value) { return value > 127 ? 127 : value; } - template - inline static T clamp(T value, T minimum, T maximum) { + template + static T clamp(T value, T minimum, T maximum) { return value > maximum ? maximum : (value < minimum ? minimum : value); } - inline static float pow(float base, float exponent) { + static float pow(const float base, const float exponent) { return powf(base, exponent); } - inline static float lerp(float v0, float v1, float t) { + static float lerp(const float v0, const float v1, const float t) { return v0 + t * (v1 - v0); } - inline static float max(float a, float b) { + static float max(const float a, const float b) { return a > b ? a : b; } - inline static float min(float a, float b) { + static float min(const float a, const float b) { return a < b ? a : b; } - inline static float abs(float value) { + static float abs(const float value) { return value > 0 ? value : -value; } - template - inline static T abs(T value) { + template + static T abs(T value) { return value > 0 ? value : -value; } - template - inline static int sign(T val) { + template + static int sign(T val) { return (T(0) < val) - (val < T(0)); } - inline static float mod(float a, float b) { + static float mod(const float a, const float b) { return a >= b ? (a - b * int(a / b)) : a; // return a >= b ? fmodf(a, b) : a; // return input >= ceil ? input % ceil : input; // from https://stackoverflow.com/questions/33333363/built-in-mod-vs-custom-mod-function-improve-the-performance-of-modulus-op } - inline static float map(float value, - float inputMin, - float inputMax, - float outputMin, - float outputMax) { - return ((value - inputMin) / (inputMax - inputMin) * (outputMax - outputMin) + outputMin); - } - - inline static int16_t map_i16(int16_t value, - int16_t inputMin, - int16_t inputMax, - int16_t outputMin, - int16_t outputMax) { + static float map(const float value, + const float inputMin, + const float inputMax, + const float outputMin, + const float outputMax) { + const float a = value - inputMin; + const float b = inputMax - inputMin; + const float c = outputMax - outputMin; + const float d = a / b; + const float e = d * c; + return e + outputMin; + } + + static int16_t map_i16(const int16_t value, + const int16_t inputMin, + const int16_t inputMax, + const int16_t outputMin, + const int16_t outputMax) { return ((value - inputMin) * (outputMax - outputMin) / (inputMax - inputMin) + outputMin); } - inline static float sin(float r) { + static float sin(const float r) { return sinf(r); } - inline static float cos(float r) { + static float cos(const float r) { return cosf(r); } - inline static float sinh(float r) { + static float sinh(const float r) { return sinhf(r); } - inline static float cosh(float r) { + static float cosh(const float r) { return coshf(r); } - inline static float sqrt(float r) { + static float sqrt(const float r) { return sqrtf(r); } - inline static float fast_sqrt(float x) { - return 1.0 / fast_inv_sqrt(x); + static float fast_sqrt(const float x) { + return 1.0f / fast_inv_sqrt(x); } #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wstrict-aliasing" - inline static float fast_inv_sqrt(float x) { - float xhalf = 0.5f * x; - int i = *(int*)&x; - i = 0x5f3759df - (i >> 1); - x = *(float*)&i; +#pragma GCC diagnostic ignored "-Wuninitialized" + static float fast_inv_sqrt(float x) { + const float xhalf = 0.5f * x; + int i = *(int *) &x; + i = 0x5f3759df - (i >> 1); + x = *(float *) &i; return x * (1.5f - xhalf * x * x); } #pragma GCC diagnostic pop - inline static float fast_sin(float x) { - if (x < -PI) { - while (x < -PI) { - x += TWO_PI; + static constexpr float PIf = (float) PI; + static constexpr float TWO_PIf = (float) TWO_PI; + + static float fast_sin(float x) { + if (x < -PIf) { + while (x < -PIf) { + x += TWO_PIf; } - } else if (x > PI) { - while (x > PI) { - x -= TWO_PI; + } else if (x > PIf) { + while (x > PIf) { + x -= TWO_PIf; } } const float x2 = x * x; @@ -285,24 +383,24 @@ namespace klangwellen { return numerator / denominator; } - inline static float fast_sin3(const float X) { + static float fast_sin3(const float X) { // Valid on interval [-PI, PI] // Sine approximation using Bhaskara I technique discovered in 7th century. // https://en.wikipedia.org/wiki/Bh%C4%81skara_I const float AbsX = abs(X); - const float Numerator = 16.0f * X * (PI - AbsX); - const float Denominator = 5.0f * PI * PI - 4.0f * AbsX * (PI - AbsX); + const float Numerator = 16.0f * X * (PIf - AbsX); + const float Denominator = 5.0f * PIf * PIf - 4.0f * AbsX * (PIf - AbsX); return Numerator / Denominator; } - inline static float fast_cos(float x) { - if (x < -PI) { - while (x < -PI) { - x += TWO_PI; + static float fast_cos(float x) { + if (x < -PIf) { + while (x < -PIf) { + x += TWO_PIf; } - } else if (x > PI) { - while (x > PI) { - x -= TWO_PI; + } else if (x > PIf) { + while (x > PIf) { + x -= TWO_PIf; } } const float x2 = x * x; @@ -311,56 +409,109 @@ namespace klangwellen { return numerator / denominator; } - inline static float fast_sinh(float x) { - auto x2 = x * x; - auto numerator = -x * (11511339840 + x2 * (1640635920 + x2 * (52785432 + x2 * 479249))); - auto denominator = -11511339840 + x2 * (277920720 + x2 * (-3177720 + x2 * 18361)); + static float fast_sinh(const float x) { + const auto x2 = x * x; + const auto numerator = -x * (11511339840 + x2 * (1640635920 + x2 * (52785432 + x2 * 479249))); + const auto denominator = -11511339840 + x2 * (277920720 + x2 * (-3177720 + x2 * 18361)); return numerator / denominator; } - inline static float fast_cosh(float x) { - auto x2 = x * x; - auto numerator = -(39251520 + x2 * (18471600 + x2 * (1075032 + 14615 * x2))); - auto denominator = -39251520 + x2 * (1154160 + x2 * (-16632 + 127 * x2)); + static float fast_cosh(const float x) { + const auto x2 = x * x; + const auto numerator = -(39251520 + x2 * (18471600 + x2 * (1075032 + 14615 * x2))); + const auto denominator = -39251520 + x2 * (1154160 + x2 * (-16632 + 127 * x2)); return numerator / denominator; } - inline static float fast_tan(float x) { - auto x2 = x * x; - auto numerator = x * (-135135 + x2 * (17325 + x2 * (-378 + x2))); - auto denominator = -135135 + x2 * (62370 + x2 * (-3150 + 28 * x2)); + static float fast_tan(const float x) { + const auto x2 = x * x; + const auto numerator = x * (-135135 + x2 * (17325 + x2 * (-378 + x2))); + const auto denominator = -135135 + x2 * (62370 + x2 * (-3150 + 28 * x2)); return numerator / denominator; } - inline static float fast_tanh(float x) { - auto x2 = x * x; - auto numerator = x * (135135 + x2 * (17325 + x2 * (378 + x2))); - auto denominator = 135135 + x2 * (62370 + x2 * (3150 + 28 * x2)); + static float fast_tanh(const float x) { + const auto x2 = x * x; + const auto numerator = x * (135135 + x2 * (17325 + x2 * (378 + x2))); + const auto denominator = 135135 + x2 * (62370 + x2 * (3150 + 28 * x2)); return numerator / denominator; } - inline static float exp_j(float x) { - auto numerator = 1680 + x * (840 + x * (180 + x * (20 + x))); - auto denominator = 1680 + x * (-840 + x * (180 + x * (-20 + x))); + static float exp_j(const float x) { + const auto numerator = 1680 + x * (840 + x * (180 + x * (20 + x))); + const auto denominator = 1680 + x * (-840 + x * (180 + x * (-20 + x))); return numerator / denominator; } - inline static float fast_atan(float x) { - static const float PI_FOURTH = (PI / 4.0); - return PI_FOURTH * x - x * (fabs(x) - 1) * (0.2447 + 0.0663 * fabs(x)); + static float fast_atan(const float x) { + static constexpr float PI_FOURTH = (PIf / 4.0f); + return PI_FOURTH * x - x * (fabs(x) - 1) * (0.2447f + 0.0663f * fabs(x)); } - inline static float fast_atan2(float x) { - static const float PI_FOURTH = (PI / 4.0); - static const float A = 0.0776509570923569; - static const float B = -0.287434475393028; - static const float C = (PI_FOURTH - A - B); - float xx = x * x; + static float fast_atan2(float x) { + static constexpr float PI_FOURTH = (PIf / 4.0f); + static const float A = 0.0776509570923569; + static const float B = -0.287434475393028; + static const float C = (PI_FOURTH - A - B); + const float xx = x * x; return ((A * xx + B) * xx + C) * x; } + /* --- interpolation --- */ + + static float linear_interpolate(const float y1, const float y2, const float mu) { + return y1 * (1.0f - mu) + y2 * mu; + } + + static float interpolate_samples_linear(const float *buffer, const size_t bufferSize, const float position) { + const int posInt = static_cast(position); + const float mu = position - posInt; + + // Ensure that the positions are valid + const int y1Pos = posInt; + const int y2Pos = (posInt + 1 >= bufferSize) ? bufferSize - 1 : posInt + 1; + + // Get the samples for interpolation + const float y1 = buffer[y1Pos]; + const float y2 = buffer[y2Pos]; + + // Return the interpolated sample + return linear_interpolate(y1, y2, mu); + } + + static float cubic_interpolate(const float y0, const float y1, const float y2, const float y3, const float mu) { + const float mu2 = mu * mu; + const float a0 = y3 - y2 - y0 + y1; + const float a1 = y0 - y1 - a0; + const float a2 = y2 - y0; + const float a3 = y1; + + return (a0 * mu * mu2 + a1 * mu2 + a2 * mu + a3); + } + + static float interpolate_samples_cubic(const float *buffer, const uint32_t bufferSize, const float position) { + const int posInt = static_cast(position); + const float mu = position - posInt; + + // Ensure that the positions are valid + const int y0Pos = (posInt - 1 < 0) ? 0 : posInt - 1; + const int y1Pos = posInt; + const int y2Pos = (posInt + 1 >= bufferSize) ? bufferSize - 1 : posInt + 1; + const int y3Pos = (posInt + 2 >= bufferSize) ? bufferSize - 1 : posInt + 2; + + // Get the samples for interpolation + const float y0 = buffer[y0Pos]; + const float y1 = buffer[y1Pos]; + const float y2 = buffer[y2Pos]; + const float y3 = buffer[y3Pos]; + + // Return the interpolated sample + return cubic_interpolate(y0, y1, y2, y3, mu); + } + /* --- buffer --- */ - inline static void fill(float* buffer, float value, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + + inline static void fill(float *buffer, float value, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer[i] = value; } @@ -369,20 +520,20 @@ namespace klangwellen { /** * copies src into dst. dst will be overwritten. */ - inline static void copy(float* src, float* dst, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void copy(float *src, float *dst, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { std::copy(src, src + length, dst); } /** * adds buffer_b to buffer_a. result will be stored in buffer_a, buffer_b will not be changed. */ - inline static void add(float* buffer_a, float* buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void add(float *buffer_a, float *buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] += buffer_b[i]; } } - inline static void add(float* buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void add(float *buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] += scalar; } @@ -391,13 +542,13 @@ namespace klangwellen { /** * subtracts buffer_b from buffer_a. result will be stored in buffer_a, buffer_b will not be changed. */ - inline static void sub(float* buffer_a, float* buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void sub(float *buffer_a, float *buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] -= buffer_b[i]; } } - inline static void sub(float* buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void sub(float *buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] -= scalar; } @@ -406,13 +557,13 @@ namespace klangwellen { /** * multiplies buffer_b with buffer_a. result will be stored in buffer_a, buffer_b will not be changed. */ - inline static void mult(float* buffer_a, float* buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void mult(float *buffer_a, float *buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] *= buffer_b[i]; } } - inline static void mult(float* buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void mult(float *buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] *= scalar; } @@ -421,16 +572,16 @@ namespace klangwellen { /** * divides buffer_b from buffer_a. result will be stored in buffer_a, buffer_b will not be changed. */ - inline static void div(float* buffer_a, float* buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void div(float *buffer_a, float *buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] /= buffer_b[i]; } } - inline static void div(float* buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void div(float *buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] /= scalar; } } }; -} // namespace klangwellen +} // namespace klangwellen diff --git a/desktop/libraries/KlangWellen/src/Note.h b/desktop/libraries/KlangWellen/src/Note.h index 3264a3cf..f5ea8045 100644 --- a/desktop/libraries/KlangWellen/src/Note.h +++ b/desktop/libraries/KlangWellen/src/Note.h @@ -1,9 +1,8 @@ - /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/desktop/libraries/KlangWellen/src/Ramp.h b/desktop/libraries/KlangWellen/src/Ramp.h index 7e0c1277..f9bf3493 100644 --- a/desktop/libraries/KlangWellen/src/Ramp.h +++ b/desktop/libraries/KlangWellen/src/Ramp.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/desktop/libraries/KlangWellen/src/Resonator.h b/desktop/libraries/KlangWellen/src/Resonator.h new file mode 100644 index 00000000..c1760be4 --- /dev/null +++ b/desktop/libraries/KlangWellen/src/Resonator.h @@ -0,0 +1,111 @@ +/* + * KlangWellen + * + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul + * + * This library is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, version 3. + * + * This library is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +/** + * PROCESSOR INTERFACE + * + * - [ ] float process() + * - [x] float process(float)‌ + * - [ ] void process(Signal&) + * - [ ] void process(float*, uint32_t) + * - [ ] void process(float*, float*, uint32_t) + */ + +#pragma once + +#include +#include + +#include "KlangWellen.h" +#include "Signal.h" + +namespace klangwellen { + class Resonator { + public: + Resonator() + : m_frequency(440.0f), m_sampleRate(KlangWellen::DEFAULT_SAMPLING_RATE), m_qFactor(1.0f) { + calculateCoefficients(); + } + + Resonator(float frequency, float sampleRate, float qFactor) + : m_frequency(frequency), m_sampleRate(sampleRate), m_qFactor(qFactor) { + calculateCoefficients(); + } + + float process(float input) { + // IIR filter processing + float output = a0 * input + a1 * x1 + a2 * x2 - b1 * y1 - b2 * y2; + + // Update filter states + x2 = x1; + x1 = input; + y2 = y1; + y1 = output; + + return output; + } + + void set_frequency(float frequency) { + m_frequency = frequency; + calculateCoefficients(); + } + + const float get_frequency() { + return m_frequency; + } + + void set_Q(float qFactor) { + m_qFactor = qFactor; + calculateCoefficients(); + } + + float get_Q() { + return m_qFactor; + } + + private: + float m_frequency; // Resonant frequency + float m_sampleRate; // Sample rate of the audio + float m_qFactor; // Quality factor + + // Filter coefficients + float a0, a1, a2, b1, b2; + + // Filter states + float x1 = 0, x2 = 0, y1 = 0, y2 = 0; + + void calculateCoefficients() { + // Calculate the coefficients for the resonator filter + float omega = 2 * M_PI * m_frequency / m_sampleRate; + float alpha = sin(omega) / (2 * m_qFactor); + + a0 = 1 + alpha; + a1 = -2 * cos(omega); + a2 = 1 - alpha; + b1 = -2 * cos(omega); + b2 = 1 - alpha; + + // Normalize the coefficients + a1 /= a0; + a2 /= a0; + b1 /= a0; + b2 /= a0; + } + }; +} // namespace klangwellen diff --git a/desktop/libraries/KlangWellen/src/Reverb.h b/desktop/libraries/KlangWellen/src/Reverb.h index fb40856a..51e59561 100644 --- a/desktop/libraries/KlangWellen/src/Reverb.h +++ b/desktop/libraries/KlangWellen/src/Reverb.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/desktop/libraries/KlangWellen/src/SAM.h b/desktop/libraries/KlangWellen/src/SAM.h index c2202151..c94cf024 100644 --- a/desktop/libraries/KlangWellen/src/SAM.h +++ b/desktop/libraries/KlangWellen/src/SAM.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/desktop/libraries/KlangWellen/src/Sampler.h b/desktop/libraries/KlangWellen/src/Sampler.h index e4981683..a5af9794 100644 --- a/desktop/libraries/KlangWellen/src/Sampler.h +++ b/desktop/libraries/KlangWellen/src/Sampler.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -41,7 +41,6 @@ #include "KlangWellen.h" namespace klangwellen { - class SamplerListener { public: virtual void is_done() = 0; @@ -50,23 +49,26 @@ namespace klangwellen { /** * plays back an array of samples at different speeds. */ - template + template class SamplerT { public: - static const int8_t NO_LOOP_POINT = -1; + static constexpr int8_t NO_LOOP_POINT = -1; - SamplerT() : SamplerT(0) {} + SamplerT() : SamplerT(0) { + } - SamplerT(int32_t buffer_length, uint32_t sampling_rate = KlangWellen::DEFAULT_SAMPLING_RATE) : SamplerT(new BUFFER_TYPE[buffer_length], buffer_length, sampling_rate) { + explicit SamplerT(int32_t buffer_length, + uint32_t sampling_rate = KlangWellen::DEFAULT_SAMPLING_RATE) : SamplerT( + new BUFFER_TYPE[buffer_length], buffer_length, sampling_rate) { fAllocatedBuffer = true; } - SamplerT(BUFFER_TYPE* buffer, - int32_t buffer_length, - uint32_t sampling_rate = KlangWellen::DEFAULT_SAMPLING_RATE) : fDirectionForward(true), - fInPoint(0), - fOutPoint(0), - fSpeed(0) { + SamplerT(BUFFER_TYPE * buffer, + const int32_t buffer_length, + const uint32_t sampling_rate = KlangWellen::DEFAULT_SAMPLING_RATE) : fDirectionForward(true), + fInPoint(0), + fOutPoint(0), + fSpeed(0) { fSamplingRate = sampling_rate; set_buffer(buffer, buffer_length); fBufferIndex = 0; @@ -88,11 +90,11 @@ namespace klangwellen { } } - void add_listener(SamplerListener* sampler_listener) { + void add_listener(SamplerListener *sampler_listener) { fSamplerListeners.push_back(sampler_listener); } - bool remove_listener(SamplerListener* sampler_listener) { + bool remove_listener(SamplerListener *sampler_listener) { for (auto it = fSamplerListeners.begin(); it != fSamplerListeners.end(); ++it) { if (*it == sampler_listener) { fSamplerListeners.erase(it); @@ -102,7 +104,7 @@ namespace klangwellen { return false; } - int32_t get_in() { + int32_t get_in() const { return fInPoint; } @@ -113,51 +115,51 @@ namespace klangwellen { fInPoint = in_point; } - int32_t get_out() { + int32_t get_out() const { return fOutPoint; } - void set_out(int32_t out_point) { + void set_out(const int32_t out_point) { fOutPoint = out_point > last_index() ? last_index() : (out_point < fInPoint ? fInPoint : out_point); } - float get_speed() { + float get_speed() const { return fSpeed; } - void set_speed(float speed) { + void set_speed(const float speed) { fSpeed = speed; fDirectionForward = speed > 0; set_frequency(KlangWellen::abs(speed) * fSamplingRate / fBufferLength); /* aka `step_size = speed` */ } - void set_frequency(float frequency) { + void set_frequency(const float frequency) { fFrequency = frequency; - fStepSize = fFrequency / fFrequencyScale * ((float)fBufferLength / fSamplingRate); + fStepSize = fFrequency / fFrequencyScale * (static_cast(fBufferLength) / fSamplingRate); } - float get_frequency() { + float get_frequency() const { return fFrequency; } - void set_amplitude(float amplitude) { + void set_amplitude(const float amplitude) { fAmplitude = amplitude; } - float get_amplitude() { + float get_amplitude() const { return fAmplitude; } - BUFFER_TYPE* get_buffer() { + BUFFER_TYPE *get_buffer() { return fBuffer; } - int32_t get_buffer_length() { + int32_t get_buffer_length() const { return fBufferLength; } - void set_buffer(BUFFER_TYPE* buffer, int32_t buffer_length) { - fAllocatedBuffer = false; // TODO huuui, this is not nice and might cause some trouble somewhere + void set_buffer(BUFFER_TYPE *buffer, const int32_t buffer_length) { + fAllocatedBuffer = false; // TODO huuui, this is not nice and might cause some trouble somewhere fBuffer = buffer; fBufferLength = buffer_length; rewind(); @@ -168,52 +170,52 @@ namespace klangwellen { fLoopOut = NO_LOOP_POINT; } - void interpolate_samples(bool interpolate_samples) { + void interpolate_samples(bool const interpolate_samples) { fInterpolateSamples = interpolate_samples; } - bool interpolate_samples() { + bool interpolate_samples() const { return fInterpolateSamples; } - int32_t get_position() { - return (int32_t)fBufferIndex; + int32_t get_position() const { + return static_cast(fBufferIndex); } - float get_position_normalized() { + float get_position_normalized() const { return fBufferLength > 0 ? fBufferIndex / fBufferLength : 0.0f; } - float get_position_fractional_part() { + float get_position_fractional_part() const { return fBufferIndex - get_position(); } - bool is_playing() { + bool is_playing() const { return fIsPlaying; } float process() { if (fBufferLength == 0) { - notifyListeners(); // "buffer is empty" + notifyListeners(); // "buffer is empty" return 0.0f; } if (!fIsPlaying) { - notifyListeners(); // "not playing" + notifyListeners(); // "not playing" return 0.0f; } validateInOutPoints(); fBufferIndex += fDirectionForward ? fStepSize : -fStepSize; - const int32_t mRoundedIndex = (int32_t)fBufferIndex; + const int32_t mRoundedIndex = static_cast(fBufferIndex); const float mFrac = fBufferIndex - mRoundedIndex; const int32_t mCurrentIndex = wrapIndex(mRoundedIndex); fBufferIndex = mCurrentIndex + mFrac; if (fDirectionForward ? (mCurrentIndex >= fOutPoint) : (mCurrentIndex <= fInPoint)) { - notifyListeners(); // "reached end" + notifyListeners(); // "reached end" return 0.0f; } else { fIsFlaggedDone = false; @@ -225,8 +227,10 @@ namespace klangwellen { if (fInterpolateSamples) { // TODO evaluate direction? const int32_t mNextIndex = wrapIndex(mCurrentIndex + 1); - const float mNextSample = fBuffer[mNextIndex]; + const float mNextSample = convert_sample(fBuffer[mNextIndex]); mSample = mSample * (1.0f - mFrac) + mNextSample * mFrac; + // mSample = interpolate_samples_linear(fBuffer, fBufferLength, fBufferIndex); + // mSample = interpolate_samples_cubic(fBuffer, fBufferLength, fBufferIndex); } mSample *= fAmplitude; @@ -234,23 +238,23 @@ namespace klangwellen { if (fEdgeFadePadding > 0) { const int32_t mRelativeIndex = fBufferLength - mCurrentIndex; if (mCurrentIndex < fEdgeFadePadding) { - const float mFadeInAmount = (float)mCurrentIndex / fEdgeFadePadding; + const float mFadeInAmount = static_cast(mCurrentIndex) / fEdgeFadePadding; mSample *= mFadeInAmount; } else if (mRelativeIndex < fEdgeFadePadding) { - const float mFadeOutAmount = (float)mRelativeIndex / fEdgeFadePadding; + const float mFadeOutAmount = static_cast(mRelativeIndex) / fEdgeFadePadding; mSample *= mFadeOutAmount; } } return mSample; } - void process(float* signal_buffer, const uint32_t buffer_length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + void process(float *signal_buffer, const uint32_t buffer_length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint16_t i = 0; i < buffer_length; i++) { signal_buffer[i] = process(); } } - int32_t get_edge_fading() { + int32_t get_edge_fading() const { return fEdgeFadePadding; } @@ -266,7 +270,7 @@ namespace klangwellen { fBufferIndex = fDirectionForward ? fOutPoint : fInPoint; } - bool is_looping() { + bool is_looping() const { return fEvaluateLoop; } @@ -311,7 +315,7 @@ namespace klangwellen { } } - void record(float* samples, int32_t num_samples) { + void record(float *samples, int32_t num_samples) { if (fIsRecording) { for (int32_t i = 0; i < num_samples; i++) { const float sample = samples[i]; @@ -320,7 +324,7 @@ namespace klangwellen { } } - bool is_recording() { + bool is_recording() const { return fIsRecording; } @@ -331,7 +335,7 @@ namespace klangwellen { uint32_t end_recording() { fIsRecording = false; const int32_t mBufferLength = fRecording.size(); - float* mBuffer = new float[mBufferLength]; + float * mBuffer = new float[mBufferLength]; for (int32_t i = 0; i < mBufferLength; i++) { mBuffer[i] = fRecording[i]; } @@ -344,42 +348,42 @@ namespace klangwellen { return mBufferLength; } - int32_t get_loop_in() { + int32_t get_loop_in() const { return fLoopIn; } - void set_loop_in(int32_t loop_in_point) { + void set_loop_in(const int32_t loop_in_point) { fLoopIn = KlangWellen::clamp(loop_in_point, NO_LOOP_POINT, fBufferLength - 1); } - float get_loop_in_normalized() { + float get_loop_in_normalized() const { if (fBufferLength < 2) { return 0.0f; } - return (float)fLoopIn / (fBufferLength - 1); + return static_cast(fLoopIn) / (fBufferLength - 1); } - void set_loop_in_normalized(float loop_in_point_normalized) { - set_loop_in((int32_t)(loop_in_point_normalized * fBufferLength - 1)); + void set_loop_in_normalized(const float loop_in_point_normalized) { + set_loop_in(static_cast(loop_in_point_normalized * fBufferLength - 1)); } - int32_t get_loop_out() { + int32_t get_loop_out() const { return fLoopOut; } - void set_loop_out(int32_t loop_out_point) { + void set_loop_out(const int32_t loop_out_point) { fLoopOut = KlangWellen::clamp(loop_out_point, NO_LOOP_POINT, fBufferLength - 1); } - float get_loop_out_normalized() { + float get_loop_out_normalized() const { if (fBufferLength < 2) { return 0.0f; } - return (float)fLoopOut / (fBufferLength - 1); + return static_cast(fLoopOut) / (fBufferLength - 1); } - void set_loop_out_normalized(float loop_out_point_normalized) { - set_loop_out((int32_t)(loop_out_point_normalized * fBufferLength - 1)); + void set_loop_out_normalized(const float loop_out_point_normalized) { + set_loop_out(static_cast(loop_out_point_normalized * fBufferLength - 1)); } void note_on() { @@ -388,7 +392,7 @@ namespace klangwellen { enable_loop(true); } - void note_on(uint8_t note, uint8_t velocity) { + void note_on(const uint8_t note, const uint8_t velocity) { fIsPlaying = true; set_frequency(KlangWellen::note_to_frequency(note)); set_amplitude(KlangWellen::clamp127(velocity) / 127.0f); @@ -403,61 +407,61 @@ namespace klangwellen { * this function can be used to tune a loaded sample to a specific frequency. after the sampler has been tuned the * method set_frequency(float) can be used to play the sample at a desired frequency. * - * @param frequency_scale the assumed frequency of the sampler buffer in Hz + * @param tune_frequency the assumed frequency of the sampler buffer in Hz */ - void tune_frequency_to(float tune_frequency) { + void tune_frequency_to(const float tune_frequency) { fFrequencyScale = tune_frequency; } - void set_duration(float seconds) { + void set_duration(const float seconds) { if (fBufferLength == 0 || seconds == 0.0f) { return; } - const float mNormDurationSec = ((float)fBufferLength / (float)fSamplingRate); + const float mNormDurationSec = (static_cast(fBufferLength) / static_cast(fSamplingRate)); const float mSpeed = mNormDurationSec / seconds; set_speed(mSpeed); } - float get_duration() { + float get_duration() const { if (fBufferLength == 0 || fSpeed == 0.0f) { return 0; } - const float mNormDurationSec = ((float)fBufferLength / (float)fSamplingRate); + const float mNormDurationSec = (static_cast(fBufferLength) / static_cast(fSamplingRate)); return mNormDurationSec / fSpeed; } private: - std::vector fSamplerListeners; - std::vector fRecording; - uint32_t fSamplingRate; - float fAmplitude; - BUFFER_TYPE* fBuffer; - int32_t fBufferLength; - float fBufferIndex; - bool fDirectionForward; - int32_t fEdgeFadePadding; - bool fEvaluateLoop; - float fFrequency; - float fFrequencyScale; - int32_t fInPoint; - int32_t fOutPoint; - int32_t fLoopIn; - int32_t fLoopOut; - bool fInterpolateSamples; - bool fIsPlaying; - float fSpeed; - float fStepSize; - bool fIsFlaggedDone; - bool fIsRecording; - bool fAllocatedBuffer; - - int32_t last_index() { + std::vector fSamplerListeners; + std::vector fRecording; + uint32_t fSamplingRate; + float fAmplitude; + BUFFER_TYPE * fBuffer; + int32_t fBufferLength; + float fBufferIndex; + bool fDirectionForward; + int32_t fEdgeFadePadding; + bool fEvaluateLoop; + float fFrequency; + float fFrequencyScale; + int32_t fInPoint; + int32_t fOutPoint; + int32_t fLoopIn; + int32_t fLoopOut; + bool fInterpolateSamples; + bool fIsPlaying; + float fSpeed; + float fStepSize; + bool fIsFlaggedDone; + bool fIsRecording; + bool fAllocatedBuffer; + + int32_t last_index() const { return fBufferLength - 1; } void notifyListeners() { if (!fIsFlaggedDone) { - for (SamplerListener* l : fSamplerListeners) { + for (SamplerListener *l: fSamplerListeners) { l->is_done(); } } @@ -486,7 +490,7 @@ namespace klangwellen { } } - int32_t wrapIndex(int32_t i) { + int32_t wrapIndex(int32_t i) const { /* check if in loop concept viable i.e loop in- and output points are set */ if (fEvaluateLoop) { if (fLoopIn != NO_LOOP_POINT && fLoopOut != NO_LOOP_POINT) { @@ -511,30 +515,30 @@ namespace klangwellen { return i; } - inline float convert_sample(const BUFFER_TYPE pRawSample) { + float convert_sample(const BUFFER_TYPE pRawSample) { return pRawSample; } }; - template <> - float klangwellen::SamplerT::convert_sample(const uint8_t pRawSample) { - const static float mScale = 1.0 / ((1 << 8) - 1); - const float mRange = pRawSample * mScale; + template<> + inline float klangwellen::SamplerT::convert_sample(const uint8_t pRawSample) { + constexpr static float mScale = 1.0 / ((1 << 8) - 1); + const float mRange = pRawSample * mScale; return mRange * 2.0 - 1.0; } - template <> - float klangwellen::SamplerT::convert_sample(const int8_t pRawSample) { - const static float mScale = 1.0 / ((1 << 8) - 1); - const float mOffset = pRawSample + (1 << 7); - const float mRange = mOffset * mScale; + template<> + inline float klangwellen::SamplerT::convert_sample(const int8_t pRawSample) { + constexpr static float mScale = 1.0 / ((1 << 8) - 1); + const float mOffset = pRawSample + (1 << 7); + const float mRange = mOffset * mScale; return mRange * 2.0 - 1.0; } - template <> - float klangwellen::SamplerT::convert_sample(const uint16_t pRawSample) { - const static float mScale = 1.0 / ((1 << 16) - 1); - const float mRange = pRawSample * mScale; + template<> + inline float klangwellen::SamplerT::convert_sample(const uint16_t pRawSample) { + constexpr static float mScale = 1.0 / ((1 << 16) - 1); + const float mRange = pRawSample * mScale; return mRange * 2.0 - 1.0; // @note(below: less precise but faster) // const float s = pRawSample; @@ -543,18 +547,18 @@ namespace klangwellen { // return a; } - template <> - float klangwellen::SamplerT::convert_sample(const int16_t pRawSample) { - const static float mScale = 1.0 / ((1 << 16) - 1); - const float mOffset = pRawSample + (1 << 15); - const float mRange = mOffset * mScale; + template<> + inline float klangwellen::SamplerT::convert_sample(const int16_t pRawSample) { + constexpr static float mScale = 1.0 / ((1 << 16) - 1); + const float mOffset = pRawSample + (1 << 15); + const float mRange = mOffset * mScale; return mRange * 2.0 - 1.0; } - using SamplerUI8 = SamplerT; - using SamplerI8 = SamplerT; + using SamplerUI8 = SamplerT; + using SamplerI8 = SamplerT; using SamplerUI16 = SamplerT; - using SamplerI16 = SamplerT; - using SamplerF32 = SamplerT; - using Sampler = SamplerT; -} // namespace klangwellen + using SamplerI16 = SamplerT; + using SamplerF32 = SamplerT; + using Sampler = SamplerT; +} // namespace klangwellen diff --git a/desktop/libraries/KlangWellen/src/Scale.cpp b/desktop/libraries/KlangWellen/src/Scale.cpp deleted file mode 100644 index 202ef75c..00000000 --- a/desktop/libraries/KlangWellen/src/Scale.cpp +++ /dev/null @@ -1,14 +0,0 @@ -#include "Scale.h" - -const klangwellen::Scale klangwellen::Scale::CHROMATIC{0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11}; -const klangwellen::Scale klangwellen::Scale::FIFTH{0, 7}; -const klangwellen::Scale klangwellen::Scale::MINOR{0, 2, 3, 5, 7, 8, 10}; -const klangwellen::Scale klangwellen::Scale::MAJOR{0, 2, 4, 5, 7, 9, 11}; -const klangwellen::Scale klangwellen::Scale::MINOR_CHORD{0, 3, 7}; -const klangwellen::Scale klangwellen::Scale::MAJOR_CHORD{0, 4, 7}; -const klangwellen::Scale klangwellen::Scale::MINOR_CHORD_7{0, 3, 7, 11}; -const klangwellen::Scale klangwellen::Scale::MAJOR_CHORD_7{0, 4, 7, 11}; -const klangwellen::Scale klangwellen::Scale::MINOR_PENTATONIC{0, 3, 5, 7, 10}; -const klangwellen::Scale klangwellen::Scale::MAJOR_PENTATONIC{0, 4, 5, 7, 11}; -const klangwellen::Scale klangwellen::Scale::OCTAVE{0}; -const klangwellen::Scale klangwellen::Scale::DIMINISHED{0, 3, 6, 9}; diff --git a/desktop/libraries/KlangWellen/src/Scale.h b/desktop/libraries/KlangWellen/src/Scale.h index 5ed8e723..ed6a6494 100644 --- a/desktop/libraries/KlangWellen/src/Scale.h +++ b/desktop/libraries/KlangWellen/src/Scale.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -26,19 +26,6 @@ namespace klangwellen { class Scale { public: - static const Scale CHROMATIC; - static const Scale FIFTH; - static const Scale MINOR; - static const Scale MAJOR; - static const Scale MINOR_CHORD; - static const Scale MAJOR_CHORD; - static const Scale MINOR_CHORD_7; - static const Scale MAJOR_CHORD_7; - static const Scale MINOR_PENTATONIC; - static const Scale MAJOR_PENTATONIC; - static const Scale OCTAVE; - static const Scale DIMINISHED; - Scale(const std::initializer_list& note_list) : length(note_list.size()) { notes = new uint8_t[length]; uint16_t i = 0; diff --git a/desktop/libraries/KlangWellen/src/ScaleCollection.h b/desktop/libraries/KlangWellen/src/ScaleCollection.h new file mode 100644 index 00000000..9ccebf7c --- /dev/null +++ b/desktop/libraries/KlangWellen/src/ScaleCollection.h @@ -0,0 +1,40 @@ +/* + * KlangWellen + * + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul + * + * This library is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, version 3. + * + * This library is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +#pragma once + +#include "Scale.h" + +namespace klangwellen { + class ScaleCollection { + public: + static inline const Scale CHROMATIC{0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11}; + static inline const Scale FIFTH{0, 7}; + static inline const Scale MINOR{0, 2, 3, 5, 7, 8, 10}; + static inline const Scale MAJOR{0, 2, 4, 5, 7, 9, 11}; + static inline const Scale MINOR_CHORD{0, 3, 7}; + static inline const Scale MAJOR_CHORD{0, 4, 7}; + static inline const Scale MINOR_CHORD_7{0, 3, 7, 11}; + static inline const Scale MAJOR_CHORD_7{0, 4, 7, 11}; + static inline const Scale MINOR_PENTATONIC{0, 3, 5, 7, 10}; + static inline const Scale MAJOR_PENTATONIC{0, 4, 5, 7, 11}; + static inline const Scale OCTAVE{0}; + static inline const Scale DIMINISHED{0, 3, 6, 9}; + }; +} // namespace klangwellen diff --git a/desktop/libraries/KlangWellen/src/Signal.h b/desktop/libraries/KlangWellen/src/Signal.h index a9acd3c6..3094dc49 100644 --- a/desktop/libraries/KlangWellen/src/Signal.h +++ b/desktop/libraries/KlangWellen/src/Signal.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/desktop/libraries/KlangWellen/src/Stream.h b/desktop/libraries/KlangWellen/src/Stream.h new file mode 100644 index 00000000..91171151 --- /dev/null +++ b/desktop/libraries/KlangWellen/src/Stream.h @@ -0,0 +1,201 @@ +/* +* KlangWellen + * + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul + * + * This library is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, version 3. + * + * This library is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +/** + * PROCESSOR INTERFACE + * + * - [x] float process() + * - [ ] float process(float) + * - [ ] void process(Signal&) + * - [*] void process(float*, uint32_t) + * - [ ] void process(float*, float*, uint32_t) + */ + +#pragma once + +namespace klangwellen { + class StreamDataProvider { + public: + virtual ~StreamDataProvider() = default; + + virtual void fill_buffer(float *buffer, uint32_t length) { + std::fill_n(buffer, length, 0.0f); + } + }; + + class Stream final { + public: + Stream(StreamDataProvider *stream_data_provider, + const uint32_t stream_buffer_size, + const uint8_t stream_buffer_division = 4, + const uint8_t stream_buffer_update_offset = 1, + const uint32_t sampling_rate = KlangWellen::DEFAULT_SAMPLING_RATE) + : fStreamDataProvider(stream_data_provider), + fBufferLength(stream_buffer_size), + fBuffer(new float[stream_buffer_size]), + fBufferDivision(stream_buffer_division), + fBufferSegmentOffset(stream_buffer_update_offset % stream_buffer_division), + fSamplingRate(sampling_rate), + fAmplitude(1.0f), + fStepSize(1.0f), + fInterpolateSamples(true), + fBufferIndex(0.0f), + fBufferIndexPrev(0.0f), + fCompleteEvent(NO_EVENT) { + fStreamDataProvider->fill_buffer(fBuffer, fBufferLength); + } + + ~Stream() { + delete[] fBuffer; + } + + float process() { + fBufferIndex += fStepSize; + const int32_t mRoundedIndex = static_cast(fBufferIndex); + const float mFrac = fBufferIndex - mRoundedIndex; + const int32_t mCurrentIndex = wrapIndex(mRoundedIndex); + fBufferIndex = mCurrentIndex + mFrac; + + float mSample = convert_sample(fBuffer[mCurrentIndex]); + + /* linear interpolation */ + if (fInterpolateSamples) { + const int32_t mNextIndex = wrapIndex(mCurrentIndex + 1); + const float mNextSample = convert_sample(fBuffer[mNextIndex]); + const float a = mSample * (1.0f - mFrac); + const float b = mNextSample * mFrac; + mSample = a + b; + } + mSample *= fAmplitude; + + /* load next block */ + int8_t mCompleteEvent = checkCompleteEvent(fBufferDivision); + if (mCompleteEvent > NO_EVENT) { + fCompleteEvent = mCompleteEvent; + mCompleteEvent -= fBufferSegmentOffset; + mCompleteEvent += fBufferDivision; + mCompleteEvent %= fBufferDivision; + replace_segment(fBufferDivision, mCompleteEvent); + } + fBufferIndexPrev = fBufferIndex; + + return mSample; + } + + void replace_segment(const uint32_t numberOfSegments, const uint32_t segmentIndex) const { + if (segmentIndex >= numberOfSegments) { + std::cerr << "Segment index out of range" << std::endl; + return; + } + + const uint32_t lengthOfSegment = fBufferLength / numberOfSegments; + float * startOfSegment = fBuffer + (segmentIndex * lengthOfSegment); + + fStreamDataProvider->fill_buffer(startOfSegment, lengthOfSegment); + } + + float *get_buffer() const { + return fBuffer; + } + + int8_t get_sector() const { + return fCompleteEvent; + } + + uint8_t num_sectors() const { + return fBufferDivision; + } + + uint32_t get_buffer_length() const { + return fBufferLength; + } + + float get_current_buffer_position() const { + return fBufferIndex; + } + + void process(float *signal_buffer, const uint32_t buffer_length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + for (uint16_t i = 0; i < buffer_length; i++) { + signal_buffer[i] = process(); + } + } + + void interpolate_samples(bool const interpolate_samples) { + fInterpolateSamples = interpolate_samples; + } + + bool interpolate_samples() const { + return fInterpolateSamples; + } + + float get_speed() const { + return fStepSize; + } + + void set_speed(const float speed) { + fStepSize = speed; + } + + private: + static constexpr int8_t NO_EVENT = -1; + + StreamDataProvider *fStreamDataProvider; + + uint32_t fBufferLength; + float * fBuffer; + const uint8_t fBufferDivision; + const uint8_t fBufferSegmentOffset; + float fSamplingRate; + float fAmplitude; + float fStepSize; + bool fInterpolateSamples; + float fBufferIndex; + float fBufferIndexPrev; + int8_t fCompleteEvent; + + int32_t wrapIndex(int32_t i) const { + if (i < 0) { + i += fBufferLength; + } else if (i >= fBufferLength) { + i -= fBufferLength; + } + return i; + } + + static float convert_sample(const float pRawSample) { + return pRawSample; + } + + int8_t checkCompleteEvent(const uint8_t num_events) const { + for (int i = 0; i < num_events; ++i) { + const float mBorder = fBufferLength * i / static_cast(fBufferDivision); + if (crossedBorder(fBufferIndexPrev, fBufferIndex, mBorder)) { + return i; + } + } + return NO_EVENT; + } + + static bool crossedBorder(const float prev, const float current, const float border) { + return (border == 0 && prev > current) || + (prev < border && current >= border) || + (prev > current && current >= border); + } + }; +} // namespace klangwellen diff --git a/desktop/libraries/KlangWellen/src/Trigger.h b/desktop/libraries/KlangWellen/src/Trigger.h index e4fe492c..a23ab6a4 100644 --- a/desktop/libraries/KlangWellen/src/Trigger.h +++ b/desktop/libraries/KlangWellen/src/Trigger.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/desktop/libraries/KlangWellen/src/Vocoder.h b/desktop/libraries/KlangWellen/src/Vocoder.h index 086b07dd..689cdafc 100644 --- a/desktop/libraries/KlangWellen/src/Vocoder.h +++ b/desktop/libraries/KlangWellen/src/Vocoder.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/desktop/libraries/KlangWellen/src/Waveshaper.h b/desktop/libraries/KlangWellen/src/Waveshaper.h index b4ed9a86..ac918d20 100644 --- a/desktop/libraries/KlangWellen/src/Waveshaper.h +++ b/desktop/libraries/KlangWellen/src/Waveshaper.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://githubiquad_com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/desktop/libraries/KlangWellen/src/Wavetable.h b/desktop/libraries/KlangWellen/src/Wavetable.h index aafad196..b98e3f6c 100644 --- a/desktop/libraries/KlangWellen/src/Wavetable.h +++ b/desktop/libraries/KlangWellen/src/Wavetable.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General License as published by @@ -110,7 +110,7 @@ namespace klangwellen { static void noise(float* wavetable, uint32_t wavetable_size) { for (uint32_t i = 0; i < wavetable_size; i++) { - wavetable[i] = KlangWellen::random() * 2.0 - 1.0; + wavetable[i] = KlangWellen::random() * 2.0f - 1.0f; } } @@ -136,7 +136,7 @@ namespace klangwellen { return fourier_table(wavetable, wavetable_size, harmonics, amps, -0.25f); } - static void sawtooth(float* wavetable, uint32_t wavetable_size, bool is_ramp_up) { + static void sawtooth_ramp(float* wavetable, uint32_t wavetable_size, bool is_ramp_up) { const float mSign = is_ramp_up ? -1.0f : 1.0f; for (uint32_t i = 0; i < wavetable_size; i++) { wavetable[i] = mSign * (2.0f * ((float)i / (float)(wavetable_size - 1)) - 1.0f); @@ -149,7 +149,7 @@ namespace klangwellen { static void sine(float* wavetable, uint32_t wavetable_size) { for (uint32_t i = 0; i < wavetable_size; i++) { - wavetable[i] = KlangWellen::fast_sin(2.0f * PI * ((float)i / (float)(wavetable_size))); + wavetable[i] = KlangWellen::fast_sin(2.0f * PIf * ((float)i / (float)(wavetable_size))); } } @@ -381,6 +381,8 @@ namespace klangwellen { } private: + static constexpr float PIf = (float)PI; + static constexpr float TWO_PIf = (float)TWO_PI; static constexpr float M_DEFAULT_AMPLITUDE = 0.75f; static constexpr float M_DEFAULT_FREQUENCY = 220.0f; float* mWavetable; @@ -407,12 +409,12 @@ namespace klangwellen { static float* fourier_table(float* pWavetable, uint32_t wavetable_size, uint8_t pHarmonics, float* pAmps, float pPhase) { float a; double w; - pPhase *= PI * 2; + pPhase *= PIf * 2; for (uint8_t i = 0; i < pHarmonics; i++) { for (uint32_t n = 0; n < wavetable_size; n++) { a = (pAmps) ? pAmps[i] : 1.f; w = (i + 1) * (n * 2 * PI / wavetable_size); - pWavetable[n] += (float)(a * KlangWellen::cos(w + pPhase)); + pWavetable[n] += (float)(a * KlangWellen::cos((float)w + pPhase)); } } normalise_table(pWavetable, wavetable_size); @@ -455,8 +457,8 @@ namespace klangwellen { } float next_sample_interpolate_cubic() { - const uint32_t mOffset = (int)(mPhaseOffset * mWavetableSize) % mWavetableSize; - const float mArrayPtrOffset = mArrayPtr + mOffset; + const uint32_t mSampleOffset = (int)(mPhaseOffset * mWavetableSize) % mWavetableSize; + const float mArrayPtrOffset = mArrayPtr + mSampleOffset; /* cubic interpolation */ const float frac = mArrayPtrOffset - (int)mArrayPtrOffset; const float a = (int)mArrayPtrOffset > 0 ? mWavetable[(int)mArrayPtrOffset - 1] : mWavetable[mWavetableSize - 1]; @@ -475,8 +477,8 @@ namespace klangwellen { } float next_sample_interpolate_linear() { - const uint32_t mOffset = (uint32_t)(mPhaseOffset * mWavetableSize) % mWavetableSize; - const float mArrayPtrOffset = mArrayPtr + mOffset; + const uint32_t mSampleOffset = (uint32_t)(mPhaseOffset * mWavetableSize) % mWavetableSize; + const float mArrayPtrOffset = mArrayPtr + mSampleOffset; /* linear interpolation */ const float mFrac = mArrayPtrOffset - (int)mArrayPtrOffset; const float a = mWavetable[(int)mArrayPtrOffset]; diff --git a/desktop/libraries/Klangstrom/examples/01.Basics/Beat/Beat.ino b/desktop/libraries/Klangstrom/examples/01.Basics/Beat/Beat.ino index 38022d97..7dcbf44d 100644 --- a/desktop/libraries/Klangstrom/examples/01.Basics/Beat/Beat.ino +++ b/desktop/libraries/Klangstrom/examples/01.Basics/Beat/Beat.ino @@ -20,6 +20,11 @@ uint32_t mBeatDurationMax = 120000; uint32_t mBeatDurationInc = 1000; void setup() { + Serial.begin(115200); + Serial.println("----"); + Serial.println("Beat"); + Serial.println("----"); + Klang::lock(); Klang::connect(mVCO, Node::CH_OUT_SIGNAL, mDAC, NodeDAC::CH_IN_SIGNAL_LEFT); mVCO.set_frequency(mFreq); diff --git a/desktop/libraries/Klangstrom/examples/01.Basics/Blink/Blink.ino b/desktop/libraries/Klangstrom/examples/01.Basics/Blink/Blink.ino index ca7aafc8..8220d4c9 100644 --- a/desktop/libraries/Klangstrom/examples/01.Basics/Blink/Blink.ino +++ b/desktop/libraries/Klangstrom/examples/01.Basics/Blink/Blink.ino @@ -8,6 +8,11 @@ NodeVCOFunction mVCO; NodeDAC mDAC; void setup() { + Serial.begin(115200); + Serial.println("-----"); + Serial.println("Blink"); + Serial.println("-----"); + Klang::lock(); Klang::connect(mVCO, Node::CH_OUT_SIGNAL, mDAC, NodeDAC::CH_IN_SIGNAL); mVCO.set_amplitude(0.5); diff --git a/desktop/libraries/Klangstrom/examples/01.Basics/PassThrough/PassThrough.ino b/desktop/libraries/Klangstrom/examples/01.Basics/PassThrough/PassThrough.ino index f76318aa..44ae1c6f 100644 --- a/desktop/libraries/Klangstrom/examples/01.Basics/PassThrough/PassThrough.ino +++ b/desktop/libraries/Klangstrom/examples/01.Basics/PassThrough/PassThrough.ino @@ -3,6 +3,11 @@ using namespace klangstrom; void setup() { + Serial.begin(115200); + Serial.println("-----------"); + Serial.println("PassThrough"); + Serial.println("-----------"); + option(KLST_OPTION_AUDIO_INPUT, KLST_MIC); } diff --git a/desktop/libraries/Klangstrom/examples/02.Peripherals/Encoders/Encoders.ino b/desktop/libraries/Klangstrom/examples/02.Peripherals/Encoders/Encoders.ino index 1231cdd8..6ff1ebf9 100644 --- a/desktop/libraries/Klangstrom/examples/02.Peripherals/Encoders/Encoders.ino +++ b/desktop/libraries/Klangstrom/examples/02.Peripherals/Encoders/Encoders.ino @@ -7,6 +7,7 @@ void setup() { Serial.println("--------"); Serial.println("Encoders"); Serial.println("--------"); + register_encoder_rotated(encoder_rotated); register_encoder_pressed(encoder_pressed); register_encoder_released(encoder_released); diff --git a/stm32/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleReadSDCard/ExampleReadSDCard.ino b/desktop/libraries/Klangstrom/examples/02.Peripherals/ExternalSDCard/ExternalSDCard.ino similarity index 80% rename from stm32/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleReadSDCard/ExampleReadSDCard.ino rename to desktop/libraries/Klangstrom/examples/02.Peripherals/ExternalSDCard/ExternalSDCard.ino index afb58bc1..c41188a6 100644 --- a/stm32/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleReadSDCard/ExampleReadSDCard.ino +++ b/desktop/libraries/Klangstrom/examples/02.Peripherals/ExternalSDCard/ExternalSDCard.ino @@ -1,9 +1,7 @@ -// -// ExampleReadSDCard -// - /* + this example requires the Arduino SD library. + pin connection map for KLST_TINYv0.1: | SD_CARD | KLST | @@ -13,27 +11,31 @@ | SCK | SPI_SCK | | CS | GPIO_00 | - pin connection map for KLST_SHEEPv0.1 on : + pin connection map for KLST_SHEEPv0.1: | SD_CARD | KLST | |---------|--------------| | MOSI | SPI_USR_MOSI | | MISO | SPI_USR_MISO | | SCK | SPI_USR_SCK | - | CS | GPIO_00 | + | CS | GPIO_00 | */ -#include "Klangstrom.h" -#include #include +#include + +#include "Klangstrom.h" -Sd2Card card; +Sd2Card card; SdVolume volume; -SdFile root; +SdFile root; void setup() { - klangstrom::begin_serial_debug(true); + Serial.begin(115200); + Serial.println("--------------"); + Serial.println("ExternalSDCard"); + Serial.println("--------------"); Serial.print("\nInitializing SD card..."); @@ -42,7 +44,8 @@ void setup() { Serial.println("* is a card inserted?"); Serial.println("* is your wiring correct?"); Serial.println("* did you change the chipSelect pin to match your shield or module?"); - while (1); + while (1) + ; } else { Serial.println("Wiring is correct and a card is present."); } @@ -67,7 +70,8 @@ void setup() { // open 'volume'/'partition' - should be FAT16 or FAT32 if (!volume.init(card)) { Serial.println("Could not find FAT16/FAT32 partition.\nMake sure you've formatted the card"); - while (1); + while (1) + ; } Serial.print("Clusters: "); @@ -84,9 +88,9 @@ void setup() { Serial.print("Volume type is: FAT"); Serial.println(volume.fatType(), DEC); - volumesize = volume.blocksPerCluster(); // clusters are collections of blocks - volumesize *= volume.clusterCount(); // we'll have a lot of clusters - volumesize /= 2; // SD card blocks are always 512 bytes (2 blocks are 1KB) + volumesize = volume.blocksPerCluster(); // clusters are collections of blocks + volumesize *= volume.clusterCount(); // we'll have a lot of clusters + volumesize /= 2; // SD card blocks are always 512 bytes (2 blocks are 1KB) Serial.print("Volume size (Kb): "); Serial.println(volumesize); Serial.print("Volume size (Mb): "); @@ -103,4 +107,3 @@ void setup() { } void loop() {} - diff --git a/desktop/libraries/Klangstrom/examples/02.Peripherals/Presets/Presets.ino b/desktop/libraries/Klangstrom/examples/02.Peripherals/Presets/Presets.ino new file mode 100644 index 00000000..ccc7f6d3 --- /dev/null +++ b/desktop/libraries/Klangstrom/examples/02.Peripherals/Presets/Presets.ino @@ -0,0 +1,56 @@ +#include + +#include "Klangstrom.h" + +using namespace klangstrom; + +/* define struct to hold presets */ +struct Presets { + float frequency = 440.0; + float amplitude = 0.4; +}; + +Presets mPresets; + +void setup() { + Serial.begin(115200); + Serial.println("-------"); + Serial.println("Presets"); + Serial.println("-------"); + + register_encoder_pressed(encoder_pressed); + + /* hold button 00 at start up to reset presets */ + if (button_state(ENCODER_00)) { + Serial.print("reset presets: "); + EEPROM.put(0, mPresets); + } else { + Serial.print("read presets: "); + EEPROM.get(0, mPresets); + } + print_presets(); +} + +void loop() {} + +void print_presets() { + Serial.print(mPresets.amplitude); + Serial.print(", "); + Serial.print(mPresets.frequency); + Serial.println(); +} + +void encoder_pressed(const uint8_t index) { + if (index == ENCODER_00) { + Serial.print("write presets 0: "); + mPresets.amplitude = 0.5; + mPresets.frequency = 440.0; + EEPROM.put(0, mPresets); + } else if (index == ENCODER_01) { + Serial.print("write presets 1: "); + mPresets.amplitude = 0.6; + mPresets.frequency = 220.0; + EEPROM.put(0, mPresets); + } + print_presets(); +} diff --git a/desktop/libraries/Klangstrom/examples/02.Peripherals/Serial/Serial.ino b/desktop/libraries/Klangstrom/examples/02.Peripherals/Serial/Serial.ino new file mode 100644 index 00000000..01398f37 --- /dev/null +++ b/desktop/libraries/Klangstrom/examples/02.Peripherals/Serial/Serial.ino @@ -0,0 +1,31 @@ +#include "Klangstrom.h" + +using namespace klangstrom; + +void setup() { + Serial.begin(115200); + Serial.println("------"); + Serial.println("Serial"); + Serial.println("------"); + + LED(LED_00, LED_ON); + LED(LED_01, LED_OFF); + beats_per_minute(120); +} + +void beat(uint32_t beat) { + LED(LED_00, LED_TOGGLE); + uint8_t data[] = "hello world\n\r"; + data_transmit(SERIAL_00, data, 13); +} + +void loop() {} + +void data_receive(const uint8_t receiver, uint8_t* data, uint8_t length) { + if (receiver == SERIAL_01) { + for (int i = 0; i < length; i++) { + Serial.print((char)data[i]); + LED(LED_01, LED_TOGGLE); + } + } +} diff --git a/desktop/libraries/Klangstrom/examples/04.Tools/ApplicationTemplate/ApplicationTemplate.ino b/desktop/libraries/Klangstrom/examples/04.Tools/ApplicationTemplate/ApplicationTemplate.ino index 764b1029..fdf9ff1d 100644 --- a/desktop/libraries/Klangstrom/examples/04.Tools/ApplicationTemplate/ApplicationTemplate.ino +++ b/desktop/libraries/Klangstrom/examples/04.Tools/ApplicationTemplate/ApplicationTemplate.ino @@ -1,7 +1,7 @@ #include "KlangNodes.hpp" #include "Klangstrom.h" -// @TODO add `TaskScheduler` +// @TODO merge with `SchedulingTasks` using namespace klang; using namespace klangstrom; @@ -10,12 +10,17 @@ NodeVCOFunction mVCO; NodeDAC mDAC; bool toggle_amplitude = false; -float output_buffer_left[KLANG_SAMPLES_PER_AUDIO_BLOCK]; -float output_buffer_right[KLANG_SAMPLES_PER_AUDIO_BLOCK]; +float output_buffer_left[KLANG_SAMPLES_PER_AUDIO_BLOCK]; +float output_buffer_right[KLANG_SAMPLES_PER_AUDIO_BLOCK]; volatile bool schedule_audio = false; volatile bool schedule_beat = false; void setup() { + Serial.begin(115200); + Serial.println("-------------------"); + Serial.println("ApplicationTemplate"); + Serial.println("-------------------"); + Klang::lock(); Klang::connect(mVCO, Node::CH_OUT_SIGNAL, mDAC, NodeDAC::CH_IN_SIGNAL); mVCO.set_amplitude(0.0); diff --git a/desktop/libraries/Klangstrom/examples/04.Tools/QueryButtonStates/QueryButtonStates.ino b/desktop/libraries/Klangstrom/examples/04.Tools/QueryButtonStates/QueryButtonStates.ino index 65854cd8..132f6e72 100644 --- a/desktop/libraries/Klangstrom/examples/04.Tools/QueryButtonStates/QueryButtonStates.ino +++ b/desktop/libraries/Klangstrom/examples/04.Tools/QueryButtonStates/QueryButtonStates.ino @@ -14,6 +14,7 @@ void setup() { Serial.println("-----------------"); Serial.println("QueryButtonStates"); Serial.println("-----------------"); + Serial.println("in `setup()`: "); Serial.println("---"); print_button_states(); diff --git a/desktop/libraries/Klangstrom/examples/04.Tools/SchedulingTasks/SchedulingTasks.ino b/desktop/libraries/Klangstrom/examples/04.Tools/SchedulingTasks/SchedulingTasks.ino index 7a992166..edeb43f6 100644 --- a/desktop/libraries/Klangstrom/examples/04.Tools/SchedulingTasks/SchedulingTasks.ino +++ b/desktop/libraries/Klangstrom/examples/04.Tools/SchedulingTasks/SchedulingTasks.ino @@ -1,6 +1,8 @@ #include "Arduino.h" #include "TaskScheduler.h" +// @TODO merge with `ApplicationTemplate` + using namespace klangstrom; TaskScheduler fTaskScheduler; @@ -10,7 +12,6 @@ void setup() { Serial.println("---------------"); Serial.println("SchedulingTasks"); Serial.println("---------------"); - Serial.println("---"); fTaskScheduler.schedule_priority_task(exec_priority_task); fTaskScheduler.schedule_task(exec_normal_task); diff --git a/desktop/libraries/Klangstrom/examples/05.Applications/TalkingTerminal/TalkingTerminal.ino b/desktop/libraries/Klangstrom/examples/05.Applications/TalkingTerminal/TalkingTerminal.ino index c24ba44b..1b5af042 100644 --- a/desktop/libraries/Klangstrom/examples/05.Applications/TalkingTerminal/TalkingTerminal.ino +++ b/desktop/libraries/Klangstrom/examples/05.Applications/TalkingTerminal/TalkingTerminal.ino @@ -24,9 +24,9 @@ NodeTextToSpeechSAM mTTS(46160); void setup() { Serial.begin(115200); - Serial.println("----------------"); - Serial.println("Talking Terminal"); - Serial.println("----------------"); + Serial.println("---------------"); + Serial.println("TalkingTerminal"); + Serial.println("---------------"); USBHost.init(); USBHost.register_key_pressed(key_pressed); @@ -62,7 +62,7 @@ void beat(uint32_t beat) { LED(LED_01, LED_TOGGLE); } -void audioblock(float** input_signal, float** output_signal) { +void audioblock(float **input_signal, float **output_signal) { mDAC.process_frame(output_signal[LEFT], output_signal[RIGHT]); } diff --git a/desktop/libraries/Klangstrom/examples/06.Experiments/MaximumMemoryUsage/MaximumMemoryUsage.ino b/desktop/libraries/Klangstrom/examples/06.Experiments/MaximumMemoryUsage/MaximumMemoryUsage.ino index 2c8242ac..b2fcfb68 100644 --- a/desktop/libraries/Klangstrom/examples/06.Experiments/MaximumMemoryUsage/MaximumMemoryUsage.ino +++ b/desktop/libraries/Klangstrom/examples/06.Experiments/MaximumMemoryUsage/MaximumMemoryUsage.ino @@ -2,9 +2,9 @@ void setup() { Serial.begin(115200); - Serial.println("-------------------------"); - Serial.println("PrintAvailableMemoryBlock"); - Serial.println("-------------------------"); + Serial.println("------------------"); + Serial.println("MAximumMemoryUsage"); + Serial.println("------------------"); } void print_available_memory_block(uint32_t size_available_memory_block) { @@ -30,7 +30,7 @@ uint8_t memory_RAM_D2[294912] __attribute__((section(".sample_buffer"))) = { uint8_t memory_RAM_D3[59392] __attribute__((section(".audio_block_buffer"))) = {0}; // RAM_D3 :: 64K > 65536 bytes - 6144 bytes ( DMA BUFFER ) = 59392 bytes /* note that this allocation scheme results into a total of 871380 bytes ( 850K ) separated into 3 * memory blocks. also note that the `.data` section is the default location for global variables. - * the section attributes `.sample_buffer` and `.audio_block_buffer` are required to use other + * the section attributes `.sample_buffer` and `.audio_block_buffer` are required to use other * memory locations. */ diff --git a/desktop/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleSavingPresets/ExampleSavingPresets.ino b/desktop/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleSavingPresets/ExampleSavingPresets.ino deleted file mode 100644 index 8ef1e3de..00000000 --- a/desktop/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleSavingPresets/ExampleSavingPresets.ino +++ /dev/null @@ -1,87 +0,0 @@ -// -// ExampleSavingPresets -// - -#include "Klangstrom.h" -#include - -#include "KlangNodes.hpp" - -using namespace klang; -using namespace klangstrom; - -NodeVCOFunction mVCO; -NodeDAC mDAC; - -/* define struct to hold presets */ -struct Presets { - float frequency = 440.0; - float amplitude = 0.4; -}; - -Presets mPresets; - -void setup() { - begin_serial_debug(true); - - /* hold button 00 at start up to reset presets */ - if (button_state(KLST_BUTTON_ENCODER_00)) { - serial_debug.println("reset presets"); - EEPROM.put(0, mPresets); - } else { - serial_debug.println("read presets"); - EEPROM.get(0, mPresets); - } - - Klang::lock(); - Klang::connect(mVCO, Node::CH_OUT_SIGNAL, mDAC, NodeDAC::CH_IN_SIGNAL); - mVCO.set_amplitude(mPresets.amplitude); - mVCO.set_frequency(mPresets.frequency); - mVCO.set_waveform(NodeVCOFunction::WAVEFORM::SINE); - Klang::unlock(); -} - -void loop() { - led(LED_00, true); - mVCO.set_amplitude(mPresets.amplitude); - delay(500); - led(LED_00, false); - mVCO.set_amplitude(0.0); - delay(500); - Serial.print(mPresets.amplitude); - Serial.print(", "); - Serial.print(mPresets.frequency); - Serial.println(); -} - -void event_receive(const EVENT_TYPE event, const void* data) { - if (event == EVENT_ENCODER_BUTTON_PRESSED) { - if (data[INDEX] == ENCODER_00) { - serial_debug.println("write presets 0"); - mPresets.amplitude = 0.4; - mPresets.frequency = 440.0; - mVCO.set_amplitude(mPresets.amplitude); - mVCO.set_frequency(mPresets.frequency); - EEPROM.put(0, mPresets); - } else if (data[INDEX] == ENCODER_01) { - serial_debug.println("write presets 1"); - mPresets.amplitude = 0.5; - mPresets.frequency = 220.0; - mVCO.set_amplitude(mPresets.amplitude); - mVCO.set_frequency(mPresets.frequency); - EEPROM.put(0, mPresets); - } else if (data[INDEX] == ENCODER_02) { - serial_debug.println("write presets 2"); - mPresets.amplitude = 0.6; - mPresets.frequency = 110.0; - mVCO.set_amplitude(mPresets.amplitude); - mVCO.set_frequency(mPresets.frequency); - EEPROM.put(0, mPresets); - } - } -} - -void audioblock(float* output_signal[LEFT], float* output_signal[RIGHT], - float* input_signal[LEFT], float* input_signal[RIGHT]) { - mDAC.process_frame(output_signal[LEFT], output_signal[RIGHT]); -} diff --git a/desktop/libraries/KlangstromCard/examples/ExampleListFiles/ExampleListFiles.ino b/desktop/libraries/KlangstromCard/examples/ExampleListFiles/ExampleListFiles.ino index 0291f8e1..d8ab3426 100644 --- a/desktop/libraries/KlangstromCard/examples/ExampleListFiles/ExampleListFiles.ino +++ b/desktop/libraries/KlangstromCard/examples/ExampleListFiles/ExampleListFiles.ino @@ -9,6 +9,10 @@ using namespace klangstrom; void setup() { Serial.begin(115200); + Serial.println("----------------"); + Serial.println("ExampleListFiles"); + Serial.println("----------------"); + Serial.println("--- ExampleList Files"); Serial.println(); diff --git a/desktop/libraries/KlangstromCard/examples/ExamplePlayWAVFile/ExamplePlayWAVFile.ino b/desktop/libraries/KlangstromCard/examples/ExamplePlayWAVE/ExamplePlayWAVE.ino similarity index 91% rename from desktop/libraries/KlangstromCard/examples/ExamplePlayWAVFile/ExamplePlayWAVFile.ino rename to desktop/libraries/KlangstromCard/examples/ExamplePlayWAVE/ExamplePlayWAVE.ino index 82a3d5e9..037693f7 100644 --- a/desktop/libraries/KlangstromCard/examples/ExamplePlayWAVFile/ExamplePlayWAVFile.ino +++ b/desktop/libraries/KlangstromCard/examples/ExamplePlayWAVE/ExamplePlayWAVE.ino @@ -3,8 +3,8 @@ * and play the first file as a WAV file. */ -#include "Klangstrom.h" #include "KlangNodes.hpp" +#include "Klangstrom.h" #include "KlangstromCard.h" using namespace klang; @@ -16,7 +16,9 @@ NodeSamplerI16 mSamplerRight; void setup() { Serial.begin(115200); - Serial.println("--- Example Play WAV File"); + Serial.println("---------------"); + Serial.println("ExamplePlayWAVE"); + Serial.println("---------------"); bool mCardOpenError = Card.begin(); if (mCardOpenError) { @@ -74,7 +76,6 @@ void setup() { void loop() {} -void audioblock(float* output_signal[LEFT], float* output_signal[RIGHT], - float* input_signal[LEFT], float* input_signal[RIGHT]) { +void audioblock(float** input_signal, float** output_signal) { mDAC.process_frame(output_signal[LEFT], output_signal[RIGHT]); } diff --git a/desktop/libraries/KlangstromCard/examples/ExampleStreamWAVE/ExampleStreamWAVE.ino b/desktop/libraries/KlangstromCard/examples/ExampleStreamWAVE/ExampleStreamWAVE.ino new file mode 100644 index 00000000..e9ed2564 --- /dev/null +++ b/desktop/libraries/KlangstromCard/examples/ExampleStreamWAVE/ExampleStreamWAVE.ino @@ -0,0 +1,71 @@ +/* + * this example demonstrates how to list all files on an SD card + * and play the first file as a WAV file. + */ + +#include "KlangNodes.hpp" +#include "Klangstrom.h" +#include "KlangstromCard.h" + +using namespace klang; +using namespace klangstrom; + +NodeDAC mDAC; +NodeSamplerI16 mSampler; + +void setup() { + Serial.begin(115200); + Serial.println("-----------------"); + Serial.println("ExampleStreamWAVE"); + Serial.println("-----------------"); + + bool mCardOpenError = Card.begin(); + if (mCardOpenError) { + Serial.println("--- opening card failed"); + return; + } + + Serial.println("--- reading file list"); + vector mFiles; + Card.get_file_list(mFiles); + + for (uint16_t i = 0; i < mFiles.size(); i++) { + Serial.print(i); + Serial.print("\t"); + Serial.println(mFiles[i]); + } + Serial.println(); + + bool mFileOpenError = Card.open(mFiles[0]); /* open first file on card */ + if (mFileOpenError) { + Serial.println("--- opening file failed"); + return; + } else { + Serial.print("--- opening file: "); + Serial.println(mFiles[0]); + } + + KlangstromWaveFile mWAV; + int mWAVOpenError = Card.load_WAV(&mWAV); + if (!mWAVOpenError) { + Serial.println("--- WAV file header: "); + Card.print_WAV_header(mWAV.header); + + Klang::connect(mSampler, Node::CH_OUT_SIGNAL, mDAC, NodeDAC::CH_IN_SIGNAL_LEFT); + mSampler.set_buffer(mWAV.get_sample_data(KlangstromWaveFile::CHANNEL_LEFT)); + mSampler.set_buffer_size(mWAV.number_of_samples); + mSampler.loop(true); + mSampler.start(); + + mDAC.set_stereo(false); + } else { + Serial.print("--- loading WAV failed : "); + Serial.println(mWAVOpenError); + } +} + +void loop() {} + +void audioblock(float** input_signal, float** output_signal) { + mDAC.process_frame(output_signal[LEFT], output_signal[RIGHT]); +} diff --git a/desktop/libraries/KlangstromCard/examples/ExampleWAVFileInfo/ExampleWAVFileInfo.ino b/desktop/libraries/KlangstromCard/examples/ExampleWAVFileInfo/ExampleWAVFileInfo.ino index bd28f109..9e6cd6d1 100644 --- a/desktop/libraries/KlangstromCard/examples/ExampleWAVFileInfo/ExampleWAVFileInfo.ino +++ b/desktop/libraries/KlangstromCard/examples/ExampleWAVFileInfo/ExampleWAVFileInfo.ino @@ -42,8 +42,9 @@ void read_WAV_header() { void setup() { Serial.begin(115200); - Serial.println("--- Example WAV File Info"); - Serial.println(); + Serial.println("------------------"); + Serial.println("ExampleWAVFileInfo"); + Serial.println("------------------"); bool mOpenCardError = Card.begin(); vector mFiles; diff --git a/desktop/libraries/KlangstromCard/examples/ExampleWriteRawData/ExampleWriteRawData.ino b/desktop/libraries/KlangstromCard/examples/ExampleWriteRawData/ExampleWriteRawData.ino index f638e1ed..cfe605a9 100644 --- a/desktop/libraries/KlangstromCard/examples/ExampleWriteRawData/ExampleWriteRawData.ino +++ b/desktop/libraries/KlangstromCard/examples/ExampleWriteRawData/ExampleWriteRawData.ino @@ -1,5 +1,5 @@ -#include "Klangstrom.h" #include "KlangNodes.hpp" +#include "Klangstrom.h" #include "KlangstromCard.h" using namespace klang; @@ -25,8 +25,9 @@ void print_file_list(vector& pFiles) { void setup() { Serial.begin(115200); - Serial.println("--- Example Write Raw Data"); - Serial.println(); + Serial.println("-------------------"); + Serial.println("ExampleWriteRawData"); + Serial.println("-------------------"); bool mOpenCardError = Card.begin(); vector mFiles; @@ -81,8 +82,7 @@ void loop() { } } -void audioblock(float* output_signal[LEFT], float* output_signal[RIGHT], - float* input_signal[LEFT], float* input_signal[RIGHT]) { +void audioblock(float** input_signal, float** output_signal) { mDAC.process_frame(output_signal[LEFT], output_signal[RIGHT]); if (!fBufferUpdated) { KLANG_COPY_AUDIO_BUFFER(mBuffer, output_signal[LEFT]); diff --git a/desktop/libraries/KlangstromCard/src/KlangstromCard.cpp b/desktop/libraries/KlangstromCard/src/KlangstromCard.cpp index cda75c2a..3e70f51a 100644 --- a/desktop/libraries/KlangstromCard/src/KlangstromCard.cpp +++ b/desktop/libraries/KlangstromCard/src/KlangstromCard.cpp @@ -19,12 +19,22 @@ #include "KlangstromDefinesArduino.h" -#if KLST_ARCH == KLST_ARCH_MCU +#define KLST_ARCH_MCU 1 +#define KLST_ARCH_CPU 2 +#define KLST_ARCH_DESKTOP 2 +#define KLST_ARCH_VCV 3 +#define KLST_ARCH_PLUGIN 3 + +#ifndef KLST_ARCH +#warning "KLST_ARCH not defined" +#endif + +#if (KLST_ARCH == KLST_ARCH_MCU) #include "KlangstromCardBSP_STM32.h" klangstrom::KlangstromCard *CardPtr = new klangstrom::KlangstromCardBSP_STM32(); #elif KLST_ARCH == KLST_ARCH_CPU #include "KlangstromCardBSP_SDL.h" klangstrom::KlangstromCard *CardPtr = new klangstrom::KlangstromCardBSP_SDL(); #else -#warning "KLST_ARCH not defined" +#warning "KLST_ARCH not defined properly" #endif diff --git a/desktop/libraries/KlangstromCard/src/KlangstromCardBSP_STM32.h b/desktop/libraries/KlangstromCard/src/KlangstromCardBSP_STM32.h index 29529a06..58736dca 100644 --- a/desktop/libraries/KlangstromCard/src/KlangstromCardBSP_STM32.h +++ b/desktop/libraries/KlangstromCard/src/KlangstromCardBSP_STM32.h @@ -82,14 +82,15 @@ namespace klangstrom { private: SPISettings m_spiSettings; SPIClass mSPI = SPIClass( - SDCARD_SPI_MOSI, - SDCARD_SPI_MISO, - SDCARD_SPI_SCK, - SDCARD_SPI_CS); + SDCARD_SPI_MOSI, + SDCARD_SPI_MISO, + SDCARD_SPI_SCK, + SDCARD_SPI_CS); }; class KlangstromCardBSP_STM32 : public KlangstromCard { public: + KlangstromCardBSP_STM32(){}; bool begin(); bool begin(const char *pFolderPath); void get_file_list(vector &pFiles, bool pIncludeHiddenFiles = false); @@ -98,7 +99,7 @@ namespace klangstrom { void close(); void print_WAV_header(WaveHeader_t *mHeader); - int create_file(const String pFileName); + int create_file(const String pFileName); protected: int BSP_read_block(uint8_t *pReadBuffer, uint32_t pReadBufferSize); diff --git a/desktop/libraries/KlangstromDisplay/examples/CommandLineInterface/CommandLineInterface.ino b/desktop/libraries/KlangstromDisplay/examples/CommandLineInterface/CommandLineInterface.ino index 625af4f2..b0ab063c 100644 --- a/desktop/libraries/KlangstromDisplay/examples/CommandLineInterface/CommandLineInterface.ino +++ b/desktop/libraries/KlangstromDisplay/examples/CommandLineInterface/CommandLineInterface.ino @@ -9,6 +9,11 @@ KlangstromDisplay &Display = KlangstromDisplay::create(); KlangstromDisplayCLI &CLI = KlangstromDisplayCLI::create(&Display, &Font_7x10); void setup() { + Serial.begin(115200); + Serial.println("--------------------"); + Serial.println("CommandLineInterface"); + Serial.println("--------------------"); + Serial.begin(115200); Display.begin(); diff --git a/desktop/libraries/KlangstromDisplay/examples/DrawBuffer/DrawBuffer.ino b/desktop/libraries/KlangstromDisplay/examples/DrawBuffer/DrawBuffer.ino index e7571172..facf817f 100644 --- a/desktop/libraries/KlangstromDisplay/examples/DrawBuffer/DrawBuffer.ino +++ b/desktop/libraries/KlangstromDisplay/examples/DrawBuffer/DrawBuffer.ino @@ -1,5 +1,5 @@ -#include "Klangstrom.h" #include "KlangNodes.hpp" +#include "Klangstrom.h" #include "KlangstromDisplay.h" #include "KlangstromDisplayDrawBuffer.h" #include "KlangstromDisplayFont_5x8.h" @@ -14,6 +14,11 @@ KlangstromDisplayDrawBuffer fDrawBuffer(32); KlangstromDisplay& Display = KlangstromDisplay::create(); void setup() { + Serial.begin(115200); + Serial.println("----------"); + Serial.println("DrawBuffer"); + Serial.println("----------"); + Display.begin(); Display.background(0, 0, 0); Display.color(255, 255, 255); @@ -53,8 +58,7 @@ void loop() { delay(300); // wait for a second } -void audioblock(float* output_signal[LEFT], float* output_signal[RIGHT], - float* input_signal[LEFT], float* input_signal[RIGHT]) { +void audioblock(float** input_signal, float** output_signal) { fDAC.process_frame(output_signal[LEFT], output_signal[RIGHT]); fDrawBuffer.update_buffer(output_signal[RIGHT], KLANG_SAMPLES_PER_AUDIO_BLOCK); } diff --git a/desktop/libraries/KlangstromDisplay/examples/DrawPrimitives/DrawPrimitives.ino b/desktop/libraries/KlangstromDisplay/examples/DrawPrimitives/DrawPrimitives.ino index 412cbe3e..d4ee1d69 100644 --- a/desktop/libraries/KlangstromDisplay/examples/DrawPrimitives/DrawPrimitives.ino +++ b/desktop/libraries/KlangstromDisplay/examples/DrawPrimitives/DrawPrimitives.ino @@ -1,5 +1,5 @@ -#include "Klangstrom.h" #include "CycleCounter.h" +#include "Klangstrom.h" #include "KlangstromDisplay.h" #include "KlangstromDisplayDrawBuffer.h" #include "KlangstromDisplayFont_5x8.h" @@ -90,6 +90,11 @@ void draw_primitves() { } void setup() { + Serial.begin(115200); + Serial.println("--------------"); + Serial.println("DrawPrimitives"); + Serial.println("--------------"); + Display.begin(); klst_enable_cycle_counter(); } diff --git a/desktop/libraries/KlangstromDisplay/examples/Fonts/Fonts.ino b/desktop/libraries/KlangstromDisplay/examples/Fonts/Fonts.ino index 2ea22c2b..c435cea9 100644 --- a/desktop/libraries/KlangstromDisplay/examples/Fonts/Fonts.ino +++ b/desktop/libraries/KlangstromDisplay/examples/Fonts/Fonts.ino @@ -40,6 +40,11 @@ void draw_primitves() { } void setup() { + Serial.begin(115200); + Serial.println("-----"); + Serial.println("Fonts"); + Serial.println("-----"); + Display.begin(); } diff --git a/desktop/libraries/KlangstromDisplay/examples/FullscreenAnimation/FullscreenAnimation.ino b/desktop/libraries/KlangstromDisplay/examples/FullscreenAnimation/FullscreenAnimation.ino index 1362df83..eb425fca 100644 --- a/desktop/libraries/KlangstromDisplay/examples/FullscreenAnimation/FullscreenAnimation.ino +++ b/desktop/libraries/KlangstromDisplay/examples/FullscreenAnimation/FullscreenAnimation.ino @@ -14,6 +14,11 @@ void draw_primitves() { } void setup() { + Serial.begin(115200); + Serial.println("-------------------"); + Serial.println("FullscreenAnimation"); + Serial.println("-------------------"); + Display.begin(); beats_per_minute(900); // == 15 Hz } diff --git a/desktop/libraries/KlangstromDisplay/examples/Terminal/Terminal.ino b/desktop/libraries/KlangstromDisplay/examples/Terminal/Terminal.ino index b0296853..fa9a7e74 100644 --- a/desktop/libraries/KlangstromDisplay/examples/Terminal/Terminal.ino +++ b/desktop/libraries/KlangstromDisplay/examples/Terminal/Terminal.ino @@ -10,7 +10,9 @@ KlangstromDisplayTerminal* mTerminal; void setup() { Serial.begin(115200); - Serial.println("--- KLST TERMINAL ---"); + Serial.println("--------"); + Serial.println("Terminal"); + Serial.println("--------"); Display.begin(); Display.background(0, 0, 0); diff --git a/desktop/libraries/USBDevice/examples/USBDeviceKeyboard/USBDeviceKeyboard.ino b/desktop/libraries/USBDevice/examples/USBDeviceKeyboard/USBDeviceKeyboard.ino index 6fb70088..40ba00de 100644 --- a/desktop/libraries/USBDevice/examples/USBDeviceKeyboard/USBDeviceKeyboard.ino +++ b/desktop/libraries/USBDevice/examples/USBDeviceKeyboard/USBDeviceKeyboard.ino @@ -13,11 +13,10 @@ char mChar = 'a'; void setup() { Serial.begin(115200); - Serial.println("-------------------"); - Serial.println("USB Device Keyboard"); - Serial.println("-------------------"); - Serial.println(__DATE__); - Serial.println(__TIME__); + Serial.println("-----------------"); + Serial.println("USBDeviceKeyboard"); + Serial.println("-----------------"); + USBDevice.init(); } diff --git a/desktop/libraries/USBDevice/examples/USBDeviceMIDI/USBDeviceMIDI.ino b/desktop/libraries/USBDevice/examples/USBDeviceMIDI/USBDeviceMIDI.ino index e0793c9d..7cdb444c 100644 --- a/desktop/libraries/USBDevice/examples/USBDeviceMIDI/USBDeviceMIDI.ino +++ b/desktop/libraries/USBDevice/examples/USBDeviceMIDI/USBDeviceMIDI.ino @@ -17,11 +17,10 @@ uint8_t mNote = 48; void setup() { Serial.begin(115200); - Serial.println("---------------"); - Serial.println("USB Device MIDI"); - Serial.println("---------------"); - Serial.println(__DATE__); - Serial.println(__TIME__); + Serial.println("-------------"); + Serial.println("USBDeviceMIDI"); + Serial.println("-------------"); + USBDevice.init(); USBDevice.register_midi_note_on(midi_note_on); USBDevice.register_midi_note_off(midi_note_off); diff --git a/desktop/libraries/USBDevice/examples/USBDeviceMouse/USBDeviceMouse.ino b/desktop/libraries/USBDevice/examples/USBDeviceMouse/USBDeviceMouse.ino index d7be27f8..cf8b4b70 100644 --- a/desktop/libraries/USBDevice/examples/USBDeviceMouse/USBDeviceMouse.ino +++ b/desktop/libraries/USBDevice/examples/USBDeviceMouse/USBDeviceMouse.ino @@ -11,11 +11,10 @@ using namespace klangstrom; void setup() { Serial.begin(115200); - Serial.println("----------------"); - Serial.println("USB Device Mouse"); - Serial.println("----------------"); - Serial.println(__DATE__); - Serial.println(__TIME__); + Serial.println("--------------"); + Serial.println("USBDeviceMouse"); + Serial.println("--------------"); + usb_device_init(); } diff --git a/desktop/libraries/USBHost/examples/USBHostMIDI/USBHostMIDI.ino b/desktop/libraries/USBHost/examples/USBHostMIDI/USBHostMIDI.ino index c909a6bd..4d23917d 100644 --- a/desktop/libraries/USBHost/examples/USBHostMIDI/USBHostMIDI.ino +++ b/desktop/libraries/USBHost/examples/USBHostMIDI/USBHostMIDI.ino @@ -8,18 +8,10 @@ using namespace klangstrom; void setup() { Serial.begin(115200); - Serial.println("-------------"); - Serial.println("USB Host MIDI"); - Serial.print("( "); - Serial.print(__DATE__); - Serial.print(" "); - Serial.print(__TIME__); - Serial.println(" )"); - Serial.println("-------------"); - printf("foo"); + Serial.println("-----------"); + Serial.println("USBHostMIDI"); + Serial.println("-----------"); - Serial.println(__DATE__); - Serial.println(__TIME__); USBHost.init(); USBHost.register_midi_note_off(midi_note_off); USBHost.register_midi_note_on(midi_note_on); diff --git a/desktop/libraries/USBHost/examples/USBHostMouseKeyboard/USBHostMouseKeyboard.ino b/desktop/libraries/USBHost/examples/USBHostMouseKeyboard/USBHostMouseKeyboard.ino index 194363cf..90d4cb13 100644 --- a/desktop/libraries/USBHost/examples/USBHostMouseKeyboard/USBHostMouseKeyboard.ino +++ b/desktop/libraries/USBHost/examples/USBHostMouseKeyboard/USBHostMouseKeyboard.ino @@ -8,14 +8,10 @@ using namespace klangstrom; void setup() { Serial.begin(115200); - Serial.println("-------------------------"); - Serial.println("USB Host Mouse + Keyboard"); - Serial.print("( "); - Serial.print(__DATE__); - Serial.print(" "); - Serial.print(__TIME__); - Serial.println(" )"); - Serial.println("-------------------------"); + Serial.println("--------------------"); + Serial.println("USBHostMouseKeyboard"); + Serial.println("--------------------"); + USBHost.init(); USBHost.register_key_pressed(key_pressed); USBHost.register_key_released(key_released); diff --git a/desktop/platform.txt b/desktop/platform.txt index f3c10a94..588bdc8a 100644 --- a/desktop/platform.txt +++ b/desktop/platform.txt @@ -17,6 +17,8 @@ version=0.0.3 recipe.hooks.prebuild.1.pattern=date +macos.architecture=arm + compiler.path.macosx=/usr/bin/ compiler.path.linux=/usr/bin/ compiler.klst_define=-DKLST_ARCH=2 -DKLANG_SAMPLES_PER_AUDIO_BLOCK={build.flags.audioblock} {build.flags.OSC} {build.flags.DISPLAY} -DKLANG_AUDIO_RATE={build.flags.samplingrate} -DKLANG_DEBUG_LEVEL=2 -DKLANG_OSC_TRANSMIT_PORT=7001 -DKLANG_OSC_RECEIVE_PORT=7001 -DKLANG_BOARD_EMULATOR={build.flags.board} -DKLST_USE_CMSIS_DSP -DTEST @@ -25,7 +27,7 @@ compiler.klst_lib_src=./Klangstrom #compiler.klst_lib_src=../../libraries/Klangstrom/src #compiler.klst_shared_lib=../../libraries/Klangstrom/lib/klangstrom.a -compiler.SDL2_lib_src.macosx=SDL2.macosx/include/ +compiler.SDL2_lib_src.macosx=SDL2.macos.{macos.architecture}/include/ compiler.SDL2_lib_src.linux=SDL2.linux/include/ compiler.SDL2_lib_src.windows=SDL2.windows/include/ @@ -62,11 +64,11 @@ recipe.ar.pattern="{compiler.path}{compiler.ar.cmd}" {compiler.ar.flags} "{archi compiler.c.linker.cmd=g++ compiler.ldflags= # @TODO try to get rif of `cores/sdl` in `libpath` -compiler.c.linker.libpath.macosx=cores/sdl/SDL2.macos -compiler.c.linker.libpath.linux=cores/sdl/SDL2.linux -compiler.c.linker.libpath.windows=cores/sdl/SDL2.windows -compiler.c.linker.SDL2flags.macosx=-lm -liconv -Wl,-framework,CoreAudio -Wl,-framework,AudioToolbox -Wl,-framework,ForceFeedback -lobjc -Wl,-framework,CoreVideo -Wl,-framework,Cocoa -Wl,-framework,Carbon -Wl,-framework,IOKit -Wl,-weak_framework,QuartzCore -Wl,-weak_framework,Metal -lSDL2 -compiler.c.linker.SDL2flags.linux=-lm -D_THREAD_SAFE -D_REENTRANT -lpthread -lSDL2 -L{runtime.platform.path}/{compiler.c.linker.libpath}/lib/ +compiler.c.linker.libpath.macosx=cores/sdl/SDL2.macos.{macos.architecture} +compiler.c.linker.libpath.linux=cores/SDL2.linux +compiler.c.linker.libpath.windows=cores/SDL2.windows +compiler.c.linker.SDL2flags.macosx=-lm -liconv -Wl,-framework,CoreAudio -Wl,-framework,AudioToolbox -Wl,-framework,ForceFeedback -lobjc -Wl,-framework,CoreVideo -Wl,-framework,Cocoa -Wl,-framework,Carbon -Wl,-framework,IOKit -Wl,-weak_framework,QuartzCore -Wl,-weak_framework,Metal -lSDL2 "-L{runtime.platform.path}/{compiler.c.linker.libpath}/lib/" +compiler.c.linker.SDL2flags.linux=-lm -D_THREAD_SAFE -D_REENTRANT -lpthread -lSDL2 "-L{runtime.platform.path}/{compiler.c.linker.libpath}/lib/" compiler.c.linker.SDL2flags.windows= recipe.c.combine.pattern="{compiler.path}{compiler.c.linker.cmd}" {object_files} {compiler.c.linker.SDL2flags} "{build.path}/{archive_file}" "-L{build.path}" -o "{build.path}/{build.project_name}.exec" {compiler.ldflags} # recipe.c.combine.pattern="{compiler.path}{compiler.c.linker.cmd}" {object_files} {compiler.c.linker.SDL2flags} {runtime.platform.path}/{compiler.c.linker.libpath} "{build.path}/{archive_file}" "-L{build.path}" -o "{build.path}/{build.project_name}.exec" {compiler.ldflags} diff --git a/docs/_includes/code/ExampleListFiles.ino b/docs/_includes/code/ExampleListFiles.ino index 0291f8e1..d8ab3426 100644 --- a/docs/_includes/code/ExampleListFiles.ino +++ b/docs/_includes/code/ExampleListFiles.ino @@ -9,6 +9,10 @@ using namespace klangstrom; void setup() { Serial.begin(115200); + Serial.println("----------------"); + Serial.println("ExampleListFiles"); + Serial.println("----------------"); + Serial.println("--- ExampleList Files"); Serial.println(); diff --git a/stm32/libraries/KlangstromCard/examples/ExamplePlayWAVFile/ExamplePlayWAVFile.ino b/docs/_includes/code/ExamplePlayWAVE.ino similarity index 91% rename from stm32/libraries/KlangstromCard/examples/ExamplePlayWAVFile/ExamplePlayWAVFile.ino rename to docs/_includes/code/ExamplePlayWAVE.ino index 82a3d5e9..037693f7 100644 --- a/stm32/libraries/KlangstromCard/examples/ExamplePlayWAVFile/ExamplePlayWAVFile.ino +++ b/docs/_includes/code/ExamplePlayWAVE.ino @@ -3,8 +3,8 @@ * and play the first file as a WAV file. */ -#include "Klangstrom.h" #include "KlangNodes.hpp" +#include "Klangstrom.h" #include "KlangstromCard.h" using namespace klang; @@ -16,7 +16,9 @@ NodeSamplerI16 mSamplerRight; void setup() { Serial.begin(115200); - Serial.println("--- Example Play WAV File"); + Serial.println("---------------"); + Serial.println("ExamplePlayWAVE"); + Serial.println("---------------"); bool mCardOpenError = Card.begin(); if (mCardOpenError) { @@ -74,7 +76,6 @@ void setup() { void loop() {} -void audioblock(float* output_signal[LEFT], float* output_signal[RIGHT], - float* input_signal[LEFT], float* input_signal[RIGHT]) { +void audioblock(float** input_signal, float** output_signal) { mDAC.process_frame(output_signal[LEFT], output_signal[RIGHT]); } diff --git a/docs/_includes/code/ExampleStreamWAVE.ino b/docs/_includes/code/ExampleStreamWAVE.ino new file mode 100644 index 00000000..e9ed2564 --- /dev/null +++ b/docs/_includes/code/ExampleStreamWAVE.ino @@ -0,0 +1,71 @@ +/* + * this example demonstrates how to list all files on an SD card + * and play the first file as a WAV file. + */ + +#include "KlangNodes.hpp" +#include "Klangstrom.h" +#include "KlangstromCard.h" + +using namespace klang; +using namespace klangstrom; + +NodeDAC mDAC; +NodeSamplerI16 mSampler; + +void setup() { + Serial.begin(115200); + Serial.println("-----------------"); + Serial.println("ExampleStreamWAVE"); + Serial.println("-----------------"); + + bool mCardOpenError = Card.begin(); + if (mCardOpenError) { + Serial.println("--- opening card failed"); + return; + } + + Serial.println("--- reading file list"); + vector mFiles; + Card.get_file_list(mFiles); + + for (uint16_t i = 0; i < mFiles.size(); i++) { + Serial.print(i); + Serial.print("\t"); + Serial.println(mFiles[i]); + } + Serial.println(); + + bool mFileOpenError = Card.open(mFiles[0]); /* open first file on card */ + if (mFileOpenError) { + Serial.println("--- opening file failed"); + return; + } else { + Serial.print("--- opening file: "); + Serial.println(mFiles[0]); + } + + KlangstromWaveFile mWAV; + int mWAVOpenError = Card.load_WAV(&mWAV); + if (!mWAVOpenError) { + Serial.println("--- WAV file header: "); + Card.print_WAV_header(mWAV.header); + + Klang::connect(mSampler, Node::CH_OUT_SIGNAL, mDAC, NodeDAC::CH_IN_SIGNAL_LEFT); + mSampler.set_buffer(mWAV.get_sample_data(KlangstromWaveFile::CHANNEL_LEFT)); + mSampler.set_buffer_size(mWAV.number_of_samples); + mSampler.loop(true); + mSampler.start(); + + mDAC.set_stereo(false); + } else { + Serial.print("--- loading WAV failed : "); + Serial.println(mWAVOpenError); + } +} + +void loop() {} + +void audioblock(float** input_signal, float** output_signal) { + mDAC.process_frame(output_signal[LEFT], output_signal[RIGHT]); +} diff --git a/docs/_includes/code/ExampleWAVFileInfo.ino b/docs/_includes/code/ExampleWAVFileInfo.ino index bd28f109..9e6cd6d1 100644 --- a/docs/_includes/code/ExampleWAVFileInfo.ino +++ b/docs/_includes/code/ExampleWAVFileInfo.ino @@ -42,8 +42,9 @@ void read_WAV_header() { void setup() { Serial.begin(115200); - Serial.println("--- Example WAV File Info"); - Serial.println(); + Serial.println("------------------"); + Serial.println("ExampleWAVFileInfo"); + Serial.println("------------------"); bool mOpenCardError = Card.begin(); vector mFiles; diff --git a/docs/_includes/code/ExampleWriteRawData.ino b/docs/_includes/code/ExampleWriteRawData.ino index f638e1ed..cfe605a9 100644 --- a/docs/_includes/code/ExampleWriteRawData.ino +++ b/docs/_includes/code/ExampleWriteRawData.ino @@ -1,5 +1,5 @@ -#include "Klangstrom.h" #include "KlangNodes.hpp" +#include "Klangstrom.h" #include "KlangstromCard.h" using namespace klang; @@ -25,8 +25,9 @@ void print_file_list(vector& pFiles) { void setup() { Serial.begin(115200); - Serial.println("--- Example Write Raw Data"); - Serial.println(); + Serial.println("-------------------"); + Serial.println("ExampleWriteRawData"); + Serial.println("-------------------"); bool mOpenCardError = Card.begin(); vector mFiles; @@ -81,8 +82,7 @@ void loop() { } } -void audioblock(float* output_signal[LEFT], float* output_signal[RIGHT], - float* input_signal[LEFT], float* input_signal[RIGHT]) { +void audioblock(float** input_signal, float** output_signal) { mDAC.process_frame(output_signal[LEFT], output_signal[RIGHT]); if (!fBufferUpdated) { KLANG_COPY_AUDIO_BUFFER(mBuffer, output_signal[LEFT]); diff --git a/stm32/boards.txt b/stm32/boards.txt index 6e337dfa..4becf8ae 100644 --- a/stm32/boards.txt +++ b/stm32/boards.txt @@ -28,13 +28,14 @@ KLST_SHEEP.name=Klangstrom Board KLST_SHEEP (STM32H743VI) KLST_SHEEP.build.board=KLST_SHEEP KLST_SHEEP.build.variant=KLST_SHEEP -KLST_SHEEP.build.cmsis_lib_gcc=arm_cortexM7lfsp_math +#KLST_SHEEP.build.cmsis_lib_gcc=arm_cortexM7lfsp_math KLST_SHEEP.build.fpu=-mfpu=fpv5-d16 #KLST_SHEEP.build.fpu=-mfpu=fpv4-sp-d16 KLST_SHEEP.build.series=STM32H7xx +KLST_SHEEP.build.flash_offset=0x0 KLST_SHEEP.build.mcu=cortex-m7 KLST_SHEEP.upload.maximum_size=2097152 -KLST_SHEEP.upload.maximum_data_size=524288 +KLST_SHEEP.upload.maximum_data_size=884736 KLST_SHEEP.build.product_line=STM32H743xx # @@ -178,9 +179,10 @@ KLST_TINY.name=Klangstrom Board KLST_TINY (STM32F446RET) KLST_TINY.build.variant=KLST_TINY KLST_TINY.build.board=KLST_TINY -KLST_TINY.build.cmsis_lib_gcc=arm_cortexM4lf_math +#KLST_TINY.build.cmsis_lib_gcc=arm_cortexM4lf_math KLST_TINY.build.fpu=-mfpu=fpv4-sp-d16 KLST_TINY.build.series=STM32F4xx +KLST_TINY.build.flash_offset=0x0 KLST_TINY.build.mcu=cortex-m4 KLST_TINY.upload.maximum_size=524288 KLST_TINY.upload.maximum_data_size=131072 @@ -321,9 +323,10 @@ KLST_CORE.name=Klangstrom Board KLST_CORE (STM32H743II) KLST_CORE.build.variant=KLST_CORE KLST_CORE.build.board=KLST_CORE -KLST_CORE.build.cmsis_lib_gcc=arm_cortexM7lfsp_math +#KLST_CORE.build.cmsis_lib_gcc=arm_cortexM7lfsp_math KLST_CORE.build.fpu=-mfpu=fpv5-d16 KLST_CORE.build.series=STM32H7xx +KLST_CORE.build.flash_offset=0x0 KLST_CORE.build.mcu=cortex-m7 KLST_CORE.upload.maximum_size=2097152 KLST_CORE.upload.maximum_data_size=884736 diff --git a/stm32/cores/arduino/HardwareSerial.cpp b/stm32/cores/arduino/HardwareSerial.cpp index a06a6275..ff9ca2d2 100644 --- a/stm32/cores/arduino/HardwareSerial.cpp +++ b/stm32/cores/arduino/HardwareSerial.cpp @@ -441,7 +441,7 @@ void HardwareSerial::begin(unsigned long baud, byte config) void HardwareSerial::end() { // wait for transmission of outgoing data - flush(); + flush(TX_TIMEOUT); uart_deinit(&_serial); @@ -487,20 +487,25 @@ int HardwareSerial::availableForWrite(void) return tail - head - 1; } -void HardwareSerial::flush() +void HardwareSerial::flush(uint32_t timeout) { // If we have never written a byte, no need to flush. This special // case is needed since there is no way to force the TXC (transmit // complete) bit to 1 during initialization - if (!_written) { - return; - } - - while ((_serial.tx_head != _serial.tx_tail)) { - // nop, the interrupt handler will free up space for us + if (_written) { + uint32_t tickstart = HAL_GetTick(); + while ((_serial.tx_head != _serial.tx_tail)) { + // the interrupt handler will free up space for us + // Only manage timeout if any + if ((timeout != 0) && ((HAL_GetTick() - tickstart) >= timeout)) { + // clear any transmit data + _serial.tx_head = _serial.tx_tail; + break; + } + } + // If we get here, nothing is queued anymore (DRIE is disabled) and + // the hardware finished transmission (TXC is set). } - // If we get here, nothing is queued anymore (DRIE is disabled) and - // the hardware finished transmission (TXC is set). } size_t HardwareSerial::write(const uint8_t *buffer, size_t size) diff --git a/stm32/cores/arduino/HardwareSerial.h b/stm32/cores/arduino/HardwareSerial.h index 54ebfe2a..b0089bab 100644 --- a/stm32/cores/arduino/HardwareSerial.h +++ b/stm32/cores/arduino/HardwareSerial.h @@ -125,7 +125,7 @@ class HardwareSerial : public Stream { virtual int peek(void); virtual int read(void); int availableForWrite(void); - virtual void flush(void); + virtual void flush(uint32_t timeout = 0); virtual size_t write(uint8_t); inline size_t write(unsigned long n) { diff --git a/stm32/cores/arduino/HardwareTimer.h b/stm32/cores/arduino/HardwareTimer.h index d975ffb1..2215743d 100644 --- a/stm32/cores/arduino/HardwareTimer.h +++ b/stm32/cores/arduino/HardwareTimer.h @@ -90,6 +90,25 @@ typedef enum { PERCENT_COMPARE_FORMAT, // used for Dutycycle } TimerCompareFormat_t; +typedef enum { + FILTER_NONE = 0, // No filter + FILTER_CKINT_N2, // Sampling rate is same as clock interrupt, n=2 events + FILTER_CKINT_N4, // Sampling rate is same as clock interrupt, n=4 events + FILTER_CKINT_N8, // Sampling rate is same as clock interrupt, n=8 events + FILTER_DTS2_N6, // Sampling rate is DTS/2, n=6 events + FILTER_DTS2_N8, // Sampling rate is DTS/2, n=8 events + FILTER_DTS4_N6, // Sampling rate is DTS/4, n=6 events + FILTER_DTS4_N8, // Sampling rate is DTS/4, n=8 events + FILTER_DTS8_N6, // Sampling rate is DTS/8, n=6 events + FILTER_DTS8_N8, // Sampling rate is DTS/8, n=8 events + FILTER_DTS16_N5, // Sampling rate is DTS/16, n=5 events + FILTER_DTS16_N6, // Sampling rate is DTS/16, n=6 events + FILTER_DTS16_N8, // Sampling rate is DTS/16, n=8 events + FILTER_DTS32_N5, // Sampling rate is DTS/32, n=5 events + FILTER_DTS32_N6, // Sampling rate is DTS/32, n=6 events + FILTER_DTS32_N8, // Sampling rate is DTS/32, n=8 events +} ChannelInputFilter_t; + #ifdef __cplusplus #include @@ -121,8 +140,8 @@ class HardwareTimer { void setCount(uint32_t val, TimerFormat_t format = TICK_FORMAT); // set timer counter to value 'val' depending on format provided uint32_t getCount(TimerFormat_t format = TICK_FORMAT); // return current counter value of timer depending on format provided - void setMode(uint32_t channel, TimerModes_t mode, PinName pin = NC); // Configure timer channel with specified mode on specified pin if available - void setMode(uint32_t channel, TimerModes_t mode, uint32_t pin); + void setMode(uint32_t channel, TimerModes_t mode, PinName pin = NC, ChannelInputFilter_t filter = FILTER_NONE); // Configure timer channel with specified mode on specified pin if available + void setMode(uint32_t channel, TimerModes_t mode, uint32_t pin, ChannelInputFilter_t filter = FILTER_NONE); TimerModes_t getMode(uint32_t channel); // Retrieve configured mode diff --git a/stm32/cores/arduino/Print.cpp b/stm32/cores/arduino/Print.cpp index 8b41f7c1..e9267405 100644 --- a/stm32/cores/arduino/Print.cpp +++ b/stm32/cores/arduino/Print.cpp @@ -129,6 +129,11 @@ size_t Print::print(unsigned long long n, int base) } } +size_t Print::print(float n, int digits) +{ + return printFloat(n, digits); +} + size_t Print::print(double n, int digits) { return printFloat(n, digits); @@ -221,6 +226,13 @@ size_t Print::println(unsigned long long num, int base) return n; } +size_t Print::println(float num, int digits) +{ + size_t n = print(num, digits); + n += println(); + return n; +} + size_t Print::println(double num, int digits) { size_t n = print(num, digits); @@ -406,7 +418,8 @@ size_t Print::printULLNumber(unsigned long long n64, uint8_t base) return bytes; } -size_t Print::printFloat(double number, uint8_t digits) +template +size_t Print::printFloat(T number, uint8_t digits) { size_t n = 0; @@ -430,7 +443,7 @@ size_t Print::printFloat(double number, uint8_t digits) } // Round correctly so that print(1.999, 2) prints as "2.00" - double rounding = 0.5; + T rounding = 0.5; for (uint8_t i = 0; i < digits; ++i) { rounding /= 10.0; } @@ -439,7 +452,7 @@ size_t Print::printFloat(double number, uint8_t digits) // Extract the integer part of the number and print it unsigned long int_part = (unsigned long)number; - double remainder = number - (double)int_part; + T remainder = number - (T)int_part; n += print(int_part); // Print the decimal point, but only if there are digits beyond diff --git a/stm32/cores/arduino/Print.h b/stm32/cores/arduino/Print.h index 036fd534..dc39d634 100644 --- a/stm32/cores/arduino/Print.h +++ b/stm32/cores/arduino/Print.h @@ -36,7 +36,8 @@ class Print { int write_error; size_t printNumber(unsigned long, uint8_t); size_t printULLNumber(unsigned long long, uint8_t); - size_t printFloat(double, uint8_t); + template + size_t printFloat(T, uint8_t); protected: void setWriteError(int err = 1) { @@ -86,6 +87,7 @@ class Print { size_t print(unsigned long, int = DEC); size_t print(long long, int = DEC); size_t print(unsigned long long, int = DEC); + size_t print(float, int = 2); size_t print(double, int = 2); size_t print(const Printable &); @@ -100,6 +102,7 @@ class Print { size_t println(unsigned long, int = DEC); size_t println(long long, int = DEC); size_t println(unsigned long long, int = DEC); + size_t println(float, int = 2); size_t println(double, int = 2); size_t println(const Printable &); size_t println(void); diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll.h index bba563cc..a9823be0 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll.h @@ -33,6 +33,7 @@ #include "stm32yyxx_ll_hrtim.h" #include "stm32yyxx_ll_hsem.h" #include "stm32yyxx_ll_i2c.h" +#include "stm32yyxx_ll_i3c.h" #include "stm32yyxx_ll_icache.h" #include "stm32yyxx_ll_ipcc.h" #include "stm32yyxx_ll_iwdg.h" diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_adc.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_adc.h index 42412d18..657b7917 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_adc.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_adc.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_adc.h" #elif STM32G4xx #include "stm32g4xx_ll_adc.h" +#elif STM32H5xx + #include "stm32h5xx_ll_adc.h" #elif STM32H7xx #include "stm32h7xx_ll_adc.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_bus.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_bus.h index 5c844851..d145450c 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_bus.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_bus.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_bus.h" #elif STM32G4xx #include "stm32g4xx_ll_bus.h" +#elif STM32H5xx + #include "stm32h5xx_ll_bus.h" #elif STM32H7xx #include "stm32h7xx_ll_bus.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_comp.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_comp.h index 207d2b49..9b2bec9b 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_comp.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_comp.h @@ -16,6 +16,8 @@ #include "stm32g0xx_ll_comp.h" #elif STM32G4xx #include "stm32g4xx_ll_comp.h" +#elif STM32H5xx + #include "stm32h5xx_ll_comp.h" #elif STM32H7xx #include "stm32h7xx_ll_comp.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_cordic.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_cordic.h index d70b4bf1..acce2257 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_cordic.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_cordic.h @@ -10,6 +10,8 @@ #ifdef STM32G4xx #include "stm32g4xx_ll_cordic.h" +#elif STM32H5xx + #include "stm32h5xx_ll_cordic.h" #elif STM32H7xx #include "stm32h7xx_ll_cordic.h" #elif STM32U5xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_cortex.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_cortex.h index 20aa4094..c7e7c490 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_cortex.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_cortex.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_cortex.h" #elif STM32G4xx #include "stm32g4xx_ll_cortex.h" +#elif STM32H5xx + #include "stm32h5xx_ll_cortex.h" #elif STM32H7xx #include "stm32h7xx_ll_cortex.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_crc.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_crc.h index fd557b95..2282b63e 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_crc.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_crc.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_crc.h" #elif STM32G4xx #include "stm32g4xx_ll_crc.h" +#elif STM32H5xx + #include "stm32h5xx_ll_crc.h" #elif STM32H7xx #include "stm32h7xx_ll_crc.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_crs.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_crs.h index 36f67b4b..8bd8236b 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_crs.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_crs.h @@ -14,6 +14,8 @@ #include "stm32g0xx_ll_crs.h" #elif STM32G4xx #include "stm32g4xx_ll_crs.h" +#elif STM32H5xx + #include "stm32h5xx_ll_crs.h" #elif STM32H7xx #include "stm32h7xx_ll_crs.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dac.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dac.h index faf76d4e..e121981c 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dac.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dac.h @@ -24,6 +24,8 @@ #include "stm32g0xx_ll_dac.h" #elif STM32G4xx #include "stm32g4xx_ll_dac.h" +#elif STM32H5xx + #include "stm32h5xx_ll_dac.h" #elif STM32H7xx #include "stm32h7xx_ll_dac.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dcache.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dcache.h index b56068f8..eacbe27c 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dcache.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dcache.h @@ -8,7 +8,9 @@ #pragma GCC diagnostic ignored "-Wregister" #endif -#ifdef STM32U5xx +#ifdef STM32H5xx + #include "stm32h5xx_ll_dcache.h" +#elif STM32U5xx #include "stm32u5xx_ll_dcache.h" #endif #pragma GCC diagnostic pop diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dlyb.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dlyb.h index 9e5c1b7b..f1ee0973 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dlyb.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dlyb.h @@ -8,7 +8,9 @@ #pragma GCC diagnostic ignored "-Wregister" #endif -#ifdef STM32U5xx +#ifdef STM32H5xx + #include "stm32h5xx_ll_dlyb.h" +#elif STM32U5xx #include "stm32u5xx_ll_dlyb.h" #endif #pragma GCC diagnostic pop diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dma.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dma.h index 5e73d672..c58eb5fc 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dma.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_dma.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_dma.h" #elif STM32G4xx #include "stm32g4xx_ll_dma.h" +#elif STM32H5xx + #include "stm32h5xx_ll_dma.h" #elif STM32H7xx #include "stm32h7xx_ll_dma.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_exti.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_exti.h index 4a6a1869..e25bc6d4 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_exti.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_exti.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_exti.h" #elif STM32G4xx #include "stm32g4xx_ll_exti.h" +#elif STM32H5xx + #include "stm32h5xx_ll_exti.h" #elif STM32H7xx #include "stm32h7xx_ll_exti.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_fmac.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_fmac.h index f4778bca..7d00f1e1 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_fmac.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_fmac.h @@ -10,6 +10,8 @@ #ifdef STM32G4xx #include "stm32g4xx_ll_fmac.h" +#elif STM32H5xx + #include "stm32h5xx_ll_fmac.h" #elif STM32H7xx #include "stm32h7xx_ll_fmac.h" #elif STM32U5xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_fmc.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_fmc.h index 13c8cc01..8a1604e4 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_fmc.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_fmc.h @@ -16,6 +16,8 @@ #include "stm32f7xx_ll_fmc.h" #elif STM32G4xx #include "stm32g4xx_ll_fmc.h" +#elif STM32H5xx + #include "stm32h5xx_ll_fmc.h" #elif STM32H7xx #include "stm32h7xx_ll_fmc.h" #elif STM32L4xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_gpio.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_gpio.h index 71357b80..b6ae1ea4 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_gpio.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_gpio.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_gpio.h" #elif STM32G4xx #include "stm32g4xx_ll_gpio.h" +#elif STM32H5xx + #include "stm32h5xx_ll_gpio.h" #elif STM32H7xx #include "stm32h7xx_ll_gpio.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_i2c.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_i2c.h index 487aaa6b..5f16edb5 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_i2c.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_i2c.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_i2c.h" #elif STM32G4xx #include "stm32g4xx_ll_i2c.h" +#elif STM32H5xx + #include "stm32h5xx_ll_i2c.h" #elif STM32H7xx #include "stm32h7xx_ll_i2c.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_i3c.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_i3c.h new file mode 100644 index 00000000..9af241a3 --- /dev/null +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_i3c.h @@ -0,0 +1,15 @@ +#ifndef _STM32YYXX_LL_I3C_H_ +#define _STM32YYXX_LL_I3C_H_ +/* LL raised several warnings, ignore them */ +#pragma GCC diagnostic push +#pragma GCC diagnostic ignored "-Wunused-parameter" +#pragma GCC diagnostic ignored "-Wstrict-aliasing" +#ifdef __cplusplus + #pragma GCC diagnostic ignored "-Wregister" +#endif + +#ifdef STM32H5xx + #include "stm32h5xx_ll_i3c.h" +#endif +#pragma GCC diagnostic pop +#endif /* _STM32YYXX_LL_I3C_H_ */ diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_icache.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_icache.h index 68fea8bb..3ad222b5 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_icache.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_icache.h @@ -8,7 +8,9 @@ #pragma GCC diagnostic ignored "-Wregister" #endif -#ifdef STM32L5xx +#ifdef STM32H5xx + #include "stm32h5xx_ll_icache.h" +#elif STM32L5xx #include "stm32l5xx_ll_icache.h" #elif STM32U5xx #include "stm32u5xx_ll_icache.h" diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_iwdg.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_iwdg.h index 7069194b..da8b0a3e 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_iwdg.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_iwdg.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_iwdg.h" #elif STM32G4xx #include "stm32g4xx_ll_iwdg.h" +#elif STM32H5xx + #include "stm32h5xx_ll_iwdg.h" #elif STM32H7xx #include "stm32h7xx_ll_iwdg.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_lptim.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_lptim.h index 924454f9..d9744006 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_lptim.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_lptim.h @@ -16,6 +16,8 @@ #include "stm32g0xx_ll_lptim.h" #elif STM32G4xx #include "stm32g4xx_ll_lptim.h" +#elif STM32H5xx + #include "stm32h5xx_ll_lptim.h" #elif STM32H7xx #include "stm32h7xx_ll_lptim.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_lpuart.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_lpuart.h index 1d30d1b7..f4bbea98 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_lpuart.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_lpuart.h @@ -12,6 +12,8 @@ #include "stm32g0xx_ll_lpuart.h" #elif STM32G4xx #include "stm32g4xx_ll_lpuart.h" +#elif STM32H5xx + #include "stm32h5xx_ll_lpuart.h" #elif STM32H7xx #include "stm32h7xx_ll_lpuart.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_opamp.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_opamp.h index 9f291e65..c13b074b 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_opamp.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_opamp.h @@ -12,6 +12,8 @@ #include "stm32f3xx_ll_opamp.h" #elif STM32G4xx #include "stm32g4xx_ll_opamp.h" +#elif STM32H5xx + #include "stm32h5xx_ll_opamp.h" #elif STM32H7xx #include "stm32h7xx_ll_opamp.h" #elif STM32L1xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_pka.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_pka.h index 2fb4ce9b..bdbc7918 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_pka.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_pka.h @@ -8,7 +8,9 @@ #pragma GCC diagnostic ignored "-Wregister" #endif -#ifdef STM32L4xx +#ifdef STM32H5xx + #include "stm32h5xx_ll_pka.h" +#elif STM32L4xx #include "stm32l4xx_ll_pka.h" #elif STM32L5xx #include "stm32l5xx_ll_pka.h" diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_pwr.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_pwr.h index 4194d10e..e92b3eb4 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_pwr.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_pwr.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_pwr.h" #elif STM32G4xx #include "stm32g4xx_ll_pwr.h" +#elif STM32H5xx + #include "stm32h5xx_ll_pwr.h" #elif STM32H7xx #include "stm32h7xx_ll_pwr.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_rcc.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_rcc.h index 0a06ae9d..701c161a 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_rcc.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_rcc.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_rcc.h" #elif STM32G4xx #include "stm32g4xx_ll_rcc.h" +#elif STM32H5xx + #include "stm32h5xx_ll_rcc.h" #elif STM32H7xx #include "stm32h7xx_ll_rcc.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_rng.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_rng.h index ed4cc972..cddaad98 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_rng.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_rng.h @@ -18,6 +18,8 @@ #include "stm32g0xx_ll_rng.h" #elif STM32G4xx #include "stm32g4xx_ll_rng.h" +#elif STM32H5xx + #include "stm32h5xx_ll_rng.h" #elif STM32H7xx #include "stm32h7xx_ll_rng.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_rtc.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_rtc.h index 28bba248..929998b3 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_rtc.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_rtc.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_rtc.h" #elif STM32G4xx #include "stm32g4xx_ll_rtc.h" +#elif STM32H5xx + #include "stm32h5xx_ll_rtc.h" #elif STM32H7xx #include "stm32h7xx_ll_rtc.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_sdmmc.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_sdmmc.h index c786d3b7..33109b70 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_sdmmc.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_sdmmc.h @@ -16,6 +16,8 @@ #include "stm32f4xx_ll_sdmmc.h" #elif STM32F7xx #include "stm32f7xx_ll_sdmmc.h" +#elif STM32H5xx + #include "stm32h5xx_ll_sdmmc.h" #elif STM32H7xx #include "stm32h7xx_ll_sdmmc.h" #elif STM32L1xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_spi.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_spi.h index ff287bd7..4a948f20 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_spi.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_spi.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_spi.h" #elif STM32G4xx #include "stm32g4xx_ll_spi.h" +#elif STM32H5xx + #include "stm32h5xx_ll_spi.h" #elif STM32H7xx #include "stm32h7xx_ll_spi.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_system.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_system.h index d7b78b14..29fc4cab 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_system.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_system.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_system.h" #elif STM32G4xx #include "stm32g4xx_ll_system.h" +#elif STM32H5xx + #include "stm32h5xx_ll_system.h" #elif STM32H7xx #include "stm32h7xx_ll_system.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_tim.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_tim.h index 68e4833a..a8164c05 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_tim.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_tim.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_tim.h" #elif STM32G4xx #include "stm32g4xx_ll_tim.h" +#elif STM32H5xx + #include "stm32h5xx_ll_tim.h" #elif STM32H7xx #include "stm32h7xx_ll_tim.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_ucpd.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_ucpd.h index 8a384150..82657a88 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_ucpd.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_ucpd.h @@ -12,6 +12,8 @@ #include "stm32g0xx_ll_ucpd.h" #elif STM32G4xx #include "stm32g4xx_ll_ucpd.h" +#elif STM32H5xx + #include "stm32h5xx_ll_ucpd.h" #elif STM32L5xx #include "stm32l5xx_ll_ucpd.h" #elif STM32U5xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_usart.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_usart.h index 97b9e293..78bce432 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_usart.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_usart.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_usart.h" #elif STM32G4xx #include "stm32g4xx_ll_usart.h" +#elif STM32H5xx + #include "stm32h5xx_ll_usart.h" #elif STM32H7xx #include "stm32h7xx_ll_usart.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_usb.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_usb.h index 538e51a4..43f41d37 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_usb.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_usb.h @@ -24,6 +24,8 @@ #include "stm32g0xx_ll_usb.h" #elif STM32G4xx #include "stm32g4xx_ll_usb.h" +#elif STM32H5xx + #include "stm32h5xx_ll_usb.h" #elif STM32H7xx #include "stm32h7xx_ll_usb.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_utils.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_utils.h index 3baed94e..a9b43262 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_utils.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_utils.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_utils.h" #elif STM32G4xx #include "stm32g4xx_ll_utils.h" +#elif STM32H5xx + #include "stm32h5xx_ll_utils.h" #elif STM32H7xx #include "stm32h7xx_ll_utils.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_wwdg.h b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_wwdg.h index 053323dd..1244c703 100644 --- a/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_wwdg.h +++ b/stm32/cores/arduino/stm32/LL/stm32yyxx_ll_wwdg.h @@ -26,6 +26,8 @@ #include "stm32g0xx_ll_wwdg.h" #elif STM32G4xx #include "stm32g4xx_ll_wwdg.h" +#elif STM32H5xx + #include "stm32h5xx_ll_wwdg.h" #elif STM32H7xx #include "stm32h7xx_ll_wwdg.h" #elif STM32L0xx diff --git a/stm32/cores/arduino/stm32/OpenAMP/openamp.c b/stm32/cores/arduino/stm32/OpenAMP/openamp.c index 351026d7..8055a552 100644 --- a/stm32/cores/arduino/stm32/OpenAMP/openamp.c +++ b/stm32/cores/arduino/stm32/OpenAMP/openamp.c @@ -119,7 +119,7 @@ int OPENAMP_Init() MAILBOX_Init(); - /* Libmetal Initilalization */ + /* Libmetal Initialization */ status = OPENAMP_shmem_init(RPMSG_REMOTE); if (status) { return status; diff --git a/stm32/cores/arduino/stm32/PeripheralPins.h b/stm32/cores/arduino/stm32/PeripheralPins.h index 9e9057f1..c1b56d83 100644 --- a/stm32/cores/arduino/stm32/PeripheralPins.h +++ b/stm32/cores/arduino/stm32/PeripheralPins.h @@ -46,6 +46,10 @@ extern const PinMap PinMap_DAC[]; extern const PinMap PinMap_I2C_SDA[]; extern const PinMap PinMap_I2C_SCL[]; +//*** I3C *** +extern const PinMap PinMap_I3C_SDA[]; +extern const PinMap PinMap_I3C_SCL[]; + //*** TIM *** /* For backward compatibility */ #define PinMap_PWM PinMap_TIM @@ -96,6 +100,20 @@ extern const PinMap PinMap_USB_OTG_FS[]; extern const PinMap PinMap_USB_OTG_HS[]; //*** SD *** -extern const PinMap PinMap_SD[]; +extern const PinMap PinMap_SD_CMD[]; +extern const PinMap PinMap_SD_CK[]; +extern const PinMap PinMap_SD_DATA0[]; +extern const PinMap PinMap_SD_DATA1[]; +extern const PinMap PinMap_SD_DATA2[]; +extern const PinMap PinMap_SD_DATA3[]; +extern const PinMap PinMap_SD_DATA4[]; +extern const PinMap PinMap_SD_DATA5[]; +extern const PinMap PinMap_SD_DATA6[]; +extern const PinMap PinMap_SD_DATA7[]; +extern const PinMap PinMap_SD_CKIN[]; +extern const PinMap PinMap_SD_CDIR[]; +extern const PinMap PinMap_SD_D0DIR[]; +extern const PinMap PinMap_SD_D123DIR[]; + #endif diff --git a/stm32/cores/arduino/stm32/backup.h b/stm32/cores/arduino/stm32/backup.h index e17c98df..9965f68c 100644 --- a/stm32/cores/arduino/stm32/backup.h +++ b/stm32/cores/arduino/stm32/backup.h @@ -86,6 +86,10 @@ static inline void enableBackupDomain(void) /* Enable BKPSRAM CLK for backup SRAM */ __HAL_RCC_BKPSRAM_CLK_ENABLE(); #endif +#if defined(TAMP_BKP0R) && defined(__HAL_RCC_RTCAPB_CLK_ENABLE) + /* Enable RTC CLK for TAMP backup registers */ + __HAL_RCC_RTCAPB_CLK_ENABLE(); +#endif } static inline void disableBackupDomain(void) @@ -95,13 +99,17 @@ static inline void disableBackupDomain(void) HAL_PWR_DisableBkUpAccess(); #endif #ifdef __HAL_RCC_BKPSRAM_CLK_DISABLE - /* Disnable BKPSRAM CLK for backup SRAM */ + /* Disable BKPSRAM CLK for backup SRAM */ __HAL_RCC_BKPSRAM_CLK_DISABLE(); #endif #ifdef __HAL_RCC_BKP_CLK_DISABLE /* Disable BKP CLK for backup registers */ __HAL_RCC_BKP_CLK_DISABLE(); #endif +#if defined(TAMP_BKP0R) && defined(__HAL_RCC_RTCAPB_CLK_DISABLE) + /* Disable RTC CLK for TAMP backup registers */ + __HAL_RCC_RTCAPB_CLK_DISABLE(); +#endif } static inline void setBackupRegister(uint32_t index, uint32_t value) @@ -111,8 +119,8 @@ static inline void setBackupRegister(uint32_t index, uint32_t value) #elif defined(RTC_BKP0R) LL_RTC_BAK_SetRegister(RTC, index, value); #elif defined(TAMP_BKP0R) -#if defined(STM32G4xx) || defined(STM32L5xx) || defined(STM32U5xx) ||\ - defined(STM32MP1xx) || defined(STM32WLxx) +#if defined(STM32G4xx) || defined(STM32H5xx) || defined(STM32L5xx) ||\ + defined(STM32U5xx) || defined(STM32MP1xx) || defined(STM32WLxx) /* For those series this API requires RTC even if it is not used and TAMP is used instead */ LL_RTC_BKP_SetRegister(RTC, index, value); @@ -134,8 +142,8 @@ static inline uint32_t getBackupRegister(uint32_t index) #elif defined(RTC_BKP0R) return LL_RTC_BAK_GetRegister(RTC, index); #elif defined(TAMP_BKP0R) -#if defined(STM32G4xx) || defined(STM32L5xx) || defined(STM32U5xx) ||\ - defined(STM32MP1xx) || defined(STM32WLxx) +#if defined(STM32G4xx) || defined(STM32H5xx) || defined(STM32L5xx) ||\ + defined(STM32U5xx) || defined(STM32MP1xx) || defined(STM32WLxx) /* For those series this API requires RTC even if it is not used and TAMP is used instead */ return LL_RTC_BKP_GetRegister(RTC, index); diff --git a/stm32/cores/arduino/stm32/stm32_def.h b/stm32/cores/arduino/stm32/stm32_def.h index 910d306e..3292b0d6 100644 --- a/stm32/cores/arduino/stm32/stm32_def.h +++ b/stm32/cores/arduino/stm32/stm32_def.h @@ -6,8 +6,8 @@ * @brief STM32 core version number */ #define STM32_CORE_VERSION_MAJOR (0x02U) /*!< [31:24] major version */ -#define STM32_CORE_VERSION_MINOR (0x05U) /*!< [23:16] minor version */ -#define STM32_CORE_VERSION_PATCH (0x00U) /*!< [15:8] patch version */ +#define STM32_CORE_VERSION_MINOR (0x07U) /*!< [23:16] minor version */ +#define STM32_CORE_VERSION_PATCH (0x01U) /*!< [15:8] patch version */ /* * Extra label for development: * 0: official release @@ -40,6 +40,8 @@ #include "stm32g0xx.h" #elif defined(STM32G4xx) #include "stm32g4xx.h" +#elif defined(STM32H5xx) + #include "stm32h5xx.h" #elif defined(STM32H7xx) #include "stm32h7xx.h" #elif defined(STM32L0xx) @@ -90,10 +92,12 @@ #if !defined(USB) && defined(USB_DRD_FS) #define USB USB_DRD_FS #define PinMap_USB PinMap_USB_DRD_FS - #if defined(STM32U5xx) + #if defined(STM32H5xx) || defined(STM32U5xx) #define USB_BASE USB_DRD_BASE - #define __HAL_RCC_USB_CLK_ENABLE __HAL_RCC_USB_FS_CLK_ENABLE - #define __HAL_RCC_USB_CLK_DISABLE __HAL_RCC_USB_FS_CLK_DISABLE + #if !defined(__HAL_RCC_USB_CLK_ENABLE) + #define __HAL_RCC_USB_CLK_ENABLE __HAL_RCC_USB_FS_CLK_ENABLE + #define __HAL_RCC_USB_CLK_DISABLE __HAL_RCC_USB_FS_CLK_DISABLE + #endif #endif #endif diff --git a/stm32/cores/arduino/stm32/stm32_def_build.h b/stm32/cores/arduino/stm32/stm32_def_build.h index acbd6d57..1f666040 100644 --- a/stm32/cores/arduino/stm32/stm32_def_build.h +++ b/stm32/cores/arduino/stm32/stm32_def_build.h @@ -220,6 +220,14 @@ #define CMSIS_STARTUP_FILE "startup_stm32g4a1xx.s" #elif defined(STM32GBK1CB) #define CMSIS_STARTUP_FILE "startup_stm32gbk1cb.s" + #elif defined(STM32H503xx) + #define CMSIS_STARTUP_FILE "startup_stm32h503xx.s" + #elif defined(STM32H562xx) + #define CMSIS_STARTUP_FILE "startup_stm32h562xx.s" + #elif defined(STM32H563xx) + #define CMSIS_STARTUP_FILE "startup_stm32h563xx.s" + #elif defined(STM32H573xx) + #define CMSIS_STARTUP_FILE "startup_stm32h573xx.s" #elif defined(STM32H723xx) #define CMSIS_STARTUP_FILE "startup_stm32h723xx.s" #elif defined(STM32H725xx) @@ -430,6 +438,14 @@ #define CMSIS_STARTUP_FILE "startup_stm32u5a5xx.s" #elif defined(STM32U5A9xx) #define CMSIS_STARTUP_FILE "startup_stm32u5a9xx.s" + #elif defined(STM32U5F7xx) + #define CMSIS_STARTUP_FILE "startup_stm32u5f7xx.s" + #elif defined(STM32U5F9xx) + #define CMSIS_STARTUP_FILE "startup_stm32u5f9xx.s" + #elif defined(STM32U5G7xx) + #define CMSIS_STARTUP_FILE "startup_stm32u5g7xx.s" + #elif defined(STM32U5G9xx) + #define CMSIS_STARTUP_FILE "startup_stm32u5g9xx.s" #elif defined(STM32WB10xx) #define CMSIS_STARTUP_FILE "startup_stm32wb10xx_cm4.s" #elif defined(STM32WB15xx) diff --git a/stm32/cores/arduino/stm32/timer.h b/stm32/cores/arduino/stm32/timer.h index 8af27406..3210f591 100644 --- a/stm32/cores/arduino/stm32/timer.h +++ b/stm32/cores/arduino/stm32/timer.h @@ -59,8 +59,9 @@ extern "C" { #define TIM1_IRQn TIM1_UP_TIM10_IRQn #define TIM1_IRQHandler TIM1_UP_TIM10_IRQHandler #endif -#elif defined(STM32H7xx) || defined(STM32L5xx) || defined(STM32MP1xx) ||\ - defined(STM32U5xx) || defined(STM32WBxx) || defined(STM32WLxx) +#elif defined(STM32H5xx) || defined(STM32H7xx) || defined(STM32L5xx) ||\ + defined(STM32MP1xx) || defined(STM32U5xx) || defined(STM32WBxx) ||\ + defined(STM32WLxx) #define TIM1_IRQn TIM1_UP_IRQn #define TIM1_IRQHandler TIM1_UP_IRQHandler #endif @@ -84,8 +85,8 @@ extern "C" { #if defined(STM32G0xx) #define TIM6_IRQn TIM6_DAC_LPTIM1_IRQn #define TIM6_IRQHandler TIM6_DAC_LPTIM1_IRQHandler -#elif !defined(STM32F1xx) && !defined(STM32L1xx) && !defined(STM32L5xx) &&\ - !defined(STM32MP1xx) && !defined(STM32U5xx) +#elif !defined(STM32F1xx) && !defined(STM32H5xx) && !defined(STM32L1xx) &&\ + !defined(STM32L5xx) && !defined(STM32MP1xx) && !defined(STM32U5xx) #define TIM6_IRQn TIM6_DAC_IRQn #define TIM6_IRQHandler TIM6_DAC_IRQHandler #endif @@ -107,8 +108,9 @@ extern "C" { || defined(STM32H7xx) #define TIM8_IRQn TIM8_UP_TIM13_IRQn #define TIM8_IRQHandler TIM8_UP_TIM13_IRQHandler -#elif defined(STM32F3xx) || defined(STM32G4xx) || defined(STM32L4xx) ||\ - defined(STM32L5xx) || defined(STM32MP1xx) || defined(STM32U5xx) +#elif defined(STM32F3xx) || defined(STM32G4xx) || defined(STM32H5xx) ||\ + defined(STM32L4xx) || defined(STM32L5xx) || defined(STM32MP1xx) ||\ + defined(STM32U5xx) #define TIM8_IRQn TIM8_UP_IRQn #define TIM8_IRQHandler TIM8_UP_IRQHandler #endif diff --git a/stm32/cores/arduino/stm32/uart.h b/stm32/cores/arduino/stm32/uart.h index 9d2da475..2a8537f4 100644 --- a/stm32/cores/arduino/stm32/uart.h +++ b/stm32/cores/arduino/stm32/uart.h @@ -86,7 +86,9 @@ struct serial_s { }; /* Exported constants --------------------------------------------------------*/ +#ifndef TX_TIMEOUT #define TX_TIMEOUT 1000 +#endif #if !defined(RCC_USART1CLKSOURCE_HSI) /* Some series like C0 have 2 derivated clock from HSI: HSIKER (for peripherals) diff --git a/stm32/cores/arduino/stm32/usb/cdc/cdc_queue.h b/stm32/cores/arduino/stm32/usb/cdc/cdc_queue.h index cd10ac2b..1ec58d02 100644 --- a/stm32/cores/arduino/stm32/usb/cdc/cdc_queue.h +++ b/stm32/cores/arduino/stm32/usb/cdc/cdc_queue.h @@ -53,8 +53,14 @@ extern "C" { #else #define CDC_QUEUE_MAX_PACKET_SIZE USB_FS_MAX_PACKET_SIZE #endif -#define CDC_TRANSMIT_QUEUE_BUFFER_SIZE ((uint16_t)(CDC_QUEUE_MAX_PACKET_SIZE * 2)) -#define CDC_RECEIVE_QUEUE_BUFFER_SIZE ((uint16_t)(CDC_QUEUE_MAX_PACKET_SIZE * 3)) +#ifndef CDC_TRANSMIT_QUEUE_BUFFER_PACKET_NUMBER +#define CDC_TRANSMIT_QUEUE_BUFFER_PACKET_NUMBER 2 +#endif +#ifndef CDC_RECEIVE_QUEUE_BUFFER_PACKET_NUMBER +#define CDC_RECEIVE_QUEUE_BUFFER_PACKET_NUMBER 3 +#endif +#define CDC_TRANSMIT_QUEUE_BUFFER_SIZE ((uint16_t)(CDC_QUEUE_MAX_PACKET_SIZE * CDC_TRANSMIT_QUEUE_BUFFER_PACKET_NUMBER)) +#define CDC_RECEIVE_QUEUE_BUFFER_SIZE ((uint16_t)(CDC_QUEUE_MAX_PACKET_SIZE * CDC_RECEIVE_QUEUE_BUFFER_PACKET_NUMBER)) typedef struct { uint8_t buffer[CDC_TRANSMIT_QUEUE_BUFFER_SIZE]; @@ -91,4 +97,4 @@ void CDC_ReceiveQueue_CommitBlock(CDC_ReceiveQueue_TypeDef *queue, uint16_t size } #endif -#endif // __CDC_QUEUE_H \ No newline at end of file +#endif // __CDC_QUEUE_H diff --git a/stm32/cores/arduino/stm32/usb/cdc/usbd_cdc.c b/stm32/cores/arduino/stm32/usb/cdc/usbd_cdc.c index 027189cd..e1c9a508 100644 --- a/stm32/cores/arduino/stm32/usb/cdc/usbd_cdc.c +++ b/stm32/cores/arduino/stm32/usb/cdc/usbd_cdc.c @@ -10,6 +10,17 @@ * - Command IN transfer (class requests management) * - Error management * + ****************************************************************************** + * @attention + * + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** * @verbatim * * =================================================================== @@ -37,17 +48,6 @@ * @endverbatim * ****************************************************************************** - * @attention - * - *

© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.

- * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** */ #ifdef USBCON @@ -103,13 +103,14 @@ static uint8_t USBD_CDC_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *re static uint8_t USBD_CDC_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum); static uint8_t USBD_CDC_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum); static uint8_t USBD_CDC_EP0_RxReady(USBD_HandleTypeDef *pdev); - -static uint8_t *USBD_CDC_GetFSCfgDesc(uint16_t *length); -static uint8_t *USBD_CDC_GetHSCfgDesc(uint16_t *length); -static uint8_t *USBD_CDC_GetOtherSpeedCfgDesc(uint16_t *length); -static uint8_t *USBD_CDC_GetOtherSpeedCfgDesc(uint16_t *length); -uint8_t *USBD_CDC_GetDeviceQualifierDescriptor(uint16_t *length); - +#ifndef USE_USBD_COMPOSITE + static uint8_t *USBD_CDC_GetFSCfgDesc(uint16_t *length); + static uint8_t *USBD_CDC_GetHSCfgDesc(uint16_t *length); + static uint8_t *USBD_CDC_GetOtherSpeedCfgDesc(uint16_t *length); + uint8_t *USBD_CDC_GetDeviceQualifierDescriptor(uint16_t *length); +#endif /* USE_USBD_COMPOSITE */ + +#ifndef USE_USBD_COMPOSITE /* USB Standard Device Descriptor */ __ALIGN_BEGIN static uint8_t USBD_CDC_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_DESC] __ALIGN_END = { USB_LEN_DEV_QUALIFIER_DESC, @@ -123,7 +124,7 @@ __ALIGN_BEGIN static uint8_t USBD_CDC_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_ 0x01, 0x00, }; - +#endif /* USE_USBD_COMPOSITE */ /** * @} */ @@ -132,6 +133,8 @@ __ALIGN_BEGIN static uint8_t USBD_CDC_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_ * @{ */ +/* Prevent dynamic allocation */ +USBD_CDC_HandleTypeDef _hcdc; /* CDC interface class callbacks structure */ USBD_ClassTypeDef USBD_CDC = { @@ -145,12 +148,20 @@ USBD_ClassTypeDef USBD_CDC = { NULL, NULL, NULL, +#ifdef USE_USBD_COMPOSITE + NULL, + NULL, + NULL, + NULL, +#else USBD_CDC_GetHSCfgDesc, USBD_CDC_GetFSCfgDesc, USBD_CDC_GetOtherSpeedCfgDesc, USBD_CDC_GetDeviceQualifierDescriptor, +#endif /* USE_USBD_COMPOSITE */ }; +#ifndef USE_USBD_COMPOSITE /* USB CDC device Configuration Descriptor */ __ALIGN_BEGIN static uint8_t USBD_CDC_CfgHSDesc[USB_CDC_CONFIG_DESC_SIZ] __ALIGN_END = { /* Configuration Descriptor */ @@ -258,12 +269,13 @@ __ALIGN_BEGIN static uint8_t USBD_CDC_CfgFSDesc[USB_CDC_CONFIG_DESC_SIZ] __ALIGN 0x00, 0x02, /* bNumInterfaces: 2 interfaces */ 0x01, /* bConfigurationValue: Configuration value */ - 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ + 0x00, /* iConfiguration: Index of string descriptor + describing the configuration */ #if (USBD_SELF_POWERED == 1U) 0xC0, /* bmAttributes: Bus Powered according to user configuration */ #else 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif +#endif /* USBD_SELF_POWERED */ USBD_MAX_POWER, /* MaxPower (mA) */ /*---------------------------------------------------------------------------*/ @@ -440,6 +452,11 @@ __ALIGN_BEGIN static uint8_t USBD_CDC_OtherSpeedCfgDesc[USB_CDC_CONFIG_DESC_SIZ] 0x00, 0x00 /* bInterval */ }; +#endif /* USE_USBD_COMPOSITE */ + +static uint8_t CDCInEpAdd = CDC_IN_EP; +static uint8_t CDCOutEpAdd = CDC_OUT_EP; +static uint8_t CDCCmdEpAdd = CDC_CMD_EP; /** * @} @@ -461,65 +478,82 @@ static uint8_t USBD_CDC_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) UNUSED(cfgidx); USBD_CDC_HandleTypeDef *hcdc; - hcdc = (USBD_CDC_HandleTypeDef *)USBD_malloc(sizeof(USBD_CDC_HandleTypeDef)); + // hcdc = (USBD_CDC_HandleTypeDef *)USBD_malloc(sizeof(USBD_CDC_HandleTypeDef)); + hcdc = &_hcdc; if (hcdc == NULL) { - pdev->pClassData = NULL; + pdev->pClassDataCmsit[pdev->classId] = NULL; return (uint8_t)USBD_EMEM; } - pdev->pClassData = (void *)hcdc; + (void)USBD_memset(hcdc, 0, sizeof(USBD_CDC_HandleTypeDef)); + + pdev->pClassDataCmsit[pdev->classId] = (void *)hcdc; + pdev->pClassData = pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + CDCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + CDCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + CDCCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (pdev->dev_speed == USBD_SPEED_HIGH) { /* Open EP IN */ - (void)USBD_LL_OpenEP(pdev, CDC_IN_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, CDCInEpAdd, USBD_EP_TYPE_BULK, CDC_DATA_HS_IN_PACKET_SIZE); - pdev->ep_in[CDC_IN_EP & 0xFU].is_used = 1U; + pdev->ep_in[CDCInEpAdd & 0xFU].is_used = 1U; /* Open EP OUT */ - (void)USBD_LL_OpenEP(pdev, CDC_OUT_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, CDCOutEpAdd, USBD_EP_TYPE_BULK, CDC_DATA_HS_OUT_PACKET_SIZE); - pdev->ep_out[CDC_OUT_EP & 0xFU].is_used = 1U; + pdev->ep_out[CDCOutEpAdd & 0xFU].is_used = 1U; /* Set bInterval for CDC CMD Endpoint */ - pdev->ep_in[CDC_CMD_EP & 0xFU].bInterval = CDC_HS_BINTERVAL; + pdev->ep_in[CDCCmdEpAdd & 0xFU].bInterval = CDC_HS_BINTERVAL; } else { /* Open EP IN */ - (void)USBD_LL_OpenEP(pdev, CDC_IN_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, CDCInEpAdd, USBD_EP_TYPE_BULK, CDC_DATA_FS_IN_PACKET_SIZE); - pdev->ep_in[CDC_IN_EP & 0xFU].is_used = 1U; + pdev->ep_in[CDCInEpAdd & 0xFU].is_used = 1U; /* Open EP OUT */ - (void)USBD_LL_OpenEP(pdev, CDC_OUT_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, CDCOutEpAdd, USBD_EP_TYPE_BULK, CDC_DATA_FS_OUT_PACKET_SIZE); - pdev->ep_out[CDC_OUT_EP & 0xFU].is_used = 1U; + pdev->ep_out[CDCOutEpAdd & 0xFU].is_used = 1U; /* Set bInterval for CMD Endpoint */ - pdev->ep_in[CDC_CMD_EP & 0xFU].bInterval = CDC_FS_BINTERVAL; + pdev->ep_in[CDCCmdEpAdd & 0xFU].bInterval = CDC_FS_BINTERVAL; } /* Open Command IN EP */ - (void)USBD_LL_OpenEP(pdev, CDC_CMD_EP, USBD_EP_TYPE_INTR, CDC_CMD_PACKET_SIZE); - pdev->ep_in[CDC_CMD_EP & 0xFU].is_used = 1U; + (void)USBD_LL_OpenEP(pdev, CDCCmdEpAdd, USBD_EP_TYPE_INTR, CDC_CMD_PACKET_SIZE); + pdev->ep_in[CDCCmdEpAdd & 0xFU].is_used = 1U; + + hcdc->RxBuffer = NULL; /* Init physical Interface components */ - ((USBD_CDC_ItfTypeDef *)pdev->pUserData)->Init(); + ((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->Init(); /* Init Xfer states */ hcdc->TxState = 0U; hcdc->RxState = 0U; + if (hcdc->RxBuffer == NULL) { + return (uint8_t)USBD_EMEM; + } + if (pdev->dev_speed == USBD_SPEED_HIGH) { /* Prepare Out endpoint to receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, CDC_OUT_EP, hcdc->RxBuffer, + (void)USBD_LL_PrepareReceive(pdev, CDCOutEpAdd, hcdc->RxBuffer, CDC_DATA_HS_OUT_PACKET_SIZE); } else { /* Prepare Out endpoint to receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, CDC_OUT_EP, hcdc->RxBuffer, + (void)USBD_LL_PrepareReceive(pdev, CDCOutEpAdd, hcdc->RxBuffer, CDC_DATA_FS_OUT_PACKET_SIZE); } @@ -537,23 +571,33 @@ static uint8_t USBD_CDC_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) { UNUSED(cfgidx); + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this CDC class instance */ + CDCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + CDCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + CDCCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + /* Close EP IN */ - (void)USBD_LL_CloseEP(pdev, CDC_IN_EP); - pdev->ep_in[CDC_IN_EP & 0xFU].is_used = 0U; + (void)USBD_LL_CloseEP(pdev, CDCInEpAdd); + pdev->ep_in[CDCInEpAdd & 0xFU].is_used = 0U; /* Close EP OUT */ - (void)USBD_LL_CloseEP(pdev, CDC_OUT_EP); - pdev->ep_out[CDC_OUT_EP & 0xFU].is_used = 0U; + (void)USBD_LL_CloseEP(pdev, CDCOutEpAdd); + pdev->ep_out[CDCOutEpAdd & 0xFU].is_used = 0U; /* Close Command IN EP */ - (void)USBD_LL_CloseEP(pdev, CDC_CMD_EP); - pdev->ep_in[CDC_CMD_EP & 0xFU].is_used = 0U; - pdev->ep_in[CDC_CMD_EP & 0xFU].bInterval = 0U; + (void)USBD_LL_CloseEP(pdev, CDCCmdEpAdd); + pdev->ep_in[CDCCmdEpAdd & 0xFU].is_used = 0U; + pdev->ep_in[CDCCmdEpAdd & 0xFU].bInterval = 0U; /* DeInit physical Interface components */ - if (pdev->pClassData != NULL) { - ((USBD_CDC_ItfTypeDef *)pdev->pUserData)->DeInit(); - (void)USBD_free(pdev->pClassData); + if (pdev->pClassDataCmsit[pdev->classId] != NULL) { + ((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->DeInit(); + /* No need to free as hhid is no more dynamically allocated */ + // (void)USBD_free(pdev->pClassDataCmsit[pdev->classId]); + pdev->pClassDataCmsit[pdev->classId] = NULL; pdev->pClassData = NULL; } @@ -570,7 +614,7 @@ static uint8_t USBD_CDC_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) static uint8_t USBD_CDC_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; uint16_t len; uint8_t ifalt = 0U; uint16_t status_info = 0U; @@ -584,9 +628,9 @@ static uint8_t USBD_CDC_Setup(USBD_HandleTypeDef *pdev, case USB_REQ_TYPE_CLASS: if (req->wLength != 0U) { if ((req->bmRequest & 0x80U) != 0U) { - ((USBD_CDC_ItfTypeDef *)pdev->pUserData)->Control(req->bRequest, - (uint8_t *)hcdc->data, - req->wLength); + ((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->Control(req->bRequest, + (uint8_t *)hcdc->data, + req->wLength); len = MIN(CDC_REQ_MAX_DATA_SIZE, req->wLength); (void)USBD_CtlSendData(pdev, (uint8_t *)hcdc->data, len); @@ -597,8 +641,8 @@ static uint8_t USBD_CDC_Setup(USBD_HandleTypeDef *pdev, (void)USBD_CtlPrepareRx(pdev, (uint8_t *)hcdc->data, hcdc->CmdLength); } } else { - ((USBD_CDC_ItfTypeDef *)pdev->pUserData)->Control(req->bRequest, - (uint8_t *)req, 0U); + ((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->Control(req->bRequest, + (uint8_t *)req, 0U); } break; @@ -657,25 +701,27 @@ static uint8_t USBD_CDC_Setup(USBD_HandleTypeDef *pdev, */ static uint8_t USBD_CDC_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) { - USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; - PCD_HandleTypeDef *hpcd = pdev->pData; + USBD_CDC_HandleTypeDef *hcdc; + PCD_HandleTypeDef *hpcd = (PCD_HandleTypeDef *)pdev->pData; - if (hcdc == NULL) { + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_FAIL; } - if ((pdev->ep_in[epnum].total_length > 0U) && - ((pdev->ep_in[epnum].total_length % hpcd->IN_ep[epnum].maxpacket) == 0U)) { + hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + + if ((pdev->ep_in[epnum & 0xFU].total_length > 0U) && + ((pdev->ep_in[epnum & 0xFU].total_length % hpcd->IN_ep[epnum & 0xFU].maxpacket) == 0U)) { /* Update the packet total length */ - pdev->ep_in[epnum].total_length = 0U; + pdev->ep_in[epnum & 0xFU].total_length = 0U; /* Send ZLP */ (void)USBD_LL_Transmit(pdev, epnum, NULL, 0U); } else { hcdc->TxState = 0U; - if (((USBD_CDC_ItfTypeDef *)pdev->pUserData)->TransmitCplt != NULL) { - ((USBD_CDC_ItfTypeDef *)pdev->pUserData)->TransmitCplt(hcdc->TxBuffer, &hcdc->TxLength, epnum); + if (((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->TransmitCplt != NULL) { + ((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->TransmitCplt(hcdc->TxBuffer, &hcdc->TxLength, epnum); } } @@ -691,9 +737,9 @@ static uint8_t USBD_CDC_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) */ static uint8_t USBD_CDC_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) { - USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; - if (hcdc == NULL) { + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_FAIL; } @@ -703,7 +749,7 @@ static uint8_t USBD_CDC_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) /* USB data will be immediately processed, this allow next USB traffic being NAKed till the end of the application Xfer */ - ((USBD_CDC_ItfTypeDef *)pdev->pUserData)->Receive(hcdc->RxBuffer, &hcdc->RxLength); + ((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->Receive(hcdc->RxBuffer, &hcdc->RxLength); return (uint8_t)USBD_OK; } @@ -716,31 +762,46 @@ static uint8_t USBD_CDC_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) */ static uint8_t USBD_CDC_EP0_RxReady(USBD_HandleTypeDef *pdev) { - USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hcdc == NULL) { return (uint8_t)USBD_FAIL; } - if ((pdev->pUserData != NULL) && (hcdc->CmdOpCode != 0xFFU)) { - ((USBD_CDC_ItfTypeDef *)pdev->pUserData)->Control(hcdc->CmdOpCode, - (uint8_t *)hcdc->data, - (uint16_t)hcdc->CmdLength); + if ((pdev->pUserData[pdev->classId] != NULL) && (hcdc->CmdOpCode != 0xFFU)) { + ((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->Control(hcdc->CmdOpCode, + (uint8_t *)hcdc->data, + (uint16_t)hcdc->CmdLength); hcdc->CmdOpCode = 0xFFU; } return (uint8_t)USBD_OK; } - +#ifndef USE_USBD_COMPOSITE /** * @brief USBD_CDC_GetFSCfgDesc * Return configuration descriptor - * @param speed : current device speed * @param length : pointer data length * @retval pointer to descriptor buffer */ static uint8_t *USBD_CDC_GetFSCfgDesc(uint16_t *length) { + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_CDC_CfgFSDesc, CDC_CMD_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CDC_CfgFSDesc, CDC_OUT_EP); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CDC_CfgFSDesc, CDC_IN_EP); + + if (pEpCmdDesc != NULL) { + pEpCmdDesc->bInterval = CDC_FS_BINTERVAL; + } + + if (pEpOutDesc != NULL) { + pEpOutDesc->wMaxPacketSize = CDC_DATA_FS_MAX_PACKET_SIZE; + } + + if (pEpInDesc != NULL) { + pEpInDesc->wMaxPacketSize = CDC_DATA_FS_MAX_PACKET_SIZE; + } + *length = (uint16_t)sizeof(USBD_CDC_CfgFSDesc); return USBD_CDC_CfgFSDesc; @@ -749,12 +810,27 @@ static uint8_t *USBD_CDC_GetFSCfgDesc(uint16_t *length) /** * @brief USBD_CDC_GetHSCfgDesc * Return configuration descriptor - * @param speed : current device speed * @param length : pointer data length * @retval pointer to descriptor buffer */ static uint8_t *USBD_CDC_GetHSCfgDesc(uint16_t *length) { + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_CDC_CfgHSDesc, CDC_CMD_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CDC_CfgHSDesc, CDC_OUT_EP); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CDC_CfgHSDesc, CDC_IN_EP); + + if (pEpCmdDesc != NULL) { + pEpCmdDesc->bInterval = CDC_HS_BINTERVAL; + } + + if (pEpOutDesc != NULL) { + pEpOutDesc->wMaxPacketSize = CDC_DATA_HS_MAX_PACKET_SIZE; + } + + if (pEpInDesc != NULL) { + pEpInDesc->wMaxPacketSize = CDC_DATA_HS_MAX_PACKET_SIZE; + } + *length = (uint16_t)sizeof(USBD_CDC_CfgHSDesc); return USBD_CDC_CfgHSDesc; @@ -763,12 +839,27 @@ static uint8_t *USBD_CDC_GetHSCfgDesc(uint16_t *length) /** * @brief USBD_CDC_GetOtherSpeedCfgDesc * Return configuration descriptor - * @param speed : current device speed * @param length : pointer data length * @retval pointer to descriptor buffer */ static uint8_t *USBD_CDC_GetOtherSpeedCfgDesc(uint16_t *length) { + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_CDC_OtherSpeedCfgDesc, CDC_CMD_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CDC_OtherSpeedCfgDesc, CDC_OUT_EP); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CDC_OtherSpeedCfgDesc, CDC_IN_EP); + + if (pEpCmdDesc != NULL) { + pEpCmdDesc->bInterval = CDC_FS_BINTERVAL; + } + + if (pEpOutDesc != NULL) { + pEpOutDesc->wMaxPacketSize = CDC_DATA_FS_MAX_PACKET_SIZE; + } + + if (pEpInDesc != NULL) { + pEpInDesc->wMaxPacketSize = CDC_DATA_FS_MAX_PACKET_SIZE; + } + *length = (uint16_t)sizeof(USBD_CDC_OtherSpeedCfgDesc); return USBD_CDC_OtherSpeedCfgDesc; @@ -786,7 +877,7 @@ uint8_t *USBD_CDC_GetDeviceQualifierDescriptor(uint16_t *length) return USBD_CDC_DeviceQualifierDesc; } - +#endif /* USE_USBD_COMPOSITE */ /** * @brief USBD_CDC_RegisterInterface * @param pdev: device instance @@ -800,7 +891,7 @@ uint8_t USBD_CDC_RegisterInterface(USBD_HandleTypeDef *pdev, return (uint8_t)USBD_FAIL; } - pdev->pUserData = fops; + pdev->pUserData[pdev->classId] = fops; return (uint8_t)USBD_OK; } @@ -809,12 +900,21 @@ uint8_t USBD_CDC_RegisterInterface(USBD_HandleTypeDef *pdev, * @brief USBD_CDC_SetTxBuffer * @param pdev: device instance * @param pbuff: Tx Buffer + * @param length: length of data to be sent + * @param ClassId: The Class ID * @retval status */ +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CDC_SetTxBuffer(USBD_HandleTypeDef *pdev, + uint8_t *pbuff, uint32_t length, uint8_t ClassId) +{ + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[ClassId]; +#else uint8_t USBD_CDC_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, uint32_t length) { - USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#endif /* USE_USBD_COMPOSITE */ if (hcdc == NULL) { return (uint8_t)USBD_FAIL; @@ -834,7 +934,7 @@ uint8_t USBD_CDC_SetTxBuffer(USBD_HandleTypeDef *pdev, */ uint8_t USBD_CDC_SetRxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff) { - USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hcdc == NULL) { return (uint8_t)USBD_FAIL; @@ -849,13 +949,26 @@ uint8_t USBD_CDC_SetRxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff) * @brief USBD_CDC_TransmitPacket * Transmit packet on IN endpoint * @param pdev: device instance + * @param ClassId: The Class ID * @retval status */ +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CDC_TransmitPacket(USBD_HandleTypeDef *pdev, uint8_t ClassId) +{ + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[ClassId]; +#else uint8_t USBD_CDC_TransmitPacket(USBD_HandleTypeDef *pdev) { - USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#endif /* USE_USBD_COMPOSITE */ + USBD_StatusTypeDef ret = USBD_BUSY; +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + CDCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, ClassId); +#endif /* USE_USBD_COMPOSITE */ + if (hcdc == NULL) { return (uint8_t)USBD_FAIL; } @@ -865,10 +978,10 @@ uint8_t USBD_CDC_TransmitPacket(USBD_HandleTypeDef *pdev) hcdc->TxState = 1U; /* Update the packet total length */ - pdev->ep_in[CDC_IN_EP & 0xFU].total_length = hcdc->TxLength; + pdev->ep_in[CDCInEpAdd & 0xFU].total_length = hcdc->TxLength; /* Transmit next packet */ - (void)USBD_LL_Transmit(pdev, CDC_IN_EP, hcdc->TxBuffer, hcdc->TxLength); + (void)USBD_LL_Transmit(pdev, CDCInEpAdd, hcdc->TxBuffer, hcdc->TxLength); ret = USBD_OK; } @@ -884,29 +997,40 @@ uint8_t USBD_CDC_TransmitPacket(USBD_HandleTypeDef *pdev) */ uint8_t USBD_CDC_ReceivePacket(USBD_HandleTypeDef *pdev) { - USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; - if (hcdc == NULL) { +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + CDCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_FAIL; } if (pdev->dev_speed == USBD_SPEED_HIGH) { /* Prepare Out endpoint to receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, CDC_OUT_EP, hcdc->RxBuffer, + (void)USBD_LL_PrepareReceive(pdev, CDCOutEpAdd, hcdc->RxBuffer, CDC_DATA_HS_OUT_PACKET_SIZE); } else { /* Prepare Out endpoint to receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, CDC_OUT_EP, hcdc->RxBuffer, + (void)USBD_LL_PrepareReceive(pdev, CDCOutEpAdd, hcdc->RxBuffer, CDC_DATA_FS_OUT_PACKET_SIZE); } return (uint8_t)USBD_OK; } - +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CDC_ClearBuffer(USBD_HandleTypeDef *pdev, uint8_t ClassId) +{ + /* Suspend or Resume USB Out process */ + if (pdev->pClassDataCmsit[classId] != NULL) { +#else uint8_t USBD_CDC_ClearBuffer(USBD_HandleTypeDef *pdev) { /* Suspend or Resume USB Out process */ - if (pdev->pClassData != NULL) { + if (pdev->pClassDataCmsit[pdev->classId] != NULL) { +#endif /* USE_USBD_COMPOSITE */ /* Prepare Out endpoint to receive next packet */ USBD_LL_PrepareReceive(pdev, CDC_OUT_EP, 0, 0); return (uint8_t)USBD_OK; diff --git a/stm32/cores/arduino/stm32/usb/cdc/usbd_cdc.h b/stm32/cores/arduino/stm32/usb/cdc/usbd_cdc.h index 4e397807..3c32c7c9 100644 --- a/stm32/cores/arduino/stm32/usb/cdc/usbd_cdc.h +++ b/stm32/cores/arduino/stm32/usb/cdc/usbd_cdc.h @@ -144,13 +144,19 @@ extern USBD_ClassTypeDef USBD_CDC; uint8_t USBD_CDC_RegisterInterface(USBD_HandleTypeDef *pdev, USBD_CDC_ItfTypeDef *fops); +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CDC_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, + uint32_t length, uint8_t ClassId); +uint8_t USBD_CDC_TransmitPacket(USBD_HandleTypeDef *pdev, uint8_t ClassId); +uint8_t USBD_CDC_ClearBuffer(USBD_HandleTypeDef *pdev, uint8_t ClassId); +#else uint8_t USBD_CDC_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, uint32_t length); - +uint8_t USBD_CDC_TransmitPacket(USBD_HandleTypeDef *pdev); +uint8_t USBD_CDC_ClearBuffer(USBD_HandleTypeDef *pdev); +#endif /* USE_USBD_COMPOSITE */ uint8_t USBD_CDC_SetRxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff); uint8_t USBD_CDC_ReceivePacket(USBD_HandleTypeDef *pdev); -uint8_t USBD_CDC_ClearBuffer(USBD_HandleTypeDef *pdev); -uint8_t USBD_CDC_TransmitPacket(USBD_HandleTypeDef *pdev); /** * @} */ diff --git a/stm32/cores/arduino/stm32/usb/cdc/usbd_cdc_if.c b/stm32/cores/arduino/stm32/usb/cdc/usbd_cdc_if.c index e19a26e0..326ef321 100644 --- a/stm32/cores/arduino/stm32/usb/cdc/usbd_cdc_if.c +++ b/stm32/cores/arduino/stm32/usb/cdc/usbd_cdc_if.c @@ -48,6 +48,9 @@ USBD_HandleTypeDef hUSBD_Device_CDC; static bool CDC_initialized = false; static bool CDC_DTR_enabled = true; +#if defined(ICACHE) && defined (HAL_ICACHE_MODULE_ENABLED) && !defined(HAL_ICACHE_MODULE_DISABLED) + static bool icache_enabled = false; +#endif /* ICACHE && HAL_ICACHE_MODULE_ENABLED && !HAL_ICACHE_MODULE_DISABLED */ /* Received Data over USB are stored in this buffer */ CDC_TransmitQueue_TypeDef TransmitQueue; @@ -270,6 +273,15 @@ static int8_t USBD_CDC_TransmitCplt(uint8_t *Buf, uint32_t *Len, uint8_t epnum) void CDC_init(void) { +#if defined(ICACHE) && defined (HAL_ICACHE_MODULE_ENABLED) && !defined(HAL_ICACHE_MODULE_DISABLED) + if (HAL_ICACHE_IsEnabled() == 1) { + icache_enabled = true; + /* Disable instruction cache prior to internal cacheable memory update */ + if (HAL_ICACHE_Disable() != HAL_OK) { + Error_Handler(); + } + } +#endif /* ICACHE && HAL_ICACHE_MODULE_ENABLED && !HAL_ICACHE_MODULE_DISABLED */ if (!CDC_initialized) { /* Init Device Library */ if (USBD_Init(&hUSBD_Device_CDC, &USBD_Desc, 0) == USBD_OK) { @@ -294,6 +306,14 @@ void CDC_deInit(void) USBD_DeInit(&hUSBD_Device_CDC); CDC_initialized = false; } +#if defined(ICACHE) && defined (HAL_ICACHE_MODULE_ENABLED) && !defined(HAL_ICACHE_MODULE_DISABLED) + if (icache_enabled) { + /* Re-enable instruction cache */ + if (HAL_ICACHE_Enable() != HAL_OK) { + Error_Handler(); + } + } +#endif /* ICACHE && HAL_ICACHE_MODULE_ENABLED && !HAL_ICACHE_MODULE_DISABLED */ } bool CDC_connected() diff --git a/stm32/cores/arduino/stm32/usb/hid/usbd_hid_composite.c b/stm32/cores/arduino/stm32/usb/hid/usbd_hid_composite.c index 51f3369e..75fee40e 100644 --- a/stm32/cores/arduino/stm32/usb/hid/usbd_hid_composite.c +++ b/stm32/cores/arduino/stm32/usb/hid/usbd_hid_composite.c @@ -90,13 +90,13 @@ static uint8_t USBD_HID_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx); static uint8_t USBD_COMPOSITE_HID_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); static uint8_t USBD_HID_MOUSE_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); static uint8_t USBD_HID_KEYBOARD_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); - -static uint8_t *USBD_HID_GetFSCfgDesc(uint16_t *length); -static uint8_t *USBD_HID_GetHSCfgDesc(uint16_t *length); -static uint8_t *USBD_HID_GetOtherSpeedCfgDesc(uint16_t *length); -static uint8_t *USBD_HID_GetDeviceQualifierDesc(uint16_t *length); static uint8_t USBD_HID_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum); - +#ifndef USE_USBD_COMPOSITE + static uint8_t *USBD_HID_GetFSCfgDesc(uint16_t *length); + static uint8_t *USBD_HID_GetHSCfgDesc(uint16_t *length); + static uint8_t *USBD_HID_GetOtherSpeedCfgDesc(uint16_t *length); + static uint8_t *USBD_HID_GetDeviceQualifierDesc(uint16_t *length); +#endif /* USE_USBD_COMPOSITE */ /** * @} */ @@ -105,6 +105,9 @@ static uint8_t USBD_HID_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum); * @{ */ +/* Prevent dynamic allocation */ +USBD_HID_HandleTypeDef _hhid; + USBD_ClassTypeDef USBD_COMPOSITE_HID = { USBD_HID_Init, USBD_HID_DeInit, @@ -116,14 +119,22 @@ USBD_ClassTypeDef USBD_COMPOSITE_HID = { NULL, /* SOF */ NULL, NULL, +#ifdef USE_USBD_COMPOSITE + NULL, + NULL, + NULL, + NULL, +#else USBD_HID_GetHSCfgDesc, USBD_HID_GetFSCfgDesc, USBD_HID_GetOtherSpeedCfgDesc, USBD_HID_GetDeviceQualifierDesc, +#endif /* USE_USBD_COMPOSITE */ }; +#ifndef USE_USBD_COMPOSITE /* USB HID device FS Configuration Descriptor */ -__ALIGN_BEGIN static uint8_t USBD_HID_CfgFSDesc[USB_COMPOSITE_HID_CONFIG_DESC_SIZ] __ALIGN_END = { +__ALIGN_BEGIN static uint8_t USBD_HID_CfgFSDesc[USB_COMPOSITE_HID_CONFIG_DESC_SIZ] __ALIGN_END = { 0x09, /* bLength: Configuration Descriptor size */ USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ LOBYTE(USB_COMPOSITE_HID_CONFIG_DESC_SIZ), /* wTotalLength: Bytes returned */ @@ -135,7 +146,7 @@ __ALIGN_BEGIN static uint8_t USBD_HID_CfgFSDesc[USB_COMPOSITE_HID_CONFIG_DESC_SI 0xC0, /* bmAttributes: Bus Powered according to user configuration */ #else 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif +#endif /* USBD_SELF_POWERED */ USBD_MAX_POWER, /* MaxPower (mA) */ /************** Descriptor of Joystick Mouse interface ****************/ @@ -219,7 +230,7 @@ __ALIGN_BEGIN static uint8_t USBD_HID_CfgHSDesc[USB_COMPOSITE_HID_CONFIG_DESC_SI 0xC0, /* bmAttributes: Bus Powered according to user configuration */ #else 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif +#endif /* USBD_SELF_POWERED */ USBD_MAX_POWER, /* MaxPower (mA) */ /************** Descriptor of Joystick Mouse interface ****************/ @@ -375,9 +386,10 @@ __ALIGN_BEGIN static uint8_t USBD_HID_OtherSpeedCfgDesc[USB_COMPOSITE_HID_CONFIG HID_FS_BINTERVAL /* bInterval: Polling Interval */ /* 59 */ } ; +#endif /* USE_USBD_COMPOSITE */ /* USB HID device Configuration Descriptor */ -__ALIGN_BEGIN static uint8_t USBD_MOUSE_HID_Desc[USB_HID_DESC_SIZ] __ALIGN_END = { +__ALIGN_BEGIN static uint8_t USBD_MOUSE_HID_Desc[USB_HID_DESC_SIZ] __ALIGN_END = { 0x09, /* bLength: HID Descriptor size */ HID_DESCRIPTOR_TYPE, /* bDescriptorType: HID */ 0x11, /* bcdHID: HID Class Spec release number */ @@ -402,6 +414,7 @@ __ALIGN_BEGIN static uint8_t USBD_KEYBOARD_HID_Desc[USB_HID_DESC_SIZ] __ALIGN_E 0x00, }; +#ifndef USE_USBD_COMPOSITE /* USB Standard Device Descriptor */ __ALIGN_BEGIN static uint8_t USBD_HID_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_DESC] __ALIGN_END = { USB_LEN_DEV_QUALIFIER_DESC, @@ -415,6 +428,7 @@ __ALIGN_BEGIN static uint8_t USBD_HID_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_ 0x01, 0x00, }; +#endif /* USE_USBD_COMPOSITE */ __ALIGN_BEGIN static uint8_t HID_MOUSE_ReportDesc[HID_MOUSE_REPORT_DESC_SIZE] __ALIGN_END = { 0x05, 0x01, /* Usage Page (Generic Desktop Ctrls) */ @@ -487,6 +501,9 @@ __ALIGN_BEGIN static uint8_t HID_KEYBOARD_ReportDesc[HID_KEYBOARD_REPORT_DESC_SI 0xC0 // End Collection }; +static uint8_t HIDMInEpAdd = HID_MOUSE_EPIN_ADDR; +static uint8_t HIDKInEpAdd = HID_KEYBOARD_EPIN_ADDR; + /** * @} */ @@ -509,33 +526,40 @@ static uint8_t USBD_HID_Init(USBD_HandleTypeDef *pdev, USBD_HID_HandleTypeDef *hhid; - hhid = (USBD_HID_HandleTypeDef *)USBD_malloc(sizeof(USBD_HID_HandleTypeDef)); + // hhid = (USBD_HID_HandleTypeDef *)USBD_malloc(sizeof(USBD_HID_HandleTypeDef)); + hhid = &_hhid; if (hhid == NULL) { - pdev->pClassData = NULL; + pdev->pClassDataCmsit[pdev->classId] = NULL; return (uint8_t)USBD_EMEM; } - pdev->pClassData = (void *)hhid; + pdev->pClassDataCmsit[pdev->classId] = (void *)hhid; + pdev->pClassData = pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + HIDInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (pdev->dev_speed == USBD_SPEED_HIGH) { - pdev->ep_in[HID_MOUSE_EPIN_ADDR & 0xFU].bInterval = HID_HS_BINTERVAL; - pdev->ep_in[HID_KEYBOARD_EPIN_ADDR & 0xFU].bInterval = HID_HS_BINTERVAL; + pdev->ep_in[HIDMInEpAdd & 0xFU].bInterval = HID_HS_BINTERVAL; + pdev->ep_in[HIDKInEpAdd & 0xFU].bInterval = HID_HS_BINTERVAL; } else { /* LOW and FULL-speed endpoints */ - pdev->ep_in[HID_MOUSE_EPIN_ADDR & 0xFU].bInterval = HID_FS_BINTERVAL; - pdev->ep_in[HID_KEYBOARD_EPIN_ADDR & 0xFU].bInterval = HID_FS_BINTERVAL; + pdev->ep_in[HIDMInEpAdd & 0xFU].bInterval = HID_FS_BINTERVAL; + pdev->ep_in[HIDKInEpAdd & 0xFU].bInterval = HID_FS_BINTERVAL; } /* Open EP IN */ - (void)USBD_LL_OpenEP(pdev, HID_MOUSE_EPIN_ADDR, USBD_EP_TYPE_INTR, + (void)USBD_LL_OpenEP(pdev, HIDMInEpAdd, USBD_EP_TYPE_INTR, HID_MOUSE_EPIN_SIZE); - pdev->ep_in[HID_MOUSE_EPIN_ADDR & 0xFU].is_used = 1U; + pdev->ep_in[HIDMInEpAdd & 0xFU].is_used = 1U; /* Open EP IN */ - (void)USBD_LL_OpenEP(pdev, HID_KEYBOARD_EPIN_ADDR, USBD_EP_TYPE_INTR, + (void)USBD_LL_OpenEP(pdev, HIDKInEpAdd, USBD_EP_TYPE_INTR, HID_KEYBOARD_EPIN_SIZE); - pdev->ep_in[HID_KEYBOARD_EPIN_ADDR & 0xFU].is_used = 1U; + pdev->ep_in[HIDKInEpAdd & 0xFU].is_used = 1U; hhid->Mousestate = HID_IDLE; hhid->Keyboardstate = HID_IDLE; @@ -554,18 +578,24 @@ static uint8_t USBD_HID_DeInit(USBD_HandleTypeDef *pdev, { UNUSED(cfgidx); +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + HIDInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + /* Close HID EPs */ - (void)USBD_LL_CloseEP(pdev, HID_MOUSE_EPIN_ADDR); - pdev->ep_in[HID_MOUSE_EPIN_ADDR & 0xFU].is_used = 0U; - pdev->ep_in[HID_MOUSE_EPIN_ADDR & 0xFU].bInterval = 0U; - USBD_LL_CloseEP(pdev, HID_KEYBOARD_EPIN_ADDR); - pdev->ep_in[HID_KEYBOARD_EPIN_ADDR & 0xFU].is_used = 0U; - pdev->ep_in[HID_KEYBOARD_EPIN_ADDR & 0xFU].bInterval = 0U; + (void)USBD_LL_CloseEP(pdev, HIDMInEpAdd); + pdev->ep_in[HIDMInEpAdd & 0xFU].is_used = 0U; + pdev->ep_in[HIDMInEpAdd & 0xFU].bInterval = 0U; + (void)USBD_LL_CloseEP(pdev, HIDKInEpAdd); + pdev->ep_in[HIDKInEpAdd & 0xFU].is_used = 0U; + pdev->ep_in[HIDKInEpAdd & 0xFU].bInterval = 0U; /* Free allocated memory */ - if (pdev->pClassData != NULL) { - (void)USBD_free(pdev->pClassData); - pdev->pClassData = NULL; + if (pdev->pClassDataCmsit[pdev->classId] != NULL) { + /* No need to free as hhid is no more dynamically allocated */ + // (void)USBD_free(pdev->pClassDataCmsit[pdev->classId]); + pdev->pClassDataCmsit[pdev->classId] = NULL; } return (uint8_t)USBD_OK; @@ -599,7 +629,7 @@ static uint8_t USBD_COMPOSITE_HID_Setup(USBD_HandleTypeDef *pdev, static uint8_t USBD_HID_MOUSE_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_HID_HandleTypeDef *hhid = (USBD_HID_HandleTypeDef *)pdev->pClassData; + USBD_HID_HandleTypeDef *hhid = (USBD_HID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; USBD_StatusTypeDef ret = USBD_OK; uint16_t len = 0U; uint8_t *pbuf = NULL; @@ -708,7 +738,7 @@ static uint8_t USBD_HID_MOUSE_Setup(USBD_HandleTypeDef *pdev, static uint8_t USBD_HID_KEYBOARD_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_HID_HandleTypeDef *hhid = (USBD_HID_HandleTypeDef *) pdev->pClassData; + USBD_HID_HandleTypeDef *hhid = (USBD_HID_HandleTypeDef *) pdev->pClassDataCmsit[pdev->classId]; uint16_t len = 0U; uint8_t *pbuf = NULL; uint16_t status_info = 0U; @@ -811,26 +841,42 @@ static uint8_t USBD_HID_KEYBOARD_Setup(USBD_HandleTypeDef *pdev, * Send HID Report * @param pdev: device instance * @param buff: pointer to report + * @param ClassId: The Class ID * @retval status */ +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_HID_MOUSE_SendReport(USBD_HandleTypeDef *pdev, + uint8_t *report, + uint16_t len, + uint8_t ClassId) +{ + USBD_HID_HandleTypeDef *hhid = (USBD_HID_HandleTypeDef *)pdev->pClassDataCmsit[ClassId]; +#else uint8_t USBD_HID_MOUSE_SendReport(USBD_HandleTypeDef *pdev, uint8_t *report, uint16_t len) { - USBD_HID_HandleTypeDef *hhid = (USBD_HID_HandleTypeDef *)pdev->pClassData; + USBD_HID_HandleTypeDef *hhid = (USBD_HID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#endif /* USE_USBD_COMPOSITE */ if (hhid == NULL) { return (uint8_t)USBD_FAIL; } +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + HIDInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, ClassId); +#endif /* USE_USBD_COMPOSITE */ + if (pdev->dev_state == USBD_STATE_CONFIGURED) { if (hhid->Mousestate == HID_IDLE) { hhid->Mousestate = HID_BUSY; - (void)USBD_LL_Transmit(pdev, HID_MOUSE_EPIN_ADDR, report, len); + (void)USBD_LL_Transmit(pdev, HIDMInEpAdd, report, len); } else { return (uint8_t)USBD_BUSY; } } + return (uint8_t)USBD_OK; } @@ -839,27 +885,42 @@ uint8_t USBD_HID_MOUSE_SendReport(USBD_HandleTypeDef *pdev, * Send HID Report * @param pdev: device instance * @param buff: pointer to report + * @param ClassId: The Class ID * @retval status */ +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_HID_KEYBOARD_SendReport(USBD_HandleTypeDef *pdev, + uint8_t *report, + uint16_t len, + uint8_t ClassId) +{ + USBD_HID_HandleTypeDef *hhid = (USBD_HID_HandleTypeDef *)pdev->pClassDataCmsit[ClassId]; +#else uint8_t USBD_HID_KEYBOARD_SendReport(USBD_HandleTypeDef *pdev, uint8_t *report, uint16_t len) { - USBD_HID_HandleTypeDef *hhid = (USBD_HID_HandleTypeDef *)pdev->pClassData; + USBD_HID_HandleTypeDef *hhid = (USBD_HID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#endif /* USE_USBD_COMPOSITE */ if (hhid == NULL) { return (uint8_t)USBD_FAIL; } +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + HIDInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, ClassId); +#endif /* USE_USBD_COMPOSITE */ if (pdev->dev_state == USBD_STATE_CONFIGURED) { if (hhid->Keyboardstate == HID_IDLE) { hhid->Keyboardstate = HID_BUSY; - (void)USBD_LL_Transmit(pdev, HID_KEYBOARD_EPIN_ADDR, report, len); + (void)USBD_LL_Transmit(pdev, HIDKInEpAdd, report, len); } else { return (uint8_t)USBD_BUSY; } } + return (uint8_t)USBD_OK; } @@ -888,6 +949,7 @@ uint32_t USBD_HID_GetPollingInterval(USBD_HandleTypeDef *pdev) return ((uint32_t)(polling_interval)); } +#ifndef USE_USBD_COMPOSITE /** * @brief USBD_HID_GetCfgFSDesc * return FS configuration descriptor @@ -897,8 +959,18 @@ uint32_t USBD_HID_GetPollingInterval(USBD_HandleTypeDef *pdev) */ static uint8_t *USBD_HID_GetFSCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_HID_CfgFSDesc); + USBD_EpDescTypeDef *pEpDesc = USBD_GetEpDesc(USBD_HID_CfgFSDesc, HID_MOUSE_EPIN_ADDR); + if (pEpDesc != NULL) { + pEpDesc->bInterval = HID_FS_BINTERVAL; + } + + pEpDesc = USBD_GetEpDesc(USBD_HID_CfgFSDesc, HID_KEYBOARD_EPIN_ADDR); + if (pEpDesc != NULL) { + pEpDesc->bInterval = HID_FS_BINTERVAL; + } + + *length = (uint16_t)sizeof(USBD_HID_CfgFSDesc); return USBD_HID_CfgFSDesc; } @@ -911,8 +983,18 @@ static uint8_t *USBD_HID_GetFSCfgDesc(uint16_t *length) */ static uint8_t *USBD_HID_GetHSCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_HID_CfgHSDesc); + USBD_EpDescTypeDef *pEpDesc = USBD_GetEpDesc(USBD_HID_CfgHSDesc, HID_MOUSE_EPIN_ADDR); + + if (pEpDesc != NULL) { + pEpDesc->bInterval = HID_HS_BINTERVAL; + } + pEpDesc = USBD_GetEpDesc(USBD_HID_CfgHSDesc, HID_KEYBOARD_EPIN_ADDR); + if (pEpDesc != NULL) { + pEpDesc->bInterval = HID_HS_BINTERVAL; + } + + *length = (uint16_t)sizeof(USBD_HID_CfgHSDesc); return USBD_HID_CfgHSDesc; } @@ -925,10 +1007,21 @@ static uint8_t *USBD_HID_GetHSCfgDesc(uint16_t *length) */ static uint8_t *USBD_HID_GetOtherSpeedCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_HID_OtherSpeedCfgDesc); + USBD_EpDescTypeDef *pEpDesc = USBD_GetEpDesc(USBD_HID_OtherSpeedCfgDesc, HID_MOUSE_EPIN_ADDR); + + if (pEpDesc != NULL) { + pEpDesc->bInterval = HID_FS_BINTERVAL; + } + pEpDesc = USBD_GetEpDesc(USBD_HID_OtherSpeedCfgDesc, HID_KEYBOARD_EPIN_ADDR); + + if (pEpDesc != NULL) { + pEpDesc->bInterval = HID_FS_BINTERVAL; + } + *length = (uint16_t)sizeof(USBD_HID_OtherSpeedCfgDesc); return USBD_HID_OtherSpeedCfgDesc; } +#endif /* USE_USBD_COMPOSITE */ /** * @brief USBD_HID_DataIn @@ -944,14 +1037,14 @@ static uint8_t USBD_HID_DataIn(USBD_HandleTypeDef *pdev, /* Ensure that the FIFO is empty before a new transfer, this condition could be caused by a new transfer before the end of the previous transfer */ if (epnum == (HID_KEYBOARD_EPIN_ADDR & 0x7F)) { - ((USBD_HID_HandleTypeDef *)pdev->pClassData)->Keyboardstate = HID_IDLE; + ((USBD_HID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId])->Keyboardstate = HID_IDLE; } else if (epnum == (HID_MOUSE_EPIN_ADDR & 0x7F)) { - ((USBD_HID_HandleTypeDef *)pdev->pClassData)->Mousestate = HID_IDLE; + ((USBD_HID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId])->Mousestate = HID_IDLE; } return (uint8_t)USBD_OK; } - +#ifndef USE_USBD_COMPOSITE /** * @brief DeviceQualifierDescriptor * return Device Qualifier descriptor @@ -964,6 +1057,7 @@ static uint8_t *USBD_HID_GetDeviceQualifierDesc(uint16_t *length) return USBD_HID_DeviceQualifierDesc; } +#endif /* USE_USBD_COMPOSITE */ #endif /* USBD_USE_HID_COMPOSITE */ #endif /* USBCON */ /************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/cores/arduino/stm32/usb/hid/usbd_hid_composite.h b/stm32/cores/arduino/stm32/usb/hid/usbd_hid_composite.h index 7ff3d586..c5d4aaab 100644 --- a/stm32/cores/arduino/stm32/usb/hid/usbd_hid_composite.h +++ b/stm32/cores/arduino/stm32/usb/hid/usbd_hid_composite.h @@ -92,6 +92,22 @@ typedef struct { HID_StateTypeDef Mousestate; HID_StateTypeDef Keyboardstate; } USBD_HID_HandleTypeDef; + +/* + * HID Class specification version 1.1 + * 6.2.1 HID Descriptor + */ + +typedef struct { + uint8_t bLength; + uint8_t bDescriptorType; + uint16_t bcdHID; + uint8_t bCountryCode; + uint8_t bNumDescriptors; + uint8_t bHIDDescriptorType; + uint16_t wItemLength; +} __PACKED USBD_HIDDescTypeDef; + /** * @} */ @@ -119,12 +135,23 @@ extern USBD_ClassTypeDef USBD_COMPOSITE_HID; /** @defgroup USB_CORE_Exported_Functions * @{ */ +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_HID_MOUSE_SendReport(USBD_HandleTypeDef *pdev, + uint8_t *report, + uint16_t len, + uint8_t ClassId); +uint8_t USBD_HID_KEYBOARD_SendReport(USBD_HandleTypeDef *pdev, + uint8_t *report, + uint16_t len, + uint8_t ClassId); +#else uint8_t USBD_HID_MOUSE_SendReport(USBD_HandleTypeDef *pdev, uint8_t *report, uint16_t len); uint8_t USBD_HID_KEYBOARD_SendReport(USBD_HandleTypeDef *pdev, uint8_t *report, uint16_t len); +#endif /* USE_USBD_COMPOSITE */ uint32_t USBD_HID_GetPollingInterval(USBD_HandleTypeDef *pdev); diff --git a/stm32/cores/arduino/stm32/usb/hid/usbd_hid_composite_if.c b/stm32/cores/arduino/stm32/usb/hid/usbd_hid_composite_if.c index 5f716e66..9355bbac 100644 --- a/stm32/cores/arduino/stm32/usb/hid/usbd_hid_composite_if.c +++ b/stm32/cores/arduino/stm32/usb/hid/usbd_hid_composite_if.c @@ -23,15 +23,14 @@ #include "usbd_hid_composite_if.h" #include "usbd_hid_composite.h" -#ifdef __cplusplus -extern "C" { -#endif - /* USB Device Core HID composite handle declaration */ USBD_HandleTypeDef hUSBD_Device_HID; static bool HID_keyboard_initialized = false; static bool HID_mouse_initialized = false; +#if defined(ICACHE) && defined (HAL_ICACHE_MODULE_ENABLED) && !defined(HAL_ICACHE_MODULE_DISABLED) + static bool icache_enabled = false; +#endif /** * @brief Initialize USB devices @@ -40,6 +39,15 @@ static bool HID_mouse_initialized = false; */ void HID_Composite_Init(HID_Interface device) { +#if defined(ICACHE) && defined (HAL_ICACHE_MODULE_ENABLED) && !defined(HAL_ICACHE_MODULE_DISABLED) + if (HAL_ICACHE_IsEnabled() == 1) { + icache_enabled = true; + /* Disable instruction cache prior to internal cacheable memory update */ + if (HAL_ICACHE_Disable() != HAL_OK) { + Error_Handler(); + } + } +#endif /* ICACHE && HAL_ICACHE_MODULE_ENABLED && !HAL_ICACHE_MODULE_DISABLED */ if (IS_HID_INTERFACE(device) && !HID_keyboard_initialized && !HID_mouse_initialized) { /* Init Device Library */ @@ -76,6 +84,14 @@ void HID_Composite_DeInit(HID_Interface device) /* DeInit Device Library */ USBD_DeInit(&hUSBD_Device_HID); } +#if defined(ICACHE) && defined (HAL_ICACHE_MODULE_ENABLED) && !defined(HAL_ICACHE_MODULE_DISABLED) + if (icache_enabled) { + /* Re-enable instruction cache */ + if (HAL_ICACHE_Enable() != HAL_OK) { + Error_Handler(); + } + } +#endif /* ICACHE && HAL_ICACHE_MODULE_ENABLED && !HAL_ICACHE_MODULE_DISABLED */ if (device == HID_KEYBOARD) { HID_keyboard_initialized = false; } @@ -106,9 +122,6 @@ void HID_Composite_keyboard_sendReport(uint8_t *report, uint16_t len) USBD_HID_KEYBOARD_SendReport(&hUSBD_Device_HID, report, len); } -#ifdef __cplusplus -} -#endif #endif /* USBD_USE_HID_COMPOSITE */ #endif /* USBCON */ /************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/cores/arduino/stm32/usb/usbd_conf.c b/stm32/cores/arduino/stm32/usb/usbd_conf.c index 48c92fca..86571852 100644 --- a/stm32/cores/arduino/stm32/usb/usbd_conf.c +++ b/stm32/cores/arduino/stm32/usb/usbd_conf.c @@ -57,15 +57,17 @@ void HAL_PCD_MspInit(PCD_HandleTypeDef *hpcd) pinMode(PIN_UCPD_TCPP, OUTPUT_OPEN_DRAIN); digitalWriteFast(digitalPinToPinName(PIN_UCPD_TCPP), LOW); #endif - -#if defined(PWR_CR2_USV) || defined(PWR_SVMCR_USV) - /* Enable VDDUSB on Pwrctrl CR2 register*/ - HAL_PWREx_EnableVddUSB(); +#if defined(PWR_CR3_USB33DEN) || defined(PWR_USBSCR_USB33DEN) + HAL_PWREx_EnableUSBVoltageDetector(); #endif -#ifdef STM32H7xx - if (!LL_PWR_IsActiveFlag_USB()) { - HAL_PWREx_EnableUSBVoltageDetector(); - } +#if defined(PWR_CR3_USB33RDY) + while (!LL_PWR_IsActiveFlag_USB()); +#elif defined(PWR_VMSR_USB33RDY) + while (!LL_PWR_IsActiveFlag_VDDUSB()); +#endif +#if defined(PWR_CR2_USV) || defined(PWR_SVMCR_USV) || defined(PWR_USBSCR_USB33SV) + /* Enable VDDUSB */ + HAL_PWREx_EnableVddUSB(); #endif #if defined (USB) if (hpcd->Instance == USB) { @@ -647,7 +649,8 @@ uint8_t USBD_LL_IsStallEP(USBD_HandleTypeDef *pdev, uint8_t ep_addr) * @param dev_addr: Endpoint Number * @retval USBD Status */ -USBD_StatusTypeDef USBD_LL_SetUSBAddress(USBD_HandleTypeDef *pdev, uint8_t dev_addr) +USBD_StatusTypeDef USBD_LL_SetUSBAddress(USBD_HandleTypeDef *pdev, + uint8_t dev_addr) { HAL_PCD_SetAddress(pdev->pData, dev_addr); return USBD_OK; @@ -661,10 +664,8 @@ USBD_StatusTypeDef USBD_LL_SetUSBAddress(USBD_HandleTypeDef *pdev, uint8_t dev_a * @param size: Data size * @retval USBD Status */ -USBD_StatusTypeDef USBD_LL_Transmit(USBD_HandleTypeDef *pdev, - uint8_t ep_addr, - uint8_t *pbuf, - uint32_t size) +USBD_StatusTypeDef USBD_LL_Transmit(USBD_HandleTypeDef *pdev, uint8_t ep_addr, + uint8_t *pbuf, uint32_t size) { HAL_PCD_EP_Transmit(pdev->pData, ep_addr, pbuf, size); return USBD_OK; @@ -679,8 +680,7 @@ USBD_StatusTypeDef USBD_LL_Transmit(USBD_HandleTypeDef *pdev, * @retval USBD Status */ USBD_StatusTypeDef USBD_LL_PrepareReceive(USBD_HandleTypeDef *pdev, - uint8_t ep_addr, - uint8_t *pbuf, + uint8_t ep_addr, uint8_t *pbuf, uint32_t size) { HAL_PCD_EP_Receive(pdev->pData, ep_addr, pbuf, size); diff --git a/stm32/cores/arduino/stm32/usb/usbd_conf.h b/stm32/cores/arduino/stm32/usb/usbd_conf.h index 2d715520..03578c3c 100644 --- a/stm32/cores/arduino/stm32/usb/usbd_conf.h +++ b/stm32/cores/arduino/stm32/usb/usbd_conf.h @@ -74,6 +74,9 @@ extern "C" { #elif defined(STM32G0xx) #define USB_IRQn USB_UCPD1_2_IRQn #define USB_IRQHandler USB_UCPD1_2_IRQHandler +#elif defined(STM32H5xx) +#define USB_IRQn USB_DRD_FS_IRQn +#define USB_IRQHandler USB_DRD_FS_IRQHandler #elif defined(STM32U5xx) && !defined(USB_DRD_FS) #define USB_IRQn OTG_FS_IRQn #define USB_IRQHandler OTG_FS_IRQHandler @@ -195,7 +198,7 @@ extern "C" { #ifndef UVC_MATRIX_COEFFICIENTS #define UVC_MATRIX_COEFFICIENTS 0x04U #endif /* UVC_MATRIX_COEFFICIENTS */ -#endif +#endif /* USBD_UVC_FORMAT_UNCOMPRESSED */ /* Video Stream frame width and height */ #ifndef UVC_WIDTH @@ -279,7 +282,7 @@ extern "C" { } while (0) #else #define USBD_UsrLog(...) do {} while (0) -#endif +#endif /* (USBD_DEBUG_LEVEL > 0U) */ #if (USBD_DEBUG_LEVEL > 1U) @@ -290,7 +293,7 @@ extern "C" { } while (0) #else #define USBD_ErrLog(...) do {} while (0) -#endif +#endif /* (USBD_DEBUG_LEVEL > 1U) */ #if (USBD_DEBUG_LEVEL > 2U) #define USBD_DbgLog(...) do { \ @@ -300,7 +303,7 @@ extern "C" { } while (0) #else #define USBD_DbgLog(...) do {} while (0) -#endif +#endif /* (USBD_DEBUG_LEVEL > 2U) */ /* Exported functions -------------------------------------------------------*/ void *USBD_static_malloc(uint32_t size); diff --git a/stm32/cores/arduino/stm32/usb/usbd_desc.c b/stm32/cores/arduino/stm32/usb/usbd_desc.c index 6474dba2..651bb9cf 100644 --- a/stm32/cores/arduino/stm32/usb/usbd_desc.c +++ b/stm32/cores/arduino/stm32/usb/usbd_desc.c @@ -110,7 +110,7 @@ uint8_t *USBD_Class_InterfaceStrDescriptor(USBD_SpeedTypeDef speed, uint16_t *le #endif /* USB_CLASS_USER_STRING_DESC */ #if ((USBD_LPM_ENABLED == 1) || (USBD_CLASS_BOS_ENABLED == 1)) uint8_t *USBD_USR_BOSDescriptor(USBD_SpeedTypeDef speed, uint16_t *length); -#endif +#endif /* (USBD_LPM_ENABLED == 1) || (USBD_CLASS_BOS_ENABLED == 1) */ /* Private variables ---------------------------------------------------------*/ USBD_DescriptorsTypeDef USBD_Desc = { USBD_Class_DeviceDescriptor, @@ -122,11 +122,11 @@ USBD_DescriptorsTypeDef USBD_Desc = { USBD_Class_InterfaceStrDescriptor, #if (USBD_CLASS_USER_STRING_DESC == 1) USBD_Class_UserStrDescriptor, -#endif +#endif /* USB_CLASS_USER_STRING_DESC */ #if ((USBD_LPM_ENABLED == 1) || (USBD_CLASS_BOS_ENABLED == 1)) USBD_USR_BOSDescriptor, -#endif +#endif /* (USBD_LPM_ENABLED == 1) || (USBD_CLASS_BOS_ENABLED == 1) */ }; #ifdef USBD_USE_HID_COMPOSITE @@ -139,7 +139,7 @@ __ALIGN_BEGIN uint8_t USBD_Class_DeviceDesc[USB_LEN_DEV_DESC] __ALIGN_END = { in order to support BOS Desc */ #else 0x00, /* bcdUSB */ -#endif +#endif /* (USBD_LPM_ENABLED == 1) || (USBD_CLASS_BOS_ENABLED == 1) */ 0x02, 0x00, /* bDeviceClass */ 0x00, /* bDeviceSubClass */ @@ -149,8 +149,8 @@ __ALIGN_BEGIN uint8_t USBD_Class_DeviceDesc[USB_LEN_DEV_DESC] __ALIGN_END = { HIBYTE(USBD_VID), /* idVendor */ LOBYTE(USBD_PID), /* idProduct */ HIBYTE(USBD_PID), /* idProduct */ - 0x00, /* bcdDevice rel. 0.00 */ - 0x00, + 0x00, /* bcdDevice rel. 2.00 */ + 0x02, USBD_IDX_MFC_STR, /* Index of manufacturer string */ USBD_IDX_PRODUCT_STR, /* Index of product string */ USBD_IDX_SERIAL_STR, /* Index of serial number string */ @@ -178,8 +178,8 @@ __ALIGN_BEGIN uint8_t USBD_Class_DeviceDesc[USB_LEN_DEV_DESC] __ALIGN_END = { HIBYTE(USBD_VID), /* idVendor */ LOBYTE(USBD_PID), /* idProduct */ HIBYTE(USBD_PID), /* idProduct */ - 0x00, /* bcdDevice rel. 0.00 */ - 0x00, + 0x00, /* bcdDevice rel. 2.00 */ + 0x02, USBD_IDX_MFC_STR, /* Index of manufacturer string */ USBD_IDX_PRODUCT_STR, /* Index of product string */ USBD_IDX_SERIAL_STR, /* Index of serial number string */ @@ -204,7 +204,7 @@ __ALIGN_BEGIN uint8_t USBD_BOSDesc[USB_SIZ_BOS_DESC] __ALIGN_END = { 0x0, 0x0 }; -#endif +#endif /* USBD_LPM_ENABLED */ /* USB Device Billboard BOS descriptor Template */ #if (USBD_CLASS_BOS_ENABLED == 1) @@ -229,14 +229,16 @@ __ALIGN_BEGIN uint8_t USBD_BOSDesc[USB_SIZ_BOS_DESC] __ALIGN_END = { 0x34, /* bLength */ 0x10, /* bDescriptorType: DEVICE CAPABILITY Type */ 0x0D, /* bDevCapabilityType: BILLBOARD_CAPABILITY */ - USBD_BB_URL_STRING_INDEX, /* iAddtionalInfoURL: Index of string descriptor providing a URL where the user can go to get more - detailed information about the product and the various Alternate Modes it supports */ + USBD_BB_URL_STRING_INDEX, /* iAddtionalInfoURL: Index of string descriptor providing a URL where the user + can go to get more detailed information about the product and the various + Alternate Modes it supports */ 0x02, /* bNumberOfAlternateModes: Number of Alternate modes supported. The maximum value that this field can be set to is MAX_NUM_ALT_MODE. */ 0x00, /* bPreferredAlternateMode: Index of the preferred Alternate Mode. System - software may use this information to provide the user with a better user experience. */ + software may use this information to provide the user with a better + user experience. */ 0x00, 0x00, /* VCONN Power needed by the adapter for full functionality 000b = 1W */ @@ -271,7 +273,7 @@ __ALIGN_BEGIN uint8_t USBD_BOSDesc[USB_SIZ_BOS_DESC] __ALIGN_END = { 0 first Mode entry 1 second Mode entry */ - USBD_BB_ALTMODE1_STRING_INDEX, /* iAlternateModeString[1]: Index of string descriptor describing protocol. + USBD_BB_ALTMODE1_STRING_INDEX, /* iAlternateModeString[1]: Index of string descriptor describing protocol. It is optional to support this string. */ /* Alternate Mode Desc */ /* ----------- Device Capability Descriptor: BillBoard Alternate Mode Desc ---------- */ @@ -279,18 +281,21 @@ __ALIGN_BEGIN uint8_t USBD_BOSDesc[USB_SIZ_BOS_DESC] __ALIGN_END = { 0x10, /* bDescriptorType: Device Descriptor Type */ 0x0F, /* bDevCapabilityType: BILLBOARD ALTERNATE MODE CAPABILITY */ 0x00, /* bIndex: Index of Alternate Mode described in the Billboard Capability Desc */ - 0x10, 0x00, 0x00, 0x00, /* dwAlternateModeVdo: contents of the Mode VDO for the alternate mode identified by bIndex */ + 0x10, 0x00, 0x00, 0x00, /* dwAlternateModeVdo: contents of the Mode VDO for the alternate mode + identified by bIndex */ 0x08, /* bLength */ 0x10, /* bDescriptorType: Device Descriptor Type */ 0x0F, /* bDevCapabilityType: BILLBOARD ALTERNATE MODE CAPABILITY */ 0x01, /* bIndex: Index of Alternate Mode described in the Billboard Capability Desc */ - 0x20, 0x00, 0x00, 0x00, /* dwAlternateModeVdo: contents of the Mode VDO for the alternate mode identified by bIndex */ + 0x20, 0x00, 0x00, 0x00, /* dwAlternateModeVdo: contents of the Mode VDO for the alternate mode + identified by bIndex */ }; -#endif +#endif /* USBD_CLASS_BOS_ENABLED */ + /* USB Standard Device Descriptor */ -__ALIGN_BEGIN uint8_t USBD_LangIDDesc[USB_LEN_LANGID_STR_DESC] __ALIGN_END = { +__ALIGN_BEGIN uint8_t USBD_LangIDDesc[USB_LEN_LANGID_STR_DESC] __ALIGN_END = { USB_LEN_LANGID_STR_DESC, USB_DESC_TYPE_STRING, LOBYTE(USBD_LANGID_STRING), @@ -372,6 +377,7 @@ uint8_t *USBD_ManufacturerStrDescriptor(USBD_SpeedTypeDef speed, uint16_t *lengt uint8_t *USBD_SerialStrDescriptor(USBD_SpeedTypeDef speed, uint16_t *length) { UNUSED(speed); + *length = USB_SIZ_STRING_SERIAL; /* Update the serial number string descriptor with the data from the unique ID*/ @@ -419,7 +425,9 @@ uint8_t *USBD_Class_InterfaceStrDescriptor(USBD_SpeedTypeDef speed, uint16_t *le */ static void Get_SerialNum(void) { - uint32_t deviceserial0, deviceserial1, deviceserial2; + uint32_t deviceserial0; + uint32_t deviceserial1; + uint32_t deviceserial2; deviceserial0 = *(uint32_t *)DEVICE_ID1; deviceserial1 = *(uint32_t *)DEVICE_ID2; @@ -448,7 +456,7 @@ uint8_t *USBD_USR_BOSDescriptor(USBD_SpeedTypeDef speed, uint16_t *length) *length = sizeof(USBD_BOSDesc); return (uint8_t *)USBD_BOSDesc; } -#endif +#endif /* (USBD_LPM_ENABLED == 1) || (USBD_CLASS_BOS_ENABLED == 1) */ #if (USBD_CLASS_USER_STRING_DESC == 1) @@ -467,7 +475,7 @@ uint8_t *USBD_Class_UserStrDescriptor(USBD_SpeedTypeDef speed, uint8_t idx, uint UNUSED(length); return USBD_StrDesc; } -#endif +#endif /* USBD_CLASS_USER_STRING_DESC */ /** diff --git a/stm32/cores/arduino/stm32/usb/usbd_desc.h b/stm32/cores/arduino/stm32/usb/usbd_desc.h index bb757e93..aa832b27 100644 --- a/stm32/cores/arduino/stm32/usb/usbd_desc.h +++ b/stm32/cores/arduino/stm32/usb/usbd_desc.h @@ -44,7 +44,7 @@ #define USBD_BB_URL_STR_DESC (uint8_t *)"www.st.com" #define USBD_BB_ALTMODE0_STR_DESC (uint8_t *)"STM32 Alternate0 Mode" #define USBD_BB_ALTMODE1_STR_DESC (uint8_t *)"STM32 Alternate1 Mode" - #endif + #endif /* USBD_CLASS_USER_STRING_DESC */ #define USB_SIZ_STRING_SERIAL 0x1AU @@ -52,7 +52,7 @@ #define USB_SIZ_BOS_DESC 0x0CU #elif (USBD_CLASS_BOS_ENABLED == 1) #define USB_SIZ_BOS_DESC 0x5DU - #endif + #endif /* USBD_LPM_ENABLED */ /* Exported macro ------------------------------------------------------------*/ /* Exported functions ------------------------------------------------------- */ diff --git a/stm32/cores/arduino/wiring_time.h b/stm32/cores/arduino/wiring_time.h index 74b1978c..4b325376 100644 --- a/stm32/cores/arduino/wiring_time.h +++ b/stm32/cores/arduino/wiring_time.h @@ -22,6 +22,7 @@ #include "clock.h" #include "dwt.h" +#include // for struct timeval #ifdef __cplusplus extern "C" { @@ -86,6 +87,27 @@ static inline void delayMicroseconds(uint32_t us) #endif } +/** + * \brief gives the number of seconds and microseconds since the Epoch + * + * based on millisecond since last power on. + * + * \note The function is declared as weak to be overwritten in case of other + * implementations in user file (using RTC values for example). + * + * \param tv argument is a struct timeval + * \param tz argument is a struct timezone (unused) + * + * \return 0 + */ +int __attribute__((weak)) _gettimeofday(struct timeval *tv, void *tz) +{ + (void)tz; + tv->tv_sec = getCurrentMillis() / 1000; + tv->tv_usec = getCurrentMicros() - (tv->tv_sec * 1000000); // get remaining microseconds + return 0; +} + #ifdef __cplusplus } #endif diff --git a/stm32/libraries/CMSIS_DSP/CMakeLists.txt b/stm32/libraries/CMSIS_DSP/CMakeLists.txt index 655fa92a..cf81b5fc 100644 --- a/stm32/libraries/CMSIS_DSP/CMakeLists.txt +++ b/stm32/libraries/CMSIS_DSP/CMakeLists.txt @@ -20,18 +20,33 @@ target_link_libraries(CMSIS_DSP INTERFACE CMSIS_DSP_usage) add_library(CMSIS_DSP_bin OBJECT EXCLUDE_FROM_ALL src/BasicMathFunctions/BasicMathFunctions.c + src/BasicMathFunctions/BasicMathFunctionsF16.c src/BayesFunctions/BayesFunctions.c + src/BayesFunctions/BayesFunctionsF16.c src/CommonTables/CommonTables.c + src/CommonTables/CommonTablesF16.c src/ComplexMathFunctions/ComplexMathFunctions.c + src/ComplexMathFunctions/ComplexMathFunctionsF16.c src/ControllerFunctions/ControllerFunctions.c src/DistanceFunctions/DistanceFunctions.c + src/DistanceFunctions/DistanceFunctionsF16.c src/FastMathFunctions/FastMathFunctions.c + src/FastMathFunctions/FastMathFunctionsF16.c src/FilteringFunctions/FilteringFunctions.c + src/FilteringFunctions/FilteringFunctionsF16.c + src/InterpolationFunctions/InterpolationFunctions.c + src/InterpolationFunctions/InterpolationFunctionsF16.c src/MatrixFunctions/MatrixFunctions.c + src/MatrixFunctions/MatrixFunctionsF16.c + src/QuaternionMathFunctions/QuaternionMathFunctions.c src/StatisticsFunctions/StatisticsFunctions.c + src/StatisticsFunctions/StatisticsFunctionsF16.c src/SupportFunctions/SupportFunctions.c + src/SupportFunctions/SupportFunctionsF16.c src/SVMFunctions/SVMFunctions.c + src/SVMFunctions/SVMFunctionsF16.c src/TransformFunctions/TransformFunctions.c + src/TransformFunctions/TransformFunctionsF16.c ) target_link_libraries(CMSIS_DSP_bin PUBLIC CMSIS_DSP_usage) diff --git a/stm32/libraries/CMSIS_DSP/src/BasicMathFunctions/BasicMathFunctionsF16.c b/stm32/libraries/CMSIS_DSP/src/BasicMathFunctions/BasicMathFunctionsF16.c new file mode 100644 index 00000000..00a8ae57 --- /dev/null +++ b/stm32/libraries/CMSIS_DSP/src/BasicMathFunctions/BasicMathFunctionsF16.c @@ -0,0 +1 @@ +#include "../Source/BasicMathFunctions/BasicMathFunctionsF16.c" diff --git a/stm32/libraries/CMSIS_DSP/src/BayesFunctions/BayesFunctionsF16.c b/stm32/libraries/CMSIS_DSP/src/BayesFunctions/BayesFunctionsF16.c new file mode 100644 index 00000000..8ece42f5 --- /dev/null +++ b/stm32/libraries/CMSIS_DSP/src/BayesFunctions/BayesFunctionsF16.c @@ -0,0 +1 @@ +#include "../Source/BayesFunctions/BayesFunctionsF16.c" diff --git a/stm32/libraries/CMSIS_DSP/src/CommonTables/CommonTablesF16.c b/stm32/libraries/CMSIS_DSP/src/CommonTables/CommonTablesF16.c new file mode 100644 index 00000000..0ce9f67a --- /dev/null +++ b/stm32/libraries/CMSIS_DSP/src/CommonTables/CommonTablesF16.c @@ -0,0 +1 @@ +#include "../Source/CommonTables/CommonTablesF16.c" diff --git a/stm32/libraries/CMSIS_DSP/src/ComplexMathFunctions/ComplexMathFunctionsF16.c b/stm32/libraries/CMSIS_DSP/src/ComplexMathFunctions/ComplexMathFunctionsF16.c new file mode 100644 index 00000000..fb12fbad --- /dev/null +++ b/stm32/libraries/CMSIS_DSP/src/ComplexMathFunctions/ComplexMathFunctionsF16.c @@ -0,0 +1 @@ +#include "../Source/ComplexMathFunctions/ComplexMathFunctionsF16.c" diff --git a/stm32/libraries/CMSIS_DSP/src/DistanceFunctions/DistanceFunctionsF16.c b/stm32/libraries/CMSIS_DSP/src/DistanceFunctions/DistanceFunctionsF16.c new file mode 100644 index 00000000..fd47d02c --- /dev/null +++ b/stm32/libraries/CMSIS_DSP/src/DistanceFunctions/DistanceFunctionsF16.c @@ -0,0 +1 @@ +#include "../Source/DistanceFunctions/DistanceFunctionsF16.c" diff --git a/stm32/libraries/CMSIS_DSP/src/FastMathFunctions/FastMathFunctionsF16.c b/stm32/libraries/CMSIS_DSP/src/FastMathFunctions/FastMathFunctionsF16.c new file mode 100644 index 00000000..b83767e2 --- /dev/null +++ b/stm32/libraries/CMSIS_DSP/src/FastMathFunctions/FastMathFunctionsF16.c @@ -0,0 +1 @@ +#include "../Source/FastMathFunctions/FastMathFunctionsF16.c" diff --git a/stm32/libraries/CMSIS_DSP/src/FilteringFunctions/FilteringFunctionsF16.c b/stm32/libraries/CMSIS_DSP/src/FilteringFunctions/FilteringFunctionsF16.c new file mode 100644 index 00000000..d5f0be09 --- /dev/null +++ b/stm32/libraries/CMSIS_DSP/src/FilteringFunctions/FilteringFunctionsF16.c @@ -0,0 +1 @@ +#include "../Source/FilteringFunctions/FilteringFunctionsF16.c" diff --git a/stm32/libraries/CMSIS_DSP/src/InterpolationFunctions/InterpolationFunctions.c b/stm32/libraries/CMSIS_DSP/src/InterpolationFunctions/InterpolationFunctions.c new file mode 100644 index 00000000..7f935a75 --- /dev/null +++ b/stm32/libraries/CMSIS_DSP/src/InterpolationFunctions/InterpolationFunctions.c @@ -0,0 +1 @@ +#include "../Source/InterpolationFunctions/InterpolationFunctions.c" diff --git a/stm32/libraries/CMSIS_DSP/src/InterpolationFunctions/InterpolationFunctionsF16.c b/stm32/libraries/CMSIS_DSP/src/InterpolationFunctions/InterpolationFunctionsF16.c new file mode 100644 index 00000000..7ddb2fa5 --- /dev/null +++ b/stm32/libraries/CMSIS_DSP/src/InterpolationFunctions/InterpolationFunctionsF16.c @@ -0,0 +1 @@ +#include "../Source/InterpolationFunctions/InterpolationFunctionsF16.c" diff --git a/stm32/libraries/CMSIS_DSP/src/MatrixFunctions/MatrixFunctionsF16.c b/stm32/libraries/CMSIS_DSP/src/MatrixFunctions/MatrixFunctionsF16.c new file mode 100644 index 00000000..76cecebf --- /dev/null +++ b/stm32/libraries/CMSIS_DSP/src/MatrixFunctions/MatrixFunctionsF16.c @@ -0,0 +1 @@ +#include "../Source/MatrixFunctions/MatrixFunctionsF16.c" diff --git a/stm32/libraries/CMSIS_DSP/src/QuaternionMathFunctions/QuaternionMathFunctions.c b/stm32/libraries/CMSIS_DSP/src/QuaternionMathFunctions/QuaternionMathFunctions.c new file mode 100644 index 00000000..f3d1c48b --- /dev/null +++ b/stm32/libraries/CMSIS_DSP/src/QuaternionMathFunctions/QuaternionMathFunctions.c @@ -0,0 +1 @@ +#include "../Source/QuaternionMathFunctions/QuaternionMathFunctions.c" diff --git a/stm32/libraries/CMSIS_DSP/src/SVMFunctions/SVMFunctionsF16.c b/stm32/libraries/CMSIS_DSP/src/SVMFunctions/SVMFunctionsF16.c new file mode 100644 index 00000000..573f6910 --- /dev/null +++ b/stm32/libraries/CMSIS_DSP/src/SVMFunctions/SVMFunctionsF16.c @@ -0,0 +1 @@ +#include "../Source/SVMFunctions/SVMFunctionsF16.c" diff --git a/stm32/libraries/CMSIS_DSP/src/StatisticsFunctions/StatisticsFunctionsF16.c b/stm32/libraries/CMSIS_DSP/src/StatisticsFunctions/StatisticsFunctionsF16.c new file mode 100644 index 00000000..9dbed439 --- /dev/null +++ b/stm32/libraries/CMSIS_DSP/src/StatisticsFunctions/StatisticsFunctionsF16.c @@ -0,0 +1 @@ +#include "../Source/StatisticsFunctions/StatisticsFunctionsF16.c" diff --git a/stm32/libraries/CMSIS_DSP/src/SupportFunctions/SupportFunctionsF16.c b/stm32/libraries/CMSIS_DSP/src/SupportFunctions/SupportFunctionsF16.c new file mode 100644 index 00000000..4c46794a --- /dev/null +++ b/stm32/libraries/CMSIS_DSP/src/SupportFunctions/SupportFunctionsF16.c @@ -0,0 +1 @@ +#include "../Source/SupportFunctions/SupportFunctionsF16.c" diff --git a/stm32/libraries/CMSIS_DSP/src/TransformFunctions/TransformFunctionsF16.c b/stm32/libraries/CMSIS_DSP/src/TransformFunctions/TransformFunctionsF16.c new file mode 100644 index 00000000..dd8f9a0d --- /dev/null +++ b/stm32/libraries/CMSIS_DSP/src/TransformFunctions/TransformFunctionsF16.c @@ -0,0 +1 @@ +#include "../Source/TransformFunctions/TransformFunctionsF16.c" diff --git a/stm32/libraries/EEPROM/src/utility/stm32_eeprom.c b/stm32/libraries/EEPROM/src/utility/stm32_eeprom.c index e37638da..68f50fb7 100644 --- a/stm32/libraries/EEPROM/src/utility/stm32_eeprom.c +++ b/stm32/libraries/EEPROM/src/utility/stm32_eeprom.c @@ -19,6 +19,7 @@ #include "stm32_eeprom.h" #include "stm32yyxx_ll_utils.h" #include +#include #ifdef __cplusplus extern "C" { @@ -37,15 +38,19 @@ extern "C" { #endif /* !FLASH_BANK_NUMBER */ /* Be able to change FLASH_DATA_SECTOR to use if relevant */ -#if defined(FLASH_SECTOR_TOTAL) +#if defined(FLASH_SECTOR_TOTAL) || defined(FLASH_SECTOR_NB) #if !defined(FLASH_DATA_SECTOR) +#if defined(FLASH_SECTOR_TOTAL) #define FLASH_DATA_SECTOR ((uint32_t)(FLASH_SECTOR_TOTAL - 1)) +#elif defined(FLASH_SECTOR_NB) +#define FLASH_DATA_SECTOR ((uint32_t)(FLASH_SECTOR_NB - 1)) +#endif #else #ifndef FLASH_BASE_ADDRESS #error "FLASH_BASE_ADDRESS have to be defined when FLASH_DATA_SECTOR is defined" #endif #endif /* !FLASH_DATA_SECTOR */ -#endif /* FLASH_SECTOR_TOTAL */ +#endif /* FLASH_SECTOR_TOTAL || FLASH_SECTOR_NB */ /* Be able to change FLASH_PAGE_NUMBER to use if relevant */ #if !defined(FLASH_PAGE_NUMBER) && defined(FLASH_PAGE_SIZE) @@ -62,6 +67,12 @@ extern "C" { #define FLASH_END FLASH_BANK2_END #elif defined (FLASH_BANK1_END) && (FLASH_BANK_NUMBER == FLASH_BANK_1) #define FLASH_END FLASH_BANK1_END +#elif defined(FLASH_DATA_SECTOR) +#if defined(FLASH_BANK_2) && (FLASH_BANK_NUMBER == FLASH_BANK_2) +#define FLASH_END ((uint32_t)(FLASH_BASE + FLASH_BANK_SIZE + (FLASH_DATA_SECTOR * FLASH_SECTOR_SIZE) + FLASH_SECTOR_SIZE - 1)) +#else +#define FLASH_END ((uint32_t)(FLASH_BASE + (FLASH_DATA_SECTOR * FLASH_SECTOR_SIZE) + FLASH_SECTOR_SIZE - 1)) +#endif /* FLASH_BANK_2 */ #elif defined(FLASH_BASE) && defined(FLASH_PAGE_NUMBER) && defined (FLASH_PAGE_SIZE) /* If FLASH_PAGE_NUMBER is defined by user, this is not really end of the flash */ #define FLASH_END ((uint32_t)(FLASH_BASE + (((FLASH_PAGE_NUMBER +1) * FLASH_PAGE_SIZE))-1)) @@ -164,7 +175,25 @@ void eeprom_buffered_write_byte(uint32_t pos, uint8_t value) */ void eeprom_buffer_fill(void) { +#if defined(ICACHE) && defined (HAL_ICACHE_MODULE_ENABLED) && !defined(HAL_ICACHE_MODULE_DISABLED) + bool icache_enabled = false; + if (HAL_ICACHE_IsEnabled() == 1) { + icache_enabled = true; + /* Disable instruction cache prior to internal cacheable memory update */ + if (HAL_ICACHE_Disable() != HAL_OK) { + Error_Handler(); + } + } +#endif /* ICACHE && HAL_ICACHE_MODULE_ENABLED && !HAL_ICACHE_MODULE_DISABLED */ memcpy(eeprom_buffer, (uint8_t *)(FLASH_BASE_ADDRESS), E2END + 1); +#if defined(ICACHE) && defined (HAL_ICACHE_MODULE_ENABLED) && !defined(HAL_ICACHE_MODULE_DISABLED) + if (icache_enabled) { + /* Re-enable instruction cache */ + if (HAL_ICACHE_Enable() != HAL_OK) { + Error_Handler(); + } + } +#endif /* ICACHE && HAL_ICACHE_MODULE_ENABLED && !HAL_ICACHE_MODULE_DISABLED */ } #if defined(EEPROM_RETRAM_MODE) @@ -188,6 +217,16 @@ void eeprom_buffer_flush(void) */ void eeprom_buffer_flush(void) { +#if defined(ICACHE) && defined (HAL_ICACHE_MODULE_ENABLED) && !defined(HAL_ICACHE_MODULE_DISABLED) + bool icache_enabled = false; + if (HAL_ICACHE_IsEnabled() == 1) { + icache_enabled = true; + /* Disable instruction cache prior to internal cacheable memory update */ + if (HAL_ICACHE_Disable() != HAL_OK) { + Error_Handler(); + } + } +#endif /* ICACHE && HAL_ICACHE_MODULE_ENABLED && !HAL_ICACHE_MODULE_DISABLED */ FLASH_EraseInitTypeDef EraseInitStruct; uint32_t offset = 0; uint32_t address = FLASH_BASE_ADDRESS; @@ -240,6 +279,8 @@ void eeprom_buffer_flush(void) uint32_t SectorError = 0; #if defined(FLASH_TYPEPROGRAM_FLASHWORD) uint64_t data[4] = {0x0000}; +#elif defined(FLASH_TYPEPROGRAM_QUADWORD) + uint32_t data[4] = {0x0000}; #else uint32_t data = 0; #endif @@ -249,7 +290,9 @@ void eeprom_buffer_flush(void) #if defined(FLASH_BANK_NUMBER) EraseInitStruct.Banks = FLASH_BANK_NUMBER; #endif +#if defined(FLASH_VOLTAGE_RANGE_3) EraseInitStruct.VoltageRange = FLASH_VOLTAGE_RANGE_3; +#endif EraseInitStruct.Sector = FLASH_DATA_SECTOR; EraseInitStruct.NbSectors = 1; @@ -263,11 +306,20 @@ void eeprom_buffer_flush(void) if (HAL_FLASH_Program(FLASH_TYPEPROGRAM_FLASHWORD, address, (uint32_t)data) == HAL_OK) { address += 32; offset += 32; -#else +#elif defined(FLASH_TYPEPROGRAM_QUADWORD) + /* 128 bits */ + memcpy(&data, eeprom_buffer + offset, 4 * sizeof(uint32_t)); + if (HAL_FLASH_Program(FLASH_TYPEPROGRAM_QUADWORD, address, (uint32_t)data) == HAL_OK) { + address += 16; + offset += 16; +#elif defined(FLASH_TYPEPROGRAM_WORD) memcpy(&data, eeprom_buffer + offset, sizeof(uint32_t)); if (HAL_FLASH_Program(FLASH_TYPEPROGRAM_WORD, address, data) == HAL_OK) { address += 4; offset += 4; +#else +#error "Unknown FLASH Program Type." + if (0) {} #endif } else { address = address_end + 1; @@ -276,6 +328,15 @@ void eeprom_buffer_flush(void) } HAL_FLASH_Lock(); #endif +#if defined(ICACHE) && defined (HAL_ICACHE_MODULE_ENABLED) && !defined(HAL_ICACHE_MODULE_DISABLED) + if (icache_enabled) + { + /* Re-enable instruction cache */ + if (HAL_ICACHE_Enable() != HAL_OK) { + Error_Handler(); + } + } +#endif /* ICACHE && HAL_ICACHE_MODULE_ENABLED && !HAL_ICACHE_MODULE_DISABLED */ } #endif /* defined(EEPROM_RETRAM_MODE) */ diff --git a/stm32/libraries/EEPROM/src/utility/stm32_eeprom.h b/stm32/libraries/EEPROM/src/utility/stm32_eeprom.h index c3acf605..fd768781 100644 --- a/stm32/libraries/EEPROM/src/utility/stm32_eeprom.h +++ b/stm32/libraries/EEPROM/src/utility/stm32_eeprom.h @@ -84,7 +84,7 @@ extern "C" { * emulation. Anyway, all the sector size will be erased. * So pay attention to not use this sector for other stuff. */ -#define FLASH_PAGE_SIZE ((uint32_t)(16*1024)) /* 16kB page */ +#define FLASH_PAGE_SIZE ((uint32_t)(8*1024)) /* 8kB page */ #endif #if defined(DATA_EEPROM_BASE) || defined(FLASH_EEPROM_BASE) diff --git a/stm32/libraries/KlangWellen/examples/01.Wavetable/01.Wavetable.ino b/stm32/libraries/KlangWellen/examples/01.Wavetable/01.Wavetable.ino index 72de4e3b..1a21d3d8 100644 --- a/stm32/libraries/KlangWellen/examples/01.Wavetable/01.Wavetable.ino +++ b/stm32/libraries/KlangWellen/examples/01.Wavetable/01.Wavetable.ino @@ -7,6 +7,11 @@ using namespace klangwellen; Wavetable fWavetable; void setup() { + Serial.begin(115200); + Serial.println("------------"); + Serial.println("01.Wavetable"); + Serial.println("------------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SAWTOOTH, 16); fWavetable.set_frequency(55); fWavetable.set_amplitude(0.5); @@ -19,4 +24,4 @@ void audioblock(float** input_signal, float** output_signal) { output_signal[LEFT][i] = fWavetable.process(); } KlangWellen::copy(output_signal[LEFT], output_signal[RIGHT]); -} \ No newline at end of file +} diff --git a/stm32/libraries/KlangWellen/examples/02.ADSR/02.ADSR.ino b/stm32/libraries/KlangWellen/examples/02.ADSR/02.ADSR.ino index adfe4cc7..67a16c88 100644 --- a/stm32/libraries/KlangWellen/examples/02.ADSR/02.ADSR.ino +++ b/stm32/libraries/KlangWellen/examples/02.ADSR/02.ADSR.ino @@ -9,6 +9,11 @@ Wavetable fWavetable; ADSR fADSR; void setup() { + Serial.begin(115200); + Serial.println("-------"); + Serial.println("02.ADSR"); + Serial.println("-------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SINE); } diff --git a/stm32/libraries/KlangWellen/examples/03.LowPassFilter/03.LowPassFilter.ino b/stm32/libraries/KlangWellen/examples/03.LowPassFilter/03.LowPassFilter.ino index 9cc9ac65..dc806717 100644 --- a/stm32/libraries/KlangWellen/examples/03.LowPassFilter/03.LowPassFilter.ino +++ b/stm32/libraries/KlangWellen/examples/03.LowPassFilter/03.LowPassFilter.ino @@ -8,6 +8,11 @@ Wavetable fWavetable; FilterLowPassMoogLadder fFilter; void setup() { + Serial.begin(115200); + Serial.println("----------------"); + Serial.println("03.LowPassFilter"); + Serial.println("----------------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SQUARE); fWavetable.set_frequency(55); fFilter.set_resonance(0.85f); diff --git a/stm32/libraries/KlangWellen/examples/04.Reverb/04.Reverb.ino b/stm32/libraries/KlangWellen/examples/04.Reverb/04.Reverb.ino index a9ee38c3..6561e9da 100644 --- a/stm32/libraries/KlangWellen/examples/04.Reverb/04.Reverb.ino +++ b/stm32/libraries/KlangWellen/examples/04.Reverb/04.Reverb.ino @@ -11,6 +11,11 @@ ADSR fADSR; Reverb fReverb; void setup() { + Serial.begin(115200); + Serial.println("---------"); + Serial.println("04.Reverb"); + Serial.println("---------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SINE); fReverb.set_roomsize(0.9); } diff --git a/stm32/libraries/KlangWellen/examples/05.FormantFilter/05.FormantFilter.ino b/stm32/libraries/KlangWellen/examples/05.FormantFilter/05.FormantFilter.ino index 0af62dd1..5e58a2df 100644 --- a/stm32/libraries/KlangWellen/examples/05.FormantFilter/05.FormantFilter.ino +++ b/stm32/libraries/KlangWellen/examples/05.FormantFilter/05.FormantFilter.ino @@ -8,6 +8,11 @@ Wavetable fWavetable; FilterVowelFormant fFilter; void setup() { + Serial.begin(115200); + Serial.println("----------------"); + Serial.println("05.FormantFilter"); + Serial.println("----------------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SAWTOOTH); fWavetable.set_frequency(35); fWavetable.set_amplitude(0.125f); diff --git a/stm32/libraries/KlangWellen/examples/06.SAM/06.SAM.ino b/stm32/libraries/KlangWellen/examples/06.SAM/06.SAM.ino index 0505aa7f..3c96c193 100644 --- a/stm32/libraries/KlangWellen/examples/06.SAM/06.SAM.ino +++ b/stm32/libraries/KlangWellen/examples/06.SAM/06.SAM.ino @@ -10,6 +10,11 @@ SAM fSAM_left(fBuffer, 48000); SAM fSAM_right(48000); void setup() { + Serial.begin(115200); + Serial.println("------"); + Serial.println("06.SAM"); + Serial.println("------"); + fSAM_right.set_speed(120); fSAM_right.set_throat(100); beats_per_minute(60); diff --git a/stm32/libraries/KlangWellen/examples/07.Sampler/07.Sampler.ino b/stm32/libraries/KlangWellen/examples/07.Sampler/07.Sampler.ino index 90ddc012..0dca2a05 100644 --- a/stm32/libraries/KlangWellen/examples/07.Sampler/07.Sampler.ino +++ b/stm32/libraries/KlangWellen/examples/07.Sampler/07.Sampler.ino @@ -6,8 +6,13 @@ using namespace klangwellen; Sampler fSampler{KlangWellen::DEFAULT_SAMPLING_RATE / 8}; void setup() { - for (size_t i = 0; i < fSampler.get_buffer_length(); i++) { - float ratio = 1.0 - (float)i / fSampler.get_buffer_length(); + Serial.begin(115200); + Serial.println("----------"); + Serial.println("07.Sampler"); + Serial.println("----------"); + + for (int32_t i = 0; i < fSampler.get_buffer_length(); i++) { + float ratio = 1.0 - (float)i / fSampler.get_buffer_length(); // fSampler.get_buffer()[i] = KlangWellen::random() * 0.2f * ratio * 127; // for SamplerI8 fSampler.get_buffer()[i] = KlangWellen::random() * 0.2f * ratio; } diff --git a/stm32/libraries/KlangWellen/examples/08.Delay/08.Delay.ino b/stm32/libraries/KlangWellen/examples/08.Delay/08.Delay.ino index 49a3abc4..e92e9746 100644 --- a/stm32/libraries/KlangWellen/examples/08.Delay/08.Delay.ino +++ b/stm32/libraries/KlangWellen/examples/08.Delay/08.Delay.ino @@ -1,3 +1,5 @@ +// @TODO Delay is broken, fix it! + #include "ADSR.h" #include "Delay.h" #include "KlangWellen.h" @@ -12,6 +14,11 @@ ADSR fADSR; Delay fDelay; void setup() { + Serial.begin(115200); + Serial.println("--------"); + Serial.println("08.Delay"); + Serial.println("--------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_TRIANGLE); fWavetable.set_amplitude(0.4); fDelay.set_echo_length(0.05); @@ -34,7 +41,7 @@ void beat(uint32_t beat_counter) { } void audioblock(float** input_signal, float** output_signal) { - // /* process as float ( mono, single sample ) ... */ + /* process as float ( mono, single sample ) ... */ // for (uint16_t i = 0; i < KLANG_SAMPLES_PER_AUDIO_BLOCK; i++) { // output_signal[LEFT][i] = fDelay.process(fADSR.process(fWavetable.process())); // output_signal[RIGHT][i] = output_signal[LEFT][i]; diff --git a/stm32/libraries/KlangWellen/examples/09.LFO/09.LFO.ino b/stm32/libraries/KlangWellen/examples/09.LFO/09.LFO.ino index 91599add..72d111c5 100644 --- a/stm32/libraries/KlangWellen/examples/09.LFO/09.LFO.ino +++ b/stm32/libraries/KlangWellen/examples/09.LFO/09.LFO.ino @@ -9,6 +9,11 @@ Wavetable fWavetable; FilterLowPassMoogLadder fFilter; void setup() { + Serial.begin(115200); + Serial.println("------"); + Serial.println("09.LFO"); + Serial.println("------"); + fLFO.set_waveform(KlangWellen::WAVEFORM_SINE); fLFO.set_oscillation_range(55, 1000); fLFO.set_frequency(0.5); diff --git a/stm32/libraries/KlangWellen/examples/10.BeatDSP/10.BeatDSP.ino b/stm32/libraries/KlangWellen/examples/10.BeatDSP/10.BeatDSP.ino index 8dff8030..4072cc9c 100644 --- a/stm32/libraries/KlangWellen/examples/10.BeatDSP/10.BeatDSP.ino +++ b/stm32/libraries/KlangWellen/examples/10.BeatDSP/10.BeatDSP.ino @@ -13,6 +13,11 @@ ADSR fADSR_DSP; BeatDSP fBeat; void setup() { + Serial.begin(115200); + Serial.println("----------"); + Serial.println("10.BeatDSP"); + Serial.println("----------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SINE); fWavetable.set_frequency(220.0); fWavetable_DSP.set_waveform(KlangWellen::WAVEFORM_TRIANGLE); diff --git a/stm32/libraries/KlangWellen/examples/11.Trigger/11.Trigger.ino b/stm32/libraries/KlangWellen/examples/11.Trigger/11.Trigger.ino index 2840b9c6..4cf5a49c 100644 --- a/stm32/libraries/KlangWellen/examples/11.Trigger/11.Trigger.ino +++ b/stm32/libraries/KlangWellen/examples/11.Trigger/11.Trigger.ino @@ -14,6 +14,11 @@ ADSR fADSR_DSP; Trigger fTrigger; void setup() { + Serial.begin(115200); + Serial.println("----------"); + Serial.println("11.Trigger"); + Serial.println("----------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SINE); fWavetable.set_frequency(220.0); fWavetable_DSP.set_waveform(KlangWellen::WAVEFORM_TRIANGLE); diff --git a/stm32/libraries/KlangWellen/examples/12.Clamp/12.Clamp.ino b/stm32/libraries/KlangWellen/examples/12.Clamp/12.Clamp.ino index 60de9f8f..a6953541 100644 --- a/stm32/libraries/KlangWellen/examples/12.Clamp/12.Clamp.ino +++ b/stm32/libraries/KlangWellen/examples/12.Clamp/12.Clamp.ino @@ -11,6 +11,11 @@ ADSR fADSR; Clamp fClamp; void setup() { + Serial.begin(115200); + Serial.println("--------"); + Serial.println("12.Clamp"); + Serial.println("--------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SINE); fClamp.set_min(-0.1); fClamp.set_max(0.1); diff --git a/stm32/libraries/KlangWellen/examples/13.Vocoder/13.Vocoder.ino b/stm32/libraries/KlangWellen/examples/13.Vocoder/13.Vocoder.ino index ac45a65c..446c7e6f 100644 --- a/stm32/libraries/KlangWellen/examples/13.Vocoder/13.Vocoder.ino +++ b/stm32/libraries/KlangWellen/examples/13.Vocoder/13.Vocoder.ino @@ -9,9 +9,14 @@ using namespace klangwellen; SAM fSAM(48000); Wavetable fWavetable; -Vocoder fVocoder{13, 3}; // this still works on KLST_SHEEP but only with `fastest` compiler option +Vocoder fVocoder{13, 3}; // this still works on KLST_SHEEP but only with `fastest` compiler option void setup() { + Serial.begin(115200); + Serial.println("----------"); + Serial.println("13.Vocoder"); + Serial.println("----------"); + fSAM.speak("hello world"); fWavetable.set_waveform(KlangWellen::WAVEFORM_SAWTOOTH); diff --git a/stm32/libraries/KlangWellen/examples/14.Filters/14.Filters.ino b/stm32/libraries/KlangWellen/examples/14.Filters/14.Filters.ino index 5401c11d..d2ffe693 100644 --- a/stm32/libraries/KlangWellen/examples/14.Filters/14.Filters.ino +++ b/stm32/libraries/KlangWellen/examples/14.Filters/14.Filters.ino @@ -8,6 +8,11 @@ Wavetable fWavetable; Filter fFilter; void setup() { + Serial.begin(115200); + Serial.println("----------"); + Serial.println("14.Filters"); + Serial.println("----------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SQUARE); fWavetable.set_frequency(55); klangstrom::beats_per_minute(120 * 4); diff --git a/stm32/libraries/KlangWellen/examples/15.Ramp/15.Ramp.ino b/stm32/libraries/KlangWellen/examples/15.Ramp/15.Ramp.ino index db8df4e4..a5fb8ace 100644 --- a/stm32/libraries/KlangWellen/examples/15.Ramp/15.Ramp.ino +++ b/stm32/libraries/KlangWellen/examples/15.Ramp/15.Ramp.ino @@ -12,6 +12,11 @@ Ramp fRampFilterFrequency; Ramp fRampFrequency; void setup() { + Serial.begin(115200); + Serial.println("-------"); + Serial.println("15.Ramp"); + Serial.println("-------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SAWTOOTH); fWavetable.set_frequency(27.5); fFilter.set_resonance(0.3); diff --git a/stm32/libraries/KlangWellen/examples/16.Envelope/16.Envelope.ino b/stm32/libraries/KlangWellen/examples/16.Envelope/16.Envelope.ino index a8d396c8..e904bd24 100644 --- a/stm32/libraries/KlangWellen/examples/16.Envelope/16.Envelope.ino +++ b/stm32/libraries/KlangWellen/examples/16.Envelope/16.Envelope.ino @@ -10,6 +10,11 @@ Envelope fFrequencyEnvelope; Envelope fAmplitudeEnvelope; void setup() { + Serial.begin(115200); + Serial.println("-----------"); + Serial.println("16.Envelope"); + Serial.println("-----------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SAWTOOTH); fWavetable.set_frequency(55.0); fWavetable.set_amplitude(0.0); diff --git a/stm32/libraries/KlangWellen/examples/17.Waveshaper/17.Waveshaper.ino b/stm32/libraries/KlangWellen/examples/17.Waveshaper/17.Waveshaper.ino index 59036be8..4b2e2693 100644 --- a/stm32/libraries/KlangWellen/examples/17.Waveshaper/17.Waveshaper.ino +++ b/stm32/libraries/KlangWellen/examples/17.Waveshaper/17.Waveshaper.ino @@ -9,6 +9,11 @@ Waveshaper fWaveshaper; uint8_t fWaveshaperType = Waveshaper::SIN; void setup() { + Serial.begin(115200); + Serial.println("-------------"); + Serial.println("17.Waveshaper"); + Serial.println("-------------"); + fWavetable.set_waveform(KlangWellen::WAVEFORM_SINE); fWavetable.set_amplitude(4.0); fWavetable.set_frequency(55); diff --git a/stm32/libraries/KlangWellen/src/ADSR.h b/stm32/libraries/KlangWellen/src/ADSR.h index 532a6565..f755b56f 100644 --- a/stm32/libraries/KlangWellen/src/ADSR.h +++ b/stm32/libraries/KlangWellen/src/ADSR.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/stm32/libraries/KlangWellen/src/BeatDSP.h b/stm32/libraries/KlangWellen/src/BeatDSP.h index 2b70c730..1240e7a3 100644 --- a/stm32/libraries/KlangWellen/src/BeatDSP.h +++ b/stm32/libraries/KlangWellen/src/BeatDSP.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/stm32/libraries/KlangWellen/src/Clamp.h b/stm32/libraries/KlangWellen/src/Clamp.h index e7ee8d90..874cf422 100644 --- a/stm32/libraries/KlangWellen/src/Clamp.h +++ b/stm32/libraries/KlangWellen/src/Clamp.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/stm32/libraries/KlangWellen/src/Delay.h b/stm32/libraries/KlangWellen/src/Delay.h index b0eaab96..d2b1c822 100644 --- a/stm32/libraries/KlangWellen/src/Delay.h +++ b/stm32/libraries/KlangWellen/src/Delay.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -27,6 +27,8 @@ * - [ ] void process(float*, float*, uint32_t) */ +// @TODO Delay is broken, fix it + #pragma once #include @@ -46,8 +48,7 @@ namespace klangwellen { * @param sample_rate the sample rate in Hz. * @param decay_rate the decay of the echo, a value between 0 and 1. 1 meaning no decay, 0 means immediate decay */ - Delay(float echo_length = 0.5, float decay_rate = 0.75, float wet = 0.8, uint32_t sample_rate = KlangWellen::DEFAULT_SAMPLING_RATE) { - fSampleRate = sample_rate; + Delay(float echo_length = 0.5, float decay_rate = 0.75, float wet = 0.8, uint32_t sample_rate = KlangWellen::DEFAULT_SAMPLING_RATE) : fSampleRate(sample_rate) { set_decay_rate(decay_rate); set_echo_length(echo_length); set_wet(wet); @@ -105,14 +106,14 @@ namespace klangwellen { } private: - int32_t fBufferPosition; - float fDecayRate; - float fWet; - float* fBuffer; - int32_t fBufferLength; - float fNewEchoLength; + int32_t fBufferPosition = 0; + float fDecayRate = 0; + float fWet = 0; + float* fBuffer = nullptr; + bool fAllocatedBuffer = false; + int32_t fBufferLength = 0; + float fNewEchoLength = 0; uint32_t fSampleRate; - bool fAllocatedBuffer; void adaptEchoLength() { if (fNewEchoLength > 0) { diff --git a/stm32/libraries/KlangWellen/src/Envelope.h b/stm32/libraries/KlangWellen/src/Envelope.h index c98ef281..49c4c3a0 100644 --- a/stm32/libraries/KlangWellen/src/Envelope.h +++ b/stm32/libraries/KlangWellen/src/Envelope.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -130,14 +130,15 @@ namespace klangwellen { */ float process() { if (!mEnvelopeDone) { - if (mEnvStage < mEnvelopeStages.size()) { + const int mNumberOfStages = mEnvelopeStages.size(); + if (mEnvStage < mNumberOfStages) { mValue += mTimeScale * mDelta; mStageDuration += mTimeScale * 1.0f / mSamplingRate; if (mStageDuration > mEnvelopeStages[mEnvStage].duration) { const float mRemainder = mStageDuration - mEnvelopeStages[mEnvStage].duration; finished_stage(mEnvStage); mEnvStage++; - if (mEnvStage < mEnvelopeStages.size() - 1) { + if (mEnvStage < mNumberOfStages - 1) { prepareNextStage(mEnvStage, mRemainder); } else { stop(); diff --git a/stm32/libraries/KlangWellen/src/Filter.h b/stm32/libraries/KlangWellen/src/Filter.h index ee76426b..c6625f1a 100644 --- a/stm32/libraries/KlangWellen/src/Filter.h +++ b/stm32/libraries/KlangWellen/src/Filter.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://githubiquad_com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/stm32/libraries/KlangWellen/src/FilterLowPassMoogLadder.h b/stm32/libraries/KlangWellen/src/FilterLowPassMoogLadder.h index 0cbe3724..b7c177c7 100644 --- a/stm32/libraries/KlangWellen/src/FilterLowPassMoogLadder.h +++ b/stm32/libraries/KlangWellen/src/FilterLowPassMoogLadder.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/stm32/libraries/KlangWellen/src/FilterVowelFormant.h b/stm32/libraries/KlangWellen/src/FilterVowelFormant.h index 976b5651..001391f4 100644 --- a/stm32/libraries/KlangWellen/src/FilterVowelFormant.h +++ b/stm32/libraries/KlangWellen/src/FilterVowelFormant.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/stm32/libraries/KlangWellen/src/Gain.h b/stm32/libraries/KlangWellen/src/Gain.h index 4d5f55ba..95eb0d16 100644 --- a/stm32/libraries/KlangWellen/src/Gain.h +++ b/stm32/libraries/KlangWellen/src/Gain.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/stm32/libraries/KlangWellen/src/KlangWellen.cpp b/stm32/libraries/KlangWellen/src/KlangWellen.cpp deleted file mode 100644 index 1e4c4ec4..00000000 --- a/stm32/libraries/KlangWellen/src/KlangWellen.cpp +++ /dev/null @@ -1,31 +0,0 @@ -#include "KlangWellen.h" - -/* xorshift32 ( ref: https://en.wikipedia.org/wiki/Xorshift ) */ -uint32_t klangwellen::KlangWellen::x32Seed = 23; -uint32_t klangwellen::KlangWellen::xorshift32() { - x32Seed ^= x32Seed << 13; - x32Seed ^= x32Seed >> 17; - x32Seed ^= x32Seed << 5; - return x32Seed; -} - -/** - * returns a random number between 0 ... 1 - */ -float klangwellen::KlangWellen::random_normalized() { - // TODO replace with mapping, without division - return ((float)xorshift32() / UINT32_MAX); -} - -float klangwellen::KlangWellen::random() { - // TODO replace with mapping, without division - return ((float)xorshift32() / UINT32_MAX) * 2 - 1; -} - -uint32_t klangwellen::KlangWellen::millis_to_samples(float pMillis, float pSamplingRate) { - return (uint32_t)(pMillis / 1000.0f * pSamplingRate); -} - -uint32_t klangwellen::KlangWellen::millis_to_samples(float pMillis) { - return (uint32_t)(pMillis / 1000.0f * (float)klangwellen::KlangWellen::DEFAULT_SAMPLING_RATE); -} diff --git a/stm32/libraries/KlangWellen/src/KlangWellen.h b/stm32/libraries/KlangWellen/src/KlangWellen.h index 41ace193..9d384dae 100755 --- a/stm32/libraries/KlangWellen/src/KlangWellen.h +++ b/stm32/libraries/KlangWellen/src/KlangWellen.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -30,6 +30,11 @@ #define KLANG_SAMPLES_PER_AUDIO_BLOCK 512 #endif +#ifndef KLANG_SAMPLING_RATE +#warning KLANG_SAMPLING_RATE not defined - using default value of 48000 +#define KLANG_SAMPLING_RATE 48000 +#endif + #ifndef PI #define PI M_PI #endif @@ -45,238 +50,331 @@ namespace klangwellen { class KlangWellen { public: - static const uint8_t BITS_PER_SAMPLE_16 = 16; - static const uint8_t BITS_PER_SAMPLE_24 = 24; - static const uint8_t BITS_PER_SAMPLE_32 = 32; - static const uint8_t BITS_PER_SAMPLE_8 = 8; - static constexpr float DEFAULT_ATTACK = 0.005f; - static const int DEFAULT_AUDIOBLOCK_SIZE = 1024; - static const int DEFAULT_AUDIO_DEVICE = -1; - static const uint8_t DEFAULT_BITS_PER_SAMPLE = BITS_PER_SAMPLE_16; - static constexpr float DEFAULT_DECAY = 0.01f; - static constexpr float DEFAULT_FILTER_BANDWIDTH = 100.0f; - static constexpr float DEFAULT_FILTER_FREQUENCY = 1000.0f; - static constexpr float DEFAULT_RELEASE = 0.075f; - static constexpr uint32_t DEFAULT_SAMPLING_RATE = 48000; - static constexpr uint32_t DEFAULT_INTERPOLATE_AMP_FREQ_DURATION = 5.f / 1000.f * (float)DEFAULT_SAMPLING_RATE; // KlangWellen::millis_to_samples(5); - static constexpr float DEFAULT_SUSTAIN = 0.5f; - static const int DEFAULT_WAVETABLE_SIZE = 512; - static const uint8_t DISTORTION_HARD_CLIPPING = 0; - static const uint8_t DISTORTION_FOLDBACK = 1; - static const uint8_t DISTORTION_FOLDBACK_SINGLE = 2; - static const uint8_t DISTORTION_FULL_WAVE_RECTIFICATION = 3; - static const uint8_t DISTORTION_HALF_WAVE_RECTIFICATION = 4; - static const uint8_t DISTORTION_INFINITE_CLIPPING = 5; - static const uint8_t DISTORTION_SOFT_CLIPPING_CUBIC = 6; - static const uint8_t DISTORTION_SOFT_CLIPPING_ARC_TANGENT = 7; - static const uint8_t DISTORTION_BIT_CRUSHING = 8; - static const int EVENT_UNDEFINED = -1; - static const int EVENT_CHANNEL = 0; - static const int EVENT_NOTE_ON = 0; - static const int EVENT_NOTE = 1; - static const int EVENT_NOTE_OFF = 1; - static const int EVENT_CONTROLCHANGE = 2; - static const int EVENT_PITCHBEND = 3; - static const int EVENT_PROGRAMCHANGE = 4; - static const int EVENT_VELOCITY = 2; - static const uint8_t FILTER_MODE_LOW_PASS = 0; - static const uint8_t FILTER_MODE_HIGH_PASS = 1; - static const uint8_t FILTER_MODE_BAND_PASS = 2; - static const uint8_t FILTER_MODE_NOTCH = 3; - static const uint8_t FILTER_MODE_PEAK = 4; - static const uint8_t FILTER_MODE_LOWSHELF = 5; - static const uint8_t FILTER_MODE_HIGHSHELF = 6; - static const uint8_t FILTER_MODE_BAND_REJECT = 7; - static const int LOOP_INFINITE = std::numeric_limits::max(); - static const uint8_t MONO = 1; - static const uint8_t NOISE_WHITE = 0; - static const uint8_t NOISE_GAUSSIAN_WHITE = 1; - static const uint8_t NOISE_GAUSSIAN_WHITE2 = 2; - static const uint8_t NOISE_PINK = 3; - static const uint8_t NOISE_PINK2 = 4; - static const uint8_t NOISE_PINK3 = 5; - static const uint8_t NOISE_SIMPLEX = 6; - static constexpr float NOTE_WHOLE = 0.25f; - static constexpr float NOTE_HALF = 0.5f; - static const uint8_t NOTE_QUARTER = 1; - static const uint8_t NOTE_EIGHTH = 2; - static const uint8_t NOTE_SIXTEENTH = 4; - static const uint8_t NOTE_THIRTYSECOND = 8; - static const int NO_AUDIO_DEVICE = -2; - static const int NO_CHANNELS = 0; - static const int NO_EVENT = -1; - static const int NO_INPOINT = 0; - static const int NO_LOOP = -2; - static const int NO_LOOP_COUNT = -1; - static const int NO_OUTPOINT = -1; - static const int NO_POSITION = -1; - static const int NO_VALUE = -1; - static const uint8_t PAN_LINEAR = 0; - static const uint8_t PAN_SINE_LAW = 2; - static const uint8_t PAN_SQUARE_LAW = 1; - static const uint8_t SIGNAL_LEFT = 0; - static constexpr float SIGNAL_MAX = 1.0f; - static constexpr float SIGNAL_MIN = -1.0f; - static const int SIGNAL_MONO = 1; - static const int SIGNAL_PROCESSING_IGNORE_IN_OUTPOINTS = -3; - static const int SIGNAL_RIGHT = 1; - static const int SIGNAL_STEREO = 2; - static const uint8_t SIG_INT16_BIG_ENDIAN = 2; - static const uint8_t SIG_INT16_LITTLE_ENDIAN = 3; - static const uint8_t SIG_INT24_3_BIG_ENDIAN = 4; - static const uint8_t SIG_INT24_3_LITTLE_ENDIAN = 5; - static const uint8_t SIG_INT24_4_BIG_ENDIAN = 6; - static const uint8_t SIG_INT24_4_LITTLE_ENDIAN = 7; - static const uint8_t SIG_INT32_BIG_ENDIAN = 8; - static const uint8_t SIG_INT32_LITTLE_ENDIAN = 9; - static const uint8_t SIG_INT8 = 0; - static const uint8_t SIG_UINT8 = 1; - static const uint8_t STEREO = 2; - static const uint8_t VERSION_MAJOR = 0; - static const uint8_t VERSION_MINOR = 8; - static const uint8_t WAVEFORM_SINE = 0; - static const uint8_t WAVEFORM_TRIANGLE = 1; - static const uint8_t WAVEFORM_SAWTOOTH = 2; - static const uint8_t WAVEFORM_SQUARE = 3; - static const uint8_t WAVEFORM_NOISE = 4; - static const uint8_t WAVESHAPE_INTERPOLATE_NONE = 0; - static const uint8_t WAVESHAPE_INTERPOLATE_LINEAR = 1; - static const uint8_t WAVESHAPE_INTERPOLATE_CUBIC = 2; - static const uint8_t WAV_FORMAT_PCM = 1; - static const uint8_t WAV_FORMAT_IEEE_FLOAT_32BIT = 3; + static const uint8_t BITS_PER_SAMPLE_16 = 16; + static const uint8_t BITS_PER_SAMPLE_24 = 24; + static const uint8_t BITS_PER_SAMPLE_32 = 32; + static const uint8_t BITS_PER_SAMPLE_8 = 8; + static constexpr float DEFAULT_ATTACK = 0.005f; + static const int DEFAULT_AUDIOBLOCK_SIZE = KLANG_SAMPLES_PER_AUDIO_BLOCK; // TODO decide for either one + static const int DEFAULT_AUDIO_DEVICE = -1; + static const uint8_t DEFAULT_BITS_PER_SAMPLE = BITS_PER_SAMPLE_16; + static constexpr float DEFAULT_DECAY = 0.01f; + static constexpr float DEFAULT_FILTER_BANDWIDTH = 100.0f; + static constexpr float DEFAULT_FILTER_FREQUENCY = 1000.0f; + static constexpr float DEFAULT_RELEASE = 0.075f; + static constexpr uint32_t DEFAULT_SAMPLING_RATE = KLANG_SAMPLING_RATE; // TODO decide for either one + static constexpr uint32_t DEFAULT_INTERPOLATE_AMP_FREQ_DURATION = 5.f / 1000.f * (float) DEFAULT_SAMPLING_RATE; + // KlangWellen::millis_to_samples(5); + static constexpr float DEFAULT_SUSTAIN = 0.5f; + static const int DEFAULT_WAVETABLE_SIZE = 512; + static const uint8_t DISTORTION_HARD_CLIPPING = 0; + static const uint8_t DISTORTION_FOLDBACK = 1; + static const uint8_t DISTORTION_FOLDBACK_SINGLE = 2; + static const uint8_t DISTORTION_FULL_WAVE_RECTIFICATION = 3; + static const uint8_t DISTORTION_HALF_WAVE_RECTIFICATION = 4; + static const uint8_t DISTORTION_INFINITE_CLIPPING = 5; + static const uint8_t DISTORTION_SOFT_CLIPPING_CUBIC = 6; + static const uint8_t DISTORTION_SOFT_CLIPPING_ARC_TANGENT = 7; + static const uint8_t DISTORTION_BIT_CRUSHING = 8; + static const int EVENT_UNDEFINED = -1; + static const int EVENT_CHANNEL = 0; + static const int EVENT_NOTE_ON = 0; + static const int EVENT_NOTE = 1; + static const int EVENT_NOTE_OFF = 1; + static const int EVENT_CONTROLCHANGE = 2; + static const int EVENT_PITCHBEND = 3; + static const int EVENT_PROGRAMCHANGE = 4; + static const int EVENT_VELOCITY = 2; + static const uint8_t FILTER_MODE_LOW_PASS = 0; + static const uint8_t FILTER_MODE_HIGH_PASS = 1; + static const uint8_t FILTER_MODE_BAND_PASS = 2; + static const uint8_t FILTER_MODE_NOTCH = 3; + static const uint8_t FILTER_MODE_PEAK = 4; + static const uint8_t FILTER_MODE_LOWSHELF = 5; + static const uint8_t FILTER_MODE_HIGHSHELF = 6; + static const uint8_t FILTER_MODE_BAND_REJECT = 7; + static const int LOOP_INFINITE = std::numeric_limits::max(); + static const uint8_t MONO = 1; + static const uint8_t NOISE_WHITE = 0; + static const uint8_t NOISE_GAUSSIAN_WHITE = 1; + static const uint8_t NOISE_GAUSSIAN_WHITE2 = 2; + static const uint8_t NOISE_PINK = 3; + static const uint8_t NOISE_PINK2 = 4; + static const uint8_t NOISE_PINK3 = 5; + static const uint8_t NOISE_SIMPLEX = 6; + static constexpr float NOTE_WHOLE = 0.25f; + static constexpr float NOTE_HALF = 0.5f; + static const uint8_t NOTE_QUARTER = 1; + static const uint8_t NOTE_EIGHTH = 2; + static const uint8_t NOTE_SIXTEENTH = 4; + static const uint8_t NOTE_THIRTYSECOND = 8; + static const int NO_AUDIO_DEVICE = -2; + static const int NO_CHANNELS = 0; + static const int NO_EVENT = -1; + static const int NO_INPOINT = 0; + static const int NO_LOOP = -2; + static const int NO_LOOP_COUNT = -1; + static const int NO_OUTPOINT = -1; + static const int NO_POSITION = -1; + static const int NO_VALUE = -1; + static const uint8_t PAN_LINEAR = 0; + static const uint8_t PAN_SINE_LAW = 2; + static const uint8_t PAN_SQUARE_LAW = 1; + static const uint8_t SIGNAL_LEFT = 0; + static constexpr float SIGNAL_MAX = 1.0f; + static constexpr float SIGNAL_MIN = -1.0f; + static const int SIGNAL_MONO = 1; + static const int SIGNAL_PROCESSING_IGNORE_IN_OUTPOINTS = -3; + static const int SIGNAL_RIGHT = 1; + static const int SIGNAL_STEREO = 2; + static const uint8_t SIG_INT16_BIG_ENDIAN = 2; + static const uint8_t SIG_INT16_LITTLE_ENDIAN = 3; + static const uint8_t SIG_INT24_3_BIG_ENDIAN = 4; + static const uint8_t SIG_INT24_3_LITTLE_ENDIAN = 5; + static const uint8_t SIG_INT24_4_BIG_ENDIAN = 6; + static const uint8_t SIG_INT24_4_LITTLE_ENDIAN = 7; + static const uint8_t SIG_INT32_BIG_ENDIAN = 8; + static const uint8_t SIG_INT32_LITTLE_ENDIAN = 9; + static const uint8_t SIG_INT8 = 0; + static const uint8_t SIG_UINT8 = 1; + static const uint8_t STEREO = 2; + static const uint8_t VERSION_MAJOR = 0; + static const uint8_t VERSION_MINOR = 8; + static const uint8_t WAVEFORM_SINE = 0; + static const uint8_t WAVEFORM_TRIANGLE = 1; + static const uint8_t WAVEFORM_SAWTOOTH = 2; + static const uint8_t WAVEFORM_SQUARE = 3; + static const uint8_t WAVEFORM_NOISE = 4; + static const uint8_t WAVESHAPE_INTERPOLATE_NONE = 0; + static const uint8_t WAVESHAPE_INTERPOLATE_LINEAR = 1; + static const uint8_t WAVESHAPE_INTERPOLATE_CUBIC = 2; + static const uint8_t WAV_FORMAT_PCM = 1; + static const uint8_t WAV_FORMAT_IEEE_FLOAT_32BIT = 3; /* --- sound --- */ static constexpr float MIDI_NOTE_CONVERSION_BASE_FREQUENCY = 440.0; static const uint8_t NOTE_OFFSET = 69; - static float note_to_frequency(uint8_t pMidiNote) { + + static float note_to_frequency(uint8_t pMidiNote) { return MIDI_NOTE_CONVERSION_BASE_FREQUENCY * pow(2, ((pMidiNote - NOTE_OFFSET) / 12.0)); } + static void normalize(float *buffer, const uint32_t numSamples) { + if (buffer == nullptr || numSamples == 0) { + return; + } + + float peakValue = 0.0f; + + // find the peak value in the buffer + for (uint32_t i = 0; i < numSamples; i++) { + const float sample = std::abs(buffer[i]); + if (sample > peakValue) { + peakValue = sample; + } + } + + // avoid division by zero + if (peakValue == 0.0f) { + return; + } + + const float normalizationFactor = 1.0f / peakValue; + + // normalize the buffer + for (uint32_t i = 0; i < numSamples; i++) { + buffer[i] *= normalizationFactor; + } + } + + static void peak(float *buffer, const uint32_t length, float &min, float &max) { + if (buffer == nullptr || length == 0) { + return; + } + + min = 0.0f; + max = 0.0f; + + for (uint32_t i = 0; i < length; i++) { + const float sample = buffer[i]; + if (sample < min) { + min = sample; + } + if (sample > max) { + max = sample; + } + } + } + /* --- math --- */ - static uint32_t millis_to_samples(float pMillis, float pSamplingRate); - static uint32_t millis_to_samples(float pMillis); - static float random_normalized(); - static uint32_t x32Seed; - static uint32_t xorshift32(); - static float random(); + // static uint32_t millis_to_samples(float pMillis, float pSamplingRate); + // static uint32_t millis_to_samples(float pMillis); + // static float random_normalized(); + // static uint32_t x32Seed; + // static uint32_t xorshift32(); + // static float random(); + inline static uint32_t x32Seed = 23; + + static uint32_t millis_to_samples(const float pMillis, const float pSamplingRate) { + return static_cast(pMillis / 1000.0f * pSamplingRate); + } + + static uint32_t millis_to_samples(const float pMillis) { + return static_cast(pMillis / 1000.0f * static_cast(DEFAULT_SAMPLING_RATE)); + } + + /* xorshift32 ( ref: https://en.wikipedia.org/wiki/Xorshift ) */ + // static uint32_t x32Seed = 23; + static uint32_t xorshift32() { + x32Seed ^= x32Seed << 13; + x32Seed ^= x32Seed >> 17; + x32Seed ^= x32Seed << 5; + return x32Seed; + } + + /** + * returns a random number between 0 ... 1 + */ + static float random_normalized() { + // TODO replace with mapping, without division + return ((float) xorshift32() / UINT32_MAX); + } + + static float random() { + // TODO replace with mapping, without division + return ((float) xorshift32() / UINT32_MAX) * 2 - 1; + } - inline static float clamp(float value, - float minimum, - float maximum) { + static float clamp(const float value, + const float minimum, + const float maximum) { return value > maximum ? maximum : (value < minimum ? minimum : value); } - inline static float clamp(float value) { + static float clip(const float value) { + return clamp(value); + } + + static float clamp(const float value) { return value > SIGNAL_MAX ? SIGNAL_MAX : (value < SIGNAL_MIN ? SIGNAL_MIN : value); } - inline static uint8_t clamp127(uint8_t value) { + static uint8_t clamp127(const uint8_t value) { return value > 127 ? 127 : value; } - template - inline static T clamp(T value, T minimum, T maximum) { + template + static T clamp(T value, T minimum, T maximum) { return value > maximum ? maximum : (value < minimum ? minimum : value); } - inline static float pow(float base, float exponent) { + static float pow(const float base, const float exponent) { return powf(base, exponent); } - inline static float lerp(float v0, float v1, float t) { + static float lerp(const float v0, const float v1, const float t) { return v0 + t * (v1 - v0); } - inline static float max(float a, float b) { + static float max(const float a, const float b) { return a > b ? a : b; } - inline static float min(float a, float b) { + static float min(const float a, const float b) { return a < b ? a : b; } - inline static float abs(float value) { + static float abs(const float value) { return value > 0 ? value : -value; } - template - inline static T abs(T value) { + template + static T abs(T value) { return value > 0 ? value : -value; } - template - inline static int sign(T val) { + template + static int sign(T val) { return (T(0) < val) - (val < T(0)); } - inline static float mod(float a, float b) { + static float mod(const float a, const float b) { return a >= b ? (a - b * int(a / b)) : a; // return a >= b ? fmodf(a, b) : a; // return input >= ceil ? input % ceil : input; // from https://stackoverflow.com/questions/33333363/built-in-mod-vs-custom-mod-function-improve-the-performance-of-modulus-op } - inline static float map(float value, - float inputMin, - float inputMax, - float outputMin, - float outputMax) { - return ((value - inputMin) / (inputMax - inputMin) * (outputMax - outputMin) + outputMin); - } - - inline static int16_t map_i16(int16_t value, - int16_t inputMin, - int16_t inputMax, - int16_t outputMin, - int16_t outputMax) { + static float map(const float value, + const float inputMin, + const float inputMax, + const float outputMin, + const float outputMax) { + const float a = value - inputMin; + const float b = inputMax - inputMin; + const float c = outputMax - outputMin; + const float d = a / b; + const float e = d * c; + return e + outputMin; + } + + static int16_t map_i16(const int16_t value, + const int16_t inputMin, + const int16_t inputMax, + const int16_t outputMin, + const int16_t outputMax) { return ((value - inputMin) * (outputMax - outputMin) / (inputMax - inputMin) + outputMin); } - inline static float sin(float r) { + static float sin(const float r) { return sinf(r); } - inline static float cos(float r) { + static float cos(const float r) { return cosf(r); } - inline static float sinh(float r) { + static float sinh(const float r) { return sinhf(r); } - inline static float cosh(float r) { + static float cosh(const float r) { return coshf(r); } - inline static float sqrt(float r) { + static float sqrt(const float r) { return sqrtf(r); } - inline static float fast_sqrt(float x) { - return 1.0 / fast_inv_sqrt(x); + static float fast_sqrt(const float x) { + return 1.0f / fast_inv_sqrt(x); } #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wstrict-aliasing" - inline static float fast_inv_sqrt(float x) { - float xhalf = 0.5f * x; - int i = *(int*)&x; - i = 0x5f3759df - (i >> 1); - x = *(float*)&i; +#pragma GCC diagnostic ignored "-Wuninitialized" + static float fast_inv_sqrt(float x) { + const float xhalf = 0.5f * x; + int i = *(int *) &x; + i = 0x5f3759df - (i >> 1); + x = *(float *) &i; return x * (1.5f - xhalf * x * x); } #pragma GCC diagnostic pop - inline static float fast_sin(float x) { - if (x < -PI) { - while (x < -PI) { - x += TWO_PI; + static constexpr float PIf = (float) PI; + static constexpr float TWO_PIf = (float) TWO_PI; + + static float fast_sin(float x) { + if (x < -PIf) { + while (x < -PIf) { + x += TWO_PIf; } - } else if (x > PI) { - while (x > PI) { - x -= TWO_PI; + } else if (x > PIf) { + while (x > PIf) { + x -= TWO_PIf; } } const float x2 = x * x; @@ -285,24 +383,24 @@ namespace klangwellen { return numerator / denominator; } - inline static float fast_sin3(const float X) { + static float fast_sin3(const float X) { // Valid on interval [-PI, PI] // Sine approximation using Bhaskara I technique discovered in 7th century. // https://en.wikipedia.org/wiki/Bh%C4%81skara_I const float AbsX = abs(X); - const float Numerator = 16.0f * X * (PI - AbsX); - const float Denominator = 5.0f * PI * PI - 4.0f * AbsX * (PI - AbsX); + const float Numerator = 16.0f * X * (PIf - AbsX); + const float Denominator = 5.0f * PIf * PIf - 4.0f * AbsX * (PIf - AbsX); return Numerator / Denominator; } - inline static float fast_cos(float x) { - if (x < -PI) { - while (x < -PI) { - x += TWO_PI; + static float fast_cos(float x) { + if (x < -PIf) { + while (x < -PIf) { + x += TWO_PIf; } - } else if (x > PI) { - while (x > PI) { - x -= TWO_PI; + } else if (x > PIf) { + while (x > PIf) { + x -= TWO_PIf; } } const float x2 = x * x; @@ -311,56 +409,109 @@ namespace klangwellen { return numerator / denominator; } - inline static float fast_sinh(float x) { - auto x2 = x * x; - auto numerator = -x * (11511339840 + x2 * (1640635920 + x2 * (52785432 + x2 * 479249))); - auto denominator = -11511339840 + x2 * (277920720 + x2 * (-3177720 + x2 * 18361)); + static float fast_sinh(const float x) { + const auto x2 = x * x; + const auto numerator = -x * (11511339840 + x2 * (1640635920 + x2 * (52785432 + x2 * 479249))); + const auto denominator = -11511339840 + x2 * (277920720 + x2 * (-3177720 + x2 * 18361)); return numerator / denominator; } - inline static float fast_cosh(float x) { - auto x2 = x * x; - auto numerator = -(39251520 + x2 * (18471600 + x2 * (1075032 + 14615 * x2))); - auto denominator = -39251520 + x2 * (1154160 + x2 * (-16632 + 127 * x2)); + static float fast_cosh(const float x) { + const auto x2 = x * x; + const auto numerator = -(39251520 + x2 * (18471600 + x2 * (1075032 + 14615 * x2))); + const auto denominator = -39251520 + x2 * (1154160 + x2 * (-16632 + 127 * x2)); return numerator / denominator; } - inline static float fast_tan(float x) { - auto x2 = x * x; - auto numerator = x * (-135135 + x2 * (17325 + x2 * (-378 + x2))); - auto denominator = -135135 + x2 * (62370 + x2 * (-3150 + 28 * x2)); + static float fast_tan(const float x) { + const auto x2 = x * x; + const auto numerator = x * (-135135 + x2 * (17325 + x2 * (-378 + x2))); + const auto denominator = -135135 + x2 * (62370 + x2 * (-3150 + 28 * x2)); return numerator / denominator; } - inline static float fast_tanh(float x) { - auto x2 = x * x; - auto numerator = x * (135135 + x2 * (17325 + x2 * (378 + x2))); - auto denominator = 135135 + x2 * (62370 + x2 * (3150 + 28 * x2)); + static float fast_tanh(const float x) { + const auto x2 = x * x; + const auto numerator = x * (135135 + x2 * (17325 + x2 * (378 + x2))); + const auto denominator = 135135 + x2 * (62370 + x2 * (3150 + 28 * x2)); return numerator / denominator; } - inline static float exp_j(float x) { - auto numerator = 1680 + x * (840 + x * (180 + x * (20 + x))); - auto denominator = 1680 + x * (-840 + x * (180 + x * (-20 + x))); + static float exp_j(const float x) { + const auto numerator = 1680 + x * (840 + x * (180 + x * (20 + x))); + const auto denominator = 1680 + x * (-840 + x * (180 + x * (-20 + x))); return numerator / denominator; } - inline static float fast_atan(float x) { - static const float PI_FOURTH = (PI / 4.0); - return PI_FOURTH * x - x * (fabs(x) - 1) * (0.2447 + 0.0663 * fabs(x)); + static float fast_atan(const float x) { + static constexpr float PI_FOURTH = (PIf / 4.0f); + return PI_FOURTH * x - x * (fabs(x) - 1) * (0.2447f + 0.0663f * fabs(x)); } - inline static float fast_atan2(float x) { - static const float PI_FOURTH = (PI / 4.0); - static const float A = 0.0776509570923569; - static const float B = -0.287434475393028; - static const float C = (PI_FOURTH - A - B); - float xx = x * x; + static float fast_atan2(float x) { + static constexpr float PI_FOURTH = (PIf / 4.0f); + static const float A = 0.0776509570923569; + static const float B = -0.287434475393028; + static const float C = (PI_FOURTH - A - B); + const float xx = x * x; return ((A * xx + B) * xx + C) * x; } + /* --- interpolation --- */ + + static float linear_interpolate(const float y1, const float y2, const float mu) { + return y1 * (1.0f - mu) + y2 * mu; + } + + static float interpolate_samples_linear(const float *buffer, const size_t bufferSize, const float position) { + const int posInt = static_cast(position); + const float mu = position - posInt; + + // Ensure that the positions are valid + const int y1Pos = posInt; + const int y2Pos = (posInt + 1 >= bufferSize) ? bufferSize - 1 : posInt + 1; + + // Get the samples for interpolation + const float y1 = buffer[y1Pos]; + const float y2 = buffer[y2Pos]; + + // Return the interpolated sample + return linear_interpolate(y1, y2, mu); + } + + static float cubic_interpolate(const float y0, const float y1, const float y2, const float y3, const float mu) { + const float mu2 = mu * mu; + const float a0 = y3 - y2 - y0 + y1; + const float a1 = y0 - y1 - a0; + const float a2 = y2 - y0; + const float a3 = y1; + + return (a0 * mu * mu2 + a1 * mu2 + a2 * mu + a3); + } + + static float interpolate_samples_cubic(const float *buffer, const uint32_t bufferSize, const float position) { + const int posInt = static_cast(position); + const float mu = position - posInt; + + // Ensure that the positions are valid + const int y0Pos = (posInt - 1 < 0) ? 0 : posInt - 1; + const int y1Pos = posInt; + const int y2Pos = (posInt + 1 >= bufferSize) ? bufferSize - 1 : posInt + 1; + const int y3Pos = (posInt + 2 >= bufferSize) ? bufferSize - 1 : posInt + 2; + + // Get the samples for interpolation + const float y0 = buffer[y0Pos]; + const float y1 = buffer[y1Pos]; + const float y2 = buffer[y2Pos]; + const float y3 = buffer[y3Pos]; + + // Return the interpolated sample + return cubic_interpolate(y0, y1, y2, y3, mu); + } + /* --- buffer --- */ - inline static void fill(float* buffer, float value, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + + inline static void fill(float *buffer, float value, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer[i] = value; } @@ -369,20 +520,20 @@ namespace klangwellen { /** * copies src into dst. dst will be overwritten. */ - inline static void copy(float* src, float* dst, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void copy(float *src, float *dst, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { std::copy(src, src + length, dst); } /** * adds buffer_b to buffer_a. result will be stored in buffer_a, buffer_b will not be changed. */ - inline static void add(float* buffer_a, float* buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void add(float *buffer_a, float *buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] += buffer_b[i]; } } - inline static void add(float* buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void add(float *buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] += scalar; } @@ -391,13 +542,13 @@ namespace klangwellen { /** * subtracts buffer_b from buffer_a. result will be stored in buffer_a, buffer_b will not be changed. */ - inline static void sub(float* buffer_a, float* buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void sub(float *buffer_a, float *buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] -= buffer_b[i]; } } - inline static void sub(float* buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void sub(float *buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] -= scalar; } @@ -406,13 +557,13 @@ namespace klangwellen { /** * multiplies buffer_b with buffer_a. result will be stored in buffer_a, buffer_b will not be changed. */ - inline static void mult(float* buffer_a, float* buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void mult(float *buffer_a, float *buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] *= buffer_b[i]; } } - inline static void mult(float* buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void mult(float *buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] *= scalar; } @@ -421,16 +572,16 @@ namespace klangwellen { /** * divides buffer_b from buffer_a. result will be stored in buffer_a, buffer_b will not be changed. */ - inline static void div(float* buffer_a, float* buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void div(float *buffer_a, float *buffer_b, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] /= buffer_b[i]; } } - inline static void div(float* buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + inline static void div(float *buffer_a, float scalar, uint32_t length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint32_t i = 0; i < length; i++) { buffer_a[i] /= scalar; } } }; -} // namespace klangwellen +} // namespace klangwellen diff --git a/stm32/libraries/KlangWellen/src/Note.h b/stm32/libraries/KlangWellen/src/Note.h index 3264a3cf..f5ea8045 100644 --- a/stm32/libraries/KlangWellen/src/Note.h +++ b/stm32/libraries/KlangWellen/src/Note.h @@ -1,9 +1,8 @@ - /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/stm32/libraries/KlangWellen/src/Ramp.h b/stm32/libraries/KlangWellen/src/Ramp.h index 7e0c1277..f9bf3493 100644 --- a/stm32/libraries/KlangWellen/src/Ramp.h +++ b/stm32/libraries/KlangWellen/src/Ramp.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/stm32/libraries/KlangWellen/src/Resonator.h b/stm32/libraries/KlangWellen/src/Resonator.h new file mode 100644 index 00000000..c1760be4 --- /dev/null +++ b/stm32/libraries/KlangWellen/src/Resonator.h @@ -0,0 +1,111 @@ +/* + * KlangWellen + * + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul + * + * This library is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, version 3. + * + * This library is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +/** + * PROCESSOR INTERFACE + * + * - [ ] float process() + * - [x] float process(float)‌ + * - [ ] void process(Signal&) + * - [ ] void process(float*, uint32_t) + * - [ ] void process(float*, float*, uint32_t) + */ + +#pragma once + +#include +#include + +#include "KlangWellen.h" +#include "Signal.h" + +namespace klangwellen { + class Resonator { + public: + Resonator() + : m_frequency(440.0f), m_sampleRate(KlangWellen::DEFAULT_SAMPLING_RATE), m_qFactor(1.0f) { + calculateCoefficients(); + } + + Resonator(float frequency, float sampleRate, float qFactor) + : m_frequency(frequency), m_sampleRate(sampleRate), m_qFactor(qFactor) { + calculateCoefficients(); + } + + float process(float input) { + // IIR filter processing + float output = a0 * input + a1 * x1 + a2 * x2 - b1 * y1 - b2 * y2; + + // Update filter states + x2 = x1; + x1 = input; + y2 = y1; + y1 = output; + + return output; + } + + void set_frequency(float frequency) { + m_frequency = frequency; + calculateCoefficients(); + } + + const float get_frequency() { + return m_frequency; + } + + void set_Q(float qFactor) { + m_qFactor = qFactor; + calculateCoefficients(); + } + + float get_Q() { + return m_qFactor; + } + + private: + float m_frequency; // Resonant frequency + float m_sampleRate; // Sample rate of the audio + float m_qFactor; // Quality factor + + // Filter coefficients + float a0, a1, a2, b1, b2; + + // Filter states + float x1 = 0, x2 = 0, y1 = 0, y2 = 0; + + void calculateCoefficients() { + // Calculate the coefficients for the resonator filter + float omega = 2 * M_PI * m_frequency / m_sampleRate; + float alpha = sin(omega) / (2 * m_qFactor); + + a0 = 1 + alpha; + a1 = -2 * cos(omega); + a2 = 1 - alpha; + b1 = -2 * cos(omega); + b2 = 1 - alpha; + + // Normalize the coefficients + a1 /= a0; + a2 /= a0; + b1 /= a0; + b2 /= a0; + } + }; +} // namespace klangwellen diff --git a/stm32/libraries/KlangWellen/src/Reverb.h b/stm32/libraries/KlangWellen/src/Reverb.h index fb40856a..51e59561 100644 --- a/stm32/libraries/KlangWellen/src/Reverb.h +++ b/stm32/libraries/KlangWellen/src/Reverb.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/stm32/libraries/KlangWellen/src/SAM.h b/stm32/libraries/KlangWellen/src/SAM.h index c2202151..c94cf024 100644 --- a/stm32/libraries/KlangWellen/src/SAM.h +++ b/stm32/libraries/KlangWellen/src/SAM.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/stm32/libraries/KlangWellen/src/Sampler.h b/stm32/libraries/KlangWellen/src/Sampler.h index e4981683..a5af9794 100644 --- a/stm32/libraries/KlangWellen/src/Sampler.h +++ b/stm32/libraries/KlangWellen/src/Sampler.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -41,7 +41,6 @@ #include "KlangWellen.h" namespace klangwellen { - class SamplerListener { public: virtual void is_done() = 0; @@ -50,23 +49,26 @@ namespace klangwellen { /** * plays back an array of samples at different speeds. */ - template + template class SamplerT { public: - static const int8_t NO_LOOP_POINT = -1; + static constexpr int8_t NO_LOOP_POINT = -1; - SamplerT() : SamplerT(0) {} + SamplerT() : SamplerT(0) { + } - SamplerT(int32_t buffer_length, uint32_t sampling_rate = KlangWellen::DEFAULT_SAMPLING_RATE) : SamplerT(new BUFFER_TYPE[buffer_length], buffer_length, sampling_rate) { + explicit SamplerT(int32_t buffer_length, + uint32_t sampling_rate = KlangWellen::DEFAULT_SAMPLING_RATE) : SamplerT( + new BUFFER_TYPE[buffer_length], buffer_length, sampling_rate) { fAllocatedBuffer = true; } - SamplerT(BUFFER_TYPE* buffer, - int32_t buffer_length, - uint32_t sampling_rate = KlangWellen::DEFAULT_SAMPLING_RATE) : fDirectionForward(true), - fInPoint(0), - fOutPoint(0), - fSpeed(0) { + SamplerT(BUFFER_TYPE * buffer, + const int32_t buffer_length, + const uint32_t sampling_rate = KlangWellen::DEFAULT_SAMPLING_RATE) : fDirectionForward(true), + fInPoint(0), + fOutPoint(0), + fSpeed(0) { fSamplingRate = sampling_rate; set_buffer(buffer, buffer_length); fBufferIndex = 0; @@ -88,11 +90,11 @@ namespace klangwellen { } } - void add_listener(SamplerListener* sampler_listener) { + void add_listener(SamplerListener *sampler_listener) { fSamplerListeners.push_back(sampler_listener); } - bool remove_listener(SamplerListener* sampler_listener) { + bool remove_listener(SamplerListener *sampler_listener) { for (auto it = fSamplerListeners.begin(); it != fSamplerListeners.end(); ++it) { if (*it == sampler_listener) { fSamplerListeners.erase(it); @@ -102,7 +104,7 @@ namespace klangwellen { return false; } - int32_t get_in() { + int32_t get_in() const { return fInPoint; } @@ -113,51 +115,51 @@ namespace klangwellen { fInPoint = in_point; } - int32_t get_out() { + int32_t get_out() const { return fOutPoint; } - void set_out(int32_t out_point) { + void set_out(const int32_t out_point) { fOutPoint = out_point > last_index() ? last_index() : (out_point < fInPoint ? fInPoint : out_point); } - float get_speed() { + float get_speed() const { return fSpeed; } - void set_speed(float speed) { + void set_speed(const float speed) { fSpeed = speed; fDirectionForward = speed > 0; set_frequency(KlangWellen::abs(speed) * fSamplingRate / fBufferLength); /* aka `step_size = speed` */ } - void set_frequency(float frequency) { + void set_frequency(const float frequency) { fFrequency = frequency; - fStepSize = fFrequency / fFrequencyScale * ((float)fBufferLength / fSamplingRate); + fStepSize = fFrequency / fFrequencyScale * (static_cast(fBufferLength) / fSamplingRate); } - float get_frequency() { + float get_frequency() const { return fFrequency; } - void set_amplitude(float amplitude) { + void set_amplitude(const float amplitude) { fAmplitude = amplitude; } - float get_amplitude() { + float get_amplitude() const { return fAmplitude; } - BUFFER_TYPE* get_buffer() { + BUFFER_TYPE *get_buffer() { return fBuffer; } - int32_t get_buffer_length() { + int32_t get_buffer_length() const { return fBufferLength; } - void set_buffer(BUFFER_TYPE* buffer, int32_t buffer_length) { - fAllocatedBuffer = false; // TODO huuui, this is not nice and might cause some trouble somewhere + void set_buffer(BUFFER_TYPE *buffer, const int32_t buffer_length) { + fAllocatedBuffer = false; // TODO huuui, this is not nice and might cause some trouble somewhere fBuffer = buffer; fBufferLength = buffer_length; rewind(); @@ -168,52 +170,52 @@ namespace klangwellen { fLoopOut = NO_LOOP_POINT; } - void interpolate_samples(bool interpolate_samples) { + void interpolate_samples(bool const interpolate_samples) { fInterpolateSamples = interpolate_samples; } - bool interpolate_samples() { + bool interpolate_samples() const { return fInterpolateSamples; } - int32_t get_position() { - return (int32_t)fBufferIndex; + int32_t get_position() const { + return static_cast(fBufferIndex); } - float get_position_normalized() { + float get_position_normalized() const { return fBufferLength > 0 ? fBufferIndex / fBufferLength : 0.0f; } - float get_position_fractional_part() { + float get_position_fractional_part() const { return fBufferIndex - get_position(); } - bool is_playing() { + bool is_playing() const { return fIsPlaying; } float process() { if (fBufferLength == 0) { - notifyListeners(); // "buffer is empty" + notifyListeners(); // "buffer is empty" return 0.0f; } if (!fIsPlaying) { - notifyListeners(); // "not playing" + notifyListeners(); // "not playing" return 0.0f; } validateInOutPoints(); fBufferIndex += fDirectionForward ? fStepSize : -fStepSize; - const int32_t mRoundedIndex = (int32_t)fBufferIndex; + const int32_t mRoundedIndex = static_cast(fBufferIndex); const float mFrac = fBufferIndex - mRoundedIndex; const int32_t mCurrentIndex = wrapIndex(mRoundedIndex); fBufferIndex = mCurrentIndex + mFrac; if (fDirectionForward ? (mCurrentIndex >= fOutPoint) : (mCurrentIndex <= fInPoint)) { - notifyListeners(); // "reached end" + notifyListeners(); // "reached end" return 0.0f; } else { fIsFlaggedDone = false; @@ -225,8 +227,10 @@ namespace klangwellen { if (fInterpolateSamples) { // TODO evaluate direction? const int32_t mNextIndex = wrapIndex(mCurrentIndex + 1); - const float mNextSample = fBuffer[mNextIndex]; + const float mNextSample = convert_sample(fBuffer[mNextIndex]); mSample = mSample * (1.0f - mFrac) + mNextSample * mFrac; + // mSample = interpolate_samples_linear(fBuffer, fBufferLength, fBufferIndex); + // mSample = interpolate_samples_cubic(fBuffer, fBufferLength, fBufferIndex); } mSample *= fAmplitude; @@ -234,23 +238,23 @@ namespace klangwellen { if (fEdgeFadePadding > 0) { const int32_t mRelativeIndex = fBufferLength - mCurrentIndex; if (mCurrentIndex < fEdgeFadePadding) { - const float mFadeInAmount = (float)mCurrentIndex / fEdgeFadePadding; + const float mFadeInAmount = static_cast(mCurrentIndex) / fEdgeFadePadding; mSample *= mFadeInAmount; } else if (mRelativeIndex < fEdgeFadePadding) { - const float mFadeOutAmount = (float)mRelativeIndex / fEdgeFadePadding; + const float mFadeOutAmount = static_cast(mRelativeIndex) / fEdgeFadePadding; mSample *= mFadeOutAmount; } } return mSample; } - void process(float* signal_buffer, const uint32_t buffer_length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + void process(float *signal_buffer, const uint32_t buffer_length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { for (uint16_t i = 0; i < buffer_length; i++) { signal_buffer[i] = process(); } } - int32_t get_edge_fading() { + int32_t get_edge_fading() const { return fEdgeFadePadding; } @@ -266,7 +270,7 @@ namespace klangwellen { fBufferIndex = fDirectionForward ? fOutPoint : fInPoint; } - bool is_looping() { + bool is_looping() const { return fEvaluateLoop; } @@ -311,7 +315,7 @@ namespace klangwellen { } } - void record(float* samples, int32_t num_samples) { + void record(float *samples, int32_t num_samples) { if (fIsRecording) { for (int32_t i = 0; i < num_samples; i++) { const float sample = samples[i]; @@ -320,7 +324,7 @@ namespace klangwellen { } } - bool is_recording() { + bool is_recording() const { return fIsRecording; } @@ -331,7 +335,7 @@ namespace klangwellen { uint32_t end_recording() { fIsRecording = false; const int32_t mBufferLength = fRecording.size(); - float* mBuffer = new float[mBufferLength]; + float * mBuffer = new float[mBufferLength]; for (int32_t i = 0; i < mBufferLength; i++) { mBuffer[i] = fRecording[i]; } @@ -344,42 +348,42 @@ namespace klangwellen { return mBufferLength; } - int32_t get_loop_in() { + int32_t get_loop_in() const { return fLoopIn; } - void set_loop_in(int32_t loop_in_point) { + void set_loop_in(const int32_t loop_in_point) { fLoopIn = KlangWellen::clamp(loop_in_point, NO_LOOP_POINT, fBufferLength - 1); } - float get_loop_in_normalized() { + float get_loop_in_normalized() const { if (fBufferLength < 2) { return 0.0f; } - return (float)fLoopIn / (fBufferLength - 1); + return static_cast(fLoopIn) / (fBufferLength - 1); } - void set_loop_in_normalized(float loop_in_point_normalized) { - set_loop_in((int32_t)(loop_in_point_normalized * fBufferLength - 1)); + void set_loop_in_normalized(const float loop_in_point_normalized) { + set_loop_in(static_cast(loop_in_point_normalized * fBufferLength - 1)); } - int32_t get_loop_out() { + int32_t get_loop_out() const { return fLoopOut; } - void set_loop_out(int32_t loop_out_point) { + void set_loop_out(const int32_t loop_out_point) { fLoopOut = KlangWellen::clamp(loop_out_point, NO_LOOP_POINT, fBufferLength - 1); } - float get_loop_out_normalized() { + float get_loop_out_normalized() const { if (fBufferLength < 2) { return 0.0f; } - return (float)fLoopOut / (fBufferLength - 1); + return static_cast(fLoopOut) / (fBufferLength - 1); } - void set_loop_out_normalized(float loop_out_point_normalized) { - set_loop_out((int32_t)(loop_out_point_normalized * fBufferLength - 1)); + void set_loop_out_normalized(const float loop_out_point_normalized) { + set_loop_out(static_cast(loop_out_point_normalized * fBufferLength - 1)); } void note_on() { @@ -388,7 +392,7 @@ namespace klangwellen { enable_loop(true); } - void note_on(uint8_t note, uint8_t velocity) { + void note_on(const uint8_t note, const uint8_t velocity) { fIsPlaying = true; set_frequency(KlangWellen::note_to_frequency(note)); set_amplitude(KlangWellen::clamp127(velocity) / 127.0f); @@ -403,61 +407,61 @@ namespace klangwellen { * this function can be used to tune a loaded sample to a specific frequency. after the sampler has been tuned the * method set_frequency(float) can be used to play the sample at a desired frequency. * - * @param frequency_scale the assumed frequency of the sampler buffer in Hz + * @param tune_frequency the assumed frequency of the sampler buffer in Hz */ - void tune_frequency_to(float tune_frequency) { + void tune_frequency_to(const float tune_frequency) { fFrequencyScale = tune_frequency; } - void set_duration(float seconds) { + void set_duration(const float seconds) { if (fBufferLength == 0 || seconds == 0.0f) { return; } - const float mNormDurationSec = ((float)fBufferLength / (float)fSamplingRate); + const float mNormDurationSec = (static_cast(fBufferLength) / static_cast(fSamplingRate)); const float mSpeed = mNormDurationSec / seconds; set_speed(mSpeed); } - float get_duration() { + float get_duration() const { if (fBufferLength == 0 || fSpeed == 0.0f) { return 0; } - const float mNormDurationSec = ((float)fBufferLength / (float)fSamplingRate); + const float mNormDurationSec = (static_cast(fBufferLength) / static_cast(fSamplingRate)); return mNormDurationSec / fSpeed; } private: - std::vector fSamplerListeners; - std::vector fRecording; - uint32_t fSamplingRate; - float fAmplitude; - BUFFER_TYPE* fBuffer; - int32_t fBufferLength; - float fBufferIndex; - bool fDirectionForward; - int32_t fEdgeFadePadding; - bool fEvaluateLoop; - float fFrequency; - float fFrequencyScale; - int32_t fInPoint; - int32_t fOutPoint; - int32_t fLoopIn; - int32_t fLoopOut; - bool fInterpolateSamples; - bool fIsPlaying; - float fSpeed; - float fStepSize; - bool fIsFlaggedDone; - bool fIsRecording; - bool fAllocatedBuffer; - - int32_t last_index() { + std::vector fSamplerListeners; + std::vector fRecording; + uint32_t fSamplingRate; + float fAmplitude; + BUFFER_TYPE * fBuffer; + int32_t fBufferLength; + float fBufferIndex; + bool fDirectionForward; + int32_t fEdgeFadePadding; + bool fEvaluateLoop; + float fFrequency; + float fFrequencyScale; + int32_t fInPoint; + int32_t fOutPoint; + int32_t fLoopIn; + int32_t fLoopOut; + bool fInterpolateSamples; + bool fIsPlaying; + float fSpeed; + float fStepSize; + bool fIsFlaggedDone; + bool fIsRecording; + bool fAllocatedBuffer; + + int32_t last_index() const { return fBufferLength - 1; } void notifyListeners() { if (!fIsFlaggedDone) { - for (SamplerListener* l : fSamplerListeners) { + for (SamplerListener *l: fSamplerListeners) { l->is_done(); } } @@ -486,7 +490,7 @@ namespace klangwellen { } } - int32_t wrapIndex(int32_t i) { + int32_t wrapIndex(int32_t i) const { /* check if in loop concept viable i.e loop in- and output points are set */ if (fEvaluateLoop) { if (fLoopIn != NO_LOOP_POINT && fLoopOut != NO_LOOP_POINT) { @@ -511,30 +515,30 @@ namespace klangwellen { return i; } - inline float convert_sample(const BUFFER_TYPE pRawSample) { + float convert_sample(const BUFFER_TYPE pRawSample) { return pRawSample; } }; - template <> - float klangwellen::SamplerT::convert_sample(const uint8_t pRawSample) { - const static float mScale = 1.0 / ((1 << 8) - 1); - const float mRange = pRawSample * mScale; + template<> + inline float klangwellen::SamplerT::convert_sample(const uint8_t pRawSample) { + constexpr static float mScale = 1.0 / ((1 << 8) - 1); + const float mRange = pRawSample * mScale; return mRange * 2.0 - 1.0; } - template <> - float klangwellen::SamplerT::convert_sample(const int8_t pRawSample) { - const static float mScale = 1.0 / ((1 << 8) - 1); - const float mOffset = pRawSample + (1 << 7); - const float mRange = mOffset * mScale; + template<> + inline float klangwellen::SamplerT::convert_sample(const int8_t pRawSample) { + constexpr static float mScale = 1.0 / ((1 << 8) - 1); + const float mOffset = pRawSample + (1 << 7); + const float mRange = mOffset * mScale; return mRange * 2.0 - 1.0; } - template <> - float klangwellen::SamplerT::convert_sample(const uint16_t pRawSample) { - const static float mScale = 1.0 / ((1 << 16) - 1); - const float mRange = pRawSample * mScale; + template<> + inline float klangwellen::SamplerT::convert_sample(const uint16_t pRawSample) { + constexpr static float mScale = 1.0 / ((1 << 16) - 1); + const float mRange = pRawSample * mScale; return mRange * 2.0 - 1.0; // @note(below: less precise but faster) // const float s = pRawSample; @@ -543,18 +547,18 @@ namespace klangwellen { // return a; } - template <> - float klangwellen::SamplerT::convert_sample(const int16_t pRawSample) { - const static float mScale = 1.0 / ((1 << 16) - 1); - const float mOffset = pRawSample + (1 << 15); - const float mRange = mOffset * mScale; + template<> + inline float klangwellen::SamplerT::convert_sample(const int16_t pRawSample) { + constexpr static float mScale = 1.0 / ((1 << 16) - 1); + const float mOffset = pRawSample + (1 << 15); + const float mRange = mOffset * mScale; return mRange * 2.0 - 1.0; } - using SamplerUI8 = SamplerT; - using SamplerI8 = SamplerT; + using SamplerUI8 = SamplerT; + using SamplerI8 = SamplerT; using SamplerUI16 = SamplerT; - using SamplerI16 = SamplerT; - using SamplerF32 = SamplerT; - using Sampler = SamplerT; -} // namespace klangwellen + using SamplerI16 = SamplerT; + using SamplerF32 = SamplerT; + using Sampler = SamplerT; +} // namespace klangwellen diff --git a/stm32/libraries/KlangWellen/src/Scale.cpp b/stm32/libraries/KlangWellen/src/Scale.cpp deleted file mode 100644 index 202ef75c..00000000 --- a/stm32/libraries/KlangWellen/src/Scale.cpp +++ /dev/null @@ -1,14 +0,0 @@ -#include "Scale.h" - -const klangwellen::Scale klangwellen::Scale::CHROMATIC{0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11}; -const klangwellen::Scale klangwellen::Scale::FIFTH{0, 7}; -const klangwellen::Scale klangwellen::Scale::MINOR{0, 2, 3, 5, 7, 8, 10}; -const klangwellen::Scale klangwellen::Scale::MAJOR{0, 2, 4, 5, 7, 9, 11}; -const klangwellen::Scale klangwellen::Scale::MINOR_CHORD{0, 3, 7}; -const klangwellen::Scale klangwellen::Scale::MAJOR_CHORD{0, 4, 7}; -const klangwellen::Scale klangwellen::Scale::MINOR_CHORD_7{0, 3, 7, 11}; -const klangwellen::Scale klangwellen::Scale::MAJOR_CHORD_7{0, 4, 7, 11}; -const klangwellen::Scale klangwellen::Scale::MINOR_PENTATONIC{0, 3, 5, 7, 10}; -const klangwellen::Scale klangwellen::Scale::MAJOR_PENTATONIC{0, 4, 5, 7, 11}; -const klangwellen::Scale klangwellen::Scale::OCTAVE{0}; -const klangwellen::Scale klangwellen::Scale::DIMINISHED{0, 3, 6, 9}; diff --git a/stm32/libraries/KlangWellen/src/Scale.h b/stm32/libraries/KlangWellen/src/Scale.h index 5ed8e723..ed6a6494 100644 --- a/stm32/libraries/KlangWellen/src/Scale.h +++ b/stm32/libraries/KlangWellen/src/Scale.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -26,19 +26,6 @@ namespace klangwellen { class Scale { public: - static const Scale CHROMATIC; - static const Scale FIFTH; - static const Scale MINOR; - static const Scale MAJOR; - static const Scale MINOR_CHORD; - static const Scale MAJOR_CHORD; - static const Scale MINOR_CHORD_7; - static const Scale MAJOR_CHORD_7; - static const Scale MINOR_PENTATONIC; - static const Scale MAJOR_PENTATONIC; - static const Scale OCTAVE; - static const Scale DIMINISHED; - Scale(const std::initializer_list& note_list) : length(note_list.size()) { notes = new uint8_t[length]; uint16_t i = 0; diff --git a/stm32/libraries/KlangWellen/src/ScaleCollection.h b/stm32/libraries/KlangWellen/src/ScaleCollection.h new file mode 100644 index 00000000..9ccebf7c --- /dev/null +++ b/stm32/libraries/KlangWellen/src/ScaleCollection.h @@ -0,0 +1,40 @@ +/* + * KlangWellen + * + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul + * + * This library is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, version 3. + * + * This library is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +#pragma once + +#include "Scale.h" + +namespace klangwellen { + class ScaleCollection { + public: + static inline const Scale CHROMATIC{0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11}; + static inline const Scale FIFTH{0, 7}; + static inline const Scale MINOR{0, 2, 3, 5, 7, 8, 10}; + static inline const Scale MAJOR{0, 2, 4, 5, 7, 9, 11}; + static inline const Scale MINOR_CHORD{0, 3, 7}; + static inline const Scale MAJOR_CHORD{0, 4, 7}; + static inline const Scale MINOR_CHORD_7{0, 3, 7, 11}; + static inline const Scale MAJOR_CHORD_7{0, 4, 7, 11}; + static inline const Scale MINOR_PENTATONIC{0, 3, 5, 7, 10}; + static inline const Scale MAJOR_PENTATONIC{0, 4, 5, 7, 11}; + static inline const Scale OCTAVE{0}; + static inline const Scale DIMINISHED{0, 3, 6, 9}; + }; +} // namespace klangwellen diff --git a/stm32/libraries/KlangWellen/src/Signal.h b/stm32/libraries/KlangWellen/src/Signal.h index a9acd3c6..3094dc49 100644 --- a/stm32/libraries/KlangWellen/src/Signal.h +++ b/stm32/libraries/KlangWellen/src/Signal.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/stm32/libraries/KlangWellen/src/Stream.h b/stm32/libraries/KlangWellen/src/Stream.h new file mode 100644 index 00000000..91171151 --- /dev/null +++ b/stm32/libraries/KlangWellen/src/Stream.h @@ -0,0 +1,201 @@ +/* +* KlangWellen + * + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul + * + * This library is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, version 3. + * + * This library is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +/** + * PROCESSOR INTERFACE + * + * - [x] float process() + * - [ ] float process(float) + * - [ ] void process(Signal&) + * - [*] void process(float*, uint32_t) + * - [ ] void process(float*, float*, uint32_t) + */ + +#pragma once + +namespace klangwellen { + class StreamDataProvider { + public: + virtual ~StreamDataProvider() = default; + + virtual void fill_buffer(float *buffer, uint32_t length) { + std::fill_n(buffer, length, 0.0f); + } + }; + + class Stream final { + public: + Stream(StreamDataProvider *stream_data_provider, + const uint32_t stream_buffer_size, + const uint8_t stream_buffer_division = 4, + const uint8_t stream_buffer_update_offset = 1, + const uint32_t sampling_rate = KlangWellen::DEFAULT_SAMPLING_RATE) + : fStreamDataProvider(stream_data_provider), + fBufferLength(stream_buffer_size), + fBuffer(new float[stream_buffer_size]), + fBufferDivision(stream_buffer_division), + fBufferSegmentOffset(stream_buffer_update_offset % stream_buffer_division), + fSamplingRate(sampling_rate), + fAmplitude(1.0f), + fStepSize(1.0f), + fInterpolateSamples(true), + fBufferIndex(0.0f), + fBufferIndexPrev(0.0f), + fCompleteEvent(NO_EVENT) { + fStreamDataProvider->fill_buffer(fBuffer, fBufferLength); + } + + ~Stream() { + delete[] fBuffer; + } + + float process() { + fBufferIndex += fStepSize; + const int32_t mRoundedIndex = static_cast(fBufferIndex); + const float mFrac = fBufferIndex - mRoundedIndex; + const int32_t mCurrentIndex = wrapIndex(mRoundedIndex); + fBufferIndex = mCurrentIndex + mFrac; + + float mSample = convert_sample(fBuffer[mCurrentIndex]); + + /* linear interpolation */ + if (fInterpolateSamples) { + const int32_t mNextIndex = wrapIndex(mCurrentIndex + 1); + const float mNextSample = convert_sample(fBuffer[mNextIndex]); + const float a = mSample * (1.0f - mFrac); + const float b = mNextSample * mFrac; + mSample = a + b; + } + mSample *= fAmplitude; + + /* load next block */ + int8_t mCompleteEvent = checkCompleteEvent(fBufferDivision); + if (mCompleteEvent > NO_EVENT) { + fCompleteEvent = mCompleteEvent; + mCompleteEvent -= fBufferSegmentOffset; + mCompleteEvent += fBufferDivision; + mCompleteEvent %= fBufferDivision; + replace_segment(fBufferDivision, mCompleteEvent); + } + fBufferIndexPrev = fBufferIndex; + + return mSample; + } + + void replace_segment(const uint32_t numberOfSegments, const uint32_t segmentIndex) const { + if (segmentIndex >= numberOfSegments) { + std::cerr << "Segment index out of range" << std::endl; + return; + } + + const uint32_t lengthOfSegment = fBufferLength / numberOfSegments; + float * startOfSegment = fBuffer + (segmentIndex * lengthOfSegment); + + fStreamDataProvider->fill_buffer(startOfSegment, lengthOfSegment); + } + + float *get_buffer() const { + return fBuffer; + } + + int8_t get_sector() const { + return fCompleteEvent; + } + + uint8_t num_sectors() const { + return fBufferDivision; + } + + uint32_t get_buffer_length() const { + return fBufferLength; + } + + float get_current_buffer_position() const { + return fBufferIndex; + } + + void process(float *signal_buffer, const uint32_t buffer_length = KLANG_SAMPLES_PER_AUDIO_BLOCK) { + for (uint16_t i = 0; i < buffer_length; i++) { + signal_buffer[i] = process(); + } + } + + void interpolate_samples(bool const interpolate_samples) { + fInterpolateSamples = interpolate_samples; + } + + bool interpolate_samples() const { + return fInterpolateSamples; + } + + float get_speed() const { + return fStepSize; + } + + void set_speed(const float speed) { + fStepSize = speed; + } + + private: + static constexpr int8_t NO_EVENT = -1; + + StreamDataProvider *fStreamDataProvider; + + uint32_t fBufferLength; + float * fBuffer; + const uint8_t fBufferDivision; + const uint8_t fBufferSegmentOffset; + float fSamplingRate; + float fAmplitude; + float fStepSize; + bool fInterpolateSamples; + float fBufferIndex; + float fBufferIndexPrev; + int8_t fCompleteEvent; + + int32_t wrapIndex(int32_t i) const { + if (i < 0) { + i += fBufferLength; + } else if (i >= fBufferLength) { + i -= fBufferLength; + } + return i; + } + + static float convert_sample(const float pRawSample) { + return pRawSample; + } + + int8_t checkCompleteEvent(const uint8_t num_events) const { + for (int i = 0; i < num_events; ++i) { + const float mBorder = fBufferLength * i / static_cast(fBufferDivision); + if (crossedBorder(fBufferIndexPrev, fBufferIndex, mBorder)) { + return i; + } + } + return NO_EVENT; + } + + static bool crossedBorder(const float prev, const float current, const float border) { + return (border == 0 && prev > current) || + (prev < border && current >= border) || + (prev > current && current >= border); + } + }; +} // namespace klangwellen diff --git a/stm32/libraries/KlangWellen/src/Trigger.h b/stm32/libraries/KlangWellen/src/Trigger.h index e4fe492c..a23ab6a4 100644 --- a/stm32/libraries/KlangWellen/src/Trigger.h +++ b/stm32/libraries/KlangWellen/src/Trigger.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/stm32/libraries/KlangWellen/src/Vocoder.h b/stm32/libraries/KlangWellen/src/Vocoder.h index 086b07dd..689cdafc 100644 --- a/stm32/libraries/KlangWellen/src/Vocoder.h +++ b/stm32/libraries/KlangWellen/src/Vocoder.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/stm32/libraries/KlangWellen/src/Waveshaper.h b/stm32/libraries/KlangWellen/src/Waveshaper.h index b4ed9a86..ac918d20 100644 --- a/stm32/libraries/KlangWellen/src/Waveshaper.h +++ b/stm32/libraries/KlangWellen/src/Waveshaper.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://githubiquad_com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by diff --git a/stm32/libraries/KlangWellen/src/Wavetable.h b/stm32/libraries/KlangWellen/src/Wavetable.h index aafad196..b98e3f6c 100644 --- a/stm32/libraries/KlangWellen/src/Wavetable.h +++ b/stm32/libraries/KlangWellen/src/Wavetable.h @@ -1,8 +1,8 @@ /* - * Wellen + * KlangWellen * - * This file is part of the *wellen* library (https://github.com/dennisppaul/wellen). - * Copyright (c) 2023 Dennis P Paul. + * This file is part of the *KlangWellen* library (https://github.com/dennisppaul/klangwellen). + * Copyright (c) 2023 Dennis P Paul * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU General License as published by @@ -110,7 +110,7 @@ namespace klangwellen { static void noise(float* wavetable, uint32_t wavetable_size) { for (uint32_t i = 0; i < wavetable_size; i++) { - wavetable[i] = KlangWellen::random() * 2.0 - 1.0; + wavetable[i] = KlangWellen::random() * 2.0f - 1.0f; } } @@ -136,7 +136,7 @@ namespace klangwellen { return fourier_table(wavetable, wavetable_size, harmonics, amps, -0.25f); } - static void sawtooth(float* wavetable, uint32_t wavetable_size, bool is_ramp_up) { + static void sawtooth_ramp(float* wavetable, uint32_t wavetable_size, bool is_ramp_up) { const float mSign = is_ramp_up ? -1.0f : 1.0f; for (uint32_t i = 0; i < wavetable_size; i++) { wavetable[i] = mSign * (2.0f * ((float)i / (float)(wavetable_size - 1)) - 1.0f); @@ -149,7 +149,7 @@ namespace klangwellen { static void sine(float* wavetable, uint32_t wavetable_size) { for (uint32_t i = 0; i < wavetable_size; i++) { - wavetable[i] = KlangWellen::fast_sin(2.0f * PI * ((float)i / (float)(wavetable_size))); + wavetable[i] = KlangWellen::fast_sin(2.0f * PIf * ((float)i / (float)(wavetable_size))); } } @@ -381,6 +381,8 @@ namespace klangwellen { } private: + static constexpr float PIf = (float)PI; + static constexpr float TWO_PIf = (float)TWO_PI; static constexpr float M_DEFAULT_AMPLITUDE = 0.75f; static constexpr float M_DEFAULT_FREQUENCY = 220.0f; float* mWavetable; @@ -407,12 +409,12 @@ namespace klangwellen { static float* fourier_table(float* pWavetable, uint32_t wavetable_size, uint8_t pHarmonics, float* pAmps, float pPhase) { float a; double w; - pPhase *= PI * 2; + pPhase *= PIf * 2; for (uint8_t i = 0; i < pHarmonics; i++) { for (uint32_t n = 0; n < wavetable_size; n++) { a = (pAmps) ? pAmps[i] : 1.f; w = (i + 1) * (n * 2 * PI / wavetable_size); - pWavetable[n] += (float)(a * KlangWellen::cos(w + pPhase)); + pWavetable[n] += (float)(a * KlangWellen::cos((float)w + pPhase)); } } normalise_table(pWavetable, wavetable_size); @@ -455,8 +457,8 @@ namespace klangwellen { } float next_sample_interpolate_cubic() { - const uint32_t mOffset = (int)(mPhaseOffset * mWavetableSize) % mWavetableSize; - const float mArrayPtrOffset = mArrayPtr + mOffset; + const uint32_t mSampleOffset = (int)(mPhaseOffset * mWavetableSize) % mWavetableSize; + const float mArrayPtrOffset = mArrayPtr + mSampleOffset; /* cubic interpolation */ const float frac = mArrayPtrOffset - (int)mArrayPtrOffset; const float a = (int)mArrayPtrOffset > 0 ? mWavetable[(int)mArrayPtrOffset - 1] : mWavetable[mWavetableSize - 1]; @@ -475,8 +477,8 @@ namespace klangwellen { } float next_sample_interpolate_linear() { - const uint32_t mOffset = (uint32_t)(mPhaseOffset * mWavetableSize) % mWavetableSize; - const float mArrayPtrOffset = mArrayPtr + mOffset; + const uint32_t mSampleOffset = (uint32_t)(mPhaseOffset * mWavetableSize) % mWavetableSize; + const float mArrayPtrOffset = mArrayPtr + mSampleOffset; /* linear interpolation */ const float mFrac = mArrayPtrOffset - (int)mArrayPtrOffset; const float a = mWavetable[(int)mArrayPtrOffset]; diff --git a/stm32/libraries/Klangstrom/examples/01.Basics/Beat/Beat.ino b/stm32/libraries/Klangstrom/examples/01.Basics/Beat/Beat.ino index 38022d97..7dcbf44d 100644 --- a/stm32/libraries/Klangstrom/examples/01.Basics/Beat/Beat.ino +++ b/stm32/libraries/Klangstrom/examples/01.Basics/Beat/Beat.ino @@ -20,6 +20,11 @@ uint32_t mBeatDurationMax = 120000; uint32_t mBeatDurationInc = 1000; void setup() { + Serial.begin(115200); + Serial.println("----"); + Serial.println("Beat"); + Serial.println("----"); + Klang::lock(); Klang::connect(mVCO, Node::CH_OUT_SIGNAL, mDAC, NodeDAC::CH_IN_SIGNAL_LEFT); mVCO.set_frequency(mFreq); diff --git a/stm32/libraries/Klangstrom/examples/01.Basics/Blink/Blink.ino b/stm32/libraries/Klangstrom/examples/01.Basics/Blink/Blink.ino index ca7aafc8..8220d4c9 100644 --- a/stm32/libraries/Klangstrom/examples/01.Basics/Blink/Blink.ino +++ b/stm32/libraries/Klangstrom/examples/01.Basics/Blink/Blink.ino @@ -8,6 +8,11 @@ NodeVCOFunction mVCO; NodeDAC mDAC; void setup() { + Serial.begin(115200); + Serial.println("-----"); + Serial.println("Blink"); + Serial.println("-----"); + Klang::lock(); Klang::connect(mVCO, Node::CH_OUT_SIGNAL, mDAC, NodeDAC::CH_IN_SIGNAL); mVCO.set_amplitude(0.5); diff --git a/stm32/libraries/Klangstrom/examples/01.Basics/PassThrough/PassThrough.ino b/stm32/libraries/Klangstrom/examples/01.Basics/PassThrough/PassThrough.ino index f76318aa..44ae1c6f 100644 --- a/stm32/libraries/Klangstrom/examples/01.Basics/PassThrough/PassThrough.ino +++ b/stm32/libraries/Klangstrom/examples/01.Basics/PassThrough/PassThrough.ino @@ -3,6 +3,11 @@ using namespace klangstrom; void setup() { + Serial.begin(115200); + Serial.println("-----------"); + Serial.println("PassThrough"); + Serial.println("-----------"); + option(KLST_OPTION_AUDIO_INPUT, KLST_MIC); } diff --git a/stm32/libraries/Klangstrom/examples/02.Peripherals/Encoders/Encoders.ino b/stm32/libraries/Klangstrom/examples/02.Peripherals/Encoders/Encoders.ino index 1231cdd8..6ff1ebf9 100644 --- a/stm32/libraries/Klangstrom/examples/02.Peripherals/Encoders/Encoders.ino +++ b/stm32/libraries/Klangstrom/examples/02.Peripherals/Encoders/Encoders.ino @@ -7,6 +7,7 @@ void setup() { Serial.println("--------"); Serial.println("Encoders"); Serial.println("--------"); + register_encoder_rotated(encoder_rotated); register_encoder_pressed(encoder_pressed); register_encoder_released(encoder_released); diff --git a/desktop/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleReadSDCard/ExampleReadSDCard.ino b/stm32/libraries/Klangstrom/examples/02.Peripherals/ExternalSDCard/ExternalSDCard.ino similarity index 80% rename from desktop/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleReadSDCard/ExampleReadSDCard.ino rename to stm32/libraries/Klangstrom/examples/02.Peripherals/ExternalSDCard/ExternalSDCard.ino index afb58bc1..c41188a6 100644 --- a/desktop/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleReadSDCard/ExampleReadSDCard.ino +++ b/stm32/libraries/Klangstrom/examples/02.Peripherals/ExternalSDCard/ExternalSDCard.ino @@ -1,9 +1,7 @@ -// -// ExampleReadSDCard -// - /* + this example requires the Arduino SD library. + pin connection map for KLST_TINYv0.1: | SD_CARD | KLST | @@ -13,27 +11,31 @@ | SCK | SPI_SCK | | CS | GPIO_00 | - pin connection map for KLST_SHEEPv0.1 on : + pin connection map for KLST_SHEEPv0.1: | SD_CARD | KLST | |---------|--------------| | MOSI | SPI_USR_MOSI | | MISO | SPI_USR_MISO | | SCK | SPI_USR_SCK | - | CS | GPIO_00 | + | CS | GPIO_00 | */ -#include "Klangstrom.h" -#include #include +#include + +#include "Klangstrom.h" -Sd2Card card; +Sd2Card card; SdVolume volume; -SdFile root; +SdFile root; void setup() { - klangstrom::begin_serial_debug(true); + Serial.begin(115200); + Serial.println("--------------"); + Serial.println("ExternalSDCard"); + Serial.println("--------------"); Serial.print("\nInitializing SD card..."); @@ -42,7 +44,8 @@ void setup() { Serial.println("* is a card inserted?"); Serial.println("* is your wiring correct?"); Serial.println("* did you change the chipSelect pin to match your shield or module?"); - while (1); + while (1) + ; } else { Serial.println("Wiring is correct and a card is present."); } @@ -67,7 +70,8 @@ void setup() { // open 'volume'/'partition' - should be FAT16 or FAT32 if (!volume.init(card)) { Serial.println("Could not find FAT16/FAT32 partition.\nMake sure you've formatted the card"); - while (1); + while (1) + ; } Serial.print("Clusters: "); @@ -84,9 +88,9 @@ void setup() { Serial.print("Volume type is: FAT"); Serial.println(volume.fatType(), DEC); - volumesize = volume.blocksPerCluster(); // clusters are collections of blocks - volumesize *= volume.clusterCount(); // we'll have a lot of clusters - volumesize /= 2; // SD card blocks are always 512 bytes (2 blocks are 1KB) + volumesize = volume.blocksPerCluster(); // clusters are collections of blocks + volumesize *= volume.clusterCount(); // we'll have a lot of clusters + volumesize /= 2; // SD card blocks are always 512 bytes (2 blocks are 1KB) Serial.print("Volume size (Kb): "); Serial.println(volumesize); Serial.print("Volume size (Mb): "); @@ -103,4 +107,3 @@ void setup() { } void loop() {} - diff --git a/stm32/libraries/Klangstrom/examples/02.Peripherals/Presets/Presets.ino b/stm32/libraries/Klangstrom/examples/02.Peripherals/Presets/Presets.ino new file mode 100644 index 00000000..ccc7f6d3 --- /dev/null +++ b/stm32/libraries/Klangstrom/examples/02.Peripherals/Presets/Presets.ino @@ -0,0 +1,56 @@ +#include + +#include "Klangstrom.h" + +using namespace klangstrom; + +/* define struct to hold presets */ +struct Presets { + float frequency = 440.0; + float amplitude = 0.4; +}; + +Presets mPresets; + +void setup() { + Serial.begin(115200); + Serial.println("-------"); + Serial.println("Presets"); + Serial.println("-------"); + + register_encoder_pressed(encoder_pressed); + + /* hold button 00 at start up to reset presets */ + if (button_state(ENCODER_00)) { + Serial.print("reset presets: "); + EEPROM.put(0, mPresets); + } else { + Serial.print("read presets: "); + EEPROM.get(0, mPresets); + } + print_presets(); +} + +void loop() {} + +void print_presets() { + Serial.print(mPresets.amplitude); + Serial.print(", "); + Serial.print(mPresets.frequency); + Serial.println(); +} + +void encoder_pressed(const uint8_t index) { + if (index == ENCODER_00) { + Serial.print("write presets 0: "); + mPresets.amplitude = 0.5; + mPresets.frequency = 440.0; + EEPROM.put(0, mPresets); + } else if (index == ENCODER_01) { + Serial.print("write presets 1: "); + mPresets.amplitude = 0.6; + mPresets.frequency = 220.0; + EEPROM.put(0, mPresets); + } + print_presets(); +} diff --git a/stm32/libraries/Klangstrom/examples/02.Peripherals/Serial/Serial.ino b/stm32/libraries/Klangstrom/examples/02.Peripherals/Serial/Serial.ino new file mode 100644 index 00000000..01398f37 --- /dev/null +++ b/stm32/libraries/Klangstrom/examples/02.Peripherals/Serial/Serial.ino @@ -0,0 +1,31 @@ +#include "Klangstrom.h" + +using namespace klangstrom; + +void setup() { + Serial.begin(115200); + Serial.println("------"); + Serial.println("Serial"); + Serial.println("------"); + + LED(LED_00, LED_ON); + LED(LED_01, LED_OFF); + beats_per_minute(120); +} + +void beat(uint32_t beat) { + LED(LED_00, LED_TOGGLE); + uint8_t data[] = "hello world\n\r"; + data_transmit(SERIAL_00, data, 13); +} + +void loop() {} + +void data_receive(const uint8_t receiver, uint8_t* data, uint8_t length) { + if (receiver == SERIAL_01) { + for (int i = 0; i < length; i++) { + Serial.print((char)data[i]); + LED(LED_01, LED_TOGGLE); + } + } +} diff --git a/stm32/libraries/Klangstrom/examples/04.Tools/ApplicationTemplate/ApplicationTemplate.ino b/stm32/libraries/Klangstrom/examples/04.Tools/ApplicationTemplate/ApplicationTemplate.ino index 764b1029..fdf9ff1d 100644 --- a/stm32/libraries/Klangstrom/examples/04.Tools/ApplicationTemplate/ApplicationTemplate.ino +++ b/stm32/libraries/Klangstrom/examples/04.Tools/ApplicationTemplate/ApplicationTemplate.ino @@ -1,7 +1,7 @@ #include "KlangNodes.hpp" #include "Klangstrom.h" -// @TODO add `TaskScheduler` +// @TODO merge with `SchedulingTasks` using namespace klang; using namespace klangstrom; @@ -10,12 +10,17 @@ NodeVCOFunction mVCO; NodeDAC mDAC; bool toggle_amplitude = false; -float output_buffer_left[KLANG_SAMPLES_PER_AUDIO_BLOCK]; -float output_buffer_right[KLANG_SAMPLES_PER_AUDIO_BLOCK]; +float output_buffer_left[KLANG_SAMPLES_PER_AUDIO_BLOCK]; +float output_buffer_right[KLANG_SAMPLES_PER_AUDIO_BLOCK]; volatile bool schedule_audio = false; volatile bool schedule_beat = false; void setup() { + Serial.begin(115200); + Serial.println("-------------------"); + Serial.println("ApplicationTemplate"); + Serial.println("-------------------"); + Klang::lock(); Klang::connect(mVCO, Node::CH_OUT_SIGNAL, mDAC, NodeDAC::CH_IN_SIGNAL); mVCO.set_amplitude(0.0); diff --git a/stm32/libraries/Klangstrom/examples/04.Tools/QueryButtonStates/QueryButtonStates.ino b/stm32/libraries/Klangstrom/examples/04.Tools/QueryButtonStates/QueryButtonStates.ino index 65854cd8..132f6e72 100644 --- a/stm32/libraries/Klangstrom/examples/04.Tools/QueryButtonStates/QueryButtonStates.ino +++ b/stm32/libraries/Klangstrom/examples/04.Tools/QueryButtonStates/QueryButtonStates.ino @@ -14,6 +14,7 @@ void setup() { Serial.println("-----------------"); Serial.println("QueryButtonStates"); Serial.println("-----------------"); + Serial.println("in `setup()`: "); Serial.println("---"); print_button_states(); diff --git a/stm32/libraries/Klangstrom/examples/04.Tools/SchedulingTasks/SchedulingTasks.ino b/stm32/libraries/Klangstrom/examples/04.Tools/SchedulingTasks/SchedulingTasks.ino index 7a992166..edeb43f6 100644 --- a/stm32/libraries/Klangstrom/examples/04.Tools/SchedulingTasks/SchedulingTasks.ino +++ b/stm32/libraries/Klangstrom/examples/04.Tools/SchedulingTasks/SchedulingTasks.ino @@ -1,6 +1,8 @@ #include "Arduino.h" #include "TaskScheduler.h" +// @TODO merge with `ApplicationTemplate` + using namespace klangstrom; TaskScheduler fTaskScheduler; @@ -10,7 +12,6 @@ void setup() { Serial.println("---------------"); Serial.println("SchedulingTasks"); Serial.println("---------------"); - Serial.println("---"); fTaskScheduler.schedule_priority_task(exec_priority_task); fTaskScheduler.schedule_task(exec_normal_task); diff --git a/stm32/libraries/Klangstrom/examples/05.Applications/TalkingTerminal/TalkingTerminal.ino b/stm32/libraries/Klangstrom/examples/05.Applications/TalkingTerminal/TalkingTerminal.ino index c24ba44b..1b5af042 100644 --- a/stm32/libraries/Klangstrom/examples/05.Applications/TalkingTerminal/TalkingTerminal.ino +++ b/stm32/libraries/Klangstrom/examples/05.Applications/TalkingTerminal/TalkingTerminal.ino @@ -24,9 +24,9 @@ NodeTextToSpeechSAM mTTS(46160); void setup() { Serial.begin(115200); - Serial.println("----------------"); - Serial.println("Talking Terminal"); - Serial.println("----------------"); + Serial.println("---------------"); + Serial.println("TalkingTerminal"); + Serial.println("---------------"); USBHost.init(); USBHost.register_key_pressed(key_pressed); @@ -62,7 +62,7 @@ void beat(uint32_t beat) { LED(LED_01, LED_TOGGLE); } -void audioblock(float** input_signal, float** output_signal) { +void audioblock(float **input_signal, float **output_signal) { mDAC.process_frame(output_signal[LEFT], output_signal[RIGHT]); } diff --git a/stm32/libraries/Klangstrom/examples/06.Experiments/MaximumMemoryUsage/MaximumMemoryUsage.ino b/stm32/libraries/Klangstrom/examples/06.Experiments/MaximumMemoryUsage/MaximumMemoryUsage.ino index 2c8242ac..b2fcfb68 100644 --- a/stm32/libraries/Klangstrom/examples/06.Experiments/MaximumMemoryUsage/MaximumMemoryUsage.ino +++ b/stm32/libraries/Klangstrom/examples/06.Experiments/MaximumMemoryUsage/MaximumMemoryUsage.ino @@ -2,9 +2,9 @@ void setup() { Serial.begin(115200); - Serial.println("-------------------------"); - Serial.println("PrintAvailableMemoryBlock"); - Serial.println("-------------------------"); + Serial.println("------------------"); + Serial.println("MAximumMemoryUsage"); + Serial.println("------------------"); } void print_available_memory_block(uint32_t size_available_memory_block) { @@ -30,7 +30,7 @@ uint8_t memory_RAM_D2[294912] __attribute__((section(".sample_buffer"))) = { uint8_t memory_RAM_D3[59392] __attribute__((section(".audio_block_buffer"))) = {0}; // RAM_D3 :: 64K > 65536 bytes - 6144 bytes ( DMA BUFFER ) = 59392 bytes /* note that this allocation scheme results into a total of 871380 bytes ( 850K ) separated into 3 * memory blocks. also note that the `.data` section is the default location for global variables. - * the section attributes `.sample_buffer` and `.audio_block_buffer` are required to use other + * the section attributes `.sample_buffer` and `.audio_block_buffer` are required to use other * memory locations. */ diff --git a/stm32/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleSavingPresets/ExampleSavingPresets.ino b/stm32/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleSavingPresets/ExampleSavingPresets.ino deleted file mode 100644 index 8ef1e3de..00000000 --- a/stm32/libraries/Klangstrom/examples/todo/klangstrom/board/ExampleSavingPresets/ExampleSavingPresets.ino +++ /dev/null @@ -1,87 +0,0 @@ -// -// ExampleSavingPresets -// - -#include "Klangstrom.h" -#include - -#include "KlangNodes.hpp" - -using namespace klang; -using namespace klangstrom; - -NodeVCOFunction mVCO; -NodeDAC mDAC; - -/* define struct to hold presets */ -struct Presets { - float frequency = 440.0; - float amplitude = 0.4; -}; - -Presets mPresets; - -void setup() { - begin_serial_debug(true); - - /* hold button 00 at start up to reset presets */ - if (button_state(KLST_BUTTON_ENCODER_00)) { - serial_debug.println("reset presets"); - EEPROM.put(0, mPresets); - } else { - serial_debug.println("read presets"); - EEPROM.get(0, mPresets); - } - - Klang::lock(); - Klang::connect(mVCO, Node::CH_OUT_SIGNAL, mDAC, NodeDAC::CH_IN_SIGNAL); - mVCO.set_amplitude(mPresets.amplitude); - mVCO.set_frequency(mPresets.frequency); - mVCO.set_waveform(NodeVCOFunction::WAVEFORM::SINE); - Klang::unlock(); -} - -void loop() { - led(LED_00, true); - mVCO.set_amplitude(mPresets.amplitude); - delay(500); - led(LED_00, false); - mVCO.set_amplitude(0.0); - delay(500); - Serial.print(mPresets.amplitude); - Serial.print(", "); - Serial.print(mPresets.frequency); - Serial.println(); -} - -void event_receive(const EVENT_TYPE event, const void* data) { - if (event == EVENT_ENCODER_BUTTON_PRESSED) { - if (data[INDEX] == ENCODER_00) { - serial_debug.println("write presets 0"); - mPresets.amplitude = 0.4; - mPresets.frequency = 440.0; - mVCO.set_amplitude(mPresets.amplitude); - mVCO.set_frequency(mPresets.frequency); - EEPROM.put(0, mPresets); - } else if (data[INDEX] == ENCODER_01) { - serial_debug.println("write presets 1"); - mPresets.amplitude = 0.5; - mPresets.frequency = 220.0; - mVCO.set_amplitude(mPresets.amplitude); - mVCO.set_frequency(mPresets.frequency); - EEPROM.put(0, mPresets); - } else if (data[INDEX] == ENCODER_02) { - serial_debug.println("write presets 2"); - mPresets.amplitude = 0.6; - mPresets.frequency = 110.0; - mVCO.set_amplitude(mPresets.amplitude); - mVCO.set_frequency(mPresets.frequency); - EEPROM.put(0, mPresets); - } - } -} - -void audioblock(float* output_signal[LEFT], float* output_signal[RIGHT], - float* input_signal[LEFT], float* input_signal[RIGHT]) { - mDAC.process_frame(output_signal[LEFT], output_signal[RIGHT]); -} diff --git a/stm32/libraries/KlangstromCard/examples/ExampleListFiles/ExampleListFiles.ino b/stm32/libraries/KlangstromCard/examples/ExampleListFiles/ExampleListFiles.ino index 0291f8e1..d8ab3426 100644 --- a/stm32/libraries/KlangstromCard/examples/ExampleListFiles/ExampleListFiles.ino +++ b/stm32/libraries/KlangstromCard/examples/ExampleListFiles/ExampleListFiles.ino @@ -9,6 +9,10 @@ using namespace klangstrom; void setup() { Serial.begin(115200); + Serial.println("----------------"); + Serial.println("ExampleListFiles"); + Serial.println("----------------"); + Serial.println("--- ExampleList Files"); Serial.println(); diff --git a/stm32/libraries/KlangstromCard/examples/ExamplePlayWAVE/ExamplePlayWAVE.ino b/stm32/libraries/KlangstromCard/examples/ExamplePlayWAVE/ExamplePlayWAVE.ino new file mode 100644 index 00000000..037693f7 --- /dev/null +++ b/stm32/libraries/KlangstromCard/examples/ExamplePlayWAVE/ExamplePlayWAVE.ino @@ -0,0 +1,81 @@ +/* + * this example demonstrates how to list all files on an SD card + * and play the first file as a WAV file. + */ + +#include "KlangNodes.hpp" +#include "Klangstrom.h" +#include "KlangstromCard.h" + +using namespace klang; +using namespace klangstrom; + +NodeDAC mDAC; +NodeSamplerI16 mSampler; +NodeSamplerI16 mSamplerRight; + +void setup() { + Serial.begin(115200); + Serial.println("---------------"); + Serial.println("ExamplePlayWAVE"); + Serial.println("---------------"); + + bool mCardOpenError = Card.begin(); + if (mCardOpenError) { + Serial.println("--- opening card failed"); + return; + } + + Serial.println("--- reading file list"); + vector mFiles; + Card.get_file_list(mFiles); + + for (uint16_t i = 0; i < mFiles.size(); i++) { + Serial.print(i); + Serial.print("\t"); + Serial.println(mFiles[i]); + } + Serial.println(); + + bool mFileOpenError = Card.open(mFiles[0]); /* open first file on card */ + if (mFileOpenError) { + Serial.println("--- opening file failed"); + return; + } else { + Serial.print("--- opening file: "); + Serial.println(mFiles[0]); + } + + KlangstromWaveFile mWAV; + int mWAVOpenError = Card.load_WAV(&mWAV); + if (!mWAVOpenError) { + Serial.println("--- WAV file header: "); + Card.print_WAV_header(mWAV.header); + + Klang::connect(mSampler, Node::CH_OUT_SIGNAL, mDAC, NodeDAC::CH_IN_SIGNAL_LEFT); + mSampler.set_buffer(mWAV.get_sample_data(KlangstromWaveFile::CHANNEL_LEFT)); + mSampler.set_buffer_size(mWAV.number_of_samples); + mSampler.loop(true); + mSampler.start(); + + if (mWAV.header->NumOfChan == 2) { + mDAC.set_stereo(true); + Klang::connect(mSamplerRight, Node::CH_OUT_SIGNAL, mDAC, NodeDAC::CH_IN_SIGNAL_RIGHT); + mSamplerRight.set_buffer(mWAV.get_sample_data(KlangstromWaveFile::CHANNEL_RIGHT)); + mSamplerRight.set_buffer_size(mWAV.number_of_samples); + mSamplerRight.loop(true); + mSamplerRight.start(); + } else { + mDAC.set_stereo(false); + } + } else { + Serial.print("--- loading WAV failed : "); + Serial.println(mWAVOpenError); + } +} + +void loop() {} + +void audioblock(float** input_signal, float** output_signal) { + mDAC.process_frame(output_signal[LEFT], output_signal[RIGHT]); +} diff --git a/stm32/libraries/KlangstromCard/examples/ExampleStreamWAVE/ExampleStreamWAVE.ino b/stm32/libraries/KlangstromCard/examples/ExampleStreamWAVE/ExampleStreamWAVE.ino new file mode 100644 index 00000000..e9ed2564 --- /dev/null +++ b/stm32/libraries/KlangstromCard/examples/ExampleStreamWAVE/ExampleStreamWAVE.ino @@ -0,0 +1,71 @@ +/* + * this example demonstrates how to list all files on an SD card + * and play the first file as a WAV file. + */ + +#include "KlangNodes.hpp" +#include "Klangstrom.h" +#include "KlangstromCard.h" + +using namespace klang; +using namespace klangstrom; + +NodeDAC mDAC; +NodeSamplerI16 mSampler; + +void setup() { + Serial.begin(115200); + Serial.println("-----------------"); + Serial.println("ExampleStreamWAVE"); + Serial.println("-----------------"); + + bool mCardOpenError = Card.begin(); + if (mCardOpenError) { + Serial.println("--- opening card failed"); + return; + } + + Serial.println("--- reading file list"); + vector mFiles; + Card.get_file_list(mFiles); + + for (uint16_t i = 0; i < mFiles.size(); i++) { + Serial.print(i); + Serial.print("\t"); + Serial.println(mFiles[i]); + } + Serial.println(); + + bool mFileOpenError = Card.open(mFiles[0]); /* open first file on card */ + if (mFileOpenError) { + Serial.println("--- opening file failed"); + return; + } else { + Serial.print("--- opening file: "); + Serial.println(mFiles[0]); + } + + KlangstromWaveFile mWAV; + int mWAVOpenError = Card.load_WAV(&mWAV); + if (!mWAVOpenError) { + Serial.println("--- WAV file header: "); + Card.print_WAV_header(mWAV.header); + + Klang::connect(mSampler, Node::CH_OUT_SIGNAL, mDAC, NodeDAC::CH_IN_SIGNAL_LEFT); + mSampler.set_buffer(mWAV.get_sample_data(KlangstromWaveFile::CHANNEL_LEFT)); + mSampler.set_buffer_size(mWAV.number_of_samples); + mSampler.loop(true); + mSampler.start(); + + mDAC.set_stereo(false); + } else { + Serial.print("--- loading WAV failed : "); + Serial.println(mWAVOpenError); + } +} + +void loop() {} + +void audioblock(float** input_signal, float** output_signal) { + mDAC.process_frame(output_signal[LEFT], output_signal[RIGHT]); +} diff --git a/stm32/libraries/KlangstromCard/examples/ExampleWAVFileInfo/ExampleWAVFileInfo.ino b/stm32/libraries/KlangstromCard/examples/ExampleWAVFileInfo/ExampleWAVFileInfo.ino index bd28f109..9e6cd6d1 100644 --- a/stm32/libraries/KlangstromCard/examples/ExampleWAVFileInfo/ExampleWAVFileInfo.ino +++ b/stm32/libraries/KlangstromCard/examples/ExampleWAVFileInfo/ExampleWAVFileInfo.ino @@ -42,8 +42,9 @@ void read_WAV_header() { void setup() { Serial.begin(115200); - Serial.println("--- Example WAV File Info"); - Serial.println(); + Serial.println("------------------"); + Serial.println("ExampleWAVFileInfo"); + Serial.println("------------------"); bool mOpenCardError = Card.begin(); vector mFiles; diff --git a/stm32/libraries/KlangstromCard/examples/ExampleWriteRawData/ExampleWriteRawData.ino b/stm32/libraries/KlangstromCard/examples/ExampleWriteRawData/ExampleWriteRawData.ino index f638e1ed..cfe605a9 100644 --- a/stm32/libraries/KlangstromCard/examples/ExampleWriteRawData/ExampleWriteRawData.ino +++ b/stm32/libraries/KlangstromCard/examples/ExampleWriteRawData/ExampleWriteRawData.ino @@ -1,5 +1,5 @@ -#include "Klangstrom.h" #include "KlangNodes.hpp" +#include "Klangstrom.h" #include "KlangstromCard.h" using namespace klang; @@ -25,8 +25,9 @@ void print_file_list(vector& pFiles) { void setup() { Serial.begin(115200); - Serial.println("--- Example Write Raw Data"); - Serial.println(); + Serial.println("-------------------"); + Serial.println("ExampleWriteRawData"); + Serial.println("-------------------"); bool mOpenCardError = Card.begin(); vector mFiles; @@ -81,8 +82,7 @@ void loop() { } } -void audioblock(float* output_signal[LEFT], float* output_signal[RIGHT], - float* input_signal[LEFT], float* input_signal[RIGHT]) { +void audioblock(float** input_signal, float** output_signal) { mDAC.process_frame(output_signal[LEFT], output_signal[RIGHT]); if (!fBufferUpdated) { KLANG_COPY_AUDIO_BUFFER(mBuffer, output_signal[LEFT]); diff --git a/stm32/libraries/KlangstromCard/src/KlangstromCard.cpp b/stm32/libraries/KlangstromCard/src/KlangstromCard.cpp index cda75c2a..3e70f51a 100644 --- a/stm32/libraries/KlangstromCard/src/KlangstromCard.cpp +++ b/stm32/libraries/KlangstromCard/src/KlangstromCard.cpp @@ -19,12 +19,22 @@ #include "KlangstromDefinesArduino.h" -#if KLST_ARCH == KLST_ARCH_MCU +#define KLST_ARCH_MCU 1 +#define KLST_ARCH_CPU 2 +#define KLST_ARCH_DESKTOP 2 +#define KLST_ARCH_VCV 3 +#define KLST_ARCH_PLUGIN 3 + +#ifndef KLST_ARCH +#warning "KLST_ARCH not defined" +#endif + +#if (KLST_ARCH == KLST_ARCH_MCU) #include "KlangstromCardBSP_STM32.h" klangstrom::KlangstromCard *CardPtr = new klangstrom::KlangstromCardBSP_STM32(); #elif KLST_ARCH == KLST_ARCH_CPU #include "KlangstromCardBSP_SDL.h" klangstrom::KlangstromCard *CardPtr = new klangstrom::KlangstromCardBSP_SDL(); #else -#warning "KLST_ARCH not defined" +#warning "KLST_ARCH not defined properly" #endif diff --git a/stm32/libraries/KlangstromCard/src/KlangstromCardBSP_STM32.h b/stm32/libraries/KlangstromCard/src/KlangstromCardBSP_STM32.h index 29529a06..58736dca 100644 --- a/stm32/libraries/KlangstromCard/src/KlangstromCardBSP_STM32.h +++ b/stm32/libraries/KlangstromCard/src/KlangstromCardBSP_STM32.h @@ -82,14 +82,15 @@ namespace klangstrom { private: SPISettings m_spiSettings; SPIClass mSPI = SPIClass( - SDCARD_SPI_MOSI, - SDCARD_SPI_MISO, - SDCARD_SPI_SCK, - SDCARD_SPI_CS); + SDCARD_SPI_MOSI, + SDCARD_SPI_MISO, + SDCARD_SPI_SCK, + SDCARD_SPI_CS); }; class KlangstromCardBSP_STM32 : public KlangstromCard { public: + KlangstromCardBSP_STM32(){}; bool begin(); bool begin(const char *pFolderPath); void get_file_list(vector &pFiles, bool pIncludeHiddenFiles = false); @@ -98,7 +99,7 @@ namespace klangstrom { void close(); void print_WAV_header(WaveHeader_t *mHeader); - int create_file(const String pFileName); + int create_file(const String pFileName); protected: int BSP_read_block(uint8_t *pReadBuffer, uint32_t pReadBufferSize); diff --git a/stm32/libraries/KlangstromDisplay/examples/CommandLineInterface/CommandLineInterface.ino b/stm32/libraries/KlangstromDisplay/examples/CommandLineInterface/CommandLineInterface.ino index 625af4f2..b0ab063c 100644 --- a/stm32/libraries/KlangstromDisplay/examples/CommandLineInterface/CommandLineInterface.ino +++ b/stm32/libraries/KlangstromDisplay/examples/CommandLineInterface/CommandLineInterface.ino @@ -9,6 +9,11 @@ KlangstromDisplay &Display = KlangstromDisplay::create(); KlangstromDisplayCLI &CLI = KlangstromDisplayCLI::create(&Display, &Font_7x10); void setup() { + Serial.begin(115200); + Serial.println("--------------------"); + Serial.println("CommandLineInterface"); + Serial.println("--------------------"); + Serial.begin(115200); Display.begin(); diff --git a/stm32/libraries/KlangstromDisplay/examples/DrawBuffer/DrawBuffer.ino b/stm32/libraries/KlangstromDisplay/examples/DrawBuffer/DrawBuffer.ino index e7571172..facf817f 100644 --- a/stm32/libraries/KlangstromDisplay/examples/DrawBuffer/DrawBuffer.ino +++ b/stm32/libraries/KlangstromDisplay/examples/DrawBuffer/DrawBuffer.ino @@ -1,5 +1,5 @@ -#include "Klangstrom.h" #include "KlangNodes.hpp" +#include "Klangstrom.h" #include "KlangstromDisplay.h" #include "KlangstromDisplayDrawBuffer.h" #include "KlangstromDisplayFont_5x8.h" @@ -14,6 +14,11 @@ KlangstromDisplayDrawBuffer fDrawBuffer(32); KlangstromDisplay& Display = KlangstromDisplay::create(); void setup() { + Serial.begin(115200); + Serial.println("----------"); + Serial.println("DrawBuffer"); + Serial.println("----------"); + Display.begin(); Display.background(0, 0, 0); Display.color(255, 255, 255); @@ -53,8 +58,7 @@ void loop() { delay(300); // wait for a second } -void audioblock(float* output_signal[LEFT], float* output_signal[RIGHT], - float* input_signal[LEFT], float* input_signal[RIGHT]) { +void audioblock(float** input_signal, float** output_signal) { fDAC.process_frame(output_signal[LEFT], output_signal[RIGHT]); fDrawBuffer.update_buffer(output_signal[RIGHT], KLANG_SAMPLES_PER_AUDIO_BLOCK); } diff --git a/stm32/libraries/KlangstromDisplay/examples/DrawPrimitives/DrawPrimitives.ino b/stm32/libraries/KlangstromDisplay/examples/DrawPrimitives/DrawPrimitives.ino index 412cbe3e..d4ee1d69 100644 --- a/stm32/libraries/KlangstromDisplay/examples/DrawPrimitives/DrawPrimitives.ino +++ b/stm32/libraries/KlangstromDisplay/examples/DrawPrimitives/DrawPrimitives.ino @@ -1,5 +1,5 @@ -#include "Klangstrom.h" #include "CycleCounter.h" +#include "Klangstrom.h" #include "KlangstromDisplay.h" #include "KlangstromDisplayDrawBuffer.h" #include "KlangstromDisplayFont_5x8.h" @@ -90,6 +90,11 @@ void draw_primitves() { } void setup() { + Serial.begin(115200); + Serial.println("--------------"); + Serial.println("DrawPrimitives"); + Serial.println("--------------"); + Display.begin(); klst_enable_cycle_counter(); } diff --git a/stm32/libraries/KlangstromDisplay/examples/Fonts/Fonts.ino b/stm32/libraries/KlangstromDisplay/examples/Fonts/Fonts.ino index 2ea22c2b..c435cea9 100644 --- a/stm32/libraries/KlangstromDisplay/examples/Fonts/Fonts.ino +++ b/stm32/libraries/KlangstromDisplay/examples/Fonts/Fonts.ino @@ -40,6 +40,11 @@ void draw_primitves() { } void setup() { + Serial.begin(115200); + Serial.println("-----"); + Serial.println("Fonts"); + Serial.println("-----"); + Display.begin(); } diff --git a/stm32/libraries/KlangstromDisplay/examples/FullscreenAnimation/FullscreenAnimation.ino b/stm32/libraries/KlangstromDisplay/examples/FullscreenAnimation/FullscreenAnimation.ino index 1362df83..eb425fca 100644 --- a/stm32/libraries/KlangstromDisplay/examples/FullscreenAnimation/FullscreenAnimation.ino +++ b/stm32/libraries/KlangstromDisplay/examples/FullscreenAnimation/FullscreenAnimation.ino @@ -14,6 +14,11 @@ void draw_primitves() { } void setup() { + Serial.begin(115200); + Serial.println("-------------------"); + Serial.println("FullscreenAnimation"); + Serial.println("-------------------"); + Display.begin(); beats_per_minute(900); // == 15 Hz } diff --git a/stm32/libraries/KlangstromDisplay/examples/Terminal/Terminal.ino b/stm32/libraries/KlangstromDisplay/examples/Terminal/Terminal.ino index b0296853..fa9a7e74 100644 --- a/stm32/libraries/KlangstromDisplay/examples/Terminal/Terminal.ino +++ b/stm32/libraries/KlangstromDisplay/examples/Terminal/Terminal.ino @@ -10,7 +10,9 @@ KlangstromDisplayTerminal* mTerminal; void setup() { Serial.begin(115200); - Serial.println("--- KLST TERMINAL ---"); + Serial.println("--------"); + Serial.println("Terminal"); + Serial.println("--------"); Display.begin(); Display.background(0, 0, 0); diff --git a/stm32/libraries/KlangstromTest/library.properties b/stm32/libraries/KlangstromTest/library.properties new file mode 100644 index 00000000..1fe4e8a1 --- /dev/null +++ b/stm32/libraries/KlangstromTest/library.properties @@ -0,0 +1,11 @@ +name=KlangstromTest +version=1.0.0 +author=dennisppaul +maintainer= +sentence=Test for Klangstrom +paragraph= +category=Test +url= +architectures=* +includes=KlangstromTest.h +depends=KlangstromTest diff --git a/stm32/libraries/KlangstromTest/src/KlangstromClass.cpp b/stm32/libraries/KlangstromTest/src/KlangstromClass.cpp new file mode 100755 index 00000000..53f5c568 --- /dev/null +++ b/stm32/libraries/KlangstromTest/src/KlangstromClass.cpp @@ -0,0 +1,3 @@ +#include "KlangstromClass.h" + +void KlangstromTestClass::init() {} diff --git a/stm32/libraries/KlangstromTest/src/KlangstromClass.h b/stm32/libraries/KlangstromTest/src/KlangstromClass.h new file mode 100755 index 00000000..d21a709c --- /dev/null +++ b/stm32/libraries/KlangstromTest/src/KlangstromClass.h @@ -0,0 +1,6 @@ +#pragma once + +class KlangstromTestClass { +public: + void init(); +}; diff --git a/stm32/libraries/KlangstromTest/src/KlangstromTest.cpp b/stm32/libraries/KlangstromTest/src/KlangstromTest.cpp new file mode 100755 index 00000000..54a25e54 --- /dev/null +++ b/stm32/libraries/KlangstromTest/src/KlangstromTest.cpp @@ -0,0 +1,3 @@ +#include "KlangstromTest.h" + +int foobar() { return 23; } diff --git a/stm32/libraries/KlangstromTest/src/KlangstromTest.h b/stm32/libraries/KlangstromTest/src/KlangstromTest.h new file mode 100755 index 00000000..9cf21589 --- /dev/null +++ b/stm32/libraries/KlangstromTest/src/KlangstromTest.h @@ -0,0 +1,6 @@ +#pragma once + +#include "KlangstromClass.h" + +int foobar(); + diff --git a/stm32/libraries/SPI/README.md b/stm32/libraries/SPI/README.md index 18fa5c63..7a0595c1 100644 --- a/stm32/libraries/SPI/README.md +++ b/stm32/libraries/SPI/README.md @@ -1,70 +1,79 @@ ## SPI -STM32 SPI library has been modified with the possibility to manage several CS pins without to stop the SPI interface. +STM32 SPI library has been modified with the possibility to manage hardware CS pin linked to the SPI peripheral. _We do not describe here the [SPI Arduino API](https://www.arduino.cc/en/Reference/SPI) but the functionalities added._ -We give to the user 3 possibilities about the management of the CS pin: -1. the CS pin is managed directly by the user code before to transfer the data (like the Arduino SPI library) -2. the user gives the CS pin number to the library API and the library manages itself the CS pin (see example below) -3. the user uses a hardware CS pin linked to the SPI peripheral +User have 2 possibilities about the management of the CS pin: +* the CS pin is managed directly by the user code before to transfer the data (like the Arduino SPI library) +* the user uses a hardware CS pin linked to the SPI peripheral -### New API functions +## New API functions -* **`SPIClass::SPIClass(uint8_t mosi, uint8_t miso, uint8_t sclk, uint8_t ssel)`**: alternative class constructor -_Params_ SPI mosi pin -_Params_ SPI miso pin -_Params_ SPI sclk pin -_Params_ (optional) SPI ssel pin. This pin must be an hardware CS pin. If you configure this pin, the chip select will be managed by the SPI peripheral. Do not use API functions with CS pin in parameter. +#### Alternative class constructor +* `SPIClass::SPIClass(uint8_t mosi, uint8_t miso, uint8_t sclk, uint8_t ssel)` -* **`void SPIClass::begin(uint8_t _pin)`**: initialize the SPI interface and add a CS pin -_Params_ spi CS pin to be managed by the SPI library +_Param_ SPI `mosi` pin -* **`void beginTransaction(uint8_t pin, SPISettings settings)`**: allows to configure the SPI with other parameter. These new parameter are saved this an associated CS pin. -_Params_ SPI CS pin to be managed by the SPI library -_Params_ SPI settings +_Param_ SPI `miso` pin -* **`void endTransaction(uint8_t pin)`**: removes a CS pin and the SPI settings associated -_Params_ SPI CS pin managed by the SPI library +_Param_ SPI `sclk` pin -**_Note 1_** The following functions must be called after initialization of the SPI instance with `begin()` or `beginTransaction()`. -If you have several device to manage, you can call `beginTransaction()` several time with different CS pin in parameter. -Then you can call the following functions with different CS pin without call again `beginTransaction()` (until you call `end()` or `endTransaction()`). +_Params_ (optional) SPI `ssel` pin. This pin must be an hardware CS pin. If you configure this pin, the chip select will be managed by the SPI peripheral. -**_Note 2_** If the mode is set to `SPI_CONTINUE`, the CS pin is kept enabled. Be careful in case you use several CS pin. +##### Example -* **`byte transfer(uint8_t pin, uint8_t _data, SPITransferMode _mode = SPI_LAST)`**: write/read one byte -_Params_ SPI CS pin managed by the SPI library -_Params_ data to write -_Params_ (optional) if `SPI_LAST` CS pin is reset, `SPI_CONTINUE` the CS pin is kept enabled. -_Return_ byte received +This is an example of the use of the hardware CS pin linked to the SPI peripheral: -* **`uint16_t transfer16(uint8_t pin, uint16_t _data, SPITransferMode _mode = SPI_LAST)`**: write/read half-word -_Params_ SPI CS pin managed by the SPI library -_Params_ 16bits data to write -_Params_ (optional) if `SPI_LAST` CS pin is reset, `SPI_CONTINUE` the CS pin is kept enabled. -_Return_ 16bits data received +```C++ +#include +// MOSI MISO SCLK SSEL +SPIClass SPI_3(PC12, PC11, PC10, PC9); + +void setup() { + SPI_3.begin(); // Enable the SPI_3 instance with default SPISsettings + SPI_3.beginTransaction(settings); // Configure the SPI_3 instance with other settings + SPI_3.transfer(0x52); // Transfers data to the first device + SPI_3.end() //SPI_3 instance is disabled +} +``` + +#### Transfer with Tx/Rx buffer + +* `void transfer(const void *tx_buf, void *rx_buf, size_t count)` :Transfer several bytes. One constant buffer used to send and one to receive data. + + _Param_ `tx_buf`: constant array of Tx bytes that is filled by the user before starting the SPI transfer. If NULL, default dummy 0xFF bytes will be clocked out. -* **`void transfer(uint8_t pin, void *_buf, size_t _count, SPITransferMode _mode = SPI_LAST)`**: write/read several bytes. Only one buffer used to write and read the data -_Params_ SPI CS pin managed by the SPI library -_Params_ pointer to data to write. The data will be replaced by the data read. -_Params_ number of data to write/read. -_Params_ (optional) if `SPI_LAST` CS pin is reset, `SPI_CONTINUE` the CS pin is kept enabled. + _Param_ `rx_buf`: array of Rx bytes that will be filled by the slave during the SPI transfer. If NULL, the received data will be discarded. -* **`void transfer(byte _pin, void *_bufout, void *_bufin, size_t _count, SPITransferMode _mode = SPI_LAST)`**: write/read several bytes. One buffer for the output data and one for the input data -_Params_ SPI CS pin managed by the SPI library -_Params_ pointer to data to write. -_Params_ pointer where to store the data read. -_Params_ number of data to write/read. -_Params_ (optional) if `SPI_LAST` CS pin is reset, `SPI_CONTINUE` the CS pin is kept enabled. + _Param_ `count`: number of bytes to send/receive. -### Example +#### Change default `SPI` instance pins +It is also possible to change the default pins used by the `SPI` instance using above API: -This is an example of the use of the CS pin management: +> [!WARNING] +> **Have to be called before `begin()`.** +* `void setMISO(uint32_t miso)` +* `void setMOSI(uint32_t mosi)` +* `void setSCLK(uint32_t sclk)` +* `void setSSEL(uint32_t ssel)` +* `void setMISO(PinName miso)` +* `void setMOSI(PinName mosi)` +* `void setSCLK(PinName sclk)` +* `void setSSEL(PinName ssel)` + +> [!NOTE] +> Using `setSSEL()` allows to enable hardware CS pin management linked to the SPI peripheral. + +##### Example: ```C++ -SPI.begin(2); //Enables the SPI instance with default settings and attaches the CS pin -SPI.beginTransaction(1, settings); //Attaches another CS pin and configure the SPI instance with other settings -SPI.transfer(1, 0x52); //Transfers data to the first device -SPI.transfer(2, 0xA4); //Transfers data to the second device. The SPI instance is configured with the right settings -SPI.end() //SPI instance is disabled + SPI.setMISO(PC_4); // using pin name PY_n + SPI.setMOSI(PC2); // using pin number PYn + SPI.begin(2); ``` + +* `SPI_HandleTypeDef *getHandle(void)`: Could be used to mix Arduino API and STM32Cube HAL API (ex: DMA). **Use at your own risk.** + +## Extended API + +* All defaustatndard `transfer()` API's have a new bool argument `skipReceive`. It allows to skip receive data after transmitting. Value can be `SPI_TRANSMITRECEIVE` or `SPI_TRANSMITONLY`. Default `SPI_TRANSMITRECEIVE`. diff --git a/stm32/libraries/SPI/library.properties b/stm32/libraries/SPI/library.properties index 0edf2975..a4bdcbd0 100644 --- a/stm32/libraries/SPI/library.properties +++ b/stm32/libraries/SPI/library.properties @@ -1,6 +1,6 @@ name=SPI -version=1.0.0 -author=Arduino, Wi6Labs +version=1.1.0 +author=Arduino, Frederic Pillon maintainer=stm32duino sentence=Enables the communication with devices that use the Serial Peripheral Interface (SPI) Bus. paragraph=This library is based on the official Arduino SPI library and adapted to STM32 boards. diff --git a/stm32/libraries/SPI/src/SPI.cpp b/stm32/libraries/SPI/src/SPI.cpp index eab07921..2c6e798b 100644 --- a/stm32/libraries/SPI/src/SPI.cpp +++ b/stm32/libraries/SPI/src/SPI.cpp @@ -16,7 +16,7 @@ SPIClass SPI; /** * @brief Default constructor. Uses pin configuration of variant.h. */ -SPIClass::SPIClass() : _CSPinConfig(NO_CONFIG) +SPIClass::SPIClass() { _spi.pin_miso = digitalPinToPinName(MISO); _spi.pin_mosi = digitalPinToPinName(MOSI); @@ -43,7 +43,7 @@ SPIClass::SPIClass() : _CSPinConfig(NO_CONFIG) * another CS pin and don't pass a CS pin as parameter to any functions * of the class. */ -SPIClass::SPIClass(uint32_t mosi, uint32_t miso, uint32_t sclk, uint32_t ssel) : _CSPinConfig(NO_CONFIG) +SPIClass::SPIClass(uint32_t mosi, uint32_t miso, uint32_t sclk, uint32_t ssel) { _spi.pin_miso = digitalPinToPinName(miso); _spi.pin_mosi = digitalPinToPinName(mosi); @@ -53,114 +53,65 @@ SPIClass::SPIClass(uint32_t mosi, uint32_t miso, uint32_t sclk, uint32_t ssel) : /** * @brief Initialize the SPI instance. - * @param _pin: chip select pin (optional). If this parameter is filled, - * it gives the management of the CS pin to the SPI class. In this case - * do not manage the CS pin outside of the SPI class. */ -void SPIClass::begin(uint8_t _pin) +void SPIClass::begin(void) { - uint8_t idx = pinIdx(_pin, ADD_NEW_PIN); - if (idx >= NB_SPI_SETTINGS) { - return; - } - - if ((_pin != CS_PIN_CONTROLLED_BY_USER) && (_spi.pin_ssel == NC)) { - pinMode(_pin, OUTPUT); - digitalWrite(_pin, HIGH); - } - _spi.handle.State = HAL_SPI_STATE_RESET; - spi_init(&_spi, spiSettings[idx].clk, - spiSettings[idx].dMode, - spiSettings[idx].bOrder); - _CSPinConfig = _pin; -#if __has_include("WiFi.h") - // Wait wifi shield initialization. - // Should be better to do in SpiDrv::begin() of WiFi library but it seems - // there is no more update on this library as shield is retired. - delay(2000); -#endif - + _spiSettings = SPISettings(); + spi_init(&_spi, _spiSettings.clockFreq, + _spiSettings.dataMode, + _spiSettings.bitOrder); } /** * @brief This function should be used to configure the SPI instance in case you * don't use the default parameters set by the begin() function. - * @param _pin: CS pin (optional). This pin will be attached with the settings. * @param settings: SPI settings(clock speed, bit order, data mode). - * @Note For each SPI instance you are able to manage until NB_SPI_SETTINGS - * devices attached to the same SPI peripheral. You need to indicate the - * CS pin used to the transfer() function and the SPI instance will be - * configured with the right settings. */ -void SPIClass::beginTransaction(uint8_t _pin, SPISettings settings) +void SPIClass::beginTransaction(SPISettings settings) { - uint8_t idx = pinIdx(_pin, ADD_NEW_PIN); - if (idx >= NB_SPI_SETTINGS) { - return; + if (_spiSettings != settings) { + _spiSettings = settings; + spi_init(&_spi, _spiSettings.clockFreq, + _spiSettings.dataMode, + _spiSettings.bitOrder); } - - spiSettings[idx].clk = settings.clk; - spiSettings[idx].dMode = settings.dMode; - spiSettings[idx].bOrder = settings.bOrder; - spiSettings[idx].noReceive = settings.noReceive; - - if ((_pin != CS_PIN_CONTROLLED_BY_USER) && (_spi.pin_ssel == NC)) { - pinMode(_pin, OUTPUT); - digitalWrite(_pin, HIGH); - } - - spi_init(&_spi, spiSettings[idx].clk, - spiSettings[idx].dMode, - spiSettings[idx].bOrder); - _CSPinConfig = _pin; } /** - * @brief Remove the CS pin and the settings associated to the SPI instance. - * @param _pin: CS pin (optional) + * @brief End the transaction after beginTransaction usage */ -void SPIClass::endTransaction(uint8_t _pin) +void SPIClass::endTransaction(void) { - RemovePin(_pin); - _CSPinConfig = NO_CONFIG; + } /** * @brief Deinitialize the SPI instance and stop it. */ -void SPIClass::end() +void SPIClass::end(void) { spi_deinit(&_spi); - RemoveAllPin(); - _CSPinConfig = NO_CONFIG; } /** * @brief Deprecated function. * Configure the bit order: MSB first or LSB first. - * @param _pin: CS pin associated to a configuration (optional). - * @param _bitOrder: MSBFIRST or LSBFIRST + * @param bitOrder: MSBFIRST or LSBFIRST */ -void SPIClass::setBitOrder(uint8_t _pin, BitOrder _bitOrder) +void SPIClass::setBitOrder(BitOrder bitOrder) { - uint8_t idx = pinIdx(_pin, GET_IDX); - if (idx >= NB_SPI_SETTINGS) { - return; - } + _spiSettings.bitOrder = bitOrder; - spiSettings[idx].bOrder = _bitOrder; - - spi_init(&_spi, spiSettings[idx].clk, - spiSettings[idx].dMode, - spiSettings[idx].bOrder); + spi_init(&_spi, _spiSettings.clockFreq, + _spiSettings.dataMode, + _spiSettings.bitOrder); } /** * @brief Deprecated function. * Configure the data mode (clock polarity and clock phase) - * @param _pin: CS pin associated to a configuration (optional). - * @param _mode: SPI_MODE0, SPI_MODE1, SPI_MODE2 or SPI_MODE3 + * @param mode: SPI_MODE0, SPI_MODE1, SPI_MODE2 or SPI_MODE3 * @note * Mode Clock Polarity (CPOL) Clock Phase (CPHA) * SPI_MODE0 0 0 @@ -168,240 +119,126 @@ void SPIClass::setBitOrder(uint8_t _pin, BitOrder _bitOrder) * SPI_MODE2 1 0 * SPI_MODE3 1 1 */ -void SPIClass::setDataMode(uint8_t _pin, uint8_t _mode) +void SPIClass::setDataMode(uint8_t mode) { - uint8_t idx = pinIdx(_pin, GET_IDX); - if (idx >= NB_SPI_SETTINGS) { - return; - } - - if (SPI_MODE0 == _mode) { - spiSettings[idx].dMode = SPI_MODE_0; - } else if (SPI_MODE1 == _mode) { - spiSettings[idx].dMode = SPI_MODE_1; - } else if (SPI_MODE2 == _mode) { - spiSettings[idx].dMode = SPI_MODE_2; - } else if (SPI_MODE3 == _mode) { - spiSettings[idx].dMode = SPI_MODE_3; - } + setDataMode((SPIMode)mode); +} - spi_init(&_spi, spiSettings[idx].clk, - spiSettings[idx].dMode, - spiSettings[idx].bOrder); +void SPIClass::setDataMode(SPIMode mode) +{ + _spiSettings.dataMode = mode; + spi_init(&_spi, _spiSettings.clockFreq, + _spiSettings.dataMode, + _spiSettings.bitOrder); } /** * @brief Deprecated function. * Configure the clock speed - * @param _pin: CS pin associated to a configuration (optional). - * @param _divider: the SPI clock can be divided by values from 1 to 255. + * @param divider: the SPI clock can be divided by values from 1 to 255. * If 0, default SPI speed is used. */ -void SPIClass::setClockDivider(uint8_t _pin, uint8_t _divider) +void SPIClass::setClockDivider(uint8_t divider) { - uint8_t idx = pinIdx(_pin, GET_IDX); - if (idx >= NB_SPI_SETTINGS) { - return; - } - if (_divider == 0) { - spiSettings[idx].clk = SPI_SPEED_CLOCK_DEFAULT; + if (divider == 0) { + _spiSettings.clockFreq = SPI_SPEED_CLOCK_DEFAULT; } else { /* Get clk freq of the SPI instance and compute it */ - spiSettings[idx].clk = spi_getClkFreq(&_spi) / _divider; + _spiSettings.clockFreq = spi_getClkFreq(&_spi) / divider; } - spi_init(&_spi, spiSettings[idx].clk, - spiSettings[idx].dMode, - spiSettings[idx].bOrder); + spi_init(&_spi, _spiSettings.clockFreq, + _spiSettings.dataMode, + _spiSettings.bitOrder); } /** * @brief Transfer one byte on the SPI bus. * begin() or beginTransaction() must be called at least once before. - * @param _pin: CS pin to select a device (optional). If the previous transfer - * used another CS pin then the SPI instance will be reconfigured. * @param data: byte to send. - * @param _mode: (optional) can be SPI_CONTINUE in case of multiple successive - * send or SPI_LAST to indicate the end of send. - * If the _mode is set to SPI_CONTINUE, keep the SPI instance alive. - * That means the CS pin is not reset. Be careful in case you use - * several CS pin. + * @param skipReceive: skip receiving data after transmit or not. + * SPI_TRANSMITRECEIVE or SPI_TRANSMITONLY. + * Optional, default: SPI_TRANSMITRECEIVE. * @return byte received from the slave. */ -byte SPIClass::transfer(uint8_t _pin, uint8_t data, SPITransferMode _mode) +uint8_t SPIClass::transfer(uint8_t data, bool skipReceive) { - uint8_t rx_buffer = 0; - - uint8_t idx = pinIdx(_pin, GET_IDX); - if (idx >= NB_SPI_SETTINGS) { - return rx_buffer; - } - - if (_pin != _CSPinConfig) { - spi_init(&_spi, spiSettings[idx].clk, - spiSettings[idx].dMode, - spiSettings[idx].bOrder); - _CSPinConfig = _pin; - } - - if ((_pin != CS_PIN_CONTROLLED_BY_USER) && (_spi.pin_ssel == NC)) { - digitalWrite(_pin, LOW); - } - - spi_transfer(&_spi, &data, &rx_buffer, sizeof(uint8_t), SPI_TRANSFER_TIMEOUT, spiSettings[idx].noReceive); - - if ((_pin != CS_PIN_CONTROLLED_BY_USER) && (_mode == SPI_LAST) && (_spi.pin_ssel == NC)) { - digitalWrite(_pin, HIGH); - } - - return rx_buffer; + spi_transfer(&_spi, &data, (!skipReceive) ? &data : NULL, sizeof(uint8_t)); + return data; } /** * @brief Transfer two bytes on the SPI bus in 16 bits format. * begin() or beginTransaction() must be called at least once before. - * @param _pin: CS pin to select a device (optional). If the previous transfer - * used another CS pin then the SPI instance will be reconfigured. * @param data: bytes to send. - * @param _mode: (optional) can be SPI_CONTINUE in case of multiple successive - * send or SPI_LAST to indicate the end of send. - * If the _mode is set to SPI_CONTINUE, keep the SPI instance alive. - * That means the CS pin is not reset. Be careful in case you use - * several CS pin. + * @param skipReceive: skip receiving data after transmit or not. + * SPI_TRANSMITRECEIVE or SPI_TRANSMITONLY. + * Optional, default: SPI_TRANSMITRECEIVE. * @return bytes received from the slave in 16 bits format. */ -uint16_t SPIClass::transfer16(uint8_t _pin, uint16_t data, SPITransferMode _mode) +uint16_t SPIClass::transfer16(uint16_t data, bool skipReceive) { - uint16_t rx_buffer = 0; uint16_t tmp; - uint8_t idx = pinIdx(_pin, GET_IDX); - if (idx >= NB_SPI_SETTINGS) { - return rx_buffer; - } - - if (_pin != _CSPinConfig) { - spi_init(&_spi, spiSettings[idx].clk, - spiSettings[idx].dMode, - spiSettings[idx].bOrder); - _CSPinConfig = _pin; - } - - if (spiSettings[idx].bOrder) { + if (_spiSettings.bitOrder) { tmp = ((data & 0xff00) >> 8) | ((data & 0xff) << 8); data = tmp; } + spi_transfer(&_spi, (uint8_t *)&data, (!skipReceive) ? (uint8_t *)&data : NULL, sizeof(uint16_t)); - if ((_pin != CS_PIN_CONTROLLED_BY_USER) && (_spi.pin_ssel == NC)) { - digitalWrite(_pin, LOW); - } - - spi_transfer(&_spi, (uint8_t *)&data, (uint8_t *)&rx_buffer, sizeof(uint16_t), - SPI_TRANSFER_TIMEOUT, spiSettings[idx].noReceive); - - if ((_pin != CS_PIN_CONTROLLED_BY_USER) && (_mode == SPI_LAST) && (_spi.pin_ssel == NC)) { - digitalWrite(_pin, HIGH); - } - - if (spiSettings[idx].bOrder) { - tmp = ((rx_buffer & 0xff00) >> 8) | ((rx_buffer & 0xff) << 8); - rx_buffer = tmp; + if (_spiSettings.bitOrder) { + tmp = ((data & 0xff00) >> 8) | ((data & 0xff) << 8); + data = tmp; } - return rx_buffer; + return data; } /** * @brief Transfer several bytes. Only one buffer used to send and receive data. * begin() or beginTransaction() must be called at least once before. - * @param _pin: CS pin to select a device (optional). If the previous transfer - * used another CS pin then the SPI instance will be reconfigured. - * @param _buf: pointer to the bytes to send. The bytes received are copy in + * @param buf: pointer to the bytes to send. The bytes received are copy in * this buffer. - * @param _count: number of bytes to send/receive. - * @param _mode: (optional) can be SPI_CONTINUE in case of multiple successive - * send or SPI_LAST to indicate the end of send. - * If the _mode is set to SPI_CONTINUE, keep the SPI instance alive. - * That means the CS pin is not reset. Be careful in case you use - * several CS pin. + * @param count: number of bytes to send/receive. + * @param skipReceive: skip receiving data after transmit or not. + * SPI_TRANSMITRECEIVE or SPI_TRANSMITONLY. + * Optional, default: SPI_TRANSMITRECEIVE. */ -void SPIClass::transfer(uint8_t _pin, void *_buf, size_t _count, SPITransferMode _mode) +void SPIClass::transfer(void *buf, size_t count, bool skipReceive) { - if ((_count == 0) || (_buf == NULL)) { - return; - } - uint8_t idx = pinIdx(_pin, GET_IDX); - if (idx >= NB_SPI_SETTINGS) { - return; - } - if (_pin != _CSPinConfig) { - - spi_init(&_spi, spiSettings[idx].clk, - spiSettings[idx].dMode, - spiSettings[idx].bOrder); - _CSPinConfig = _pin; - } - - if ((_pin != CS_PIN_CONTROLLED_BY_USER) && (_spi.pin_ssel == NC)) { - digitalWrite(_pin, LOW); - } - - spi_transfer(&_spi, ((uint8_t *)_buf), ((uint8_t *)_buf), _count, - SPI_TRANSFER_TIMEOUT, spiSettings[idx].noReceive); + spi_transfer(&_spi, (uint8_t *)buf, (!skipReceive) ? (uint8_t *)buf : NULL, count); - if ((_pin != CS_PIN_CONTROLLED_BY_USER) && (_mode == SPI_LAST) && (_spi.pin_ssel == NC)) { - digitalWrite(_pin, HIGH); - } } /** - * @brief Transfer several bytes. One buffer contains the data to send and - * another one will contains the data received. begin() or - * beginTransaction() must be called at least once before. - * @param _pin: CS pin to select a device (optional). If the previous transfer - * used another CS pin then the SPI instance will be reconfigured. - * @param _bufout: pointer to the bytes to send. - * @param _bufin: pointer to the bytes received. - * @param _count: number of bytes to send/receive. - * @param _mode: (optional) can be SPI_CONTINUE in case of multiple successive - * send or SPI_LAST to indicate the end of send. - * If the _mode is set to SPI_CONTINUE, keep the SPI instance alive. - * That means the CS pin is not reset. Be careful in case you use - * several CS pin. + * @brief Transfer several bytes. One constant buffer used to send and + * one to receive data. + * begin() or beginTransaction() must be called at least once before. + * @param tx_buf: array of Tx bytes that is filled by the user before starting + * the SPI transfer. If NULL, default dummy 0xFF bytes will be + * clocked out. + * @param rx_buf: array of Rx bytes that will be filled by the slave during + * the SPI transfer. If NULL, the received data will be discarded. + * @param count: number of bytes to send/receive. */ -void SPIClass::transfer(byte _pin, void *_bufout, void *_bufin, size_t _count, SPITransferMode _mode) +void SPIClass::transfer(const void *tx_buf, void *rx_buf, size_t count) { - if ((_count == 0) || (_bufout == NULL) || (_bufin == NULL)) { - return; - } - uint8_t idx = pinIdx(_pin, GET_IDX); - if (idx >= NB_SPI_SETTINGS) { - return; - } - - if (_pin != _CSPinConfig) { - spi_init(&_spi, spiSettings[idx].clk, - spiSettings[idx].dMode, - spiSettings[idx].bOrder); - _CSPinConfig = _pin; - } - - if ((_pin != CS_PIN_CONTROLLED_BY_USER) && (_spi.pin_ssel == NC)) { - digitalWrite(_pin, LOW); - } + spi_transfer(&_spi, ((const uint8_t *)tx_buf), ((uint8_t *)rx_buf), count); +} - spi_transfer(&_spi, ((uint8_t *)_bufout), ((uint8_t *)_bufin), _count, - SPI_TRANSFER_TIMEOUT, spiSettings[idx].noReceive); - if ((_pin != CS_PIN_CONTROLLED_BY_USER) && (_mode == SPI_LAST) && (_spi.pin_ssel == NC)) { - digitalWrite(_pin, HIGH); - } +/** + * @brief Not implemented. + */ +void SPIClass::usingInterrupt(int interruptNumber) +{ + UNUSED(interruptNumber); } /** * @brief Not implemented. */ -void SPIClass::usingInterrupt(uint8_t interruptNumber) +void SPIClass::notUsingInterrupt(int interruptNumber) { UNUSED(interruptNumber); } @@ -423,70 +260,6 @@ void SPIClass::detachInterrupt(void) } #if defined(SUBGHZSPI_BASE) -void SUBGHZSPIClass::begin(uint8_t _pin) -{ - if (_pin != CS_PIN_CONTROLLED_BY_USER) { - LL_PWR_UnselectSUBGHZSPI_NSS(); - } - SPIClass::begin(CS_PIN_CONTROLLED_BY_USER); -} - -void SUBGHZSPIClass::beginTransaction(uint8_t _pin, SPISettings settings) -{ - if (_pin != CS_PIN_CONTROLLED_BY_USER) { - LL_PWR_UnselectSUBGHZSPI_NSS(); - } - SPIClass::beginTransaction(CS_PIN_CONTROLLED_BY_USER, settings); -} - -byte SUBGHZSPIClass::transfer(uint8_t _pin, uint8_t _data, SPITransferMode _mode) -{ - byte res; - if (_pin != CS_PIN_CONTROLLED_BY_USER) { - LL_PWR_SelectSUBGHZSPI_NSS(); - } - res = SPIClass::transfer(CS_PIN_CONTROLLED_BY_USER, _data, _mode); - if (_pin != CS_PIN_CONTROLLED_BY_USER) { - LL_PWR_UnselectSUBGHZSPI_NSS(); - } - return res; -} - -uint16_t SUBGHZSPIClass::transfer16(uint8_t _pin, uint16_t _data, SPITransferMode _mode) -{ - uint16_t rx_buffer = 0; - if (_pin != CS_PIN_CONTROLLED_BY_USER) { - LL_PWR_SelectSUBGHZSPI_NSS(); - } - SPIClass::transfer16(CS_PIN_CONTROLLED_BY_USER, _data, _mode); - if (_pin != CS_PIN_CONTROLLED_BY_USER) { - LL_PWR_UnselectSUBGHZSPI_NSS(); - } - return rx_buffer; -} - -void SUBGHZSPIClass::transfer(uint8_t _pin, void *_buf, size_t _count, SPITransferMode _mode) -{ - if (_pin != CS_PIN_CONTROLLED_BY_USER) { - LL_PWR_SelectSUBGHZSPI_NSS(); - } - SPIClass::transfer(CS_PIN_CONTROLLED_BY_USER, _buf, _count, _mode); - if (_pin != CS_PIN_CONTROLLED_BY_USER) { - LL_PWR_UnselectSUBGHZSPI_NSS(); - } -} - -void SUBGHZSPIClass::transfer(byte _pin, void *_bufout, void *_bufin, size_t _count, SPITransferMode _mode) -{ - if (_pin != CS_PIN_CONTROLLED_BY_USER) { - LL_PWR_SelectSUBGHZSPI_NSS(); - } - SPIClass::transfer(CS_PIN_CONTROLLED_BY_USER, _bufout, _bufin, _count, _mode); - if (_pin != CS_PIN_CONTROLLED_BY_USER) { - LL_PWR_UnselectSUBGHZSPI_NSS(); - } -} - void SUBGHZSPIClass::enableDebugPins(uint32_t mosi, uint32_t miso, uint32_t sclk, uint32_t ssel) { /* Configure SPI GPIO pins */ diff --git a/stm32/libraries/SPI/src/SPI.h b/stm32/libraries/SPI/src/SPI.h index 7bde7903..1105991d 100644 --- a/stm32/libraries/SPI/src/SPI.h +++ b/stm32/libraries/SPI/src/SPI.h @@ -38,74 +38,48 @@ extern "C" { #define SPI_CLOCK_DIV64 64 #define SPI_CLOCK_DIV128 128 -// SPI mode parameters for SPISettings -#define SPI_MODE0 0x00 -#define SPI_MODE1 0x01 -#define SPI_MODE2 0x02 -#define SPI_MODE3 0x03 - -#define SPI_TRANSMITRECEIVE 0x0 -#define SPI_TRANSMITONLY 0x1 - -// Transfer mode -enum SPITransferMode { - SPI_CONTINUE, /* Transfer not finished: CS pin kept active */ - SPI_LAST /* Transfer ended: CS pin released */ -}; - -// Indicates the user controls himself the CS pin outside of the spi class -#define CS_PIN_CONTROLLED_BY_USER NUM_DIGITAL_PINS - -// Indicates there is no configuration selected -#define NO_CONFIG ((int16_t)(-1)) - -// Defines a default timeout delay in milliseconds for the SPI transfer -#ifndef SPI_TRANSFER_TIMEOUT - #define SPI_TRANSFER_TIMEOUT 1000 -#endif - -/* - * Defines the number of settings saved per SPI instance. Must be in range 1 to 254. - * Can be redefined in variant.h - */ -#ifndef NB_SPI_SETTINGS - #define NB_SPI_SETTINGS 4 -#endif +#define SPI_TRANSMITRECEIVE false +#define SPI_TRANSMITONLY true class SPISettings { public: - constexpr SPISettings(uint32_t clock, BitOrder bitOrder, uint8_t dataMode, bool noRecv = SPI_TRANSMITRECEIVE) - : pinCS(-1), - clk(clock), - bOrder(bitOrder), - dMode((spi_mode_e)( - (SPI_MODE0 == dataMode) ? SPI_MODE_0 : - (SPI_MODE1 == dataMode) ? SPI_MODE_1 : - (SPI_MODE2 == dataMode) ? SPI_MODE_2 : - (SPI_MODE3 == dataMode) ? SPI_MODE_3 : - SPI_MODE0 - )), - noReceive(noRecv) + constexpr SPISettings(uint32_t clock, BitOrder bitOrder, uint8_t dataMode) + : clockFreq(clock), + bitOrder(bitOrder), + dataMode((SPIMode)dataMode) + { } + constexpr SPISettings(uint32_t clock, BitOrder bitOrder, SPIMode dataMode) + : clockFreq(clock), + bitOrder(bitOrder), + dataMode(dataMode) { } constexpr SPISettings() - : pinCS(-1), - clk(SPI_SPEED_CLOCK_DEFAULT), - bOrder(MSBFIRST), - dMode(SPI_MODE_0), - noReceive(SPI_TRANSMITRECEIVE) + : clockFreq(SPI_SPEED_CLOCK_DEFAULT), + bitOrder(MSBFIRST), + dataMode(SPI_MODE0) { } + + bool operator==(const SPISettings &rhs) const + { + if ((this->clockFreq == rhs.clockFreq) && + (this->bitOrder == rhs.bitOrder) && + (this->dataMode == rhs.dataMode)) { + return true; + } + return false; + } + + bool operator!=(const SPISettings &rhs) const + { + return !(*this == rhs); + } + private: - int16_t pinCS; //CS pin associated to the configuration - uint32_t clk; //specifies the spi bus maximum clock speed - BitOrder bOrder; //bit order (MSBFirst or LSBFirst) - spi_mode_e dMode; //one of the data mode - //Mode Clock Polarity (CPOL) Clock Phase (CPHA) - //SPI_MODE0 0 0 - //SPI_MODE1 0 1 - //SPI_MODE2 1 0 - //SPI_MODE3 1 1 + uint32_t clockFreq; //specifies the spi bus maximum clock speed + BitOrder bitOrder; //bit order (MSBFirst or LSBFirst) + SPIMode dataMode; //one of the data mode + friend class SPIClass; - bool noReceive; }; class SPIClass { @@ -148,79 +122,38 @@ class SPIClass { _spi.pin_ssel = (ssel); }; - virtual void begin(uint8_t _pin = CS_PIN_CONTROLLED_BY_USER); + void begin(void); void end(void); /* This function should be used to configure the SPI instance in case you * don't use default parameters. - * You can attach another CS pin to the SPI instance and each CS pin can be - * attach with specific SPI settings. */ - virtual void beginTransaction(uint8_t pin, SPISettings settings); - void beginTransaction(SPISettings settings) - { - beginTransaction(CS_PIN_CONTROLLED_BY_USER, settings); - } - - void endTransaction(uint8_t pin); - void endTransaction(void) - { - endTransaction(CS_PIN_CONTROLLED_BY_USER); - } + void beginTransaction(SPISettings settings); + void endTransaction(void); /* Transfer functions: must be called after initialization of the SPI * instance with begin() or beginTransaction(). - * You can specify the CS pin to use. */ - virtual byte transfer(uint8_t pin, uint8_t _data, SPITransferMode _mode = SPI_LAST); - virtual uint16_t transfer16(uint8_t pin, uint16_t _data, SPITransferMode _mode = SPI_LAST); - virtual void transfer(uint8_t pin, void *_buf, size_t _count, SPITransferMode _mode = SPI_LAST); - virtual void transfer(byte _pin, void *_bufout, void *_bufin, size_t _count, SPITransferMode _mode = SPI_LAST); - - // Transfer functions when user controls himself the CS pin. - byte transfer(uint8_t _data, SPITransferMode _mode = SPI_LAST) - { - return transfer(CS_PIN_CONTROLLED_BY_USER, _data, _mode); - } - - uint16_t transfer16(uint16_t _data, SPITransferMode _mode = SPI_LAST) - { - return transfer16(CS_PIN_CONTROLLED_BY_USER, _data, _mode); - } - - void transfer(void *_buf, size_t _count, SPITransferMode _mode = SPI_LAST) - { - transfer(CS_PIN_CONTROLLED_BY_USER, _buf, _count, _mode); - } + uint8_t transfer(uint8_t data, bool skipReceive = SPI_TRANSMITRECEIVE); + uint16_t transfer16(uint16_t data, bool skipReceive = SPI_TRANSMITRECEIVE); + void transfer(void *buf, size_t count, bool skipReceive = SPI_TRANSMITRECEIVE); - void transfer(void *_bufout, void *_bufin, size_t _count, SPITransferMode _mode = SPI_LAST) - { - transfer(CS_PIN_CONTROLLED_BY_USER, _bufout, _bufin, _count, _mode); - } + /* Expand SPI API + * https://github.com/arduino/ArduinoCore-API/discussions/189 + */ + void transfer(const void *tx_buf, void *rx_buf, size_t count); /* These methods are deprecated and kept for compatibility. * Use SPISettings with SPI.beginTransaction() to configure SPI parameters. */ - void setBitOrder(uint8_t _pin, BitOrder); - void setBitOrder(BitOrder _order) - { - setBitOrder(CS_PIN_CONTROLLED_BY_USER, _order); - } - - void setDataMode(uint8_t _pin, uint8_t); - void setDataMode(uint8_t _mode) - { - setDataMode(CS_PIN_CONTROLLED_BY_USER, _mode); - } - - void setClockDivider(uint8_t _pin, uint8_t); - void setClockDivider(uint8_t _div) - { - setClockDivider(CS_PIN_CONTROLLED_BY_USER, _div); - } - - // Not implemented functions. Kept for backward compatibility. - void usingInterrupt(uint8_t interruptNumber); + void setBitOrder(BitOrder); + void setDataMode(uint8_t); + void setDataMode(SPIMode); + void setClockDivider(uint8_t); + + // Not implemented functions. Kept for compatibility. + void usingInterrupt(int interruptNumber); + void notUsingInterrupt(int interruptNumber); void attachInterrupt(void); void detachInterrupt(void); @@ -235,68 +168,8 @@ class SPIClass { spi_t _spi; private: - /* Contains various spiSettings for the same spi instance. Each spi spiSettings - is associated to a CS pin. */ - SPISettings spiSettings[NB_SPI_SETTINGS]; - - // Use to know which configuration is selected. - int16_t _CSPinConfig; - - typedef enum { - GET_IDX = 0, - ADD_NEW_PIN = 1 - } pin_option_t; - - uint8_t pinIdx(uint8_t _pin, pin_option_t option) - { - uint8_t i; - - if ((_pin > NUM_DIGITAL_PINS) && (!digitalPinIsValid(_pin))) { - return NB_SPI_SETTINGS; - } - - for (i = 0; i < NB_SPI_SETTINGS; i++) { - if (_pin == spiSettings[i].pinCS) { - return i; - } - } - - if (option == ADD_NEW_PIN) { - for (i = 0; i < NB_SPI_SETTINGS; i++) { - if (spiSettings[i].pinCS == -1) { - spiSettings[i].pinCS = _pin; - return i; - } - } - } - return i; - } - - void RemovePin(uint8_t _pin) - { - if ((_pin > NUM_DIGITAL_PINS) && (!digitalPinIsValid(_pin))) { - return; - } - - for (uint8_t i = 0; i < NB_SPI_SETTINGS; i++) { - if (spiSettings[i].pinCS == _pin) { - spiSettings[i].pinCS = -1; - spiSettings[i].clk = SPI_SPEED_CLOCK_DEFAULT; - spiSettings[i].bOrder = MSBFIRST; - spiSettings[i].dMode = SPI_MODE_0; - } - } - } - - void RemoveAllPin(void) - { - for (uint8_t i = 0; i < NB_SPI_SETTINGS; i++) { - spiSettings[i].pinCS = -1; - spiSettings[i].clk = SPI_SPEED_CLOCK_DEFAULT; - spiSettings[i].bOrder = MSBFIRST; - spiSettings[i].dMode = SPI_MODE_0; - } - } + /* Current SPISettings */ + SPISettings _spiSettings = SPISettings(); }; extern SPIClass SPI; @@ -309,17 +182,7 @@ class SUBGHZSPIClass : public SPIClass { _spi.spi = SUBGHZSPI; } - void begin(uint8_t _pin = CS_PIN_CONTROLLED_BY_USER); - void beginTransaction(uint8_t pin, SPISettings settings); - byte transfer(uint8_t pin, uint8_t _data, SPITransferMode _mode = SPI_LAST); - uint16_t transfer16(uint8_t pin, uint16_t _data, SPITransferMode _mode = SPI_LAST); - void transfer(uint8_t pin, void *_buf, size_t _count, SPITransferMode _mode = SPI_LAST); - void transfer(byte _pin, void *_bufout, void *_bufin, size_t _count, SPITransferMode _mode = SPI_LAST); void enableDebugPins(uint32_t mosi = DEBUG_SUBGHZSPI_MOSI, uint32_t miso = DEBUG_SUBGHZSPI_MISO, uint32_t sclk = DEBUG_SUBGHZSPI_SCLK, uint32_t ssel = DEBUG_SUBGHZSPI_SS); - - using SPIClass::beginTransaction; - using SPIClass::transfer; - using SPIClass::transfer16; }; #endif diff --git a/stm32/libraries/SPI/src/utility/spi_com.c b/stm32/libraries/SPI/src/utility/spi_com.c index a67813e6..4adeb9d4 100644 --- a/stm32/libraries/SPI/src/utility/spi_com.c +++ b/stm32/libraries/SPI/src/utility/spi_com.c @@ -1,40 +1,13 @@ -/** - ****************************************************************************** - * @file spi_com.c - * @author WI6LABS - * @version V1.0.0 - * @date 01-August-2016 - * @brief provide the SPI interface - * - ****************************************************************************** - * @attention - * - *

© COPYRIGHT(c) 2016 STMicroelectronics

- * - * Redistribution and use in source and binary forms, with or without modification, - * are permitted provided that the following conditions are met: - * 1. Redistributions of source code must retain the above copyright notice, - * this list of conditions and the following disclaimer. - * 2. Redistributions in binary form must reproduce the above copyright notice, - * this list of conditions and the following disclaimer in the documentation - * and/or other materials provided with the distribution. - * 3. Neither the name of STMicroelectronics nor the names of its contributors - * may be used to endorse or promote products derived from this software - * without specific prior written permission. - * - * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" - * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE - * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE - * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE - * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL - * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR - * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER - * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, - * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE - * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - * - ****************************************************************************** - */ +/* + * Copyright (c) 2016 Frederic Pillon for + * STMicroelectronics. All right reserved. + * Interface utility of the spi module for arduino. + * + * This file is free software; you can redistribute it and/or modify + * it under the terms of either the GNU General Public License version 2 + * or the GNU Lesser General Public License version 2.1, both as + * published by the Free Software Foundation. + */ #include "wiring_time.h" #include "core_debug.h" #include "stm32_def.h" @@ -214,7 +187,7 @@ static uint32_t compute_disable_delay(spi_t *obj) SPI_HandleTypeDef *handle = &(obj->handle); prescaler = 1 << ((handle->Init.BaudRatePrescaler >> SPI_CFG1_MBR_Pos) + 1); - disable_delay = ((prescaler * 1000000) / spi_freq) / 2; + disable_delay = (((prescaler * 1000000) / spi_freq) / 2) + 1; return disable_delay; } #endif @@ -227,7 +200,7 @@ static uint32_t compute_disable_delay(spi_t *obj) * @param msb : set to 1 in msb first * @retval None */ -void spi_init(spi_t *obj, uint32_t speed, spi_mode_e mode, uint8_t msb) +void spi_init(spi_t *obj, uint32_t speed, SPIMode mode, uint8_t msb) { if (obj == NULL) { return; @@ -313,13 +286,13 @@ void spi_init(spi_t *obj, uint32_t speed, spi_mode_e mode, uint8_t msb) handle->Init.Direction = SPI_DIRECTION_2LINES; - if ((mode == SPI_MODE_0) || (mode == SPI_MODE_2)) { + if ((mode == SPI_MODE0) || (mode == SPI_MODE2)) { handle->Init.CLKPhase = SPI_PHASE_1EDGE; } else { handle->Init.CLKPhase = SPI_PHASE_2EDGE; } - if ((mode == SPI_MODE_0) || (mode == SPI_MODE_1)) { + if ((mode == SPI_MODE0) || (mode == SPI_MODE1)) { handle->Init.CLKPolarity = SPI_POLARITY_LOW; } else { handle->Init.CLKPolarity = SPI_POLARITY_HIGH; @@ -496,88 +469,77 @@ void spi_deinit(spi_t *obj) #endif } -/** - * @brief This function is implemented by user to send data over SPI interface - * @param obj : pointer to spi_t structure - * @param Data : data to be sent - * @param len : length in bytes of the data to be sent - * @param Timeout: Timeout duration in tick - * @retval status of the send operation (0) in case of error - */ -spi_status_e spi_send(spi_t *obj, uint8_t *Data, uint16_t len, uint32_t Timeout) -{ - return spi_transfer(obj, Data, Data, len, Timeout, 1 /* SPI_TRANSMITONLY */); -} - /** * @brief This function is implemented by user to send/receive data over * SPI interface * @param obj : pointer to spi_t structure * @param tx_buffer : tx data to send before reception - * @param rx_buffer : data to receive + * @param rx_buffer : rx data to receive if not numm * @param len : length in byte of the data to send and receive - * @param Timeout: Timeout duration in tick - * @param skipReceive: skip receiving data after transmit or not * @retval status of the send operation (0) in case of error */ -spi_status_e spi_transfer(spi_t *obj, uint8_t *tx_buffer, uint8_t *rx_buffer, - uint16_t len, uint32_t Timeout, bool skipReceive) +spi_status_e spi_transfer(spi_t *obj, const uint8_t *tx_buffer, uint8_t *rx_buffer, + uint16_t len) { spi_status_e ret = SPI_OK; uint32_t tickstart, size = len; SPI_TypeDef *_SPI = obj->handle.Instance; + uint8_t *tx_buf = (uint8_t *)tx_buffer; - if ((obj == NULL) || (len == 0) || (Timeout == 0U)) { - return Timeout > 0U ? SPI_ERROR : SPI_TIMEOUT; - } - tickstart = HAL_GetTick(); + if (len == 0) { + ret = SPI_ERROR; + } else { + tickstart = HAL_GetTick(); #if defined(SPI_CR2_TSIZE) - /* Start transfer */ - LL_SPI_SetTransferSize(_SPI, size); - LL_SPI_Enable(_SPI); - LL_SPI_StartMasterTransfer(_SPI); + /* Start transfer */ + LL_SPI_SetTransferSize(_SPI, size); + LL_SPI_Enable(_SPI); + LL_SPI_StartMasterTransfer(_SPI); #endif - while (size--) { + while (size--) { #if defined(SPI_SR_TXP) - while (!LL_SPI_IsActiveFlag_TXP(_SPI)); + while (!LL_SPI_IsActiveFlag_TXP(_SPI)); #else - while (!LL_SPI_IsActiveFlag_TXE(_SPI)); + while (!LL_SPI_IsActiveFlag_TXE(_SPI)); #endif - LL_SPI_TransmitData8(_SPI, *tx_buffer++); + LL_SPI_TransmitData8(_SPI, tx_buf ? *tx_buf++ : 0XFF); - if (!skipReceive) { #if defined(SPI_SR_RXP) while (!LL_SPI_IsActiveFlag_RXP(_SPI)); #else while (!LL_SPI_IsActiveFlag_RXNE(_SPI)); #endif - *rx_buffer++ = LL_SPI_ReceiveData8(_SPI); - } - if ((Timeout != HAL_MAX_DELAY) && (HAL_GetTick() - tickstart >= Timeout)) { - ret = SPI_TIMEOUT; - break; + if (rx_buffer) { + *rx_buffer++ = LL_SPI_ReceiveData8(_SPI); + } else { + LL_SPI_ReceiveData8(_SPI); + } + if ((SPI_TRANSFER_TIMEOUT != HAL_MAX_DELAY) && + (HAL_GetTick() - tickstart >= SPI_TRANSFER_TIMEOUT)) { + ret = SPI_TIMEOUT; + break; + } } - } #if defined(SPI_IFCR_EOTC) - // Add a delay before disabling SPI otherwise last-bit/last-clock may be truncated - // See https://github.com/stm32duino/Arduino_Core_STM32/issues/1294 - // Computed delay is half SPI clock - delayMicroseconds(obj->disable_delay); - - /* Close transfer */ - /* Clear flags */ - LL_SPI_ClearFlag_EOT(_SPI); - LL_SPI_ClearFlag_TXTF(_SPI); - /* Disable SPI peripheral */ - LL_SPI_Disable(_SPI); + // Add a delay before disabling SPI otherwise last-bit/last-clock may be truncated + // See https://github.com/stm32duino/Arduino_Core_STM32/issues/1294 + // Computed delay is half SPI clock + delayMicroseconds(obj->disable_delay); + + /* Close transfer */ + /* Clear flags */ + LL_SPI_ClearFlag_EOT(_SPI); + LL_SPI_ClearFlag_TXTF(_SPI); + /* Disable SPI peripheral */ + LL_SPI_Disable(_SPI); #else - /* Wait for end of transfer */ - while (LL_SPI_IsActiveFlag_BSY(_SPI)); + /* Wait for end of transfer */ + while (LL_SPI_IsActiveFlag_BSY(_SPI)); #endif - + } return ret; } diff --git a/stm32/libraries/SPI/src/utility/spi_com.h b/stm32/libraries/SPI/src/utility/spi_com.h index daba244b..4d145ff7 100644 --- a/stm32/libraries/SPI/src/utility/spi_com.h +++ b/stm32/libraries/SPI/src/utility/spi_com.h @@ -1,39 +1,13 @@ -/** - ****************************************************************************** - * @file spi_com.h - * @author WI6LABS - * @version V1.0.0 - * @date 01-August-2016 - * @brief Header for spi module - ****************************************************************************** - * @attention - * - *

© COPYRIGHT(c) 2016 STMicroelectronics

- * - * Redistribution and use in source and binary forms, with or without modification, - * are permitted provided that the following conditions are met: - * 1. Redistributions of source code must retain the above copyright notice, - * this list of conditions and the following disclaimer. - * 2. Redistributions in binary form must reproduce the above copyright notice, - * this list of conditions and the following disclaimer in the documentation - * and/or other materials provided with the distribution. - * 3. Neither the name of STMicroelectronics nor the names of its contributors - * may be used to endorse or promote products derived from this software - * without specific prior written permission. - * - * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" - * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE - * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE - * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE - * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL - * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR - * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER - * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, - * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE - * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - * - ****************************************************************************** - */ +/* + * Copyright (c) 2016 Frederic Pillon for + * STMicroelectronics. All right reserved. + * Header utility of the spi module for arduino. + * + * This file is free software; you can redistribute it and/or modify + * it under the terms of either the GNU General Public License version 2 + * or the GNU Lesser General Public License version 2.1, both as + * published by the Free Software Foundation. + */ /* Define to prevent recursive inclusion -------------------------------------*/ #ifndef __SPI_COM_H @@ -78,6 +52,13 @@ typedef struct spi_s spi_t; #define SPI_SPEED_CLOCK_DIV128_MHZ ((uint32_t)128) #define SPI_SPEED_CLOCK_DIV256_MHZ ((uint32_t)256) +// Defines a default timeout delay in milliseconds for the SPI transfer +#ifndef SPI_TRANSFER_TIMEOUT +#define SPI_TRANSFER_TIMEOUT 1000 +#elif SPI_TRANSFER_TIMEOUT <= 0 +#error "SPI_TRANSFER_TIMEOUT cannot be less or equal to 0!" +#endif + ///@brief specifies the SPI mode to use //Mode Clock Polarity (CPOL) Clock Phase (CPHA) //SPI_MODE0 0 0 @@ -85,12 +66,13 @@ typedef struct spi_s spi_t; //SPI_MODE2 1 0 //SPI_MODE3 1 1 //enum definitions coming from SPI.h of SAM +// SPI mode parameters for SPISettings typedef enum { - SPI_MODE_0 = 0x00, - SPI_MODE_1 = 0x01, - SPI_MODE_2 = 0x02, - SPI_MODE_3 = 0x03 -} spi_mode_e; + SPI_MODE0 = 0, + SPI_MODE1 = 1, + SPI_MODE2 = 2, + SPI_MODE3 = 3, +} SPIMode; ///@brief SPI errors typedef enum { @@ -100,11 +82,9 @@ typedef enum { } spi_status_e; /* Exported functions ------------------------------------------------------- */ -void spi_init(spi_t *obj, uint32_t speed, spi_mode_e mode, uint8_t msb); +void spi_init(spi_t *obj, uint32_t speed, SPIMode mode, uint8_t msb); void spi_deinit(spi_t *obj); -spi_status_e spi_send(spi_t *obj, uint8_t *Data, uint16_t len, uint32_t Timeout); -spi_status_e spi_transfer(spi_t *obj, uint8_t *tx_buffer, - uint8_t *rx_buffer, uint16_t len, uint32_t Timeout, bool skipReceive); +spi_status_e spi_transfer(spi_t *obj, const uint8_t *tx_buffer, uint8_t *rx_buffer, uint16_t len); uint32_t spi_getClkFreq(spi_t *obj); #ifdef __cplusplus diff --git a/stm32/libraries/SrcWrapper/CMakeLists.txt b/stm32/libraries/SrcWrapper/CMakeLists.txt index b441d27d..888647fe 100644 --- a/stm32/libraries/SrcWrapper/CMakeLists.txt +++ b/stm32/libraries/SrcWrapper/CMakeLists.txt @@ -58,6 +58,7 @@ add_library(SrcWrapper_bin OBJECT EXCLUDE_FROM_ALL src/HAL/stm32yyxx_hal_fmpsmbus.c src/HAL/stm32yyxx_hal_fmpsmbus_ex.c src/HAL/stm32yyxx_hal_gfxmmu.c + src/HAL/stm32yyxx_hal_gfxtim.c src/HAL/stm32yyxx_hal_gpio.c src/HAL/stm32yyxx_hal_gpio_ex.c src/HAL/stm32yyxx_hal_gpu2d.c @@ -71,6 +72,7 @@ add_library(SrcWrapper_bin OBJECT EXCLUDE_FROM_ALL src/HAL/stm32yyxx_hal_i2c_ex.c src/HAL/stm32yyxx_hal_i2s.c src/HAL/stm32yyxx_hal_i2s_ex.c + src/HAL/stm32yyxx_hal_i3c.c src/HAL/stm32yyxx_hal_icache.c src/HAL/stm32yyxx_hal_ipcc.c src/HAL/stm32yyxx_hal_irda.c @@ -152,6 +154,7 @@ add_library(SrcWrapper_bin OBJECT EXCLUDE_FROM_ALL src/LL/stm32yyxx_ll_gpio.c src/LL/stm32yyxx_ll_hrtim.c src/LL/stm32yyxx_ll_i2c.c + src/LL/stm32yyxx_ll_i3c.c src/LL/stm32yyxx_ll_icache.c src/LL/stm32yyxx_ll_lpgpio.c src/LL/stm32yyxx_ll_lptim.c diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal.c index dff9d46e..9093c2e5 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal.c @@ -20,6 +20,10 @@ #include "stm32g0xx_hal.c" #elif STM32G4xx #include "stm32g4xx_hal.c" +#elif STM32H5xx + #include "stm32h5xx_hal.c" +#elif STM32H5xx + #include "stm32h5xx_hal.c" #elif STM32H7xx #include "stm32h7xx_hal.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_adc.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_adc.c index 090b8903..ac30bf41 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_adc.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_adc.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_adc.c" #elif STM32G4xx #include "stm32g4xx_hal_adc.c" +#elif STM32H5xx + #include "stm32h5xx_hal_adc.c" #elif STM32H7xx #include "stm32h7xx_hal_adc.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_adc_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_adc_ex.c index 859fddfe..c900ddda 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_adc_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_adc_ex.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_adc_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_adc_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_adc_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_adc_ex.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cec.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cec.c index 8489e44b..a5aaeff5 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cec.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cec.c @@ -14,6 +14,8 @@ #include "stm32f7xx_hal_cec.c" #elif STM32G0xx #include "stm32g0xx_hal_cec.c" +#elif STM32H5xx + #include "stm32h5xx_hal_cec.c" #elif STM32H7xx #include "stm32h7xx_hal_cec.c" #elif STM32MP1xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_comp.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_comp.c index 2ad017b1..5c061f1f 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_comp.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_comp.c @@ -10,6 +10,8 @@ #include "stm32g0xx_hal_comp.c" #elif STM32G4xx #include "stm32g4xx_hal_comp.c" +#elif STM32H5xx + #include "stm32h5xx_hal_comp.c" #elif STM32H7xx #include "stm32h7xx_hal_comp.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cordic.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cordic.c index bed0e3bc..5bb9a100 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cordic.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cordic.c @@ -4,6 +4,8 @@ #ifdef STM32G4xx #include "stm32g4xx_hal_cordic.c" +#elif STM32H5xx + #include "stm32h5xx_hal_cordic.c" #elif STM32H7xx #include "stm32h7xx_hal_cordic.c" #elif STM32U5xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cortex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cortex.c index 063e9dea..36f35eeb 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cortex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cortex.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_cortex.c" #elif STM32G4xx #include "stm32g4xx_hal_cortex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_cortex.c" #elif STM32H7xx #include "stm32h7xx_hal_cortex.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_crc.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_crc.c index 90ef5e41..d58c3b22 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_crc.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_crc.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_crc.c" #elif STM32G4xx #include "stm32g4xx_hal_crc.c" +#elif STM32H5xx + #include "stm32h5xx_hal_crc.c" #elif STM32H7xx #include "stm32h7xx_hal_crc.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_crc_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_crc_ex.c index 3940fea1..df9c0459 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_crc_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_crc_ex.c @@ -14,6 +14,8 @@ #include "stm32g0xx_hal_crc_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_crc_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_crc_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_crc_ex.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cryp.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cryp.c index 26309bec..2113baaa 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cryp.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cryp.c @@ -12,6 +12,8 @@ #include "stm32g0xx_hal_cryp.c" #elif STM32G4xx #include "stm32g4xx_hal_cryp.c" +#elif STM32H5xx + #include "stm32h5xx_hal_cryp.c" #elif STM32H7xx #include "stm32h7xx_hal_cryp.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cryp_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cryp_ex.c index 16c3774b..df46a519 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cryp_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_cryp_ex.c @@ -10,6 +10,8 @@ #include "stm32g0xx_hal_cryp_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_cryp_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_cryp_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_cryp_ex.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dac.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dac.c index d4f21bca..c4d0bfc1 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dac.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dac.c @@ -18,6 +18,8 @@ #include "stm32g0xx_hal_dac.c" #elif STM32G4xx #include "stm32g4xx_hal_dac.c" +#elif STM32H5xx + #include "stm32h5xx_hal_dac.c" #elif STM32H7xx #include "stm32h7xx_hal_dac.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dac_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dac_ex.c index 1f25d912..822ed08d 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dac_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dac_ex.c @@ -18,6 +18,8 @@ #include "stm32g0xx_hal_dac_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_dac_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_dac_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_dac_ex.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dcache.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dcache.c index aadf9db3..a19aed53 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dcache.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dcache.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32U5xx +#ifdef STM32H5xx + #include "stm32h5xx_hal_dcache.c" +#elif STM32U5xx #include "stm32u5xx_hal_dcache.c" #endif #pragma GCC diagnostic pop diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dcmi.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dcmi.c index 935fa966..cf6c79a6 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dcmi.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dcmi.c @@ -8,6 +8,8 @@ #include "stm32f4xx_hal_dcmi.c" #elif STM32F7xx #include "stm32f7xx_hal_dcmi.c" +#elif STM32H5xx + #include "stm32h5xx_hal_dcmi.c" #elif STM32H7xx #include "stm32h7xx_hal_dcmi.c" #elif STM32L4xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dma.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dma.c index e8a88625..9f4d6bb6 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dma.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dma.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_dma.c" #elif STM32G4xx #include "stm32g4xx_hal_dma.c" +#elif STM32H5xx + #include "stm32h5xx_hal_dma.c" #elif STM32H7xx #include "stm32h7xx_hal_dma.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dma_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dma_ex.c index 9fc28126..07518e9d 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dma_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dma_ex.c @@ -14,6 +14,8 @@ #include "stm32g0xx_hal_dma_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_dma_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_dma_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_dma_ex.c" #elif STM32L4xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dts.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dts.c index e43637c1..d797630b 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dts.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_dts.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32H7xx +#ifdef STM32H5xx + #include "stm32h5xx_hal_dts.c" +#elif STM32H7xx #include "stm32h7xx_hal_dts.c" #endif #pragma GCC diagnostic pop diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_eth.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_eth.c index f441dac3..8af8e551 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_eth.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_eth.c @@ -12,6 +12,8 @@ #elif STM32F7xx #include "Legacy/stm32f7xx_hal_eth.c" #include "stm32f7xx_hal_eth.c" +#elif STM32H5xx + #include "stm32h5xx_hal_eth.c" #elif STM32H7xx #include "Legacy/stm32h7xx_hal_eth.c" #include "stm32h7xx_hal_eth.c" diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_eth_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_eth_ex.c index 0f7f9bcd..2a0e9a69 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_eth_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_eth_ex.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32H7xx +#ifdef STM32H5xx + #include "stm32h5xx_hal_eth_ex.c" +#elif STM32H7xx #include "Legacy/stm32h7xx_hal_eth_ex.c" #include "stm32h7xx_hal_eth_ex.c" #endif diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_exti.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_exti.c index 2314f8f8..6eb44ba3 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_exti.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_exti.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_exti.c" #elif STM32G4xx #include "stm32g4xx_hal_exti.c" +#elif STM32H5xx + #include "stm32h5xx_hal_exti.c" #elif STM32H7xx #include "stm32h7xx_hal_exti.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_fdcan.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_fdcan.c index a501b948..e64f1e3f 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_fdcan.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_fdcan.c @@ -6,6 +6,8 @@ #include "stm32g0xx_hal_fdcan.c" #elif STM32G4xx #include "stm32g4xx_hal_fdcan.c" +#elif STM32H5xx + #include "stm32h5xx_hal_fdcan.c" #elif STM32H7xx #include "stm32h7xx_hal_fdcan.c" #elif STM32L5xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_flash.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_flash.c index 09919ff9..44649287 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_flash.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_flash.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_flash.c" #elif STM32G4xx #include "stm32g4xx_hal_flash.c" +#elif STM32H5xx + #include "stm32h5xx_hal_flash.c" #elif STM32H7xx #include "stm32h7xx_hal_flash.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_flash_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_flash_ex.c index 4b39fc1d..228dc847 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_flash_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_flash_ex.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_flash_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_flash_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_flash_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_flash_ex.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_fmac.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_fmac.c index f4a8a865..3231f942 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_fmac.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_fmac.c @@ -4,6 +4,8 @@ #ifdef STM32G4xx #include "stm32g4xx_hal_fmac.c" +#elif STM32H5xx + #include "stm32h5xx_hal_fmac.c" #elif STM32H7xx #include "stm32h7xx_hal_fmac.c" #elif STM32U5xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_gfxtim.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_gfxtim.c new file mode 100644 index 00000000..af5bf9d8 --- /dev/null +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_gfxtim.c @@ -0,0 +1,8 @@ +/* HAL raised several warnings, ignore them */ +#pragma GCC diagnostic push +#pragma GCC diagnostic ignored "-Wunused-parameter" + +#ifdef STM32U5xx + #include "stm32u5xx_hal_gfxtim.c" +#endif +#pragma GCC diagnostic pop diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_gpio.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_gpio.c index 39c75082..25efe43a 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_gpio.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_gpio.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_gpio.c" #elif STM32G4xx #include "stm32g4xx_hal_gpio.c" +#elif STM32H5xx + #include "stm32h5xx_hal_gpio.c" #elif STM32H7xx #include "stm32h7xx_hal_gpio.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_gtzc.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_gtzc.c index 0d9a5930..bbed4555 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_gtzc.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_gtzc.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32L5xx +#ifdef STM32H5xx + #include "stm32h5xx_hal_gtzc.c" +#elif STM32L5xx #include "stm32l5xx_hal_gtzc.c" #elif STM32U5xx #include "stm32u5xx_hal_gtzc.c" diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_hash.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_hash.c index 89f1d695..c95f79a2 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_hash.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_hash.c @@ -8,6 +8,8 @@ #include "stm32f4xx_hal_hash.c" #elif STM32F7xx #include "stm32f7xx_hal_hash.c" +#elif STM32H5xx + #include "stm32h5xx_hal_hash.c" #elif STM32H7xx #include "stm32h7xx_hal_hash.c" #elif STM32L4xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_hcd.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_hcd.c index 705aba97..7a82631a 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_hcd.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_hcd.c @@ -12,6 +12,8 @@ #include "stm32f7xx_hal_hcd.c" #elif STM32G0xx #include "stm32g0xx_hal_hcd.c" +#elif STM32H5xx + #include "stm32h5xx_hal_hcd.c" #elif STM32H7xx #include "stm32h7xx_hal_hcd.c" #elif STM32L4xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2c.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2c.c index 3e1dfe76..2819ed0a 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2c.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2c.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_i2c.c" #elif STM32G4xx #include "stm32g4xx_hal_i2c.c" +#elif STM32H5xx + #include "stm32h5xx_hal_i2c.c" #elif STM32H7xx #include "stm32h7xx_hal_i2c.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2c_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2c_ex.c index a57b1461..67d36e26 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2c_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2c_ex.c @@ -16,6 +16,8 @@ #include "stm32g0xx_hal_i2c_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_i2c_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_i2c_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_i2c_ex.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2s.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2s.c index 54d346cf..2c2c9925 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2s.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2s.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_i2s.c" #elif STM32G4xx #include "stm32g4xx_hal_i2s.c" +#elif STM32H5xx + #include "stm32h5xx_hal_i2s.c" #elif STM32H7xx #include "stm32h7xx_hal_i2s.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2s_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2s_ex.c index 2a66b1bb..3d82a0e6 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2s_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i2s_ex.c @@ -6,6 +6,8 @@ #include "stm32f3xx_hal_i2s_ex.c" #elif STM32F4xx #include "stm32f4xx_hal_i2s_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_i2s_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_i2s_ex.c" #endif diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i3c.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i3c.c new file mode 100644 index 00000000..c259588d --- /dev/null +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_i3c.c @@ -0,0 +1,8 @@ +/* HAL raised several warnings, ignore them */ +#pragma GCC diagnostic push +#pragma GCC diagnostic ignored "-Wunused-parameter" + +#ifdef STM32H5xx + #include "stm32h5xx_hal_i3c.c" +#endif +#pragma GCC diagnostic pop diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_icache.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_icache.c index 8aae1d97..9bafb147 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_icache.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_icache.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32L5xx +#ifdef STM32H5xx + #include "stm32h5xx_hal_icache.c" +#elif STM32L5xx #include "stm32l5xx_hal_icache.c" #elif STM32U5xx #include "stm32u5xx_hal_icache.c" diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_irda.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_irda.c index 14c6e7f3..d4518e26 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_irda.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_irda.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_irda.c" #elif STM32G4xx #include "stm32g4xx_hal_irda.c" +#elif STM32H5xx + #include "stm32h5xx_hal_irda.c" #elif STM32H7xx #include "stm32h7xx_hal_irda.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_iwdg.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_iwdg.c index 7efc51f3..fec85566 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_iwdg.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_iwdg.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_iwdg.c" #elif STM32G4xx #include "stm32g4xx_hal_iwdg.c" +#elif STM32H5xx + #include "stm32h5xx_hal_iwdg.c" #elif STM32H7xx #include "stm32h7xx_hal_iwdg.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_jpeg.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_jpeg.c index b6b80145..a12960b1 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_jpeg.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_jpeg.c @@ -6,5 +6,7 @@ #include "stm32f7xx_hal_jpeg.c" #elif STM32H7xx #include "stm32h7xx_hal_jpeg.c" +#elif STM32U5xx + #include "stm32u5xx_hal_jpeg.c" #endif #pragma GCC diagnostic pop diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_lptim.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_lptim.c index 23d7db98..5515eb09 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_lptim.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_lptim.c @@ -10,6 +10,8 @@ #include "stm32g0xx_hal_lptim.c" #elif STM32G4xx #include "stm32g4xx_hal_lptim.c" +#elif STM32H5xx + #include "stm32h5xx_hal_lptim.c" #elif STM32H7xx #include "stm32h7xx_hal_lptim.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_mmc.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_mmc.c index 858085d7..937a1ccc 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_mmc.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_mmc.c @@ -10,6 +10,8 @@ #include "stm32f4xx_hal_mmc.c" #elif STM32F7xx #include "stm32f7xx_hal_mmc.c" +#elif STM32H5xx + #include "stm32h5xx_hal_mmc.c" #elif STM32H7xx #include "stm32h7xx_hal_mmc.c" #elif STM32L4xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_mmc_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_mmc_ex.c index 6ed94dc7..48862371 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_mmc_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_mmc_ex.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32H7xx +#ifdef STM32H5xx + #include "stm32h5xx_hal_mmc_ex.c" +#elif STM32H7xx #include "stm32h7xx_hal_mmc_ex.c" #elif STM32L4xx #include "stm32l4xx_hal_mmc_ex.c" diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_nand.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_nand.c index c2473af9..d21869a7 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_nand.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_nand.c @@ -14,6 +14,8 @@ #include "stm32f7xx_hal_nand.c" #elif STM32G4xx #include "stm32g4xx_hal_nand.c" +#elif STM32H5xx + #include "stm32h5xx_hal_nand.c" #elif STM32H7xx #include "stm32h7xx_hal_nand.c" #elif STM32L4xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_nor.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_nor.c index e886bb1d..9dc31d59 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_nor.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_nor.c @@ -14,6 +14,8 @@ #include "stm32f7xx_hal_nor.c" #elif STM32G4xx #include "stm32g4xx_hal_nor.c" +#elif STM32H5xx + #include "stm32h5xx_hal_nor.c" #elif STM32H7xx #include "stm32h7xx_hal_nor.c" #elif STM32L1xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_opamp.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_opamp.c index 1c5463d6..361a4820 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_opamp.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_opamp.c @@ -6,6 +6,8 @@ #include "stm32f3xx_hal_opamp.c" #elif STM32G4xx #include "stm32g4xx_hal_opamp.c" +#elif STM32H5xx + #include "stm32h5xx_hal_opamp.c" #elif STM32H7xx #include "stm32h7xx_hal_opamp.c" #elif STM32L1xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_opamp_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_opamp_ex.c index e51c2778..57a2422e 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_opamp_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_opamp_ex.c @@ -6,6 +6,8 @@ #include "stm32f3xx_hal_opamp_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_opamp_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_opamp_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_opamp_ex.c" #elif STM32L1xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_otfdec.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_otfdec.c index a27c7ccb..a71b64f9 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_otfdec.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_otfdec.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32H7xx +#ifdef STM32H5xx + #include "stm32h5xx_hal_otfdec.c" +#elif STM32H7xx #include "stm32h7xx_hal_otfdec.c" #elif STM32L5xx #include "stm32l5xx_hal_otfdec.c" diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pcd.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pcd.c index f5d72e77..700ef004 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pcd.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pcd.c @@ -18,6 +18,8 @@ #include "stm32g0xx_hal_pcd.c" #elif STM32G4xx #include "stm32g4xx_hal_pcd.c" +#elif STM32H5xx + #include "stm32h5xx_hal_pcd.c" #elif STM32H7xx #include "stm32h7xx_hal_pcd.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pcd_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pcd_ex.c index c0f4aca3..bd144c5c 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pcd_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pcd_ex.c @@ -18,6 +18,8 @@ #include "stm32g0xx_hal_pcd_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_pcd_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_pcd_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_pcd_ex.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pka.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pka.c index 66a059e4..a72ea147 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pka.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pka.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32L4xx +#ifdef STM32H5xx + #include "stm32h5xx_hal_pka.c" +#elif STM32L4xx #include "stm32l4xx_hal_pka.c" #elif STM32L5xx #include "stm32l5xx_hal_pka.c" diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pssi.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pssi.c index 6a0e499a..24db2c6b 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pssi.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pssi.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32H7xx +#ifdef STM32H5xx + #include "stm32h5xx_hal_pssi.c" +#elif STM32H7xx #include "stm32h7xx_hal_pssi.c" #elif STM32L4xx #include "stm32l4xx_hal_pssi.c" diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pwr.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pwr.c index 509ab24f..df4ea283 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pwr.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pwr.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_pwr.c" #elif STM32G4xx #include "stm32g4xx_hal_pwr.c" +#elif STM32H5xx + #include "stm32h5xx_hal_pwr.c" #elif STM32H7xx #include "stm32h7xx_hal_pwr.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pwr_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pwr_ex.c index bb6ee48d..070aef7e 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pwr_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_pwr_ex.c @@ -18,6 +18,8 @@ #include "stm32g0xx_hal_pwr_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_pwr_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_pwr_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_pwr_ex.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_ramcfg.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_ramcfg.c index edaa6037..fdf54967 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_ramcfg.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_ramcfg.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32U5xx +#ifdef STM32H5xx + #include "stm32h5xx_hal_ramcfg.c" +#elif STM32U5xx #include "stm32u5xx_hal_ramcfg.c" #endif #pragma GCC diagnostic pop diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rcc.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rcc.c index 12fd2468..217f9158 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rcc.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rcc.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_rcc.c" #elif STM32G4xx #include "stm32g4xx_hal_rcc.c" +#elif STM32H5xx + #include "stm32h5xx_hal_rcc.c" #elif STM32H7xx #include "stm32h7xx_hal_rcc.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rcc_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rcc_ex.c index 9d25bf60..ca46a0bb 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rcc_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rcc_ex.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_rcc_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_rcc_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_rcc_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_rcc_ex.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rng.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rng.c index 141ba7d7..41bedadb 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rng.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rng.c @@ -12,6 +12,8 @@ #include "stm32g0xx_hal_rng.c" #elif STM32G4xx #include "stm32g4xx_hal_rng.c" +#elif STM32H5xx + #include "stm32h5xx_hal_rng.c" #elif STM32H7xx #include "stm32h7xx_hal_rng.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rng_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rng_ex.c index 20bad616..a6cf27e6 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rng_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rng_ex.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32H7xx +#ifdef STM32H5xx + #include "stm32h5xx_hal_rng_ex.c" +#elif STM32H7xx #include "stm32h7xx_hal_rng_ex.c" #elif STM32L4xx #include "stm32l4xx_hal_rng_ex.c" diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rtc.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rtc.c index 66499f57..5f8e6c1a 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rtc.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rtc.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_rtc.c" #elif STM32G4xx #include "stm32g4xx_hal_rtc.c" +#elif STM32H5xx + #include "stm32h5xx_hal_rtc.c" #elif STM32H7xx #include "stm32h7xx_hal_rtc.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rtc_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rtc_ex.c index 654a8f56..570d63aa 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rtc_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_rtc_ex.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_rtc_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_rtc_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_rtc_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_rtc_ex.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sai.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sai.c index 6f8a3ac4..29e4aa56 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sai.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sai.c @@ -8,6 +8,8 @@ #include "stm32f7xx_hal_sai.c" #elif STM32G4xx #include "stm32g4xx_hal_sai.c" +#elif STM32H5xx + #include "stm32h5xx_hal_sai.c" #elif STM32H7xx #include "stm32h7xx_hal_sai.c" #elif STM32L4xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sai_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sai_ex.c index 449d99ea..bea93ee0 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sai_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sai_ex.c @@ -8,6 +8,8 @@ #include "stm32f7xx_hal_sai_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_sai_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_sai_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_sai_ex.c" #elif STM32L4xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sd.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sd.c index 3bbd4973..6a7087ad 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sd.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sd.c @@ -10,6 +10,8 @@ #include "stm32f4xx_hal_sd.c" #elif STM32F7xx #include "stm32f7xx_hal_sd.c" +#elif STM32H5xx + #include "stm32h5xx_hal_sd.c" #elif STM32H7xx #include "stm32h7xx_hal_sd.c" #elif STM32L1xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sd_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sd_ex.c index 881a90c7..4d520697 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sd_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sd_ex.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32H7xx +#ifdef STM32H5xx + #include "stm32h5xx_hal_sd_ex.c" +#elif STM32H7xx #include "stm32h7xx_hal_sd_ex.c" #elif STM32L4xx #include "stm32l4xx_hal_sd_ex.c" diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sdram.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sdram.c index d4252633..ceb9a4ed 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sdram.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sdram.c @@ -6,6 +6,8 @@ #include "stm32f4xx_hal_sdram.c" #elif STM32F7xx #include "stm32f7xx_hal_sdram.c" +#elif STM32H5xx + #include "stm32h5xx_hal_sdram.c" #elif STM32H7xx #include "stm32h7xx_hal_sdram.c" #endif diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smartcard.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smartcard.c index 98a44370..f72baaf3 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smartcard.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smartcard.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_smartcard.c" #elif STM32G4xx #include "stm32g4xx_hal_smartcard.c" +#elif STM32H5xx + #include "stm32h5xx_hal_smartcard.c" #elif STM32H7xx #include "stm32h7xx_hal_smartcard.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smartcard_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smartcard_ex.c index 5c49fbdf..c5bf0377 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smartcard_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smartcard_ex.c @@ -14,6 +14,8 @@ #include "stm32g0xx_hal_smartcard_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_smartcard_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_smartcard_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_smartcard_ex.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smbus.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smbus.c index f37b8967..688a0563 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smbus.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smbus.c @@ -16,10 +16,14 @@ #include "stm32g0xx_hal_smbus.c" #elif STM32G4xx #include "stm32g4xx_hal_smbus.c" +#elif STM32H5xx + #include "stm32h5xx_hal_smbus.c" #elif STM32H7xx #include "stm32h7xx_hal_smbus.c" #elif STM32L0xx #include "stm32l0xx_hal_smbus.c" +#elif STM32L1xx + #include "stm32l1xx_hal_smbus.c" #elif STM32L4xx #include "stm32l4xx_hal_smbus.c" #elif STM32L5xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smbus_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smbus_ex.c index b37baba3..869115fd 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smbus_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_smbus_ex.c @@ -8,6 +8,8 @@ #include "stm32g0xx_hal_smbus_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_smbus_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_smbus_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_smbus_ex.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_spi.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_spi.c index 7efe9f7b..d54977b4 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_spi.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_spi.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_spi.c" #elif STM32G4xx #include "stm32g4xx_hal_spi.c" +#elif STM32H5xx + #include "stm32h5xx_hal_spi.c" #elif STM32H7xx #include "stm32h7xx_hal_spi.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_spi_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_spi_ex.c index 3f82f33a..a5e02b61 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_spi_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_spi_ex.c @@ -14,6 +14,8 @@ #include "stm32g0xx_hal_spi_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_spi_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_spi_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_spi_ex.c" #elif STM32L4xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sram.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sram.c index 0950d7ab..93f83a41 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sram.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_sram.c @@ -14,6 +14,8 @@ #include "stm32f7xx_hal_sram.c" #elif STM32G4xx #include "stm32g4xx_hal_sram.c" +#elif STM32H5xx + #include "stm32h5xx_hal_sram.c" #elif STM32H7xx #include "stm32h7xx_hal_sram.c" #elif STM32L1xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_tim.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_tim.c index 6e9b38e5..e99fefa0 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_tim.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_tim.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_tim.c" #elif STM32G4xx #include "stm32g4xx_hal_tim.c" +#elif STM32H5xx + #include "stm32h5xx_hal_tim.c" #elif STM32H7xx #include "stm32h7xx_hal_tim.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_tim_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_tim_ex.c index 7a4a6112..79fd6e40 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_tim_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_tim_ex.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_tim_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_tim_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_tim_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_tim_ex.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_uart.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_uart.c index 3ad369e2..5bff5cab 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_uart.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_uart.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_uart.c" #elif STM32G4xx #include "stm32g4xx_hal_uart.c" +#elif STM32H5xx + #include "stm32h5xx_hal_uart.c" #elif STM32H7xx #include "stm32h7xx_hal_uart.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_uart_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_uart_ex.c index f9923dbc..28d98e94 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_uart_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_uart_ex.c @@ -14,6 +14,8 @@ #include "stm32g0xx_hal_uart_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_uart_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_uart_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_uart_ex.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_usart.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_usart.c index b95b7e00..23364343 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_usart.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_usart.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_usart.c" #elif STM32G4xx #include "stm32g4xx_hal_usart.c" +#elif STM32H5xx + #include "stm32h5xx_hal_usart.c" #elif STM32H7xx #include "stm32h7xx_hal_usart.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_usart_ex.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_usart_ex.c index 0692a903..b2d5165f 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_usart_ex.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_usart_ex.c @@ -12,6 +12,8 @@ #include "stm32g0xx_hal_usart_ex.c" #elif STM32G4xx #include "stm32g4xx_hal_usart_ex.c" +#elif STM32H5xx + #include "stm32h5xx_hal_usart_ex.c" #elif STM32H7xx #include "stm32h7xx_hal_usart_ex.c" #elif STM32L4xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_wwdg.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_wwdg.c index dfa2d2a6..615bc465 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_wwdg.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_wwdg.c @@ -20,6 +20,8 @@ #include "stm32g0xx_hal_wwdg.c" #elif STM32G4xx #include "stm32g4xx_hal_wwdg.c" +#elif STM32H5xx + #include "stm32h5xx_hal_wwdg.c" #elif STM32H7xx #include "stm32h7xx_hal_wwdg.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_xspi.c b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_xspi.c index 69d077cf..84a04746 100644 --- a/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_xspi.c +++ b/stm32/libraries/SrcWrapper/src/HAL/stm32yyxx_hal_xspi.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32U5xx +#ifdef STM32H5xx + #include "stm32h5xx_hal_xspi.c" +#elif STM32U5xx #include "stm32u5xx_hal_xspi.c" #endif #pragma GCC diagnostic pop diff --git a/stm32/libraries/SrcWrapper/src/HardwareTimer.cpp b/stm32/libraries/SrcWrapper/src/HardwareTimer.cpp index 231f53f4..f65a4f4f 100644 --- a/stm32/libraries/SrcWrapper/src/HardwareTimer.cpp +++ b/stm32/libraries/SrcWrapper/src/HardwareTimer.cpp @@ -619,9 +619,9 @@ void HardwareTimer::setCount(uint32_t counter, TimerFormat_t format) * @param pin: Arduino pin number, ex: D1, 1 or PA1 * @retval None */ -void HardwareTimer::setMode(uint32_t channel, TimerModes_t mode, uint32_t pin) +void HardwareTimer::setMode(uint32_t channel, TimerModes_t mode, uint32_t pin, ChannelInputFilter_t filter) { - setMode(channel, mode, digitalPinToPinName(pin)); + setMode(channel, mode, digitalPinToPinName(pin), filter); } /** @@ -631,7 +631,7 @@ void HardwareTimer::setMode(uint32_t channel, TimerModes_t mode, uint32_t pin) * @param pin: pin name, ex: PB_0 * @retval None */ -void HardwareTimer::setMode(uint32_t channel, TimerModes_t mode, PinName pin) +void HardwareTimer::setMode(uint32_t channel, TimerModes_t mode, PinName pin, ChannelInputFilter_t filter) { int timChannel = getChannel(channel); int timAssociatedInputChannel; @@ -659,7 +659,7 @@ void HardwareTimer::setMode(uint32_t channel, TimerModes_t mode, PinName pin) channelIC.ICPolarity = TIM_ICPOLARITY_RISING; channelIC.ICSelection = TIM_ICSELECTION_DIRECTTI; channelIC.ICPrescaler = TIM_ICPSC_DIV1; - channelIC.ICFilter = 0; + channelIC.ICFilter = filter; switch (mode) { case TIMER_DISABLED: diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_adc.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_adc.c index e3f19830..5150ef44 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_adc.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_adc.c @@ -20,6 +20,8 @@ #include "stm32g0xx_ll_adc.c" #elif STM32G4xx #include "stm32g4xx_ll_adc.c" +#elif STM32H5xx + #include "stm32h5xx_ll_adc.c" #elif STM32H7xx #include "stm32h7xx_ll_adc.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_comp.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_comp.c index 4678b5eb..751c5540 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_comp.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_comp.c @@ -10,6 +10,8 @@ #include "stm32g0xx_ll_comp.c" #elif STM32G4xx #include "stm32g4xx_ll_comp.c" +#elif STM32H5xx + #include "stm32h5xx_ll_comp.c" #elif STM32H7xx #include "stm32h7xx_ll_comp.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_cordic.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_cordic.c index b8582561..765dc164 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_cordic.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_cordic.c @@ -4,6 +4,8 @@ #ifdef STM32G4xx #include "stm32g4xx_ll_cordic.c" +#elif STM32H5xx + #include "stm32h5xx_ll_cordic.c" #elif STM32H7xx #include "stm32h7xx_ll_cordic.c" #elif STM32U5xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_crc.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_crc.c index 52f7d7fc..fe5362aa 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_crc.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_crc.c @@ -20,6 +20,8 @@ #include "stm32g0xx_ll_crc.c" #elif STM32G4xx #include "stm32g4xx_ll_crc.c" +#elif STM32H5xx + #include "stm32h5xx_ll_crc.c" #elif STM32H7xx #include "stm32h7xx_ll_crc.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_crs.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_crs.c index d8e5e0d9..39154154 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_crs.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_crs.c @@ -8,6 +8,8 @@ #include "stm32g0xx_ll_crs.c" #elif STM32G4xx #include "stm32g4xx_ll_crs.c" +#elif STM32H5xx + #include "stm32h5xx_ll_crs.c" #elif STM32H7xx #include "stm32h7xx_ll_crs.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_dac.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_dac.c index 4b9340b1..4ac1c51b 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_dac.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_dac.c @@ -18,6 +18,8 @@ #include "stm32g0xx_ll_dac.c" #elif STM32G4xx #include "stm32g4xx_ll_dac.c" +#elif STM32H5xx + #include "stm32h5xx_ll_dac.c" #elif STM32H7xx #include "stm32h7xx_ll_dac.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_dlyb.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_dlyb.c index ed0b90ee..21be9915 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_dlyb.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_dlyb.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32U5xx +#ifdef STM32H5xx + #include "stm32h5xx_ll_dlyb.c" +#elif STM32U5xx #include "stm32u5xx_ll_dlyb.c" #endif #pragma GCC diagnostic pop diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_dma.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_dma.c index 145d3f60..499829e9 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_dma.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_dma.c @@ -20,6 +20,8 @@ #include "stm32g0xx_ll_dma.c" #elif STM32G4xx #include "stm32g4xx_ll_dma.c" +#elif STM32H5xx + #include "stm32h5xx_ll_dma.c" #elif STM32H7xx #include "stm32h7xx_ll_dma.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_exti.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_exti.c index 750f6fe1..3b745db4 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_exti.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_exti.c @@ -20,6 +20,8 @@ #include "stm32g0xx_ll_exti.c" #elif STM32G4xx #include "stm32g4xx_ll_exti.c" +#elif STM32H5xx + #include "stm32h5xx_ll_exti.c" #elif STM32H7xx #include "stm32h7xx_ll_exti.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_fmac.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_fmac.c index d1529da3..ee65c4cf 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_fmac.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_fmac.c @@ -4,6 +4,8 @@ #ifdef STM32G4xx #include "stm32g4xx_ll_fmac.c" +#elif STM32H5xx + #include "stm32h5xx_ll_fmac.c" #elif STM32H7xx #include "stm32h7xx_ll_fmac.c" #elif STM32U5xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_fmc.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_fmc.c index 3416400d..927bea3e 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_fmc.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_fmc.c @@ -10,6 +10,8 @@ #include "stm32f7xx_ll_fmc.c" #elif STM32G4xx #include "stm32g4xx_ll_fmc.c" +#elif STM32H5xx + #include "stm32h5xx_ll_fmc.c" #elif STM32H7xx #include "stm32h7xx_ll_fmc.c" #elif STM32L4xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_gpio.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_gpio.c index d24eb63a..80020540 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_gpio.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_gpio.c @@ -20,6 +20,8 @@ #include "stm32g0xx_ll_gpio.c" #elif STM32G4xx #include "stm32g4xx_ll_gpio.c" +#elif STM32H5xx + #include "stm32h5xx_ll_gpio.c" #elif STM32H7xx #include "stm32h7xx_ll_gpio.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_i2c.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_i2c.c index 9f898fbd..cf0c0a4e 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_i2c.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_i2c.c @@ -20,6 +20,8 @@ #include "stm32g0xx_ll_i2c.c" #elif STM32G4xx #include "stm32g4xx_ll_i2c.c" +#elif STM32H5xx + #include "stm32h5xx_ll_i2c.c" #elif STM32H7xx #include "stm32h7xx_ll_i2c.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_i3c.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_i3c.c new file mode 100644 index 00000000..fa64ec4c --- /dev/null +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_i3c.c @@ -0,0 +1,8 @@ +/* LL raised several warnings, ignore them */ +#pragma GCC diagnostic push +#pragma GCC diagnostic ignored "-Wunused-parameter" + +#ifdef STM32H5xx + #include "stm32h5xx_ll_i3c.c" +#endif +#pragma GCC diagnostic pop diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_icache.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_icache.c index dd90e690..a4398a2e 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_icache.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_icache.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32L5xx +#ifdef STM32H5xx + #include "stm32h5xx_ll_icache.c" +#elif STM32L5xx #include "stm32l5xx_ll_icache.c" #elif STM32U5xx #include "stm32u5xx_ll_icache.c" diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_lptim.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_lptim.c index c0b17db0..ddfe613d 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_lptim.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_lptim.c @@ -10,6 +10,8 @@ #include "stm32g0xx_ll_lptim.c" #elif STM32G4xx #include "stm32g4xx_ll_lptim.c" +#elif STM32H5xx + #include "stm32h5xx_ll_lptim.c" #elif STM32H7xx #include "stm32h7xx_ll_lptim.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_lpuart.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_lpuart.c index b656bf77..51954ec1 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_lpuart.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_lpuart.c @@ -6,6 +6,8 @@ #include "stm32g0xx_ll_lpuart.c" #elif STM32G4xx #include "stm32g4xx_ll_lpuart.c" +#elif STM32H5xx + #include "stm32h5xx_ll_lpuart.c" #elif STM32H7xx #include "stm32h7xx_ll_lpuart.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_opamp.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_opamp.c index 382b5a3f..58908335 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_opamp.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_opamp.c @@ -6,6 +6,8 @@ #include "stm32f3xx_ll_opamp.c" #elif STM32G4xx #include "stm32g4xx_ll_opamp.c" +#elif STM32H5xx + #include "stm32h5xx_ll_opamp.c" #elif STM32H7xx #include "stm32h7xx_ll_opamp.c" #elif STM32L1xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_pka.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_pka.c index 97dabc9f..f4af2d69 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_pka.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_pka.c @@ -2,7 +2,9 @@ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" -#ifdef STM32L4xx +#ifdef STM32H5xx + #include "stm32h5xx_ll_pka.c" +#elif STM32L4xx #include "stm32l4xx_ll_pka.c" #elif STM32L5xx #include "stm32l5xx_ll_pka.c" diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_pwr.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_pwr.c index 90cb4e3e..84281ba3 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_pwr.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_pwr.c @@ -20,6 +20,8 @@ #include "stm32g0xx_ll_pwr.c" #elif STM32G4xx #include "stm32g4xx_ll_pwr.c" +#elif STM32H5xx + #include "stm32h5xx_ll_pwr.c" #elif STM32H7xx #include "stm32h7xx_ll_pwr.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_rcc.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_rcc.c index 4c411303..267edc09 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_rcc.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_rcc.c @@ -20,6 +20,8 @@ #include "stm32g0xx_ll_rcc.c" #elif STM32G4xx #include "stm32g4xx_ll_rcc.c" +#elif STM32H5xx + #include "stm32h5xx_ll_rcc.c" #elif STM32H7xx #include "stm32h7xx_ll_rcc.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_rng.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_rng.c index 85257d65..cd3b1ec7 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_rng.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_rng.c @@ -12,6 +12,8 @@ #include "stm32g0xx_ll_rng.c" #elif STM32G4xx #include "stm32g4xx_ll_rng.c" +#elif STM32H5xx + #include "stm32h5xx_ll_rng.c" #elif STM32H7xx #include "stm32h7xx_ll_rng.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_rtc.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_rtc.c index f545a4c0..e8132a5d 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_rtc.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_rtc.c @@ -20,6 +20,8 @@ #include "stm32g0xx_ll_rtc.c" #elif STM32G4xx #include "stm32g4xx_ll_rtc.c" +#elif STM32H5xx + #include "stm32h5xx_ll_rtc.c" #elif STM32H7xx #include "stm32h7xx_ll_rtc.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_sdmmc.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_sdmmc.c index b41d28b2..19dc1360 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_sdmmc.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_sdmmc.c @@ -10,6 +10,8 @@ #include "stm32f4xx_ll_sdmmc.c" #elif STM32F7xx #include "stm32f7xx_ll_sdmmc.c" +#elif STM32H5xx + #include "stm32h5xx_ll_sdmmc.c" #elif STM32H7xx #include "stm32h7xx_ll_sdmmc.c" #elif STM32L1xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_spi.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_spi.c index 8867f410..58b1dae2 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_spi.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_spi.c @@ -20,6 +20,8 @@ #include "stm32g0xx_ll_spi.c" #elif STM32G4xx #include "stm32g4xx_ll_spi.c" +#elif STM32H5xx + #include "stm32h5xx_ll_spi.c" #elif STM32H7xx #include "stm32h7xx_ll_spi.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_tim.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_tim.c index d9eb9138..4417d88a 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_tim.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_tim.c @@ -20,6 +20,8 @@ #include "stm32g0xx_ll_tim.c" #elif STM32G4xx #include "stm32g4xx_ll_tim.c" +#elif STM32H5xx + #include "stm32h5xx_ll_tim.c" #elif STM32H7xx #include "stm32h7xx_ll_tim.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_ucpd.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_ucpd.c index e8de5b26..fbe90291 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_ucpd.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_ucpd.c @@ -6,6 +6,8 @@ #include "stm32g0xx_ll_ucpd.c" #elif STM32G4xx #include "stm32g4xx_ll_ucpd.c" +#elif STM32H5xx + #include "stm32h5xx_ll_ucpd.c" #elif STM32L5xx #include "stm32l5xx_ll_ucpd.c" #elif STM32U5xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_usart.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_usart.c index 4ff054c5..1c51985d 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_usart.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_usart.c @@ -20,6 +20,8 @@ #include "stm32g0xx_ll_usart.c" #elif STM32G4xx #include "stm32g4xx_ll_usart.c" +#elif STM32H5xx + #include "stm32h5xx_ll_usart.c" #elif STM32H7xx #include "stm32h7xx_ll_usart.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_usb.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_usb.c index e3b8050d..5672d4d9 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_usb.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_usb.c @@ -18,6 +18,8 @@ #include "stm32g0xx_ll_usb.c" #elif STM32G4xx #include "stm32g4xx_ll_usb.c" +#elif STM32H5xx + #include "stm32h5xx_ll_usb.c" #elif STM32H7xx #include "stm32h7xx_ll_usb.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_utils.c b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_utils.c index 56dd17ea..18546e82 100644 --- a/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_utils.c +++ b/stm32/libraries/SrcWrapper/src/LL/stm32yyxx_ll_utils.c @@ -20,6 +20,8 @@ #include "stm32g0xx_ll_utils.c" #elif STM32G4xx #include "stm32g4xx_ll_utils.c" +#elif STM32H5xx + #include "stm32h5xx_ll_utils.c" #elif STM32H7xx #include "stm32h7xx_ll_utils.c" #elif STM32L0xx diff --git a/stm32/libraries/SrcWrapper/src/stm32/analog.cpp b/stm32/libraries/SrcWrapper/src/stm32/analog.cpp index da96456b..63ff690c 100644 --- a/stm32/libraries/SrcWrapper/src/stm32/analog.cpp +++ b/stm32/libraries/SrcWrapper/src/stm32/analog.cpp @@ -861,8 +861,9 @@ uint16_t adc_read_value(PinName pin, uint32_t resolution) #endif #if !defined(STM32F1xx) && !defined(STM32F2xx) && !defined(STM32F3xx) && \ !defined(STM32F4xx) && !defined(STM32F7xx) && !defined(STM32G4xx) && \ - !defined(STM32H7xx) && !defined(STM32L4xx) && !defined(STM32L5xx) && \ - !defined(STM32MP1xx) && !defined(STM32WBxx) || defined(ADC_SUPPORT_2_5_MSPS) + !defined(STM32H5xx) && !defined(STM32H7xx) && !defined(STM32L4xx) && \ + !defined(STM32L5xx) && !defined(STM32MP1xx) && !defined(STM32WBxx) || \ + defined(ADC_SUPPORT_2_5_MSPS) AdcHandle.Init.LowPowerAutoPowerOff = DISABLE; /* ADC automatically powers-off after a conversion and automatically wakes-up when a new conversion is triggered */ #endif #ifdef ADC_CHANNELS_BANK_B @@ -939,8 +940,8 @@ uint16_t adc_read_value(PinName pin, uint32_t resolution) AdcChannelConf.Channel = channel; /* Specifies the channel to configure into ADC */ -#if defined(STM32G4xx) || defined(STM32L4xx) || defined(STM32L5xx) || \ - defined(STM32WBxx) +#if defined(STM32G4xx) || defined(STM32H5xx) || defined(STM32L4xx) || \ + defined(STM32L5xx) || defined(STM32WBxx) if (!IS_ADC_CHANNEL(&AdcHandle, AdcChannelConf.Channel)) { #else if (!IS_ADC_CHANNEL(AdcChannelConf.Channel)) { diff --git a/stm32/libraries/SrcWrapper/src/stm32/clock.c b/stm32/libraries/SrcWrapper/src/stm32/clock.c index 208b9707..b0ef8e32 100644 --- a/stm32/libraries/SrcWrapper/src/stm32/clock.c +++ b/stm32/libraries/SrcWrapper/src/stm32/clock.c @@ -95,9 +95,11 @@ void enableClock(sourceClock_t source) switch (source) { case LSI_CLOCK: #ifdef RCC_FLAG_LSI1RDY + __HAL_RCC_LSI1_ENABLE(); if (__HAL_RCC_GET_FLAG(RCC_FLAG_LSI1RDY) == RESET) { RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_LSI1; #else + __HAL_RCC_LSI_ENABLE(); if (__HAL_RCC_GET_FLAG(RCC_FLAG_LSIRDY) == RESET) { RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_LSI; #endif @@ -105,6 +107,7 @@ void enableClock(sourceClock_t source) } break; case HSI_CLOCK: + __HAL_RCC_HSI_ENABLE(); if (__HAL_RCC_GET_FLAG(RCC_FLAG_HSIRDY) == RESET) { RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI; RCC_OscInitStruct.HSIState = RCC_HSI_ON; @@ -116,6 +119,7 @@ void enableClock(sourceClock_t source) } break; case LSE_CLOCK: + __HAL_RCC_LSE_CONFIG(RCC_LSE_ON); if (__HAL_RCC_GET_FLAG(RCC_FLAG_LSERDY) == RESET) { #ifdef __HAL_RCC_LSEDRIVE_CONFIG __HAL_RCC_LSEDRIVE_CONFIG(RCC_LSEDRIVE_LOW); @@ -124,10 +128,17 @@ void enableClock(sourceClock_t source) RCC_OscInitStruct.LSEState = RCC_LSE_ON; } break; - case HSE_CLOCK: - if (__HAL_RCC_GET_FLAG(RCC_FLAG_HSERDY) == RESET) { - RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE; - RCC_OscInitStruct.HSEState = RCC_HSE_ON; + case HSE_CLOCK: { +#if defined(RCC_HSE_BYPASS_PWR) && defined(LORAWAN_BOARD_HAS_TCXO) && (LORAWAN_BOARD_HAS_TCXO == 1) + uint32_t HSEState = RCC_HSE_BYPASS_PWR; +#else + uint32_t HSEState = RCC_HSE_ON; +#endif + __HAL_RCC_HSE_CONFIG(HSEState); + if (__HAL_RCC_GET_FLAG(RCC_FLAG_HSERDY) == RESET) { + RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE; + RCC_OscInitStruct.HSEState = HSEState; + } } break; default: diff --git a/stm32/libraries/SrcWrapper/src/stm32/interrupt.cpp b/stm32/libraries/SrcWrapper/src/stm32/interrupt.cpp index 96a29685..c11780ab 100644 --- a/stm32/libraries/SrcWrapper/src/stm32/interrupt.cpp +++ b/stm32/libraries/SrcWrapper/src/stm32/interrupt.cpp @@ -70,7 +70,7 @@ static gpio_irq_conf_str gpio_irq_conf[NB_EXTI] = { {.irqnb = EXTI4_15_IRQn, .callback = NULL}, //GPIO_PIN_13 {.irqnb = EXTI4_15_IRQn, .callback = NULL}, //GPIO_PIN_14 {.irqnb = EXTI4_15_IRQn, .callback = NULL} //GPIO_PIN_15 -#elif defined (STM32MP1xx) || defined (STM32L5xx) || defined (STM32U5xx) +#elif defined (STM32H5xx) || defined (STM32MP1xx) || defined (STM32L5xx) || defined (STM32U5xx) {.irqnb = EXTI0_IRQn, .callback = NULL}, //GPIO_PIN_0 {.irqnb = EXTI1_IRQn, .callback = NULL}, //GPIO_PIN_1 {.irqnb = EXTI2_IRQn, .callback = NULL}, //GPIO_PIN_2 @@ -250,7 +250,8 @@ void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin) } } -#if defined(STM32C0xx) || defined(STM32G0xx) || defined(STM32MP1xx) || defined(STM32L5xx) || defined(STM32U5xx) +#if defined(STM32C0xx) || defined(STM32G0xx) || defined(STM32H5xx) || \ + defined(STM32MP1xx) || defined(STM32L5xx) || defined(STM32U5xx) /** * @brief EXTI line detection callback. * @param GPIO_Pin Specifies the port pin connected to corresponding EXTI line. @@ -373,7 +374,7 @@ void EXTI4_IRQHandler(void) HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_4); } -#if !defined(STM32MP1xx) && !defined(STM32L5xx) && !defined(STM32U5xx) +#if !defined(STM32H5xx) && !defined(STM32MP1xx) && !defined(STM32L5xx) && !defined(STM32U5xx) /** * @brief This function handles external line 5 to 9 interrupt request. * @param None @@ -399,7 +400,7 @@ void EXTI15_10_IRQHandler(void) HAL_GPIO_EXTI_IRQHandler(pin); } } -#else /* STM32MP1xx && STM32L5xx && STM32U5xx */ +#else /* STM32L5xx && STM32MP1xx && STM32L5xx && STM32U5xx */ /** * @brief This function handles external line 5 interrupt request. diff --git a/stm32/libraries/SrcWrapper/src/stm32/system_stm32yyxx.c b/stm32/libraries/SrcWrapper/src/stm32/system_stm32yyxx.c index 1ca7f7aa..fd194446 100644 --- a/stm32/libraries/SrcWrapper/src/stm32/system_stm32yyxx.c +++ b/stm32/libraries/SrcWrapper/src/stm32/system_stm32yyxx.c @@ -18,6 +18,8 @@ #include "system_stm32g0xx.c" #elif STM32G4xx #include "system_stm32g4xx.c" +#elif STM32H5xx + #include "system_stm32h5xx.c" #elif STM32H7xx #include "system_stm32h7xx.c" #elif STM32L0xx diff --git a/stm32/libraries/USBDevice/examples/USBDeviceKeyboard/USBDeviceKeyboard.ino b/stm32/libraries/USBDevice/examples/USBDeviceKeyboard/USBDeviceKeyboard.ino index 6fb70088..40ba00de 100644 --- a/stm32/libraries/USBDevice/examples/USBDeviceKeyboard/USBDeviceKeyboard.ino +++ b/stm32/libraries/USBDevice/examples/USBDeviceKeyboard/USBDeviceKeyboard.ino @@ -13,11 +13,10 @@ char mChar = 'a'; void setup() { Serial.begin(115200); - Serial.println("-------------------"); - Serial.println("USB Device Keyboard"); - Serial.println("-------------------"); - Serial.println(__DATE__); - Serial.println(__TIME__); + Serial.println("-----------------"); + Serial.println("USBDeviceKeyboard"); + Serial.println("-----------------"); + USBDevice.init(); } diff --git a/stm32/libraries/USBDevice/examples/USBDeviceMIDI/USBDeviceMIDI.ino b/stm32/libraries/USBDevice/examples/USBDeviceMIDI/USBDeviceMIDI.ino index e0793c9d..7cdb444c 100644 --- a/stm32/libraries/USBDevice/examples/USBDeviceMIDI/USBDeviceMIDI.ino +++ b/stm32/libraries/USBDevice/examples/USBDeviceMIDI/USBDeviceMIDI.ino @@ -17,11 +17,10 @@ uint8_t mNote = 48; void setup() { Serial.begin(115200); - Serial.println("---------------"); - Serial.println("USB Device MIDI"); - Serial.println("---------------"); - Serial.println(__DATE__); - Serial.println(__TIME__); + Serial.println("-------------"); + Serial.println("USBDeviceMIDI"); + Serial.println("-------------"); + USBDevice.init(); USBDevice.register_midi_note_on(midi_note_on); USBDevice.register_midi_note_off(midi_note_off); diff --git a/stm32/libraries/USBDevice/examples/USBDeviceMouse/USBDeviceMouse.ino b/stm32/libraries/USBDevice/examples/USBDeviceMouse/USBDeviceMouse.ino index d7be27f8..cf8b4b70 100644 --- a/stm32/libraries/USBDevice/examples/USBDeviceMouse/USBDeviceMouse.ino +++ b/stm32/libraries/USBDevice/examples/USBDeviceMouse/USBDeviceMouse.ino @@ -11,11 +11,10 @@ using namespace klangstrom; void setup() { Serial.begin(115200); - Serial.println("----------------"); - Serial.println("USB Device Mouse"); - Serial.println("----------------"); - Serial.println(__DATE__); - Serial.println(__TIME__); + Serial.println("--------------"); + Serial.println("USBDeviceMouse"); + Serial.println("--------------"); + usb_device_init(); } diff --git a/stm32/libraries/USBHost/examples/USBHostMIDI/USBHostMIDI.ino b/stm32/libraries/USBHost/examples/USBHostMIDI/USBHostMIDI.ino index c909a6bd..4d23917d 100644 --- a/stm32/libraries/USBHost/examples/USBHostMIDI/USBHostMIDI.ino +++ b/stm32/libraries/USBHost/examples/USBHostMIDI/USBHostMIDI.ino @@ -8,18 +8,10 @@ using namespace klangstrom; void setup() { Serial.begin(115200); - Serial.println("-------------"); - Serial.println("USB Host MIDI"); - Serial.print("( "); - Serial.print(__DATE__); - Serial.print(" "); - Serial.print(__TIME__); - Serial.println(" )"); - Serial.println("-------------"); - printf("foo"); + Serial.println("-----------"); + Serial.println("USBHostMIDI"); + Serial.println("-----------"); - Serial.println(__DATE__); - Serial.println(__TIME__); USBHost.init(); USBHost.register_midi_note_off(midi_note_off); USBHost.register_midi_note_on(midi_note_on); diff --git a/stm32/libraries/USBHost/examples/USBHostMouseKeyboard/USBHostMouseKeyboard.ino b/stm32/libraries/USBHost/examples/USBHostMouseKeyboard/USBHostMouseKeyboard.ino index 194363cf..90d4cb13 100644 --- a/stm32/libraries/USBHost/examples/USBHostMouseKeyboard/USBHostMouseKeyboard.ino +++ b/stm32/libraries/USBHost/examples/USBHostMouseKeyboard/USBHostMouseKeyboard.ino @@ -8,14 +8,10 @@ using namespace klangstrom; void setup() { Serial.begin(115200); - Serial.println("-------------------------"); - Serial.println("USB Host Mouse + Keyboard"); - Serial.print("( "); - Serial.print(__DATE__); - Serial.print(" "); - Serial.print(__TIME__); - Serial.println(" )"); - Serial.println("-------------------------"); + Serial.println("--------------------"); + Serial.println("USBHostMouseKeyboard"); + Serial.println("--------------------"); + USBHost.init(); USBHost.register_key_pressed(key_pressed); USBHost.register_key_released(key_released); diff --git a/stm32/libraries/Wire/src/Wire.cpp b/stm32/libraries/Wire/src/Wire.cpp index 1d8d3ee5..055bdec2 100644 --- a/stm32/libraries/Wire/src/Wire.cpp +++ b/stm32/libraries/Wire/src/Wire.cpp @@ -35,16 +35,27 @@ static const uint8_t MASTER_ADDRESS = 0x01; TwoWire::TwoWire() { + memset((void *)&_i2c, 0, sizeof(_i2c)); _i2c.sda = digitalPinToPinName(SDA); _i2c.scl = digitalPinToPinName(SCL); } TwoWire::TwoWire(uint32_t sda, uint32_t scl) { + memset((void *)&_i2c, 0, sizeof(_i2c)); _i2c.sda = digitalPinToPinName(sda); _i2c.scl = digitalPinToPinName(scl); } +/** + * @brief TwoWire destructor + * @retval None + */ +TwoWire::~TwoWire() +{ + end(); +} + // Public Methods ////////////////////////////////////////////////////////////// void TwoWire::begin(uint32_t sda, uint32_t scl) @@ -106,11 +117,15 @@ void TwoWire::begin(int address, bool generalCall, bool NoStretchMode) void TwoWire::end(void) { i2c_deinit(&_i2c); - free(txBuffer); - txBuffer = nullptr; + if (txBuffer != nullptr) { + free(txBuffer); + txBuffer = nullptr; + } txBufferAllocated = 0; - free(rxBuffer); - rxBuffer = nullptr; + if (rxBuffer != nullptr) { + free(rxBuffer); + rxBuffer = nullptr; + } rxBufferAllocated = 0; } diff --git a/stm32/libraries/Wire/src/Wire.h b/stm32/libraries/Wire/src/Wire.h index e5a5d8d4..fafea29d 100644 --- a/stm32/libraries/Wire/src/Wire.h +++ b/stm32/libraries/Wire/src/Wire.h @@ -78,6 +78,7 @@ class TwoWire : public Stream { public: TwoWire(); TwoWire(uint32_t sda, uint32_t scl); + ~TwoWire(); // setSCL/SDA have to be called before begin() void setSCL(uint32_t scl) { diff --git a/stm32/platform.txt b/stm32/platform.txt index 1d145e49..41cc6af2 100644 --- a/stm32/platform.txt +++ b/stm32/platform.txt @@ -17,6 +17,12 @@ compiler.warning_flags.default= compiler.warning_flags.more=-Wall compiler.warning_flags.all=-Wall -Wextra +# EXPERIMENTAL feature: optimization flags +# - this is alpha and may be subject to change without notice +compiler.optimization_flags={compiler.optimization_flags} +compiler.optimization_flags.release={build.flags.optimize} {build.flags.debug} +compiler.optimization_flags.debug=-Og -g + compiler.path={runtime.tools.xpack-arm-none-eabi-gcc-12.2.1-1.2.path}/bin/ compiler.S.cmd=arm-none-eabi-gcc @@ -28,24 +34,24 @@ compiler.objcopy.cmd=arm-none-eabi-objcopy compiler.elf2hex.cmd=arm-none-eabi-objcopy compiler.libraries.ldflags= -compiler.extra_flags=-mcpu={build.mcu} {build.fpu} {build.float-abi} -DUSE_FULL_LL_DRIVER -mthumb "@{build.opt.path}" +compiler.extra_flags=-mcpu={build.mcu} {build.fpu} {build.float-abi} -DVECT_TAB_OFFSET={build.flash_offset} -DUSE_FULL_LL_DRIVER -mthumb "@{build.opt.path}" compiler.S.flags={compiler.extra_flags} -c -x assembler-with-cpp {compiler.stm.extra_include} -compiler.c.flags={compiler.extra_flags} -c {build.flags.optimize} {build.flags.debug} {compiler.warning_flags} -std={compiler.c.std} -ffunction-sections -fdata-sections --param max-inline-insns-single=500 -MMD {compiler.stm.extra_include} +compiler.c.flags={compiler.extra_flags} -c {compiler.optimization_flags} {compiler.warning_flags} -std={compiler.c.std} -ffunction-sections -fdata-sections --param max-inline-insns-single=500 -MMD {compiler.stm.extra_include} -compiler.cpp.flags={compiler.extra_flags} -c {build.flags.optimize} {build.flags.debug} {compiler.warning_flags} -std={compiler.cpp.std} -ffunction-sections -fdata-sections -fno-threadsafe-statics --param max-inline-insns-single=500 -fno-rtti -fno-exceptions -fno-use-cxa-atexit -MMD {compiler.stm.extra_include} +compiler.cpp.flags={compiler.extra_flags} -c {compiler.optimization_flags} {compiler.warning_flags} -std={compiler.cpp.std} -ffunction-sections -fdata-sections -fno-threadsafe-statics --param max-inline-insns-single=500 -fno-rtti -fno-exceptions -fno-use-cxa-atexit -MMD {compiler.stm.extra_include} compiler.ar.flags=rcs -compiler.c.elf.flags=-mcpu={build.mcu} {build.fpu} {build.float-abi} -mthumb {build.flags.optimize} {build.flags.debug} {build.flags.ldspecs} -Wl,--defsym=LD_FLASH_OFFSET={build.flash_offset} -Wl,--defsym=LD_MAX_SIZE={upload.maximum_size} -Wl,--defsym=LD_MAX_DATA_SIZE={upload.maximum_data_size} -Wl,--cref -Wl,--check-sections -Wl,--gc-sections -Wl,--entry=Reset_Handler -Wl,--unresolved-symbols=report-all -Wl,--warn-common +compiler.c.elf.flags=-mcpu={build.mcu} {build.fpu} {build.float-abi} -mthumb {compiler.optimization_flags} {build.flags.ldspecs} -Wl,--defsym=LD_FLASH_OFFSET={build.flash_offset} -Wl,--defsym=LD_MAX_SIZE={upload.maximum_size} -Wl,--defsym=LD_MAX_DATA_SIZE={upload.maximum_data_size} -Wl,--cref -Wl,--check-sections -Wl,--gc-sections -Wl,--entry=Reset_Handler -Wl,--unresolved-symbols=report-all -Wl,--warn-common compiler.objcopy.eep.flags=-O ihex -j .eeprom --set-section-flags=.eeprom=alloc,load --no-change-warnings --change-section-lma .eeprom=0 compiler.elf2bin.flags=-O binary compiler.elf2hex.flags=-O ihex -compiler.ldflags= +compiler.ldflags=-Wl,--no-warn-rwx-segments compiler.size.cmd=arm-none-eabi-size compiler.define=-DARDUINO= @@ -69,8 +75,7 @@ compiler.ar.extra_flags= compiler.elf2bin.extra_flags= compiler.elf2hex.extra_flags= -compiler.arm.cmsis.c.flags="-I{runtime.tools.CMSIS-5.7.0.path}/CMSIS/Core/Include/" "-I{build.system.path}/Drivers/CMSIS/Device/ST/{build.series}/Include/" "-I{build.system.path}/Drivers/CMSIS/Device/ST/{build.series}/Source/Templates/gcc/" "-I{runtime.tools.CMSIS-5.7.0.path}/CMSIS/DSP/Include" "-I{runtime.tools.CMSIS-5.7.0.path}/CMSIS/DSP/PrivateInclude" -compiler.arm.cmsis.ldflags="-L{runtime.tools.CMSIS-5.7.0.path}/CMSIS/DSP/Lib/GCC/" -l{build.cmsis_lib_gcc} +compiler.arm.cmsis.c.flags="-I{runtime.tools.CMSIS-5.9.0.path}/CMSIS/Core/Include/" "-I{build.system.path}/Drivers/CMSIS/Device/ST/{build.series}/Include/" "-I{build.system.path}/Drivers/CMSIS/Device/ST/{build.series}/Source/Templates/gcc/" "-I{runtime.tools.CMSIS-5.9.0.path}/CMSIS/DSP/Include" "-I{runtime.tools.CMSIS-5.9.0.path}/CMSIS/DSP/PrivateInclude" # USB Flags # --------- @@ -108,7 +113,6 @@ build.float-abi= build.flags.optimize=-Os build.flags.debug=-DNDEBUG build.flags.ldspecs=--specs=nano.specs -build.flash_offset=0 # Pre and post build hooks build.opt.name=build.opt @@ -138,7 +142,7 @@ recipe.S.o.pattern="{compiler.path}{compiler.S.cmd}" {compiler.S.flags} {build.i recipe.ar.pattern="{compiler.path}{compiler.ar.cmd}" {compiler.ar.flags} {compiler.ar.extra_flags} "{archive_file_path}" "{object_file}" ## Combine gc-sections, archives, and objects -recipe.c.combine.pattern="{compiler.path}{compiler.c.elf.cmd}" {compiler.c.elf.flags} "-Wl,--default-script={build.variant.path}/{build.ldscript}" "-Wl,--script={build.system.path}/ldscript.ld" "-Wl,-Map,{build.path}/{build.project_name}.map" {compiler.c.elf.extra_flags} {compiler.ldflags} {compiler.arm.cmsis.ldflags} -o "{build.path}/{build.project_name}.elf" "-L{build.path}" -Wl,--start-group {object_files} {compiler.libraries.ldflags} "{archive_file_path}" -lc -Wl,--end-group -lm -lgcc -lstdc++ +recipe.c.combine.pattern="{compiler.path}{compiler.c.elf.cmd}" {compiler.c.elf.flags} "-Wl,--default-script={build.variant.path}/{build.ldscript}" "-Wl,--script={build.system.path}/ldscript.ld" "-Wl,-Map,{build.path}/{build.project_name}.map" {compiler.c.elf.extra_flags} {compiler.ldflags} -o "{build.path}/{build.project_name}.elf" "-L{build.path}" -Wl,--start-group {object_files} {compiler.libraries.ldflags} "{archive_file_path}" -lc -Wl,--end-group -lm -lgcc -lstdc++ ## Create output (.bin file) recipe.objcopy.bin.pattern="{compiler.path}{compiler.objcopy.cmd}" {compiler.elf2bin.flags} {compiler.elf2bin.extra_flags} "{build.path}/{build.project_name}.elf" "{build.path}/{build.project_name}.bin" @@ -179,7 +183,7 @@ tools.stm32CubeProg.busybox.windows={path}/win/busybox.exe tools.stm32CubeProg.cmd=stm32CubeProg.sh tools.stm32CubeProg.upload.params.verbose= tools.stm32CubeProg.upload.params.quiet= -tools.stm32CubeProg.upload.pattern="{busybox}" sh "{path}/{cmd}" {upload.protocol} "{build.path}/{build.project_name}.bin" {upload.options} +tools.stm32CubeProg.upload.pattern="{busybox}" sh "{path}/{cmd}" {upload.protocol} "{build.path}/{build.project_name}.bin" {build.flash_offset} {upload.options} # blackmagic upload for generic STM32 tools.bmp_upload.cmd=arm-none-eabi-gdb diff --git a/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/Include/stm32f446xx.h b/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/Include/stm32f446xx.h index acc00c17..7658cefc 100644 --- a/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/Include/stm32f446xx.h +++ b/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/Include/stm32f446xx.h @@ -7,7 +7,7 @@ * This file contains: * - Data structures and the address mapping for all peripherals * - peripherals registers declarations and bits definition - * - Macros to access peripheral’s registers hardware + * - Macros to access peripheral's registers hardware * ****************************************************************************** * @attention @@ -34,7 +34,7 @@ #define __STM32F446xx_H #ifdef __cplusplus - extern "C" { +extern "C" { #endif /* __cplusplus */ /** @addtogroup Configuration_section_for_CMSIS @@ -64,7 +64,7 @@ */ typedef enum { -/****** Cortex-M4 Processor Exceptions Numbers ****************************************************************/ + /****** Cortex-M4 Processor Exceptions Numbers ****************************************************************/ NonMaskableInt_IRQn = -14, /*!< 2 Non Maskable Interrupt */ MemoryManagement_IRQn = -12, /*!< 4 Cortex-M4 Memory Management Interrupt */ BusFault_IRQn = -11, /*!< 5 Cortex-M4 Bus Fault Interrupt */ @@ -73,7 +73,7 @@ typedef enum DebugMonitor_IRQn = -4, /*!< 12 Cortex-M4 Debug Monitor Interrupt */ PendSV_IRQn = -2, /*!< 14 Cortex-M4 Pend SV Interrupt */ SysTick_IRQn = -1, /*!< 15 Cortex-M4 System Tick Interrupt */ -/****** STM32 specific Interrupt Numbers **********************************************************************/ + /****** STM32 specific Interrupt Numbers **********************************************************************/ WWDG_IRQn = 0, /*!< Window WatchDog Interrupt */ PVD_IRQn = 1, /*!< PVD through EXTI Line detection Interrupt */ TAMP_STAMP_IRQn = 2, /*!< Tamper and TimeStamp interrupts through the EXTI line */ @@ -152,14 +152,14 @@ typedef enum OTG_HS_IRQn = 77, /*!< USB OTG HS global interrupt */ DCMI_IRQn = 78, /*!< DCMI global interrupt */ FPU_IRQn = 81, /*!< FPU global interrupt */ - SPI4_IRQn = 84, /*!< SPI4 global Interrupt */ - SAI1_IRQn = 87, /*!< SAI1 global Interrupt */ - SAI2_IRQn = 91, /*!< SAI2 global Interrupt */ - QUADSPI_IRQn = 92, /*!< QuadSPI global Interrupt */ - CEC_IRQn = 93, /*!< CEC global Interrupt */ - SPDIF_RX_IRQn = 94, /*!< SPDIF-RX global Interrupt */ - FMPI2C1_EV_IRQn = 95, /*!< FMPI2C1 Event Interrupt */ - FMPI2C1_ER_IRQn = 96 /*!< FMPI2C1 Error Interrupt */ + SPI4_IRQn = 84, /*!< SPI4 global Interrupt */ + SAI1_IRQn = 87, /*!< SAI1 global Interrupt */ + SAI2_IRQn = 91, /*!< SAI2 global Interrupt */ + QUADSPI_IRQn = 92, /*!< QuadSPI global Interrupt */ + CEC_IRQn = 93, /*!< CEC global Interrupt */ + SPDIF_RX_IRQn = 94, /*!< SPDIF-RX global Interrupt */ + FMPI2C1_EV_IRQn = 95, /*!< FMPI2C1 Event Interrupt */ + FMPI2C1_ER_IRQn = 96 /*!< FMPI2C1 Error Interrupt */ } IRQn_Type; /** @@ -288,7 +288,7 @@ typedef struct __IO uint32_t RXDR; /*!< CEC Rx Data Register, Address offset:0x0C */ __IO uint32_t ISR; /*!< CEC Interrupt and Status Register, Address offset:0x10 */ __IO uint32_t IER; /*!< CEC interrupt enable register, Address offset:0x14 */ -}CEC_TypeDef; +} CEC_TypeDef; /** * @brief CRC calculation unit */ @@ -334,7 +334,7 @@ typedef struct __IO uint32_t CR; /*!< Debug MCU configuration register, Address offset: 0x04 */ __IO uint32_t APB1FZ; /*!< Debug MCU APB1 freeze register, Address offset: 0x08 */ __IO uint32_t APB2FZ; /*!< Debug MCU APB2 freeze register, Address offset: 0x0C */ -}DBGMCU_TypeDef; +} DBGMCU_TypeDef; /** * @brief DCMI @@ -734,7 +734,7 @@ typedef struct uint16_t RESERVED1; /*!< Reserved, 0x0E */ __IO uint32_t DR; /*!< Data input register, Address offset: 0x10 */ __IO uint32_t CSR; /*!< Channel Status register, Address offset: 0x14 */ - __IO uint32_t DIR; /*!< Debug Information register, Address offset: 0x18 */ + __IO uint32_t DIR; /*!< Debug Information register, Address offset: 0x18 */ uint16_t RESERVED2; /*!< Reserved, 0x1A */ } SPDIFRX_TypeDef; @@ -1173,9 +1173,9 @@ typedef struct * @} */ - /** @addtogroup Peripheral_Registers_Bits_Definition - * @{ - */ +/** @addtogroup Peripheral_Registers_Bits_Definition +* @{ +*/ /******************************************************************************/ /* Peripheral Registers_Bits_Definition */ @@ -1187,7 +1187,7 @@ typedef struct /* */ /******************************************************************************/ /* - * @brief Specific device feature definitions (not present on all devices in the STM32F4 serie) + * @brief Specific device feature definitions (not present on all devices in the STM32F4 series) */ #define ADC_MULTIMODE_SUPPORT /*! - - - - - - - Release Notes for STM32F4xx CMSIS - - - - - -
-


-

-
- - - - - - -
- - - - - - - - - -
Back - - - to Release page
-

Release - Notes for STM32F4xx CMSIS

-

Copyright - 2017 STMicroelectronics

-

-
-

 

- - - - - - - - - -
-

Update History

- -

V2.6.8 / 11-February-2022

- -     -   Main - - - Changes
-
- - -
    - -
  • All source files: update disclaimer to add reference to the new license agreement.
  • - -
  • Correct ETH bits definitions to be in line with naming used in the STM32F4 reference manual documents.
  • -
-

V2.6.7 / 16-July-2021

- -     -   Main - - - Changes
-
- -
    -
  • Add missing definition - FLASH_CR_ERRIE to the CMSIS header file.
  • - -
  • Remove unsupported “GPIOF_BASE” - and “GPIOG_BASE” defines from STM32F412Vx - device.
  • - -
  • Add new atomic register access - macros in stm32f4xx.h file.
  • - -
  • Add LSI maximum startup time - datasheet value: LSI_STARTUP_TIME.
  • - -
  • Fix a typo in CMSIS STM32F4xx - version macro (__STM32F4xx_CMSIS_VERSION).
    -
  • -
-

V2.6.6 / 12-February-2021

-     -   Main - - - Changes
-
- -
    -
  • system_stm32f4xx.c:
  • -
      -
    • Protect -Vector - - - table modification following SRAM or - FLASH preprocessor directive by a - generic preprocessor directive : - USER_VECT_TAB_ADDRESS
    • -
    • Update - - - SystemInit_ExtMemCtl() API to initialize - the tmpreg variable before each time out - loop condition.
      -
    • -
    -
  • Add License.md and - Readme.md files required for GitHub - publication
  • -
  • Improve GCC startup - files robustness.
  • -
  • Fix wrong value for - GPIO_MODER_MODE8_Msk and - GPIO_MODER_MODE2_Pos.
  • -
  • Update max number of - host channels in FS for STM32F446:
  • -
      -
    • Update - - - USB_OTG_FS_HOST_MAX_CHANNEL_NBR value - from 8 to 12.
    • -
    -
  • Add SMBDEN and SMBHEN - bit definition for STM32F410Tx device.
    -
  • -
-

V2.6.5 / 10-February-2020

-     -   Main - - - Changes
-
- -
    -
  • All header files
  • -
      -
    • Update - - - to use new BSD License format
      -
    • -
    -
  • MDK-ARM startup files
  • -
      -
    • Update - - - to fix invalid config wizard annotations
    • -
    -
-

V2.6.4 / 06-December-2019

-     -   Main - - - Changes
-
- -
    -
  • stm32f446xx.h file
  • -
      -
    • Update - - - to support HW flow control on UART4 and - UART5 instances
    • -
    -
  • stm32f412xx.h, stm32f413xx.h and stm32f423xx.h files
  • -
      -
    • Remove - - - unused IS_USB_ALL_INSTANCE() assert macro
    • -
    -
  • All header files
  • -
      -
    • Remove - - - unused IS_TIM_SYNCHRO_INSTANCE() assert - macro
    • -
    -
  • system_stm32f4xx.c file
  • -
      -
    • Update - - - SystemInit() API to don't reset RCC - registers to its reset values
      -
    • -
    -
-

V2.6.3 / 08-February-2019

-     -   Main - - - Changes
-
- -
    -
  • CRYP:
  • -
      -
    • Update CMSIS - devices with correct CRYP data input - register name: DIN instead of DR
    • -
    • Add Bits - definition for CRYP CR ALGOMODE AES - GCM/CCM
    • -
    -
  • HASH:
  • -
      -
    • Update - HASH_DIGEST_TypeDef structure: resize - the HR register
      -
    • -
    • Remove MDMAT Bits - definition
    • -
    -
  • TIM:
  • -
      -
    • Add requires TIM - assert macros:
    • -
        -
      • IS_TIM_SYNCHRO_INSTANCE()
      • -
      • IS_TIM_CLOCKSOURCE_TIX_INSTANCE()
      • -
      • IS_TIM_CLOCKSOURCE_ITRX_INSTANCE()
      • -
      -
    -
  • RCC
  • -
      -
    • Add - - - RCC_CSR_BORRSTF bits definition
    • -
    -
  • GPIO
    -
  • -
      -
    • Fix - - - GPIO BRR bits definition
    • -
    • Adjust - - - the GPIO present on STM32F412 devices
      -
    • -
    -
  • SAI
  • -
      -
    • Fix - - - frame length in SAI_xFRCR_FSALL & SAI_xFRCR_FRL bits - description
      -
    • -
    -
  • USB:
  • -
      -
    • Add - - - missing Bits Definitions in - USB_OTG_DOEPMSK register
    • -
    -
      -
        -
      • USB_OTG_DOEPMSK_AHBERRM
      • -
      • USB_OTG_DOEPMSK_OTEPSPRM
      • -
      • USB_OTG_DOEPMSK_BERRM
      • -
      • USB_OTG_DOEPMSK_NAKM
      • -
      • USB_OTG_DOEPMSK_NYETM
      • -
      -
    -
      -
    • Add - - - missing Bits Definitions in - USB_OTG_DIEPINT register
    • -
    -
      -
        -
      • USB_OTG_DIEPINT_INEPNM
      • -
      • USB_OTG_DIEPINT_AHBERR
      • -
      • USB_OTG_DOEPINT_OUTPKTERR
      • -
      •  USB_OTG_DOEPINT_NAK
      • -
      • USB_OTG_DOEPINT_STPKTRX
      • -
      -
    • Add - - - missing Bits Definitions in USB_OTG_DCFG - register
    • -
        -
      • USB_OTG_DCFG_XCVRDLY
      • -
      • USB_OTG_DCFG_ERRATIM
      • -
      -
    -
      -
    • Update - - - USB OTG max number of endpoints (6 FS - and 9 HS instead of 5 and 8)
    • -
    -
  • I2C/FMPI2C
  • -
      -
    • Align - - - Bit naming for FMPI2C_CR1 register: - FMPI2C_CR1_DFN--> FMPI2C_CR1_DNF
    • -
    • Add - - - IS_SMBUS_ALL_INSTANCE() - - - define
    • -
    -
  • DFSDM
  • -
      -
    • Align - - - Bit naming for DFSDM_FLTICR register: - DFSDM_FLTICR_CLRSCSDF--> - DFSDM_FLTICR_CLRSCDF
    • -
    -
  • PWR
  • -
      -
    • Remove PWR_CSR_WUPP - define: feature not available on - STM32F469xx/479xx devices
    • -
    -
-

V2.6.2 / 06-October-2017

-     -   Main - - - Changes
-
- -
    -
      -
    • Remove Date and - Version from all header files
    • -
    • USB_OTG - - - register clean up: remove - duplicated bits definitions
    • -
    • stm32f401xc.h, stm32f401xe.h, stm32f411xe.h files
    • -
        -
      • Remove - - - BKPSRAM_BASE define: feature not - available
      • -
      -
    • stm32f405xx.h, stm32f407xx.h files
    • -
        -
      • Rename - - - HASH_RNG_IRQn to RNG_IRQn: HASH - instance not available 
      • -
      -
    • stm32f410xx.h, stm32f412xx.h, stm32f413xx.h,  stm32f423xx.h files
    • -
        -
      • Add - - - missing wake-up pins defines
      • -
      -
    • stm32f412cx.h files
    • -
        -
      •  Add - support of USART3 instance
      • -
      -
    -
-

V2.6.1 / 14-February-2017

-     -   Main - - - Changes
-
- -
    -
  • General updates in - header files to support LL drivers
  • -
      -
    • Align - - - Bit naming for RCC_CSR - register (ex: RCC_CSR_PADRSTF - --> RCC_CSR_PINRSTF)
    • -
    • Add - - - new defines for RCC features support:
    • -
        -
      • RCC - - - PLLI2S and RCC PLLSAI support
      • -
      • RCC - - - PLLR I2S clock source and RCC PLLR - system clock support
      • -
      • RCC - - - SAI1A PLL source and RCC SAI1B PLL - source support
      • -
      • RCC - - - AHB2 support
        -
      • -
      -
    • Add - - - RCC_DCKCFGR_PLLI2SDIVQ_X - and  RCC_DCKCFGR_PLLSAIDIVQ_X - bits definition
    • -
    • Add new defines for - RCC_PLLI2SCFGR_RST_VALUE, - RCC_PLLSAICFGR_RST_VALUE and - RCC_PLLCFGR_RST_VALUE
    • -
    • Add - - - new defines for RTC features support:
    • -
        -
      • RTC - - - Tamper 2 support
      • -
      • RTC - - - AF2 mapping support
        -
      • -
      -
    • Align - - - Bit naming for RTC_CR and RTC_TAFCR - registers (ex: RTC_CR_BCK --> RTC_CR_BKP)
    • -
    • Add - - - new define to manage RTC backup register - number: RTC_BKP_NUMBER
    • -
    • Rename - - - IS_UART_INSTANCE() macro to - IS_UART_HALFDUPLEX_INSTANCE()
    • -
    • Add new defines to check - LIN instance: IS_UART_LIN_INSTANCE
    • -
    • Remove - - - USART6 instance from STM32F410Tx header - file
    • -
    • Rename - IS_I2S_ALL_INSTANCE_EXT() macro to - IS_I2S_EXT_ALL_INSTANCEE()
    • -
    • Add - - - IS_I2S_APB1_INSTANCE() macro to check if - I2S instance mapping: API1 or APB2
      -
    • -
    • Remove - - - SPI_I2S_SUPPORT define for SPI I2S - features support: I2S feature is - available on all STM32F4xx devices
    • -
    • Add - - - SPI_I2S_FULLDUPLEX_SUPPORT define for - STM32F413xx/423xx devices
    • -
    • Align - - - SPI_I2SCFGR bit - naming: SPI_I2SCFGR_ASTRTEN bit is - missing for STM32F412xx devices
    • -
    • Add - - - new I2S_APB1_APB2_FEATURE - - - define - - - for STM32F4xx devices where I2S IP's are - splited between RCC APB1 and APB2 - interfaces
      -
    • -
    • Add new FLASH_SR_RDERR define in FLASH_SR register
    • -
    • Add FLASH_OTP_BASE and  FLASH_OTP_END defnes - to manage FLASH OPT area
      -
    • -
    • Add - - - bit definitions for ETH_MACDBGR - register
    • -
    • Add - - - new defines ADC1_COMMON_BASE and - ADC123_COMMON_BASE to replace ADC_BASE - define
    • -
    • Add - - - new defines ADC1_COMMON and - ADC123_COMMON to replace ADC define
    • -
    • Add - - - new ADC macros: IS_ADC_COMMON_INSTANCE() -and IS_ADC_MULTIMODE_MASTER_INSTANCE()
      -
    • -
    • Add - - - new defines for ADC multi mode features - support
    • -
    • Add - - - new ADC aliases ADC_CDR_RDATA_MST and - ADC_CDR_RDATA_SLV for compatibilities - with all STM32 Families
    • -
    • Update - - - TIM CNT and ARR register mask on 32-bits
    • -
    • Add new TIM_OR_TI1_RMP define in TIM_OR register
    • -
    • Add - - - new TIM macros to check TIM feature - instance support:
    • -
        -
      • IS_TIM_COUNTER_MODE_SELECT_INSTANCE()
      • -
      • IS_TIM_CLOCK_DIVISION_INSTANCE()
      • -
      • IS_TIM_COMMUTATION_EVENT_INSTANCE()
      • -
      • IS_TIM_OCXREF_CLEAR_INSTANCE()
      • -
      • IS_TIM_CLOCKSOURCE_ETRMODE1_INSTANCE()
      • -
      • IS_TIM_CLOCKSOURCE_ETRMODE2_INSTANCE()
      • -
      • IS_TIM_REPETITION_COUNTER_INSTANCE()
      • -
      • IS_TIM_ENCODER_INTERFACE_INSTANCE()
      • -
      • IS_TIM_HALL_SENSOR_INTERFACE_INSTANCE()
      • -
      • IS_TIM_BREAK_INSTANCE()
      • -
      -
    • CAN_IER - - - register clean up: remove duplicated bit - definitions
    • -
    • USB_OTG - - - register: fix the wrong defined values - for USB_OTG_GAHBCFG bits
    • -
    -
-

V2.6.0 / 04-November-2016

-     -   Main - - - Changes
-
- -
    -
  • Add support of STM32F413xx and STM32F423xx devices 
  • -
      -
    • Add - - - "stm32f413xx.h" and "stm32f423xx.h" - - - files
    • -
    -
      -
    • -

      Add - - - startup files  "startup_stm32f413xx.s" - - - and "startup_stm32f423xx.s" - - - for EWARM, MDK-ARM and SW4STM32 - toolchains

      -
    • -
    • Add - - - Linker files "stm32f413xx_flash.icf", - - - "stm32f413xx_sram.icf", - - - "stm32f423xx_flash.icf" - - - and "stm32f423xx_sram.icf" - - - used - - - within EWARM Workspaces
    • -
    -
  • All header files
  • -
      -
    • Use - - - _Pos and _Mask macro for all Bit - Definitions
    • -
    • Update - - - LPTIM_OR Bit Definition
    • -
    • Update - - - the defined frequencies by scale for USB - exported constants
    • -
    • Add - - - UID_BASE, FLASHSIZE_BASE and - PACKAGE_BASE defines
    • -
    • Add - - - new define DAC_CHANNEL2_SUPPORT to - manage DAC channel2 support
    • -
    • Use - - - new DAC1 naming
    • -
    • Rename - - - PWR_CSR_UDSWRDY define to PWR_CSR_UDRDY - in PWR_CSR register
    • -
    • Align - - - Bit naming for EXTI_IMR and EXTI_EMR - registers (ex: EXTI_IMR_MR0 --> - EXTI_IMR_IM0)
      -
    • -
    • Add - - - new EXTI_IMR_IM define in EXTI_IMR - register
    • -
    • Add - - - missing DMA registers definition
    • -
    • Add - - - macro to check SMBUS instance support
      -
    • -
    -
  • stm32f412cx.h, stm32f412zx.h, stm32f412vx.h, stm32f412rx.h files
  • -
      -
    • Add missing SYSCFG - register: CFGR2
    • -
    -
  • stm32f405xx.h, stm32f407xx.h, stm32f427xx.h, stm32f429xx.h files
  • -
      -
    • Remove - - - HASH_RNG_IRQn in IRQn_Type enumeration
    • -
    -
  • stm32f405xx.h, stm32f407xx.h, stm32f415xx.h, stm32f417xx.h files
  • -
      -
    • Remove - - - I2C FLTR register as not supported
      -
    • -
    -
  • stm32f407xx.h, stm32f417xx.h, stm32f427xx.h, stm32f429xx.h, stm32f437xx.h, stm32f439xx.h, stm32f469xx.h, stm32f479xx.h files
  • -
      -
    • Add - - - missing Bit Definition of ETH_MACDBGR - register
    • -
    -
  • system_stm32f4xx.c file
  • -
      -
    • Add - - - APBPrescTable declaration
    • -
    -
-

V2.5.1 / 28-June-2016

-     -   Main - - - Changes
-
- -
    -
  • stm32f412rx.h, - stm32f412vx.h and - - - stm32f412zx.h files:
  • -
      -
    • -

      Add QSPI1_V2_1L - define to manage the QSPI DMA2 - limitation
      -

      -
    • -
    -
-

V2.5.0 / 22-April-2016

-     -   Main - - - Changes
-
- -
    -
  • Add support of STM32F412Cx, STM32F412RxSTM32F412Vx and STM32F412Zx devices
  • -
      -
    • -

      Add - "stm32f412Cx.h", "stm32f412Rx.h", "stm32f412Vx.h" and "stm32f412Zx.h" files

      -
    • -
    • -

      Add startup - files  "startup_stm32f412cx.s", "startup_stm32f412rx.s", "startup_stm32f412vx.s" and - - - "startup_stm32f412zx.s" - - - for EWARM, MDK-ARM and SW4STM32 - toolchains

      -
    • -
    • Add Linker files "stm32f412cx_flash.icf", "stm32f412cx_sram.icf", "stm32f412rx_flash.icf", "stm32f412rx_sram.icf", "stm32f412vx_flash.icf", "stm32f412vx_sram.icf", "stm32f412zx_flash.icf" and "stm32f412zx_sram.icf" used within EWARM - Workspaces
    • -
    -
  • Header files for all - STM32 devices
  • -
      -
    • Remove uint32_t cast - and keep only Misra Cast (U) to avoid - two types cast duplication
    • -
    • Correct some bits - definition to be in line with naming - used in the Reference Manual
    • -
        -
      • WWDG_CR_Tx changed - - - to WWDG_CR_T_x
      • -
      • WWDG_CFR_Wx changed - - - to WWDG_CFR_W_x
      • -
      • WWDG_CFR_WDGTBx changed - - - to WWDG_CFR_WDGTB_x
      • -
      -
    -
  • stm32f407xx.h, - stm32f417xx.h, stm32f427xx.h, - - - stm32f429xx.h, stm32f437xx.h, - stm32f439xx.h, stm32f446xx.h, stm32f469xx.h, stm32f479xx.h files
  • -
      -
    • Correct some bits - definition to be in line with naming - used in the Reference Manual
    • -
    -
      -
        -
      • DCMI_RISR_x changed - - - to DCMI_RIS_x
      • -
      • DCMI_RISR_OVF_RIS - - - changed to DCMI_RIS_OVR_RIS
      • -
      • DCMI_IER_OVF_IE - - - changed to DCMI_IER_OVR_IE
      • -
      -
    -
  • stm32f427xx.h, - stm32f429xx.h, stm32f437xx.h, - stm32f439xx.h, stm32f469xx.h, stm32f479xx.h, stm32f446xx.h files
  • -
      -
    • Correct some bits - definition to be in line with naming - used in the Reference Manual
    • -
        -
      • SAI_xFRCR_FSPO changed - - - to SAI_xFRCR_FSPOL
      • -
      -
    • Rename - IS_SAI_BLOCK_PERIPH to - IS_SAI_ALL_INSTANCE
      -
    • -
    -
  • stm32f410cx.h, - stm32f410rx.h, stm32f410tx.h files - - - and stm32f446xx.h
  • -
      -
    • Remove - FMPI2C_CR1_SWRST and FMPI2C_CR1_WUPEN - Bit definition for I2C_CR1 register
    • -
    -
  • stm32f407xx.h, - stm32f417xx.h, stm32f427xx.h, - stm32f437xx.h, stm32f439xx.h, - stm32f469xx.h, stm32f479xx.h files
  • -
      -
    • Add missing bits - definitions for DMA2D_CR, - DMA2D_FGPFCCR, DMA2D_BGPFCCR, - DMA2D_OPFCCR registers
    • -
    -
  • stm32f401xc.h, stm32f401xe.h, stm32f411xe.h files
  • -
      -
    • Add missing RCC_DCKCFGR register - - - in RCC_TypeDef structure
      -
    • -
    • Add - missing Bit definition for RCC_DCKCFGR - register
    • -
    -
  • system_stm32f4xx.c
    -
  • -
      -
    • Update - SystemInit_ExtMemCtl() API to fix delay - optimization problem with GCC compiler: index variable is - declared as volatile 
    • -
    -
  • stm32f4xx.h
  • -
      -
    • Rename - __STM32F4xx_CMSIS_DEVICE_VERSION_xx - defines to - __STM32F4_CMSIS_VERSION_xx (MISRA-C 2004 rule 5.1)
    • -
    -
- -

V2.4.3 / 29-January-2016

-     -   Main - - - Changes
-
-
-
    -
  • -

    Header file for all STM32 devices -

    -
  • -
      -
    • Rename -ADC - - - overrun flags definitions : - ADC_CSR_DOVR1, ADC_CSR_DOVR2 and - ADC_CSR_DOVR3 are replaced respectively - by ADC_CSR_OVR1, ADC_CSR_OVR2 and - ADC_CSR_OVR3 to be aligned with - reference manuals
    • -
    • Add - - - missing bits definitions for DAC : - DAC_CR_DMAUDRIE1 and DAC_CR_DMAUDRIE2
    • -
    • Update - - - CMSIS driver to be compliant with MISRA - C 2004 rule 10.6
    • -
    • Remove - - - the double definition of - USB_OTG_HS_MAX_IN_ENDPOINTS and add a - new one for  - USB_OTG_HS_MAX_OUT_ENDPOINTS
    • -
    -
  • stm32f446xx.h, - - - stm32f469xx.h, stm32f479xx.h files 
  • -
      -
    • Change - - - the bit definition value of - QUADSPI_CR_FTHRES
    • -
    -
  • - stm32f446xx.h, stm32f469xx.h, - stm32f479xx.h, stm32f429xx.h, - stm32f439xx.h files
  • -
      -
    • Rename - - - the LTDC_GCR_DTEN to LTDC_GCR_DEN in - order to be aligned with the reference - manual
    • -
    • Rename - - - DCMI_MISR bit definitions to DCMI_MIS
    • -
    • Rename - - - DCMI_ICR_OVF_ISC to DCMI_ICR_OVR_ISC
    • -
    • Add - - - missing bits definitions for DCMI_ESCR, - DCMI_ESUR, DCMI_CWSTRT, DCMI_CWSIZE, - DCMI_DR registers
    • -
    -
  • - stm32f407xx.h, stm32f417xx.h, - stm32f427xx.h, stm32f437xx.h files
  • -
      -
    • Rename - - - DCMI_MISR bit definitions to DCMI_MIS
    • -
    • Rename - - - DCMI_ICR_OVF_ISC to DCMI_ICR_OVR_ISC
    • -
    • Add - - - missing bits definitions for DCMI_ESCR, - DCMI_ESUR, DCMI_CWSTRT, DCMI_CWSIZE, - DCMI_DR registers
    • -
    -
  • - stm32f410cx.h, stm32f410rx.h, - stm32f410tx.h files
  • -
      -
    • Update - - - the - - - LPTIM SNGSTRT defined value
    • -
    -
  • stm32f427xx.h, - stm32f429xx.h, stm32f437xx.h, - stm32f439xx.h, stm32f469xx.h, stm32f479xx.h - - - files
  • -
      -
    • Rename - - - the DMA2D_IFSR bit definitions to - DMA2D_IFCR
    • -
    -
  • stm32f427xx.h, - stm32f429xx.h, stm32f437xx.h, - stm32f439xx.h, stm32f469xx.h, stm32f479xx.h, stm32f446xx.h - - - files 
  • -
      -
    • Correct - - - a wrong value of SAI_xCR2_CPL definition - bit 
      -
    • -
    -
-

V2.4.2 / 13-November-2015

-

Main Changes

-
    -
  • -

    system_stm32f4xx.c file -

    -
  • -
      -
    • update -SystemInit_ExtMemCtl() - - - function implementation to allow the - possibility of simultaneous use of - DATA_IN_ExtSRAM and DATA_IN_ExtSDRAM
    • -
    -
  • stm32f4xx.h - - - file
  • -
      -
    • add - - - symbols for STM32F411xC devices
      -
    • -
    -
  • stm32f405xx.h, - - - stm32f407xx.h, stm32f415xx.h, - stm32f417xx.h files
  • -
      -
    • add - - - FSMC_BCRx_CPSIZE bits definitions
    • -
    • remove - - - FSMC_BWTRx_CLKDIV and FSMC_BWTRx_DATLAT - bits definitions
    • -
    -
  • stm32f429xx.h, - - - stm32f427xx.h, stm32f437xx.h files
  • -
      -
    • add - - - FMC_BCRx_CPSIZE bits definitions
    • -
    • remove - - - FMC_BWTRx_CLKDIV and FMC_BWTRx_DATLAT - bits definitions
    • -
    -
  • stm32f446xx.h, - - - stm32f469xx.h and stm32f479xx.h
  • -
      -
    • update - - - USB_OTG_GlobalTypeDef registers - structure to remove ADP control - registers
    • -
    • add - - - USB_OTG_DOEPMSK_OTEPSPRM and - USB_OTG_DOEPINT_OTEPSPR bits definitions
    • -
    • Remove - - - ADP related bits definitions
    • -
    • add - - - IS_PCD_ALL_INSTANCE() and - IS_HCD_ALL_INSTANCE() macros
      -
    • -
    -
-

V2.4.1 / 09-October-2015

-

Main Changes

-
    -
  • "stm32f469xx.h", - - - "stm32f479xx.h" -
      -
    • Update bits definition for - DSI_WPCR and DSI_TCCR registers
      -
    • -
    -
  • -
-

V2.4.0 / 14-August-2015

-

Main Changes

-
    -
  • -

    Add - - - support of STM32F469xx - and STM32F479xx devices
    -

    -
  • -
      -
    • -

      Add - - - "stm32f469xx.h" and "stm32f479xx.h" - - - files

      -
    • -
    -
      -
    • -

      Add - - - startup files  "startup_stm32f469xx.s" - - - and "startup_stm32f479xx.s" - - - for EWARM, MDK-ARM and SW4STM32 - toolchains

      -
    • -
    • Add - - - Linker files "stm32f469xx_flash.icf", - - - "stm32f469xx_sram.icf", - - - "stm32f479xx_flash.icf" - - - and "stm32f479xx_sram.icf" - - - used - - - within EWARM Workspaces
    • -
    -
  • -

    Add - - - support of STM32F410xx - devices
    -

    -
  • -
      -
    • -

      Add - - - "stm32f410cx.h", "stm32f410tx.h" - and "stm32f410rx.h" - - - files

      -
    • -
    -
      -
    • -

      Add - - - startup files  "startup_stm32f410cx.s", - - - "startup_stm32f410rx.s" and "startup_stm32f410tx.s" - - - for EWARM, MDK-ARM and SW4STM32 - toolchains

      -
    • -
    • Add - - - Linker files "stm32f410cx_flash.icf", - - - "stm32f410cx_sram.icf", - - - "stm32f410rx_flash.icf", "stm32f410tx_sram.icf", - - - "stm32f410tx_flash.icf",  - and "stm32f410rx_sram.icf" - - - used - - - within EWARM Workspaces
    • -
    -
-

V2.3.2 / 26-June-2015

-

Main Changes

-
    -
  • "stm32f405xx.h", - - - "stm32f407xx.h", "stm32f415xx.h" and - "stm32f417xx.h" -
      -
    • Update FSMC_BTRx_DATAST - - - and FSMC_BWTRx_DATAST (where x can - be 1, 2, 3 and 4) mask on 8bits - instead of 4bits
    • -
    -
  • -
  • "stm32f427xx.h", - - - "stm32f437xx.h", "stm32f429xx.h" and - "stm32f439xx.h"
    -
      -
    • Update the defined mask value - for SAI_xSR_FLVL_2
    • -
    -
  • -
  • "stm32f415xx.h", - - - "stm32f417xx.h", "stm32f437xx.h" and - "stm32f439xx.h"
  • -
      -
    • HASH alignement with bits namming - used in documentation -
        -
          -
        • Rename HASH_IMR_DINIM to - HASH_IMR_DINIE
        • -
        -
          -
        • Rename HASH_IMR_DCIM to - HASH_IMR_DCIE
        • -
        -
          -
        • Rename HASH_STR_NBW to - HASH_STR_NBW
        • -
        -
      -
    • -
    -
  • system_stm32f4xx.c
  • -
      -
    • Remove __IO on constant table - declaration
    • -
    -
      -
    • Implement workaround to cover - RCC limitation regarding peripheral - enable delay
    • -
    -
      -
    • SystemInit_ExtMemCtl() - - - update GPIO configuration when external - SDRAM is used 
    • -
    -
- -

V2.3.1 / 03-April-2015

- Main Changes -
    -
  • -

    Header file for all STM32 devices

    -
      -
    • Update SRAM2, SRAM3 and - BKPSRAM Bit-Banding base address - defined values
    • -
    • Keep reference to - SRAM3 only for STM32F42xx and - STM32F43xx devices
    • -
    • Remove CCMDATARAM_BB_BASE: the - CCM Data RAM region is not accessible - via Bit-Banding
      -
    • -
    • Update the RTC_PRER_PREDIV_S defined value to 0x00007FFF instead of - 0x00001FFF
      -
    • -
    -
  • -
-

V2.3.0 / 02-March-2015

-

Main Changes

- -
    -
  • -

    Add - - - support of STM32F446xx - devices
    -

    -
  • -
      -
    • -

      Add - - - "stm32f446xx.h" file

      -
    • -
    -
      -
    • -

      Add - - - startup file "startup_stm32f446xx.s" - for EWARM, MDK-ARM and TrueSTUDIO - toolchains

      -
    • -
    • Add - - - Linker files "stm32f446xx_flash.icf" and - "stm32f446xx_sram.icf" used within EWARM - Workspaces
    • -
    -
  • -

    Header file for all STM32 devices

    -
      -
    • Add missing bits - definition in the EXTI IMR, EMR, RTSR, - FTSR, SWIER and PR registers
    • -
    • Update - RCC_AHB1RSTR_OTGHRST bit definition
    • -
    • Update PWR_CR_VOS - bits definition for STM32F40xx - - - and STM32F41xx - - - devices
    • -
    • -

      update SAI_xCR1_MCKDIV bit - definition
      -

      -
    • -
    -
  • -
-

V2.2.0 / 15-December-2014

-

Main Changes

- -
    -
  • -

    stm32f4xx.h

    -
      -
    • -

      Add new constant definition STM32F4 -
      -

      -
    • -
    -
  • -
  • -

    system_stm32f4xx.c
    -

    -
      -
    • -

      Fix SDRAM configuration in - SystemInit_ExtMemCtl(): change RowBitsNumber - from 11 to 12 (for MT48LC4M32B2 - available on STM324x9I_EVAL board)
      -

      -
    • -
    -
  • -
  • -

    Header file for all STM32 devices

    -
      -
    • -

      Add missing bits definition - for CAN, FMC and USB peripherals

      -
    • -
    • -

      GPIO_TypeDef: change the BSRR - register definition, the two 16-bits - definition BSRRH and BSRRL are - merged in a single 32-bits - definition BSRR

      -
    • -
    -
  • -
-

V2.1.0 / 19-June-2014

-

Main Changes

- - - -
    -
  • -

    Add - - - support of STM32F411xExx - devices
    -

    -
  • -
      -
    • -

      Add - - - "stm32f411xe.h" file

      -
    • -
    -
      -
    • -

      Add - - - startup file "startup_stm32f411xx.s" - for EWARM, MDK-ARM and TrueSTUDIO - toolchains

      -
    • -
    -
  • -

    All - - - header files

    -
  • -
      -
    • -

      Add - - - missing defines for GPIO LCKR Register

      -
    • -
    • -

      Add defines for - memories base and end addresses: - FLASH, SRAM, BKPSRAM and CCMRAM.

      -
    • -
    • -

      Add -the - - - following aliases for IRQ number and - handler definition to ensure - compatibility across the product lines - of STM32F4 Series;
      -

      -
    • -
        -
      • -

        example for STM32F405xx.h

        -
      • -
      -
    -
-
#define - - - FMC_IRQn              - - - FSMC_IRQn
- #define - - - FMC_IRQHandler     - FSMC_IRQHandler
-
    -
      -
        -
      • -

        and for STM32F427xx.h

        -
      • -
      -
    -
-
#define - - - FSMC_IRQn            - - - FMC_IRQn
- #define - - - FSMC_IRQHandler   FMC_IRQHandler
-
    -
  • "stm32f401xc.h" - - - and "stm32f401xe.h": update to be in line - with latest version of the Reference - - - manual
  • -
      -
    • Remove - - - RNG registers structures and the - corresponding bit definitions
    • -
    • Remove - - - any occurrence to RNG (clock enable, - clock reset,…)
    • -
    • Add - - - the following bit definition for PWR CR - register
    • -
        -
      • #define  PWR_CR_ADCDC1      - - - ((uint32_t)0x00002000)
      • -
      -
        -
      • #define  PWR_CR_LPLVDS       - - - ((uint32_t)0x00000400)     
      • -
      -
        -
      • #define  - - - PWR_CR_MRLVDS   -    ((uint32_t)0x00000800)     
      • -
      -
    -
  • "stm32f427xx.h", - - - "stm32f437xx.h", "stm32f429xx.h" and - "stm32f439xx.h"
  • -
      -
    • Add - - - a new legacy bit definition for PWR to - be in line with latest version of the - Reference manual
    • -
        -
      • #define  PWR_CR_LPUDS        PWR_CR_LPLVDS
      • -
      -
        -
      • #define  -PWR_CR_MRUDS      - - - PWR_CR_MRLVDS
      • -
      -
    -
  • -

    Update startup - files for EWARM toolchain to cope with - compiler enhancement of the V7.10 - version

    -
  • -
  • -

    system_stm32f4xx.c

    -
  • -
      -
    • Remove - - - dependency vs. the HAL, to allow using - this file without the need to have the - HAL drivers
      -
    • -
        -
      • Include - - - stm32f4xx.h - - - instead of stm32f4xx_hal.h
      • -
      • Add -definition - - - of HSE_VALUE and HSI_VALUE, if they - are not yet defined in the compilation - scope (these values are defined in - stm32f4xx_hal_conf).
        -
      • -
      -
    • -

      Use “__IO const” - instead of “__I”, to avoid any - compilation issue when __cplusplus - switch is defined

      -
    • -
    -
-

V2.0.0 / 18-February-2014

-

Main Changes

- - - -
    -
  • Update based on STM32Cube - specification
    -
  • -
  • This version and later has to be - used only with STM32CubeF4 based development
  • -
-

V1.3.0 / - 08-November-2013

-

Main Changes

-
    -
  • -

    Add -support - - - of STM32F401xExx - devices

    -
  • -
  • Update startup files "startup_stm32f401xx.s" for EWARM, MDK-ARM, TrueSTUDIO and - Ride toolchains: Add SPI4 interrupt - handler entry in the vector table
  • -
-

V1.2.1 / - 19-September-2013

-

Main Changes

-
    -
  • -

    system_stm32f4xx.c : Update FMC SDRAM - configuration (RBURST mode activation)
    -

    -
  • -
  • Update startup files "startup_stm32f427_437xx.s" and "startup_stm32f429_439xx.s"  for - TrueSTUDIO and Ride toolchains and - maintain the old name of startup files for - legacy purpose
  • -
-

V1.2.0 / - 11-September-2013

-

Main Changes

-
    -
  • -

    Add -support - - - of STM32F429/439xx - and STM32F401xCxx - devices

    -
  • -
  • Update - - - definition of STM32F427/437xx devices : - extension -of - - - the features to include system clock up to - 180MHz, dual bank Flash, reduced STOP Mode - current, SAI, PCROP, SDRAM and DMA2D
  • -
  • stm32f4xx.h
    -
    -
      -
    • Add the - following device defines :
    • -
        -
      • "#define - STM32F40_41xxx" for all STM32405/415/407/417xx devices
      • -
      • "#define - STM32F427_437xx" for all STM32F427/437xx devices
      • -
      • "#define - STM32F429_439xx" for all STM32F429/439xx devices
      • -
      • "#define - STM32F401xx" for all STM32F401xx devices
      • -
      -
    • Maintain the - old device define for legacy purpose
    • -
    • Update IRQ - handler enumeration structure to - support all STM32F4xx Family devices. -  
    • -
    -
  • -
  • Add new startup files "startup_stm32f40_41xxx.s","startup_stm32f427_437xx.s""startup_stm32f429_439xx.s" and "startup_stm32f401xx.s" for all toolchains and maintain - the old name for startup files for legacy - purpose
  • -
  • system_stm32f4xx.c -
      -
    • Update - the system configuration to - support all STM32F4xx Family devices. -  
    • -
    -
  • -
-

V1.1.0 / - 11-January-2013

-

Main Changes

-
    -
  • Official release for STM32F427x/437x - devices.
  • -
  • stm32f4xx.h
    -
    -
      -
    • Update product - define: replace "#define STM32F4XX" by - "#define STM32F40XX" for STM32F40x/41x - devices
    • -
    •  Add new - product define: "#define STM32F427X" - for STM32F427x/437x devices.
    • -
    -
  • -
  • Add new startup files "startup_stm32f427x.s" for all - toolchains
  • -
  • rename startup files "startup_stm32f4xx.s" by "startup_stm32f40xx.s" for all - toolchains
  • -
  • system_stm32f4xx.c -
      -
    • Prefetch Buffer - enabled
    • -
    • Add reference - to STM32F427x/437x devices and - STM324x7I_EVAL board
    • -
    • SystemInit_ExtMemCtl() - - - function
      -
      -
        -
      • Add - configuration of missing FSMC - address and data lines
        -
      • -
      -
        -
      • Change - memory type to SRAM instead of - PSRAM (PSRAM is available only on - STM324xG-EVAL RevA) and update - timing values
      • -
      -
    • -
    -
  • -
- -

V1.0.2 / 05-March-2012

-

Main Changes

-
    -
  • All source files: license - disclaimer text update and add link to the - License file on ST Internet.
  • -
-

V1.0.1 / 28-December-2011

-

Main Changes

-
    -
  • All source files: update - disclaimer to add reference to - the new license agreement
  • -
  • stm32f4xx.h
  • -
      -
    • Correct bit - - - definition: RCC_AHB2RSTR_HSAHRST changed - - - to RCC_AHB2RSTR_HASHRST
    • -
    -
-

V1.0.0 / 30-September-2011

-

Main Changes

-
    -
  • First official release for STM32F40x/41x - devices
  • -
  • Add startup file for TASKING - toolchain
  • -
  • system_stm32f4xx.c: - driver's comments update
  • -
-

V1.0.0RC2 / 26-September-2011

-

Main Changes

-
    -
  • Official version (V1.0.0) - Release Candidate2 for STM32F40x/41x - devices
  • -
  • stm32f4xx.h
  • -
      -
    • Add define for Cortex-M4 - revision __CM4_REV
    • -
    • Correct RCC_CFGR_PPRE2_DIV16 - bit (in RCC_CFGR register) value - to 0x0000E000
    • -
    • Correct some bits - definition to be in line with naming - used in the Reference Manual (RM0090)
    • -
        -
      • GPIO_OTYPER_IDR_x - changed to GPIO_IDR_IDR_x
      • -
      • GPIO_OTYPER_ODR_x - changed to GPIO_ODR_ODR_x
      • -
      • SYSCFG_PMC_MII_RMII - changed to SYSCFG_PMC_MII_RMII_SEL
      • -
      • RCC_APB2RSTR_SPI1 - changed to RCC_APB2RSTR_SPI1RST
      • -
      • DBGMCU_APB1_FZ_DBG_IWDEG_STOP - changed to DBGMCU_APB1_FZ_DBG_IWDG_STOP
      • -
      • PWR_CR_PMODE - changed to PWR_CR_VOS
      • -
      • PWR_CSR_REGRDY - changed to PWR_CSR_VOSRDY
      • -
      • Add new define - RCC_AHB1ENR_CCMDATARAMEN
      • -
      • Add new - defines SRAM2_BASE, CCMDATARAM_BASE - and BKPSRAM_BASE
      • -
      -
    • GPIO_TypeDef structure: in the - comment change AFR[2] address - mapping to 0x20-0x24 - instead of 0x24-0x28
    • -
    -
  • system_stm32f4xx.c
  • -
      -
    • SystemInit(): add code - to enable the FPU
    • -
    • SetSysClock(): change - PWR_CR_PMODE - by PWR_CR_VOS
    • -
    • SystemInit_ExtMemCtl(): - remove commented values
    • -
    -
  • startup (for all compilers)
  • -
      -
    • Delete code used to enable the - FPU (moved to system_stm32f4xx.c file)
    • -
    • File’s header updated
    • -
    -
-

V1.0.0RC1 / 25-August-2011

-

Main Changes

-
    -
  • Official version (V1.0.0) - Release Candidate1 for STM32F4xx devices
  • -
- -
    -
-
-
-

For - complete documentation on STM32 - Microcontrollers visit www.st.com/STM32

-

-
-

-
-
-

 

-
- \ No newline at end of file + + + + + + + Release Notes for STM32F4xx CMSIS + + + + + + +
+ + + diff --git a/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/Source/Templates/gcc/startup_stm32f446xx.s b/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/Source/Templates/gcc/startup_stm32f446xx.s index 41fb9f45..eef07834 100644 --- a/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/Source/Templates/gcc/startup_stm32f446xx.s +++ b/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/Source/Templates/gcc/startup_stm32f446xx.s @@ -60,6 +60,9 @@ defined in linker script */ Reset_Handler: ldr sp, =_estack /* set stack pointer */ +/* Call the clock system initialization function.*/ + bl SystemInit + /* Copy the data segment initializers from flash to SRAM */ ldr r0, =_sdata ldr r1, =_edata @@ -91,8 +94,6 @@ LoopFillZerobss: cmp r2, r4 bcc FillZerobss -/* Call the clock system initialization function.*/ - bl SystemInit /* Call static constructors */ bl __libc_init_array /* Call the application's entry point.*/ @@ -121,7 +122,6 @@ Infinite_Loop: *******************************************************************************/ .section .isr_vector,"a",%progbits .type g_pfnVectors, %object - .size g_pfnVectors, .-g_pfnVectors g_pfnVectors: @@ -242,6 +242,9 @@ g_pfnVectors: .word FMPI2C1_EV_IRQHandler /* FMPI2C 1 Event */ .word FMPI2C1_ER_IRQHandler /* FMPI2C 1 Error */ + + .size g_pfnVectors, .-g_pfnVectors + /******************************************************************************* * * Provide weak aliases for each Exception handler to the Default_Handler. diff --git a/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/_htmresc/favicon.png b/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/_htmresc/favicon.png new file mode 100644 index 0000000000000000000000000000000000000000..06713eec4974e141c6e9b4156d34e61e89f282ca GIT binary patch literal 4126 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 literal 0 HcmV?d00001 diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/mini-st.css b/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/_htmresc/mini-st_2020.css similarity index 77% rename from stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/mini-st.css rename to stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/_htmresc/mini-st_2020.css index 3caf11c3..db8b406a 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/mini-st.css +++ b/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/_htmresc/mini-st_2020.css @@ -1,39 +1,39 @@ @charset "UTF-8"; /* - Flavor name: Default (mini-default) - Author: Angelos Chalaris (chalarangelo@gmail.com) - Maintainers: Angelos Chalaris - mini.css version: v3.0.0-alpha.3 + Flavor name: Custom (mini-custom) + Generated online - https://minicss.org/flavors + mini.css version: v3.0.1 */ /* Browsers resets and base typography. */ /* Core module CSS variable definitions */ :root { - --fore-color: #111; - --secondary-fore-color: #444; - --back-color: #f8f8f8; - --secondary-back-color: #f0f0f0; - --blockquote-color: #f57c00; - --pre-color: #1565c0; - --border-color: #aaa; - --secondary-border-color: #ddd; - --heading-ratio: 1.19; + --fore-color: #03234b; + --secondary-fore-color: #03234b; + --back-color: #ffffff; + --secondary-back-color: #ffffff; + --blockquote-color: #e6007e; + --pre-color: #e6007e; + --border-color: #3cb4e6; + --secondary-border-color: #3cb4e6; + --heading-ratio: 1.2; --universal-margin: 0.5rem; - --universal-padding: 0.125rem; - --universal-border-radius: 0.125rem; - --a-link-color: #0277bd; - --a-visited-color: #01579b; } + --universal-padding: 0.25rem; + --universal-border-radius: 0.075rem; + --background-margin: 1.5%; + --a-link-color: #3cb4e6; + --a-visited-color: #8c0078; } html { - font-size: 14px; } + font-size: 13.5px; } a, b, del, em, i, ins, q, span, strong, u { font-size: 1em; } html, * { - font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Ubuntu, "Helvetica Neue", Helvetica, sans-serif; - line-height: 1.4; + font-family: -apple-system, BlinkMacSystemFont, Helvetica, arial, sans-serif; + line-height: 1.25; -webkit-text-size-adjust: 100%; } * { @@ -42,7 +42,10 @@ html, * { body { margin: 0; color: var(--fore-color); - background: var(--back-color); } + @background: var(--back-color); + background: var(--back-color) linear-gradient(#ffd200, #ffd200) repeat-y left top; + background-size: var(--background-margin); + } details { display: block; } @@ -62,9 +65,9 @@ img { height: auto; } h1, h2, h3, h4, h5, h6 { - line-height: 1.2; + line-height: 1.25; margin: calc(1.5 * var(--universal-margin)) var(--universal-margin); - font-weight: 500; } + font-weight: 400; } h1 small, h2 small, h3 small, h4 small, h5 small, h6 small { color: var(--secondary-fore-color); display: block; @@ -74,21 +77,15 @@ h1 { font-size: calc(1rem * var(--heading-ratio) * var(--heading-ratio) * var(--heading-ratio)); } h2 { - font-size: calc(1rem * var(--heading-ratio) * var(--heading-ratio); ); - background: var(--mark-back-color); - font-weight: 600; - padding: 0.1em 0.5em 0.2em 0.5em; - color: var(--mark-fore-color); } - + font-size: calc(1rem * var(--heading-ratio) * var(--heading-ratio) ); + border-style: none none solid none ; + border-width: thin; + border-color: var(--border-color); } h3 { - font-size: calc(1rem * var(--heading-ratio)); - padding-left: calc(2 * var(--universal-margin)); - /* background: var(--border-color); */ - } + font-size: calc(1rem * var(--heading-ratio) ); } h4 { - font-size: 1rem;); - padding-left: calc(4 * var(--universal-margin)); } + font-size: calc(1rem * var(--heading-ratio)); } h5 { font-size: 1rem; } @@ -101,7 +98,7 @@ p { ol, ul { margin: var(--universal-margin); - padding-left: calc(6 * var(--universal-margin)); } + padding-left: calc(3 * var(--universal-margin)); } b, strong { font-weight: 700; } @@ -111,7 +108,7 @@ hr { border: 0; line-height: 1.25em; margin: var(--universal-margin); - height: 0.0625rem; + height: 0.0714285714rem; background: linear-gradient(to right, transparent, var(--border-color) 20%, var(--border-color) 80%, transparent); } blockquote { @@ -121,16 +118,16 @@ blockquote { color: var(--secondary-fore-color); margin: var(--universal-margin); padding: calc(3 * var(--universal-padding)); - border: 0.0625rem solid var(--secondary-border-color); - border-left: 0.375rem solid var(--blockquote-color); + border: 0.0714285714rem solid var(--secondary-border-color); + border-left: 0.3rem solid var(--blockquote-color); border-radius: 0 var(--universal-border-radius) var(--universal-border-radius) 0; } blockquote:before { position: absolute; top: calc(0rem - var(--universal-padding)); left: 0; font-family: sans-serif; - font-size: 3rem; - font-weight: 700; + font-size: 2rem; + font-weight: 800; content: "\201c"; color: var(--blockquote-color); } blockquote[cite]:after { @@ -160,8 +157,8 @@ pre { background: var(--secondary-back-color); padding: calc(1.5 * var(--universal-padding)); margin: var(--universal-margin); - border: 0.0625rem solid var(--secondary-border-color); - border-left: 0.25rem solid var(--pre-color); + border: 0.0714285714rem solid var(--secondary-border-color); + border-left: 0.2857142857rem solid var(--pre-color); border-radius: 0 var(--universal-border-radius) var(--universal-border-radius) 0; } sup, sub, code, kbd { @@ -204,7 +201,8 @@ a { box-sizing: border-box; display: flex; flex: 0 1 auto; - flex-flow: row wrap; } + flex-flow: row wrap; + margin: 0 0 0 var(--background-margin); } .col-sm, [class^='col-sm-'], @@ -565,9 +563,9 @@ a { order: 999; } } /* Card component CSS variable definitions */ :root { - --card-back-color: #f8f8f8; - --card-fore-color: #111; - --card-border-color: #ddd; } + --card-back-color: #3cb4e6; + --card-fore-color: #03234b; + --card-border-color: #03234b; } .card { display: flex; @@ -578,7 +576,7 @@ a { width: 100%; background: var(--card-back-color); color: var(--card-fore-color); - border: 0.0625rem solid var(--card-border-color); + border: 0.0714285714rem solid var(--card-border-color); border-radius: var(--universal-border-radius); margin: var(--universal-margin); overflow: hidden; } @@ -592,7 +590,7 @@ a { margin: 0; border: 0; border-radius: 0; - border-bottom: 0.0625rem solid var(--card-border-color); + border-bottom: 0.0714285714rem solid var(--card-border-color); padding: var(--universal-padding); width: 100%; } .card > .sectione.media { @@ -617,17 +615,18 @@ a { width: auto; } .card.warning { -/* --card-back-color: #ffca28; */ --card-back-color: #e5b8b7; - --card-border-color: #e8b825; } + --card-fore-color: #3b234b; + --card-border-color: #8c0078; } .card.error { - --card-back-color: #b71c1c; - --card-fore-color: #f8f8f8; - --card-border-color: #a71a1a; } + --card-back-color: #464650; + --card-fore-color: #ffffff; + --card-border-color: #8c0078; } .card > .sectione.dark { - --card-back-color: #e0e0e0; } + --card-back-color: #3b234b; + --card-fore-color: #ffffff; } .card > .sectione.double-padded { padding: calc(1.5 * var(--universal-padding)); } @@ -637,12 +636,12 @@ a { */ /* Input_control module CSS variable definitions */ :root { - --form-back-color: #f0f0f0; - --form-fore-color: #111; - --form-border-color: #ddd; - --input-back-color: #f8f8f8; - --input-fore-color: #111; - --input-border-color: #ddd; + --form-back-color: #ffe97f; + --form-fore-color: #03234b; + --form-border-color: #3cb4e6; + --input-back-color: #ffffff; + --input-fore-color: #03234b; + --input-border-color: #3cb4e6; --input-focus-color: #0288d1; --input-invalid-color: #d32f2f; --button-back-color: #e2e2e2; @@ -655,13 +654,13 @@ a { form { background: var(--form-back-color); color: var(--form-fore-color); - border: 0.0625rem solid var(--form-border-color); + border: 0.0714285714rem solid var(--form-border-color); border-radius: var(--universal-border-radius); margin: var(--universal-margin); padding: calc(2 * var(--universal-padding)) var(--universal-padding); } fieldset { - border: 0.0625rem solid var(--form-border-color); + border: 0.0714285714rem solid var(--form-border-color); border-radius: var(--universal-border-radius); margin: calc(var(--universal-margin) / 4); padding: var(--universal-padding); } @@ -671,7 +670,7 @@ legend { display: table; max-width: 100%; white-space: normal; - font-weight: 700; + font-weight: 500; padding: calc(var(--universal-padding) / 2); } label { @@ -716,7 +715,7 @@ input:not([type]), [type="text"], [type="email"], [type="number"], [type="search box-sizing: border-box; background: var(--input-back-color); color: var(--input-fore-color); - border: 0.0625rem solid var(--input-border-color); + border: 0.0714285714rem solid var(--input-border-color); border-radius: var(--universal-border-radius); margin: calc(var(--universal-margin) / 2); padding: var(--universal-padding) calc(1.5 * var(--universal-padding)); } @@ -763,8 +762,8 @@ option { [type="radio"]:checked:before { border-radius: 100%; content: ''; - top: calc(0.0625rem + var(--universal-padding) / 2); - left: calc(0.0625rem + var(--universal-padding) / 2); + top: calc(0.0714285714rem + var(--universal-padding) / 2); + left: calc(0.0714285714rem + var(--universal-padding) / 2); background: var(--input-fore-color); width: 0.5rem; height: 0.5rem; } @@ -793,7 +792,7 @@ a[role="button"], label[role="button"], [role="button"] { display: inline-block; background: var(--button-back-color); color: var(--button-fore-color); - border: 0.0625rem solid var(--button-border-color); + border: 0.0714285714rem solid var(--button-border-color); border-radius: var(--universal-border-radius); padding: var(--universal-padding) calc(1.5 * var(--universal-padding)); margin: var(--universal-margin); @@ -814,7 +813,7 @@ input:disabled, input[disabled], textarea:disabled, textarea[disabled], select:d .button-group { display: flex; - border: 0.0625rem solid var(--button-group-border-color); + border: 0.0714285714rem solid var(--button-group-border-color); border-radius: var(--universal-border-radius); margin: var(--universal-margin); } .button-group > button, .button-group [type="button"], .button-group > [type="submit"], .button-group > [type="reset"], .button-group > .button, .button-group > [role="button"] { @@ -826,13 +825,13 @@ input:disabled, input[disabled], textarea:disabled, textarea[disabled], select:d border-radius: 0; box-shadow: none; } .button-group > :not(:first-child) { - border-left: 0.0625rem solid var(--button-group-border-color); } + border-left: 0.0714285714rem solid var(--button-group-border-color); } @media screen and (max-width: 499px) { .button-group { flex-direction: column; } .button-group > :not(:first-child) { border: 0; - border-top: 0.0625rem solid var(--button-group-border-color); } } + border-top: 0.0714285714rem solid var(--button-group-border-color); } } /* Custom elements for forms and input elements. @@ -874,29 +873,29 @@ button.large, [type="button"].large, [type="submit"].large, [type="reset"].large */ /* Navigation module CSS variable definitions */ :root { - --header-back-color: #f8f8f8; - --header-hover-back-color: #f0f0f0; - --header-fore-color: #444; - --header-border-color: #ddd; - --nav-back-color: #f8f8f8; - --nav-hover-back-color: #f0f0f0; - --nav-fore-color: #444; - --nav-border-color: #ddd; - --nav-link-color: #0277bd; - --footer-fore-color: #444; - --footer-back-color: #f8f8f8; - --footer-border-color: #ddd; - --footer-link-color: #0277bd; - --drawer-back-color: #f8f8f8; - --drawer-hover-back-color: #f0f0f0; - --drawer-border-color: #ddd; - --drawer-close-color: #444; } + --header-back-color: #03234b; + --header-hover-back-color: #ffd200; + --header-fore-color: #ffffff; + --header-border-color: #3cb4e6; + --nav-back-color: #ffffff; + --nav-hover-back-color: #ffe97f; + --nav-fore-color: #e6007e; + --nav-border-color: #3cb4e6; + --nav-link-color: #3cb4e6; + --footer-fore-color: #ffffff; + --footer-back-color: #03234b; + --footer-border-color: #3cb4e6; + --footer-link-color: #3cb4e6; + --drawer-back-color: #ffffff; + --drawer-hover-back-color: #ffe97f; + --drawer-border-color: #3cb4e6; + --drawer-close-color: #e6007e; } header { - height: 3.1875rem; + height: 2.75rem; background: var(--header-back-color); color: var(--header-fore-color); - border-bottom: 0.0625rem solid var(--header-border-color); + border-bottom: 0.0714285714rem solid var(--header-border-color); padding: calc(var(--universal-padding) / 4) 0; white-space: nowrap; overflow-x: auto; @@ -927,7 +926,7 @@ header { nav { background: var(--nav-back-color); color: var(--nav-fore-color); - border: 0.0625rem solid var(--nav-border-color); + border: 0.0714285714rem solid var(--nav-border-color); border-radius: var(--universal-border-radius); margin: var(--universal-margin); } nav * { @@ -946,10 +945,10 @@ nav { nav .sublink-1:before { position: absolute; left: calc(var(--universal-padding) - 1 * var(--universal-padding)); - top: -0.0625rem; + top: -0.0714285714rem; content: ''; height: 100%; - border: 0.0625rem solid var(--nav-border-color); + border: 0.0714285714rem solid var(--nav-border-color); border-left: 0; } nav .sublink-2 { position: relative; @@ -957,16 +956,16 @@ nav { nav .sublink-2:before { position: absolute; left: calc(var(--universal-padding) - 3 * var(--universal-padding)); - top: -0.0625rem; + top: -0.0714285714rem; content: ''; height: 100%; - border: 0.0625rem solid var(--nav-border-color); + border: 0.0714285714rem solid var(--nav-border-color); border-left: 0; } footer { background: var(--footer-back-color); color: var(--footer-fore-color); - border-top: 0.0625rem solid var(--footer-border-color); + border-top: 0.0714285714rem solid var(--footer-border-color); padding: calc(2 * var(--universal-padding)) var(--universal-padding); font-size: 0.875rem; } footer a, footer a:visited { @@ -1013,7 +1012,7 @@ footer.sticky { height: 100vh; overflow-y: auto; background: var(--drawer-back-color); - border: 0.0625rem solid var(--drawer-border-color); + border: 0.0714285714rem solid var(--drawer-border-color); border-radius: 0; margin: 0; z-index: 1110; @@ -1060,38 +1059,36 @@ footer.sticky { */ /* Table module CSS variable definitions. */ :root { - --table-border-color: #aaa; - --table-border-separator-color: #666; - --table-head-back-color: #e6e6e6; - --table-head-fore-color: #111; - --table-body-back-color: #f8f8f8; - --table-body-fore-color: #111; - --table-body-alt-back-color: #eee; } + --table-border-color: #03234b; + --table-border-separator-color: #03234b; + --table-head-back-color: #03234b; + --table-head-fore-color: #ffffff; + --table-body-back-color: #ffffff; + --table-body-fore-color: #03234b; + --table-body-alt-back-color: #f4f4f4; } table { border-collapse: separate; border-spacing: 0; - : margin: calc(1.5 * var(--universal-margin)) var(--universal-margin); + margin: 0; display: flex; flex: 0 1 auto; flex-flow: row wrap; padding: var(--universal-padding); - padding-top: 0; - margin: calc(1.5 * var(--universal-margin)) var(--universal-margin); } + padding-top: 0; } table caption { - font-size: 1.25 * rem; + font-size: 1rem; margin: calc(2 * var(--universal-margin)) 0; max-width: 100%; - flex: 0 0 100%; - text-align: left;} + flex: 0 0 100%; } table thead, table tbody { display: flex; flex-flow: row wrap; - border: 0.0625rem solid var(--table-border-color); } + border: 0.0714285714rem solid var(--table-border-color); } table thead { z-index: 999; border-radius: var(--universal-border-radius) var(--universal-border-radius) 0 0; - border-bottom: 0.0625rem solid var(--table-border-separator-color); } + border-bottom: 0.0714285714rem solid var(--table-border-separator-color); } table tbody { border-top: 0; margin-top: calc(0 - var(--universal-margin)); @@ -1109,11 +1106,11 @@ table { table td { background: var(--table-body-back-color); color: var(--table-body-fore-color); - border-top: 0.0625rem solid var(--table-border-color); } + border-top: 0.0714285714rem solid var(--table-border-color); } table:not(.horizontal) { overflow: auto; - max-height: 850px; } + max-height: 100%; } table:not(.horizontal) thead, table:not(.horizontal) tbody { max-width: 100%; flex: 0 0 100%; } @@ -1134,32 +1131,33 @@ table.horizontal { border: 0; } table.horizontal thead, table.horizontal tbody { border: 0; + flex: .2 0 0; flex-flow: row nowrap; } table.horizontal tbody { overflow: auto; justify-content: space-between; - flex: 1 0 0; - margin-left: calc( 4 * var(--universal-margin)); + flex: .8 0 0; + margin-left: 0; padding-bottom: calc(var(--universal-padding) / 4); } table.horizontal tr { flex-direction: column; flex: 1 0 auto; } table.horizontal th, table.horizontal td { - width: 100%; + width: auto; border: 0; - border-bottom: 0.0625rem solid var(--table-border-color); } + border-bottom: 0.0714285714rem solid var(--table-border-color); } table.horizontal th:not(:first-child), table.horizontal td:not(:first-child) { border-top: 0; } table.horizontal th { text-align: right; - border-left: 0.0625rem solid var(--table-border-color); - border-right: 0.0625rem solid var(--table-border-separator-color); } + border-left: 0.0714285714rem solid var(--table-border-color); + border-right: 0.0714285714rem solid var(--table-border-separator-color); } table.horizontal thead tr:first-child { padding-left: 0; } table.horizontal th:first-child, table.horizontal td:first-child { - border-top: 0.0625rem solid var(--table-border-color); } + border-top: 0.0714285714rem solid var(--table-border-color); } table.horizontal tbody tr:last-child td { - border-right: 0.0625rem solid var(--table-border-color); } + border-right: 0.0714285714rem solid var(--table-border-color); } table.horizontal tbody tr:last-child td:first-child { border-top-right-radius: 0.25rem; } table.horizontal tbody tr:last-child td:last-child { @@ -1191,12 +1189,12 @@ table.horizontal { display: table-row-group; } table tr, table.horizontal tr { display: block; - border: 0.0625rem solid var(--table-border-color); + border: 0.0714285714rem solid var(--table-border-color); border-radius: var(--universal-border-radius); - background: #fafafa; + background: #ffffff; padding: var(--universal-padding); margin: var(--universal-margin); - margin-bottom: calc(2 * var(--universal-margin)); } + margin-bottom: calc(1 * var(--universal-margin)); } table th, table td, table.horizontal th, table.horizontal td { width: auto; } table td, table.horizontal td { @@ -1211,9 +1209,6 @@ table.horizontal { border-top: 0; } table tbody tr:last-child td, table.horizontal tbody tr:last-child td { border-right: 0; } } -:root { - --table-body-alt-back-color: #eee; } - table tr:nth-of-type(2n) > td { background: var(--table-body-alt-back-color); } @@ -1234,8 +1229,8 @@ table.hoverable tr:hover, table.hoverable tr:hover > td, table.hoverable tr:focu */ /* Contextual module CSS variable definitions */ :root { - --mark-back-color: #0277bd; - --mark-fore-color: #fafafa; } + --mark-back-color: #3cb4e6; + --mark-fore-color: #ffffff; } mark { background: var(--mark-back-color); @@ -1243,11 +1238,11 @@ mark { font-size: 0.95em; line-height: 1em; border-radius: var(--universal-border-radius); - padding: calc(var(--universal-padding) / 4) calc(var(--universal-padding) / 2); } + padding: calc(var(--universal-padding) / 4) var(--universal-padding); } mark.inline-block { display: inline-block; font-size: 1em; - line-height: 1.5; + line-height: 1.4; padding: calc(var(--universal-padding) / 2) var(--universal-padding); } :root { @@ -1314,8 +1309,8 @@ mark { :root { --modal-overlay-color: rgba(0, 0, 0, 0.45); - --modal-close-color: #444; - --modal-close-hover-color: #f0f0f0; } + --modal-close-color: #e6007e; + --modal-close-hover-color: #ffe97f; } [type="checkbox"].modal { height: 1px; @@ -1368,13 +1363,14 @@ mark { z-index: 1211; } :root { - --collapse-label-back-color: #e8e8e8; - --collapse-label-fore-color: #212121; - --collapse-label-hover-back-color: #f0f0f0; - --collapse-selected-label-back-color: #ececec; - --collapse-border-color: #ddd; - --collapse-content-back-color: #fafafa; - --collapse-selected-label-border-color: #0277bd; } + --collapse-label-back-color: #03234b; + --collapse-label-fore-color: #ffffff; + --collapse-label-hover-back-color: #3cb4e6; + --collapse-selected-label-back-color: #3cb4e6; + --collapse-border-color: var(--collapse-label-back-color); + --collapse-selected-border-color: #ceecf8; + --collapse-content-back-color: #ffffff; + --collapse-selected-label-border-color: #3cb4e6; } .collapse { width: calc(100% - 2 * var(--universal-margin)); @@ -1395,13 +1391,13 @@ mark { .collapse > label { flex-grow: 1; display: inline-block; - height: 1.5rem; + height: 1.25rem; cursor: pointer; - transition: background 0.3s; + transition: background 0.2s; color: var(--collapse-label-fore-color); background: var(--collapse-label-back-color); - border: 0.0625rem solid var(--collapse-border-color); - padding: calc(1.5 * var(--universal-padding)); } + border: 0.0714285714rem solid var(--collapse-selected-border-color); + padding: calc(1.25 * var(--universal-padding)); } .collapse > label:hover, .collapse > label:focus { background: var(--collapse-label-hover-back-color); } .collapse > label + div { @@ -1418,7 +1414,7 @@ mark { max-height: 1px; } .collapse > :checked + label { background: var(--collapse-selected-label-back-color); - border-bottom-color: var(--collapse-selected-label-border-color); } + border-color: var(--collapse-selected-label-border-color); } .collapse > :checked + label + div { box-sizing: border-box; position: relative; @@ -1427,13 +1423,13 @@ mark { overflow: auto; margin: 0; background: var(--collapse-content-back-color); - border: 0.0625rem solid var(--collapse-border-color); + border: 0.0714285714rem solid var(--collapse-selected-border-color); border-top: 0; padding: var(--universal-padding); clip: auto; -webkit-clip-path: inset(0%); clip-path: inset(0%); - max-height: 850px; } + max-height: 100%; } .collapse > label:not(:first-of-type) { border-top: 0; } .collapse > label:first-of-type { @@ -1450,11 +1446,8 @@ mark { /* Custom elements for contextual background elements, toasts and tooltips. */ -mark.secondary { - --mark-back-color: #d32f2f; } - mark.tertiary { - --mark-back-color: #308732; } + --mark-back-color: #3cb4e6; } mark.tag { padding: calc(var(--universal-padding)/2) var(--universal-padding); @@ -1463,9 +1456,9 @@ mark.tag { /* Definitions for progress elements and spinners. */ -/* Progess module CSS variable definitions */ +/* Progress module CSS variable definitions */ :root { - --progress-back-color: #ddd; + --progress-back-color: #3cb4e6; --progress-fore-color: #555; } progress { @@ -1558,45 +1551,53 @@ span[class^='icon-'] { filter: invert(100%); } span.icon-alert { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='8' x2='12' y2='12'%3E%3C/line%3E%3Cline x1='12' y1='16' x2='12' y2='16'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='8' x2='12' y2='12'%3E%3C/line%3E%3Cline x1='12' y1='16' x2='12' y2='16'%3E%3C/line%3E%3C/svg%3E"); } span.icon-bookmark { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M19 21l-7-5-7 5V5a2 2 0 0 1 2-2h10a2 2 0 0 1 2 2z'%3E%3C/path%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M19 21l-7-5-7 5V5a2 2 0 0 1 2-2h10a2 2 0 0 1 2 2z'%3E%3C/path%3E%3C/svg%3E"); } span.icon-calendar { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='3' y='4' width='18' height='18' rx='2' ry='2'%3E%3C/rect%3E%3Cline x1='16' y1='2' x2='16' y2='6'%3E%3C/line%3E%3Cline x1='8' y1='2' x2='8' y2='6'%3E%3C/line%3E%3Cline x1='3' y1='10' x2='21' y2='10'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='3' y='4' width='18' height='18' rx='2' ry='2'%3E%3C/rect%3E%3Cline x1='16' y1='2' x2='16' y2='6'%3E%3C/line%3E%3Cline x1='8' y1='2' x2='8' y2='6'%3E%3C/line%3E%3Cline x1='3' y1='10' x2='21' y2='10'%3E%3C/line%3E%3C/svg%3E"); } span.icon-credit { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='1' y='4' width='22' height='16' rx='2' ry='2'%3E%3C/rect%3E%3Cline x1='1' y1='10' x2='23' y2='10'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='1' y='4' width='22' height='16' rx='2' ry='2'%3E%3C/rect%3E%3Cline x1='1' y1='10' x2='23' y2='10'%3E%3C/line%3E%3C/svg%3E"); } span.icon-edit { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M20 14.66V20a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V6a2 2 0 0 1 2-2h5.34'%3E%3C/path%3E%3Cpolygon points='18 2 22 6 12 16 8 16 8 12 18 2'%3E%3C/polygon%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M20 14.66V20a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V6a2 2 0 0 1 2-2h5.34'%3E%3C/path%3E%3Cpolygon points='18 2 22 6 12 16 8 16 8 12 18 2'%3E%3C/polygon%3E%3C/svg%3E"); } span.icon-link { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6'%3E%3C/path%3E%3Cpolyline points='15 3 21 3 21 9'%3E%3C/polyline%3E%3Cline x1='10' y1='14' x2='21' y2='3'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6'%3E%3C/path%3E%3Cpolyline points='15 3 21 3 21 9'%3E%3C/polyline%3E%3Cline x1='10' y1='14' x2='21' y2='3'%3E%3C/line%3E%3C/svg%3E"); } span.icon-help { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M9.09 9a3 3 0 0 1 5.83 1c0 2-3 3-3 3'%3E%3C/path%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='17' x2='12' y2='17'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M9.09 9a3 3 0 0 1 5.83 1c0 2-3 3-3 3'%3E%3C/path%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='17' x2='12' y2='17'%3E%3C/line%3E%3C/svg%3E"); } span.icon-home { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M3 9l9-7 9 7v11a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2z'%3E%3C/path%3E%3Cpolyline points='9 22 9 12 15 12 15 22'%3E%3C/polyline%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M3 9l9-7 9 7v11a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2z'%3E%3C/path%3E%3Cpolyline points='9 22 9 12 15 12 15 22'%3E%3C/polyline%3E%3C/svg%3E"); } span.icon-info { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='16' x2='12' y2='12'%3E%3C/line%3E%3Cline x1='12' y1='8' x2='12' y2='8'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='16' x2='12' y2='12'%3E%3C/line%3E%3Cline x1='12' y1='8' x2='12' y2='8'%3E%3C/line%3E%3C/svg%3E"); } span.icon-lock { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='3' y='11' width='18' height='11' rx='2' ry='2'%3E%3C/rect%3E%3Cpath d='M7 11V7a5 5 0 0 1 10 0v4'%3E%3C/path%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='3' y='11' width='18' height='11' rx='2' ry='2'%3E%3C/rect%3E%3Cpath d='M7 11V7a5 5 0 0 1 10 0v4'%3E%3C/path%3E%3C/svg%3E"); } span.icon-mail { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M4 4h16c1.1 0 2 .9 2 2v12c0 1.1-.9 2-2 2H4c-1.1 0-2-.9-2-2V6c0-1.1.9-2 2-2z'%3E%3C/path%3E%3Cpolyline points='22,6 12,13 2,6'%3E%3C/polyline%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M4 4h16c1.1 0 2 .9 2 2v12c0 1.1-.9 2-2 2H4c-1.1 0-2-.9-2-2V6c0-1.1.9-2 2-2z'%3E%3C/path%3E%3Cpolyline points='22,6 12,13 2,6'%3E%3C/polyline%3E%3C/svg%3E"); } span.icon-location { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M21 10c0 7-9 13-9 13s-9-6-9-13a9 9 0 0 1 18 0z'%3E%3C/path%3E%3Ccircle cx='12' cy='10' r='3'%3E%3C/circle%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M21 10c0 7-9 13-9 13s-9-6-9-13a9 9 0 0 1 18 0z'%3E%3C/path%3E%3Ccircle cx='12' cy='10' r='3'%3E%3C/circle%3E%3C/svg%3E"); } span.icon-phone { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M22 16.92v3a2 2 0 0 1-2.18 2 19.79 19.79 0 0 1-8.63-3.07 19.5 19.5 0 0 1-6-6 19.79 19.79 0 0 1-3.07-8.67A2 2 0 0 1 4.11 2h3a2 2 0 0 1 2 1.72 12.84 12.84 0 0 0 .7 2.81 2 2 0 0 1-.45 2.11L8.09 9.91a16 16 0 0 0 6 6l1.27-1.27a2 2 0 0 1 2.11-.45 12.84 12.84 0 0 0 2.81.7A2 2 0 0 1 22 16.92z'%3E%3C/path%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M22 16.92v3a2 2 0 0 1-2.18 2 19.79 19.79 0 0 1-8.63-3.07 19.5 19.5 0 0 1-6-6 19.79 19.79 0 0 1-3.07-8.67A2 2 0 0 1 4.11 2h3a2 2 0 0 1 2 1.72 12.84 12.84 0 0 0 .7 2.81 2 2 0 0 1-.45 2.11L8.09 9.91a16 16 0 0 0 6 6l1.27-1.27a2 2 0 0 1 2.11-.45 12.84 12.84 0 0 0 2.81.7A2 2 0 0 1 22 16.92z'%3E%3C/path%3E%3C/svg%3E"); } span.icon-rss { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M4 11a9 9 0 0 1 9 9'%3E%3C/path%3E%3Cpath d='M4 4a16 16 0 0 1 16 16'%3E%3C/path%3E%3Ccircle cx='5' cy='19' r='1'%3E%3C/circle%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M4 11a9 9 0 0 1 9 9'%3E%3C/path%3E%3Cpath d='M4 4a16 16 0 0 1 16 16'%3E%3C/path%3E%3Ccircle cx='5' cy='19' r='1'%3E%3C/circle%3E%3C/svg%3E"); } span.icon-search { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='11' cy='11' r='8'%3E%3C/circle%3E%3Cline x1='21' y1='21' x2='16.65' y2='16.65'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='11' cy='11' r='8'%3E%3C/circle%3E%3Cline x1='21' y1='21' x2='16.65' y2='16.65'%3E%3C/line%3E%3C/svg%3E"); } span.icon-settings { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='3'%3E%3C/circle%3E%3Cpath d='M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z'%3E%3C/path%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='3'%3E%3C/circle%3E%3Cpath d='M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z'%3E%3C/path%3E%3C/svg%3E"); } span.icon-share { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='18' cy='5' r='3'%3E%3C/circle%3E%3Ccircle cx='6' cy='12' r='3'%3E%3C/circle%3E%3Ccircle cx='18' cy='19' r='3'%3E%3C/circle%3E%3Cline x1='8.59' y1='13.51' x2='15.42' y2='17.49'%3E%3C/line%3E%3Cline x1='15.41' y1='6.51' x2='8.59' y2='10.49'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='18' cy='5' r='3'%3E%3C/circle%3E%3Ccircle cx='6' cy='12' r='3'%3E%3C/circle%3E%3Ccircle cx='18' cy='19' r='3'%3E%3C/circle%3E%3Cline x1='8.59' y1='13.51' x2='15.42' y2='17.49'%3E%3C/line%3E%3Cline x1='15.41' y1='6.51' x2='8.59' y2='10.49'%3E%3C/line%3E%3C/svg%3E"); } span.icon-cart { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='9' cy='21' r='1'%3E%3C/circle%3E%3Ccircle cx='20' cy='21' r='1'%3E%3C/circle%3E%3Cpath d='M1 1h4l2.68 13.39a2 2 0 0 0 2 1.61h9.72a2 2 0 0 0 2-1.61L23 6H6'%3E%3C/path%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='9' cy='21' r='1'%3E%3C/circle%3E%3Ccircle cx='20' cy='21' r='1'%3E%3C/circle%3E%3Cpath d='M1 1h4l2.68 13.39a2 2 0 0 0 2 1.61h9.72a2 2 0 0 0 2-1.61L23 6H6'%3E%3C/path%3E%3C/svg%3E"); } span.icon-upload { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4'%3E%3C/path%3E%3Cpolyline points='17 8 12 3 7 8'%3E%3C/polyline%3E%3Cline x1='12' y1='3' x2='12' y2='15'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4'%3E%3C/path%3E%3Cpolyline points='17 8 12 3 7 8'%3E%3C/polyline%3E%3Cline x1='12' y1='3' x2='12' y2='15'%3E%3C/line%3E%3C/svg%3E"); } span.icon-user { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M20 21v-2a4 4 0 0 0-4-4H8a4 4 0 0 0-4 4v2'%3E%3C/path%3E%3Ccircle cx='12' cy='7' r='4'%3E%3C/circle%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M20 21v-2a4 4 0 0 0-4-4H8a4 4 0 0 0-4 4v2'%3E%3C/path%3E%3Ccircle cx='12' cy='7' r='4'%3E%3C/circle%3E%3C/svg%3E"); } + +/* + Definitions for STMicroelectronics icons (https://brandportal.st.com/document/26). +*/ +span.icon-st-update { + background-image: url("Update.svg"); } +span.icon-st-add { + background-image: url("Add button.svg"); } /* Definitions for utilities and helper classes. @@ -1604,7 +1605,7 @@ span.icon-user { /* Utility module CSS variable definitions */ :root { --generic-border-color: rgba(0, 0, 0, 0.3); - --generic-box-shadow: 0 0.25rem 0.25rem 0 rgba(0, 0, 0, 0.125), 0 0.125rem 0.125rem -0.125rem rgba(0, 0, 0, 0.25); } + --generic-box-shadow: 0 0.2857142857rem 0.2857142857rem 0 rgba(0, 0, 0, 0.125), 0 0.1428571429rem 0.1428571429rem -0.1428571429rem rgba(0, 0, 0, 0.125); } .hidden { display: none !important; } @@ -1622,7 +1623,7 @@ span.icon-user { overflow: hidden !important; } .bordered { - border: 0.0625rem solid var(--generic-border-color) !important; } + border: 0.0714285714rem solid var(--generic-border-color) !important; } .rounded { border-radius: var(--universal-border-radius) !important; } @@ -1697,4 +1698,14 @@ span.icon-user { clip-path: inset(100%) !important; overflow: hidden !important; } } -/*# sourceMappingURL=mini-default.css.map */ +/*# sourceMappingURL=mini-custom.css.map */ + +img[alt="ST logo"] { display: block; margin: auto; width: 75%; max-width: 250px; min-width: 71px; } +img[alt="Cube logo"] { float: right; width: 30%; max-width: 10rem; min-width: 8rem; padding-right: 1rem;} + +.figure { + display: block; + margin-left: auto; + margin-right: auto; + text-align: center; +} \ No newline at end of file diff --git a/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/_htmresc/st_logo.png b/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/_htmresc/st_logo.png deleted file mode 100644 index 8b80057fd3a454a97de1c9d732b7fede82c83227..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 18616 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 diff --git a/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/_htmresc/st_logo_2020.png b/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/_htmresc/st_logo_2020.png new file mode 100644 index 0000000000000000000000000000000000000000..d6cebb5ac70e0594cbe37a6b60036a80ef3bed37 GIT binary patch literal 7520 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"NJQ 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 b^}%0^qYmp2WD(pA|JtZ4>nPPKybJpu3@X?! literal 0 HcmV?d00001 diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/Legacy/stm32_hal_legacy.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/Legacy/stm32_hal_legacy.h index 934f1f97..aa00ff4d 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/Legacy/stm32_hal_legacy.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/Legacy/stm32_hal_legacy.h @@ -37,16 +37,12 @@ extern "C" { #define AES_CLEARFLAG_CCF CRYP_CLEARFLAG_CCF #define AES_CLEARFLAG_RDERR CRYP_CLEARFLAG_RDERR #define AES_CLEARFLAG_WRERR CRYP_CLEARFLAG_WRERR -#if defined(STM32U5) || defined(STM32H7) || defined(STM32MP1) +#if defined(STM32H7) || defined(STM32MP1) #define CRYP_DATATYPE_32B CRYP_NO_SWAP #define CRYP_DATATYPE_16B CRYP_HALFWORD_SWAP #define CRYP_DATATYPE_8B CRYP_BYTE_SWAP #define CRYP_DATATYPE_1B CRYP_BIT_SWAP -#if defined(STM32U5) -#define CRYP_CCF_CLEAR CRYP_CLEAR_CCF -#define CRYP_ERR_CLEAR CRYP_CLEAR_RWEIF -#endif /* STM32U5 */ -#endif /* STM32U5 || STM32H7 || STM32MP1 */ +#endif /* STM32H7 || STM32MP1 */ /** * @} */ @@ -113,6 +109,9 @@ extern "C" { #define ADC4_SAMPLETIME_160CYCLES_5 ADC4_SAMPLETIME_814CYCLES_5 #endif /* STM32U5 */ +#if defined(STM32H5) +#define ADC_CHANNEL_VCORE ADC_CHANNEL_VDDCORE +#endif /* STM32H5 */ /** * @} */ @@ -140,7 +139,8 @@ extern "C" { #define COMP_EXTI_LINE_COMP6_EVENT COMP_EXTI_LINE_COMP6 #define COMP_EXTI_LINE_COMP7_EVENT COMP_EXTI_LINE_COMP7 #if defined(STM32L0) -#define COMP_LPTIMCONNECTION_ENABLED ((uint32_t)0x00000003U) /*!< COMPX output generic naming: connected to LPTIM input 1 for COMP1, LPTIM input 2 for COMP2 */ +#define COMP_LPTIMCONNECTION_ENABLED ((uint32_t)0x00000003U) /*!< COMPX output generic naming: connected to LPTIM + input 1 for COMP1, LPTIM input 2 for COMP2 */ #endif #define COMP_OUTPUT_COMP6TIM2OCREFCLR COMP_OUTPUT_COMP6_TIM2OCREFCLR #if defined(STM32F373xC) || defined(STM32F378xx) @@ -214,6 +214,11 @@ extern "C" { #endif #endif + +#if defined(STM32U5) +#define __HAL_COMP_COMP1_EXTI_CLEAR_RASING_FLAG __HAL_COMP_COMP1_EXTI_CLEAR_RISING_FLAG +#endif + /** * @} */ @@ -234,10 +239,12 @@ extern "C" { /** @defgroup CRC_Aliases CRC API aliases * @{ */ -#if defined(STM32C0) +#if defined(STM32H5) || defined(STM32C0) #else -#define HAL_CRC_Input_Data_Reverse HAL_CRCEx_Input_Data_Reverse /*!< Aliased to HAL_CRCEx_Input_Data_Reverse for inter STM32 series compatibility */ -#define HAL_CRC_Output_Data_Reverse HAL_CRCEx_Output_Data_Reverse /*!< Aliased to HAL_CRCEx_Output_Data_Reverse for inter STM32 series compatibility */ +#define HAL_CRC_Input_Data_Reverse HAL_CRCEx_Input_Data_Reverse /*!< Aliased to HAL_CRCEx_Input_Data_Reverse for + inter STM32 series compatibility */ +#define HAL_CRC_Output_Data_Reverse HAL_CRCEx_Output_Data_Reverse /*!< Aliased to HAL_CRCEx_Output_Data_Reverse for + inter STM32 series compatibility */ #endif /** * @} @@ -280,7 +287,13 @@ extern "C" { #define DAC_TRIGGER_LPTIM3_OUT DAC_TRIGGER_LPTIM3_CH1 #endif -#if defined(STM32L1) || defined(STM32L4) || defined(STM32G0) || defined(STM32L5) || defined(STM32H7) || defined(STM32F4) || defined(STM32G4) +#if defined(STM32H5) +#define DAC_TRIGGER_LPTIM1_OUT DAC_TRIGGER_LPTIM1_CH1 +#define DAC_TRIGGER_LPTIM2_OUT DAC_TRIGGER_LPTIM2_CH1 +#endif + +#if defined(STM32L1) || defined(STM32L4) || defined(STM32G0) || defined(STM32L5) || defined(STM32H7) || \ + defined(STM32F4) || defined(STM32G4) #define HAL_DAC_MSP_INIT_CB_ID HAL_DAC_MSPINIT_CB_ID #define HAL_DAC_MSP_DEINIT_CB_ID HAL_DAC_MSPDEINIT_CB_ID #endif @@ -345,7 +358,8 @@ extern "C" { #define HAL_DMAMUX_REQUEST_GEN_FALLING HAL_DMAMUX_REQ_GEN_FALLING #define HAL_DMAMUX_REQUEST_GEN_RISING_FALLING HAL_DMAMUX_REQ_GEN_RISING_FALLING -#if defined(STM32L4R5xx) || defined(STM32L4R9xx) || defined(STM32L4R9xx) || defined(STM32L4S5xx) || defined(STM32L4S7xx) || defined(STM32L4S9xx) +#if defined(STM32L4R5xx) || defined(STM32L4R9xx) || defined(STM32L4R9xx) || defined(STM32L4S5xx) || \ + defined(STM32L4S7xx) || defined(STM32L4S9xx) #define DMA_REQUEST_DCMI_PSSI DMA_REQUEST_DCMI #endif @@ -530,6 +544,9 @@ extern "C" { #define OB_USER_nBOOT0 OB_USER_NBOOT0 #define OB_nBOOT0_RESET OB_NBOOT0_RESET #define OB_nBOOT0_SET OB_NBOOT0_SET +#define OB_USER_SRAM134_RST OB_USER_SRAM_RST +#define OB_SRAM134_RST_ERASE OB_SRAM_RST_ERASE +#define OB_SRAM134_RST_NOT_ERASE OB_SRAM_RST_NOT_ERASE #endif /* STM32U5 */ /** @@ -574,6 +591,106 @@ extern "C" { #define HAL_SYSCFG_DisableIOAnalogSwitchVDD HAL_SYSCFG_DisableIOSwitchVDD #endif /* STM32G4 */ +#if defined(STM32H5) +#define SYSCFG_IT_FPU_IOC SBS_IT_FPU_IOC +#define SYSCFG_IT_FPU_DZC SBS_IT_FPU_DZC +#define SYSCFG_IT_FPU_UFC SBS_IT_FPU_UFC +#define SYSCFG_IT_FPU_OFC SBS_IT_FPU_OFC +#define SYSCFG_IT_FPU_IDC SBS_IT_FPU_IDC +#define SYSCFG_IT_FPU_IXC SBS_IT_FPU_IXC + +#define SYSCFG_BREAK_FLASH_ECC SBS_BREAK_FLASH_ECC +#define SYSCFG_BREAK_PVD SBS_BREAK_PVD +#define SYSCFG_BREAK_SRAM_ECC SBS_BREAK_SRAM_ECC +#define SYSCFG_BREAK_LOCKUP SBS_BREAK_LOCKUP + +#define SYSCFG_VREFBUF_VOLTAGE_SCALE0 VREFBUF_VOLTAGE_SCALE0 +#define SYSCFG_VREFBUF_VOLTAGE_SCALE1 VREFBUF_VOLTAGE_SCALE1 +#define SYSCFG_VREFBUF_VOLTAGE_SCALE2 VREFBUF_VOLTAGE_SCALE2 +#define SYSCFG_VREFBUF_VOLTAGE_SCALE3 VREFBUF_VOLTAGE_SCALE3 + +#define SYSCFG_VREFBUF_HIGH_IMPEDANCE_DISABLE VREFBUF_HIGH_IMPEDANCE_DISABLE +#define SYSCFG_VREFBUF_HIGH_IMPEDANCE_ENABLE VREFBUF_HIGH_IMPEDANCE_ENABLE + +#define SYSCFG_FASTMODEPLUS_PB6 SBS_FASTMODEPLUS_PB6 +#define SYSCFG_FASTMODEPLUS_PB7 SBS_FASTMODEPLUS_PB7 +#define SYSCFG_FASTMODEPLUS_PB8 SBS_FASTMODEPLUS_PB8 +#define SYSCFG_FASTMODEPLUS_PB9 SBS_FASTMODEPLUS_PB9 + +#define SYSCFG_ETH_MII SBS_ETH_MII +#define SYSCFG_ETH_RMII SBS_ETH_RMII +#define IS_SYSCFG_ETHERNET_CONFIG IS_SBS_ETHERNET_CONFIG + +#define SYSCFG_MEMORIES_ERASE_FLAG_IPMEE SBS_MEMORIES_ERASE_FLAG_IPMEE +#define SYSCFG_MEMORIES_ERASE_FLAG_MCLR SBS_MEMORIES_ERASE_FLAG_MCLR +#define IS_SYSCFG_MEMORIES_ERASE_FLAG IS_SBS_MEMORIES_ERASE_FLAG + +#define IS_SYSCFG_CODE_CONFIG IS_SBS_CODE_CONFIG + +#define SYSCFG_MPU_NSEC SBS_MPU_NSEC +#define SYSCFG_VTOR_NSEC SBS_VTOR_NSEC +#if defined (__ARM_FEATURE_CMSE) && (__ARM_FEATURE_CMSE == 3U) +#define SYSCFG_SAU SBS_SAU +#define SYSCFG_MPU_SEC SBS_MPU_SEC +#define SYSCFG_VTOR_AIRCR_SEC SBS_VTOR_AIRCR_SEC +#define SYSCFG_LOCK_ALL SBS_LOCK_ALL +#else +#define SYSCFG_LOCK_ALL SBS_LOCK_ALL +#endif /* __ARM_FEATURE_CMSE */ + +#define SYSCFG_CLK SBS_CLK +#define SYSCFG_CLASSB SBS_CLASSB +#define SYSCFG_FPU SBS_FPU +#define SYSCFG_ALL SBS_ALL + +#define SYSCFG_SEC SBS_SEC +#define SYSCFG_NSEC SBS_NSEC + +#define __HAL_SYSCFG_FPU_INTERRUPT_ENABLE __HAL_SBS_FPU_INTERRUPT_ENABLE +#define __HAL_SYSCFG_FPU_INTERRUPT_DISABLE __HAL_SBS_FPU_INTERRUPT_DISABLE + +#define __HAL_SYSCFG_BREAK_ECC_LOCK __HAL_SBS_BREAK_ECC_LOCK +#define __HAL_SYSCFG_BREAK_LOCKUP_LOCK __HAL_SBS_BREAK_LOCKUP_LOCK +#define __HAL_SYSCFG_BREAK_PVD_LOCK __HAL_SBS_BREAK_PVD_LOCK +#define __HAL_SYSCFG_BREAK_SRAM_ECC_LOCK __HAL_SBS_BREAK_SRAM_ECC_LOCK + +#define __HAL_SYSCFG_FASTMODEPLUS_ENABLE __HAL_SBS_FASTMODEPLUS_ENABLE +#define __HAL_SYSCFG_FASTMODEPLUS_DISABLE __HAL_SBS_FASTMODEPLUS_DISABLE + +#define __HAL_SYSCFG_GET_MEMORIES_ERASE_STATUS __HAL_SBS_GET_MEMORIES_ERASE_STATUS +#define __HAL_SYSCFG_CLEAR_MEMORIES_ERASE_STATUS __HAL_SBS_CLEAR_MEMORIES_ERASE_STATUS + +#define IS_SYSCFG_FPU_INTERRUPT IS_SBS_FPU_INTERRUPT +#define IS_SYSCFG_BREAK_CONFIG IS_SBS_BREAK_CONFIG +#define IS_SYSCFG_VREFBUF_VOLTAGE_SCALE IS_VREFBUF_VOLTAGE_SCALE +#define IS_SYSCFG_VREFBUF_HIGH_IMPEDANCE IS_VREFBUF_HIGH_IMPEDANCE +#define IS_SYSCFG_VREFBUF_TRIMMING IS_VREFBUF_TRIMMING +#define IS_SYSCFG_FASTMODEPLUS IS_SBS_FASTMODEPLUS +#define IS_SYSCFG_ITEMS_ATTRIBUTES IS_SBS_ITEMS_ATTRIBUTES +#define IS_SYSCFG_ATTRIBUTES IS_SBS_ATTRIBUTES +#define IS_SYSCFG_LOCK_ITEMS IS_SBS_LOCK_ITEMS + +#define HAL_SYSCFG_VREFBUF_VoltageScalingConfig HAL_VREFBUF_VoltageScalingConfig +#define HAL_SYSCFG_VREFBUF_HighImpedanceConfig HAL_VREFBUF_HighImpedanceConfig +#define HAL_SYSCFG_VREFBUF_TrimmingConfig HAL_VREFBUF_TrimmingConfig +#define HAL_SYSCFG_EnableVREFBUF HAL_EnableVREFBUF +#define HAL_SYSCFG_DisableVREFBUF HAL_DisableVREFBUF + +#define HAL_SYSCFG_EnableIOAnalogSwitchBooster HAL_SBS_EnableIOAnalogSwitchBooster +#define HAL_SYSCFG_DisableIOAnalogSwitchBooster HAL_SBS_DisableIOAnalogSwitchBooster +#define HAL_SYSCFG_ETHInterfaceSelect HAL_SBS_ETHInterfaceSelect + +#define HAL_SYSCFG_Lock HAL_SBS_Lock +#define HAL_SYSCFG_GetLock HAL_SBS_GetLock + +#if defined (__ARM_FEATURE_CMSE) && (__ARM_FEATURE_CMSE == 3U) +#define HAL_SYSCFG_ConfigAttributes HAL_SBS_ConfigAttributes +#define HAL_SYSCFG_GetConfigAttributes HAL_SBS_GetConfigAttributes +#endif /* __ARM_FEATURE_CMSE */ + +#endif /* STM32H5 */ + + /** * @} */ @@ -641,14 +758,16 @@ extern "C" { #define GPIO_AF10_OTG2_HS GPIO_AF10_OTG2_FS #define GPIO_AF10_OTG1_FS GPIO_AF10_OTG1_HS #define GPIO_AF12_OTG2_FS GPIO_AF12_OTG1_FS -#endif /*STM32H743xx || STM32H753xx || STM32H750xx || STM32H742xx || STM32H745xx || STM32H755xx || STM32H747xx || STM32H757xx */ +#endif /*STM32H743xx || STM32H753xx || STM32H750xx || STM32H742xx || STM32H745xx || STM32H755xx || STM32H747xx || \ + STM32H757xx */ #endif /* STM32H7 */ #define GPIO_AF0_LPTIM GPIO_AF0_LPTIM1 #define GPIO_AF1_LPTIM GPIO_AF1_LPTIM1 #define GPIO_AF2_LPTIM GPIO_AF2_LPTIM1 -#if defined(STM32L0) || defined(STM32L4) || defined(STM32F4) || defined(STM32F2) || defined(STM32F7) || defined(STM32G4) || defined(STM32H7) || defined(STM32WB) || defined(STM32U5) +#if defined(STM32L0) || defined(STM32L4) || defined(STM32F4) || defined(STM32F2) || defined(STM32F7) || \ + defined(STM32G4) || defined(STM32H7) || defined(STM32WB) || defined(STM32U5) #define GPIO_SPEED_LOW GPIO_SPEED_FREQ_LOW #define GPIO_SPEED_MEDIUM GPIO_SPEED_FREQ_MEDIUM #define GPIO_SPEED_FAST GPIO_SPEED_FREQ_HIGH @@ -670,9 +789,9 @@ extern "C" { #define GPIO_AF6_DFSDM GPIO_AF6_DFSDM1 -#if defined(STM32U5) +#if defined(STM32U5) || defined(STM32H5) #define GPIO_AF0_RTC_50Hz GPIO_AF0_RTC_50HZ -#endif /* STM32U5 */ +#endif /* STM32U5 || STM32H5 */ #if defined(STM32U5) #define GPIO_AF0_S2DSTOP GPIO_AF0_SRDSTOP #define GPIO_AF11_LPGPIO GPIO_AF11_LPGPIO1 @@ -686,7 +805,25 @@ extern "C" { */ #if defined(STM32U5) #define GTZC_PERIPH_DCMI GTZC_PERIPH_DCMI_PSSI +#define GTZC_PERIPH_LTDC GTZC_PERIPH_LTDCUSB #endif /* STM32U5 */ +#if defined(STM32H5) +#define GTZC_PERIPH_DAC12 GTZC_PERIPH_DAC1 +#define GTZC_PERIPH_ADC12 GTZC_PERIPH_ADC +#define GTZC_PERIPH_USBFS GTZC_PERIPH_USB +#endif /* STM32H5 */ +#if defined(STM32H5) || defined(STM32U5) +#define GTZC_MCPBB_NB_VCTR_REG_MAX GTZC_MPCBB_NB_VCTR_REG_MAX +#define GTZC_MCPBB_NB_LCK_VCTR_REG_MAX GTZC_MPCBB_NB_LCK_VCTR_REG_MAX +#define GTZC_MCPBB_SUPERBLOCK_UNLOCKED GTZC_MPCBB_SUPERBLOCK_UNLOCKED +#define GTZC_MCPBB_SUPERBLOCK_LOCKED GTZC_MPCBB_SUPERBLOCK_LOCKED +#define GTZC_MCPBB_BLOCK_NSEC GTZC_MPCBB_BLOCK_NSEC +#define GTZC_MCPBB_BLOCK_SEC GTZC_MPCBB_BLOCK_SEC +#define GTZC_MCPBB_BLOCK_NPRIV GTZC_MPCBB_BLOCK_NPRIV +#define GTZC_MCPBB_BLOCK_PRIV GTZC_MPCBB_BLOCK_PRIV +#define GTZC_MCPBB_LOCK_OFF GTZC_MPCBB_LOCK_OFF +#define GTZC_MCPBB_LOCK_ON GTZC_MPCBB_LOCK_ON +#endif /* STM32H5 || STM32U5 */ /** * @} */ @@ -867,7 +1004,8 @@ extern "C" { #define I2C_NOSTRETCH_ENABLED I2C_NOSTRETCH_ENABLE #define I2C_ANALOGFILTER_ENABLED I2C_ANALOGFILTER_ENABLE #define I2C_ANALOGFILTER_DISABLED I2C_ANALOGFILTER_DISABLE -#if defined(STM32F0) || defined(STM32F1) || defined(STM32F3) || defined(STM32G0) || defined(STM32L4) || defined(STM32L1) || defined(STM32F7) +#if defined(STM32F0) || defined(STM32F1) || defined(STM32F3) || defined(STM32G0) || defined(STM32L4) || \ + defined(STM32L1) || defined(STM32F7) #define HAL_I2C_STATE_MEM_BUSY_TX HAL_I2C_STATE_BUSY_TX #define HAL_I2C_STATE_MEM_BUSY_RX HAL_I2C_STATE_BUSY_RX #define HAL_I2C_STATE_MASTER_BUSY_TX HAL_I2C_STATE_BUSY_TX @@ -1005,7 +1143,7 @@ extern "C" { #define OPAMP_PGACONNECT_VM0 OPAMP_PGA_CONNECT_INVERTINGINPUT_IO0 #define OPAMP_PGACONNECT_VM1 OPAMP_PGA_CONNECT_INVERTINGINPUT_IO1 -#if defined(STM32L1) || defined(STM32L4) || defined(STM32L5) || defined(STM32H7) || defined(STM32G4) +#if defined(STM32L1) || defined(STM32L4) || defined(STM32L5) || defined(STM32H7) || defined(STM32G4) || defined(STM32U5) #define HAL_OPAMP_MSP_INIT_CB_ID HAL_OPAMP_MSPINIT_CB_ID #define HAL_OPAMP_MSP_DEINIT_CB_ID HAL_OPAMP_MSPDEINIT_CB_ID #endif @@ -1101,6 +1239,26 @@ extern "C" { #define RTC_TAMPERPIN_PA0 RTC_TAMPERPIN_POS1 #define RTC_TAMPERPIN_PI8 RTC_TAMPERPIN_POS1 +#if defined(STM32H5) +#define TAMP_SECRETDEVICE_ERASE_NONE TAMP_DEVICESECRETS_ERASE_NONE +#define TAMP_SECRETDEVICE_ERASE_BKP_SRAM TAMP_DEVICESECRETS_ERASE_BKPSRAM +#endif /* STM32H5 */ + +#if defined(STM32WBA) +#define TAMP_SECRETDEVICE_ERASE_NONE TAMP_DEVICESECRETS_ERASE_NONE +#define TAMP_SECRETDEVICE_ERASE_SRAM2 TAMP_DEVICESECRETS_ERASE_SRAM2 +#define TAMP_SECRETDEVICE_ERASE_RHUK TAMP_DEVICESECRETS_ERASE_RHUK +#define TAMP_SECRETDEVICE_ERASE_ICACHE TAMP_DEVICESECRETS_ERASE_ICACHE +#define TAMP_SECRETDEVICE_ERASE_SAES_AES_HASH TAMP_DEVICESECRETS_ERASE_SAES_AES_HASH +#define TAMP_SECRETDEVICE_ERASE_PKA_SRAM TAMP_DEVICESECRETS_ERASE_PKA_SRAM +#define TAMP_SECRETDEVICE_ERASE_ALL TAMP_DEVICESECRETS_ERASE_ALL +#endif /* STM32WBA */ + +#if defined(STM32H5) || defined(STM32WBA) +#define TAMP_SECRETDEVICE_ERASE_DISABLE TAMP_DEVICESECRETS_ERASE_NONE +#define TAMP_SECRETDEVICE_ERASE_ENABLE TAMP_SECRETDEVICE_ERASE_ALL +#endif /* STM32H5 || STM32WBA */ + #if defined(STM32F7) #define RTC_TAMPCR_TAMPXE RTC_TAMPER_ENABLE_BITS_MASK #define RTC_TAMPCR_TAMPXIE RTC_TAMPER_IT_ENABLE_BITS_MASK @@ -1111,12 +1269,12 @@ extern "C" { #define RTC_TAMPCR_TAMPXIE RTC_TAMPER_X_INTERRUPT #endif /* STM32H7 */ -#if defined(STM32F7) || defined(STM32H7) +#if defined(STM32F7) || defined(STM32H7) || defined(STM32L0) #define RTC_TAMPER1_INTERRUPT RTC_IT_TAMP1 #define RTC_TAMPER2_INTERRUPT RTC_IT_TAMP2 #define RTC_TAMPER3_INTERRUPT RTC_IT_TAMP3 #define RTC_ALL_TAMPER_INTERRUPT RTC_IT_TAMP -#endif /* STM32F7 || STM32H7 */ +#endif /* STM32F7 || STM32H7 || STM32L0 */ /** * @} @@ -1283,7 +1441,7 @@ extern "C" { #define TIM_TIM3_TI1_COMP1COMP2_OUT TIM_TIM3_TI1_COMP1_COMP2 #endif -#if defined(STM32U5) || defined(STM32MP2) +#if defined(STM32U5) #define OCREF_CLEAR_SELECT_Pos OCREF_CLEAR_SELECT_POS #define OCREF_CLEAR_SELECT_Msk OCREF_CLEAR_SELECT_MSK #endif @@ -1396,30 +1554,40 @@ extern "C" { #define ETH_MMCRFAECR 0x00000198U #define ETH_MMCRGUFCR 0x000001C4U -#define ETH_MAC_TXFIFO_FULL 0x02000000U /* Tx FIFO full */ -#define ETH_MAC_TXFIFONOT_EMPTY 0x01000000U /* Tx FIFO not empty */ -#define ETH_MAC_TXFIFO_WRITE_ACTIVE 0x00400000U /* Tx FIFO write active */ -#define ETH_MAC_TXFIFO_IDLE 0x00000000U /* Tx FIFO read status: Idle */ -#define ETH_MAC_TXFIFO_READ 0x00100000U /* Tx FIFO read status: Read (transferring data to the MAC transmitter) */ -#define ETH_MAC_TXFIFO_WAITING 0x00200000U /* Tx FIFO read status: Waiting for TxStatus from MAC transmitter */ -#define ETH_MAC_TXFIFO_WRITING 0x00300000U /* Tx FIFO read status: Writing the received TxStatus or flushing the TxFIFO */ -#define ETH_MAC_TRANSMISSION_PAUSE 0x00080000U /* MAC transmitter in pause */ -#define ETH_MAC_TRANSMITFRAMECONTROLLER_IDLE 0x00000000U /* MAC transmit frame controller: Idle */ -#define ETH_MAC_TRANSMITFRAMECONTROLLER_WAITING 0x00020000U /* MAC transmit frame controller: Waiting for Status of previous frame or IFG/backoff period to be over */ -#define ETH_MAC_TRANSMITFRAMECONTROLLER_GENRATING_PCF 0x00040000U /* MAC transmit frame controller: Generating and transmitting a Pause control frame (in full duplex mode) */ -#define ETH_MAC_TRANSMITFRAMECONTROLLER_TRANSFERRING 0x00060000U /* MAC transmit frame controller: Transferring input frame for transmission */ +#define ETH_MAC_TXFIFO_FULL 0x02000000U /* Tx FIFO full */ +#define ETH_MAC_TXFIFONOT_EMPTY 0x01000000U /* Tx FIFO not empty */ +#define ETH_MAC_TXFIFO_WRITE_ACTIVE 0x00400000U /* Tx FIFO write active */ +#define ETH_MAC_TXFIFO_IDLE 0x00000000U /* Tx FIFO read status: Idle */ +#define ETH_MAC_TXFIFO_READ 0x00100000U /* Tx FIFO read status: Read (transferring data to + the MAC transmitter) */ +#define ETH_MAC_TXFIFO_WAITING 0x00200000U /* Tx FIFO read status: Waiting for TxStatus from + MAC transmitter */ +#define ETH_MAC_TXFIFO_WRITING 0x00300000U /* Tx FIFO read status: Writing the received TxStatus + or flushing the TxFIFO */ +#define ETH_MAC_TRANSMISSION_PAUSE 0x00080000U /* MAC transmitter in pause */ +#define ETH_MAC_TRANSMITFRAMECONTROLLER_IDLE 0x00000000U /* MAC transmit frame controller: Idle */ +#define ETH_MAC_TRANSMITFRAMECONTROLLER_WAITING 0x00020000U /* MAC transmit frame controller: Waiting for Status + of previous frame or IFG/backoff period to be over */ +#define ETH_MAC_TRANSMITFRAMECONTROLLER_GENRATING_PCF 0x00040000U /* MAC transmit frame controller: Generating and + transmitting a Pause control frame (in full duplex mode) */ +#define ETH_MAC_TRANSMITFRAMECONTROLLER_TRANSFERRING 0x00060000U /* MAC transmit frame controller: Transferring input + frame for transmission */ #define ETH_MAC_MII_TRANSMIT_ACTIVE 0x00010000U /* MAC MII transmit engine active */ #define ETH_MAC_RXFIFO_EMPTY 0x00000000U /* Rx FIFO fill level: empty */ -#define ETH_MAC_RXFIFO_BELOW_THRESHOLD 0x00000100U /* Rx FIFO fill level: fill-level below flow-control de-activate threshold */ -#define ETH_MAC_RXFIFO_ABOVE_THRESHOLD 0x00000200U /* Rx FIFO fill level: fill-level above flow-control activate threshold */ +#define ETH_MAC_RXFIFO_BELOW_THRESHOLD 0x00000100U /* Rx FIFO fill level: fill-level below flow-control + de-activate threshold */ +#define ETH_MAC_RXFIFO_ABOVE_THRESHOLD 0x00000200U /* Rx FIFO fill level: fill-level above flow-control + activate threshold */ #define ETH_MAC_RXFIFO_FULL 0x00000300U /* Rx FIFO fill level: full */ #if defined(STM32F1) #else #define ETH_MAC_READCONTROLLER_IDLE 0x00000000U /* Rx FIFO read controller IDLE state */ #define ETH_MAC_READCONTROLLER_READING_DATA 0x00000020U /* Rx FIFO read controller Reading frame data */ -#define ETH_MAC_READCONTROLLER_READING_STATUS 0x00000040U /* Rx FIFO read controller Reading frame status (or time-stamp) */ +#define ETH_MAC_READCONTROLLER_READING_STATUS 0x00000040U /* Rx FIFO read controller Reading frame status + (or time-stamp) */ #endif -#define ETH_MAC_READCONTROLLER_FLUSHING 0x00000060U /* Rx FIFO read controller Flushing the frame data and status */ +#define ETH_MAC_READCONTROLLER_FLUSHING 0x00000060U /* Rx FIFO read controller Flushing the frame data and + status */ #define ETH_MAC_RXFIFO_WRITE_ACTIVE 0x00000010U /* Rx FIFO write controller active */ #define ETH_MAC_SMALL_FIFO_NOTACTIVE 0x00000000U /* MAC small FIFO read / write controllers not active */ #define ETH_MAC_SMALL_FIFO_READ_ACTIVE 0x00000002U /* MAC small FIFO read controller active */ @@ -1427,6 +1595,8 @@ extern "C" { #define ETH_MAC_SMALL_FIFO_RW_ACTIVE 0x00000006U /* MAC small FIFO read / write controllers active */ #define ETH_MAC_MII_RECEIVE_PROTOCOL_ACTIVE 0x00000001U /* MAC MII receive protocol engine active */ +#define ETH_TxPacketConfig ETH_TxPacketConfig_t /* Transmit Packet Configuration structure definition */ + /** * @} */ @@ -1590,7 +1760,8 @@ extern "C" { #define HAL_EnableDBGStandbyMode HAL_DBGMCU_EnableDBGStandbyMode #define HAL_DisableDBGStandbyMode HAL_DBGMCU_DisableDBGStandbyMode #define HAL_DBG_LowPowerConfig(Periph, cmd) (((cmd\ - )==ENABLE)? HAL_DBGMCU_DBG_EnableLowPowerConfig(Periph) : HAL_DBGMCU_DBG_DisableLowPowerConfig(Periph)) + )==ENABLE)? HAL_DBGMCU_DBG_EnableLowPowerConfig(Periph) : \ + HAL_DBGMCU_DBG_DisableLowPowerConfig(Periph)) #define HAL_VREFINT_OutputSelect HAL_SYSCFG_VREFINT_OutputSelect #define HAL_Lock_Cmd(cmd) (((cmd)==ENABLE) ? HAL_SYSCFG_Enable_Lock_VREFINT() : HAL_SYSCFG_Disable_Lock_VREFINT()) #if defined(STM32L0) @@ -1599,8 +1770,10 @@ extern "C" { #endif #define HAL_ADC_EnableBuffer_Cmd(cmd) (((cmd)==ENABLE) ? HAL_ADCEx_EnableVREFINT() : HAL_ADCEx_DisableVREFINT()) #define HAL_ADC_EnableBufferSensor_Cmd(cmd) (((cmd\ - )==ENABLE) ? HAL_ADCEx_EnableVREFINTTempSensor() : HAL_ADCEx_DisableVREFINTTempSensor()) -#if defined(STM32H7A3xx) || defined(STM32H7B3xx) || defined(STM32H7B0xx) || defined(STM32H7A3xxQ) || defined(STM32H7B3xxQ) || defined(STM32H7B0xxQ) + )==ENABLE) ? HAL_ADCEx_EnableVREFINTTempSensor() : \ + HAL_ADCEx_DisableVREFINTTempSensor()) +#if defined(STM32H7A3xx) || defined(STM32H7B3xx) || defined(STM32H7B0xx) || defined(STM32H7A3xxQ) || \ + defined(STM32H7B3xxQ) || defined(STM32H7B0xxQ) #define HAL_EnableSRDomainDBGStopMode HAL_EnableDomain3DBGStopMode #define HAL_DisableSRDomainDBGStopMode HAL_DisableDomain3DBGStopMode #define HAL_EnableSRDomainDBGStandbyMode HAL_EnableDomain3DBGStandbyMode @@ -1634,16 +1807,21 @@ extern "C" { #define HAL_FMPI2CEx_AnalogFilter_Config HAL_FMPI2CEx_ConfigAnalogFilter #define HAL_FMPI2CEx_DigitalFilter_Config HAL_FMPI2CEx_ConfigDigitalFilter -#define HAL_I2CFastModePlusConfig(SYSCFG_I2CFastModePlus, cmd) (((cmd\ - )==ENABLE)? HAL_I2CEx_EnableFastModePlus(SYSCFG_I2CFastModePlus): HAL_I2CEx_DisableFastModePlus(SYSCFG_I2CFastModePlus)) +#define HAL_I2CFastModePlusConfig(SYSCFG_I2CFastModePlus, cmd) ((cmd == ENABLE)? \ + HAL_I2CEx_EnableFastModePlus(SYSCFG_I2CFastModePlus): \ + HAL_I2CEx_DisableFastModePlus(SYSCFG_I2CFastModePlus)) -#if defined(STM32H7) || defined(STM32WB) || defined(STM32G0) || defined(STM32F0) || defined(STM32F1) || defined(STM32F2) || defined(STM32F3) || defined(STM32F4) || defined(STM32F7) || defined(STM32L0) || defined(STM32L4) || defined(STM32L5) || defined(STM32G4) || defined(STM32L1) +#if defined(STM32H7) || defined(STM32WB) || defined(STM32G0) || defined(STM32F0) || defined(STM32F1) || \ + defined(STM32F2) || defined(STM32F3) || defined(STM32F4) || defined(STM32F7) || defined(STM32L0) || \ + defined(STM32L4) || defined(STM32L5) || defined(STM32G4) || defined(STM32L1) #define HAL_I2C_Master_Sequential_Transmit_IT HAL_I2C_Master_Seq_Transmit_IT #define HAL_I2C_Master_Sequential_Receive_IT HAL_I2C_Master_Seq_Receive_IT #define HAL_I2C_Slave_Sequential_Transmit_IT HAL_I2C_Slave_Seq_Transmit_IT #define HAL_I2C_Slave_Sequential_Receive_IT HAL_I2C_Slave_Seq_Receive_IT -#endif /* STM32H7 || STM32WB || STM32G0 || STM32F0 || STM32F1 || STM32F2 || STM32F3 || STM32F4 || STM32F7 || STM32L0 || STM32L4 || STM32L5 || STM32G4 || STM32L1 */ -#if defined(STM32H7) || defined(STM32WB) || defined(STM32G0) || defined(STM32F4) || defined(STM32F7) || defined(STM32L0) || defined(STM32L4) || defined(STM32L5) || defined(STM32G4)|| defined(STM32L1) +#endif /* STM32H7 || STM32WB || STM32G0 || STM32F0 || STM32F1 || STM32F2 || STM32F3 || STM32F4 || STM32F7 || STM32L0 || + STM32L4 || STM32L5 || STM32G4 || STM32L1 */ +#if defined(STM32H7) || defined(STM32WB) || defined(STM32G0) || defined(STM32F4) || defined(STM32F7) || \ + defined(STM32L0) || defined(STM32L4) || defined(STM32L5) || defined(STM32G4)|| defined(STM32L1) #define HAL_I2C_Master_Sequential_Transmit_DMA HAL_I2C_Master_Seq_Transmit_DMA #define HAL_I2C_Master_Sequential_Receive_DMA HAL_I2C_Master_Seq_Receive_DMA #define HAL_I2C_Slave_Sequential_Transmit_DMA HAL_I2C_Slave_Seq_Transmit_DMA @@ -1768,6 +1946,17 @@ extern "C" { #define PWR_SRAM5_PAGE13_STOP_RETENTION PWR_SRAM5_PAGE13_STOP #define PWR_SRAM5_FULL_STOP_RETENTION PWR_SRAM5_FULL_STOP +#define PWR_SRAM6_PAGE1_STOP_RETENTION PWR_SRAM6_PAGE1_STOP +#define PWR_SRAM6_PAGE2_STOP_RETENTION PWR_SRAM6_PAGE2_STOP +#define PWR_SRAM6_PAGE3_STOP_RETENTION PWR_SRAM6_PAGE3_STOP +#define PWR_SRAM6_PAGE4_STOP_RETENTION PWR_SRAM6_PAGE4_STOP +#define PWR_SRAM6_PAGE5_STOP_RETENTION PWR_SRAM6_PAGE5_STOP +#define PWR_SRAM6_PAGE6_STOP_RETENTION PWR_SRAM6_PAGE6_STOP +#define PWR_SRAM6_PAGE7_STOP_RETENTION PWR_SRAM6_PAGE7_STOP +#define PWR_SRAM6_PAGE8_STOP_RETENTION PWR_SRAM6_PAGE8_STOP +#define PWR_SRAM6_FULL_STOP_RETENTION PWR_SRAM6_FULL_STOP + + #define PWR_ICACHE_FULL_STOP_RETENTION PWR_ICACHE_FULL_STOP #define PWR_DCACHE1_FULL_STOP_RETENTION PWR_DCACHE1_FULL_STOP #define PWR_DCACHE2_FULL_STOP_RETENTION PWR_DCACHE2_FULL_STOP @@ -1776,6 +1965,8 @@ extern "C" { #define PWR_PKA32RAM_FULL_STOP_RETENTION PWR_PKA32RAM_FULL_STOP #define PWR_GRAPHICPRAM_FULL_STOP_RETENTION PWR_GRAPHICPRAM_FULL_STOP #define PWR_DSIRAM_FULL_STOP_RETENTION PWR_DSIRAM_FULL_STOP +#define PWR_JPEGRAM_FULL_STOP_RETENTION PWR_JPEGRAM_FULL_STOP + #define PWR_SRAM2_PAGE1_STANDBY_RETENTION PWR_SRAM2_PAGE1_STANDBY #define PWR_SRAM2_PAGE2_STANDBY_RETENTION PWR_SRAM2_PAGE2_STANDBY @@ -1786,6 +1977,7 @@ extern "C" { #define PWR_SRAM3_FULL_RUN_RETENTION PWR_SRAM3_FULL_RUN #define PWR_SRAM4_FULL_RUN_RETENTION PWR_SRAM4_FULL_RUN #define PWR_SRAM5_FULL_RUN_RETENTION PWR_SRAM5_FULL_RUN +#define PWR_SRAM6_FULL_RUN_RETENTION PWR_SRAM6_FULL_RUN #define PWR_ALL_RAM_RUN_RETENTION_MASK PWR_ALL_RAM_RUN_MASK #endif @@ -1794,6 +1986,20 @@ extern "C" { * @} */ +/** @defgroup HAL_RTC_Aliased_Functions HAL RTC Aliased Functions maintained for legacy purpose + * @{ + */ +#if defined(STM32H5) || defined(STM32WBA) +#define HAL_RTCEx_SetBoothardwareKey HAL_RTCEx_LockBootHardwareKey +#define HAL_RTCEx_BKUPBlock_Enable HAL_RTCEx_BKUPBlock +#define HAL_RTCEx_BKUPBlock_Disable HAL_RTCEx_BKUPUnblock +#define HAL_RTCEx_Erase_SecretDev_Conf HAL_RTCEx_ConfigEraseDeviceSecrets +#endif /* STM32H5 || STM32WBA */ + +/** + * @} + */ + /** @defgroup HAL_SMBUS_Aliased_Functions HAL SMBUS Aliased Functions maintained for legacy purpose * @{ */ @@ -1819,7 +2025,8 @@ extern "C" { #define HAL_TIM_DMAError TIM_DMAError #define HAL_TIM_DMACaptureCplt TIM_DMACaptureCplt #define HAL_TIMEx_DMACommutationCplt TIMEx_DMACommutationCplt -#if defined(STM32H7) || defined(STM32G0) || defined(STM32F0) || defined(STM32F1) || defined(STM32F2) || defined(STM32F3) || defined(STM32F4) || defined(STM32F7) || defined(STM32L0) || defined(STM32L4) +#if defined(STM32H7) || defined(STM32G0) || defined(STM32F0) || defined(STM32F1) || defined(STM32F2) || \ + defined(STM32F3) || defined(STM32F4) || defined(STM32F7) || defined(STM32L0) || defined(STM32L4) #define HAL_TIM_SlaveConfigSynchronization HAL_TIM_SlaveConfigSynchro #define HAL_TIM_SlaveConfigSynchronization_IT HAL_TIM_SlaveConfigSynchro_IT #define HAL_TIMEx_CommutationCallback HAL_TIMEx_CommutCallback @@ -2076,7 +2283,8 @@ extern "C" { #define COMP_STOP __HAL_COMP_DISABLE #define COMP_LOCK __HAL_COMP_LOCK -#if defined(STM32F301x8) || defined(STM32F302x8) || defined(STM32F318xx) || defined(STM32F303x8) || defined(STM32F334x8) || defined(STM32F328xx) +#if defined(STM32F301x8) || defined(STM32F302x8) || defined(STM32F318xx) || defined(STM32F303x8) || \ + defined(STM32F334x8) || defined(STM32F328xx) #define __HAL_COMP_EXTI_RISING_IT_ENABLE(__EXTILINE__) (((__EXTILINE__) == COMP_EXTI_LINE_COMP2) ? __HAL_COMP_COMP2_EXTI_ENABLE_RISING_EDGE() : \ ((__EXTILINE__) == COMP_EXTI_LINE_COMP4) ? __HAL_COMP_COMP4_EXTI_ENABLE_RISING_EDGE() : \ __HAL_COMP_COMP6_EXTI_ENABLE_RISING_EDGE()) @@ -2248,8 +2456,10 @@ extern "C" { /** @defgroup HAL_COMP_Aliased_Functions HAL COMP Aliased Functions maintained for legacy purpose * @{ */ -#define HAL_COMP_Start_IT HAL_COMP_Start /* Function considered as legacy as EXTI event or IT configuration is done into HAL_COMP_Init() */ -#define HAL_COMP_Stop_IT HAL_COMP_Stop /* Function considered as legacy as EXTI event or IT configuration is done into HAL_COMP_Init() */ +#define HAL_COMP_Start_IT HAL_COMP_Start /* Function considered as legacy as EXTI event or IT configuration is + done into HAL_COMP_Init() */ +#define HAL_COMP_Stop_IT HAL_COMP_Stop /* Function considered as legacy as EXTI event or IT configuration is + done into HAL_COMP_Init() */ /** * @} */ @@ -2408,7 +2618,9 @@ extern "C" { #define __HAL_PWR_INTERNALWAKEUP_ENABLE HAL_PWREx_EnableInternalWakeUpLine #define __HAL_PWR_PULL_UP_DOWN_CONFIG_DISABLE HAL_PWREx_DisablePullUpPullDownConfig #define __HAL_PWR_PULL_UP_DOWN_CONFIG_ENABLE HAL_PWREx_EnablePullUpPullDownConfig -#define __HAL_PWR_PVD_EXTI_CLEAR_EGDE_TRIGGER() do { __HAL_PWR_PVD_EXTI_DISABLE_RISING_EDGE();__HAL_PWR_PVD_EXTI_DISABLE_FALLING_EDGE(); } while(0) +#define __HAL_PWR_PVD_EXTI_CLEAR_EGDE_TRIGGER() do { __HAL_PWR_PVD_EXTI_DISABLE_RISING_EDGE(); \ + __HAL_PWR_PVD_EXTI_DISABLE_FALLING_EDGE(); \ + } while(0) #define __HAL_PWR_PVD_EXTI_EVENT_DISABLE __HAL_PWR_PVD_EXTI_DISABLE_EVENT #define __HAL_PWR_PVD_EXTI_EVENT_ENABLE __HAL_PWR_PVD_EXTI_ENABLE_EVENT #define __HAL_PWR_PVD_EXTI_FALLINGTRIGGER_DISABLE __HAL_PWR_PVD_EXTI_DISABLE_FALLING_EDGE @@ -2417,8 +2629,12 @@ extern "C" { #define __HAL_PWR_PVD_EXTI_RISINGTRIGGER_ENABLE __HAL_PWR_PVD_EXTI_ENABLE_RISING_EDGE #define __HAL_PWR_PVD_EXTI_SET_FALLING_EGDE_TRIGGER __HAL_PWR_PVD_EXTI_ENABLE_FALLING_EDGE #define __HAL_PWR_PVD_EXTI_SET_RISING_EDGE_TRIGGER __HAL_PWR_PVD_EXTI_ENABLE_RISING_EDGE -#define __HAL_PWR_PVM_DISABLE() do { HAL_PWREx_DisablePVM1();HAL_PWREx_DisablePVM2();HAL_PWREx_DisablePVM3();HAL_PWREx_DisablePVM4(); } while(0) -#define __HAL_PWR_PVM_ENABLE() do { HAL_PWREx_EnablePVM1();HAL_PWREx_EnablePVM2();HAL_PWREx_EnablePVM3();HAL_PWREx_EnablePVM4(); } while(0) +#define __HAL_PWR_PVM_DISABLE() do { HAL_PWREx_DisablePVM1();HAL_PWREx_DisablePVM2(); \ + HAL_PWREx_DisablePVM3();HAL_PWREx_DisablePVM4(); \ + } while(0) +#define __HAL_PWR_PVM_ENABLE() do { HAL_PWREx_EnablePVM1();HAL_PWREx_EnablePVM2(); \ + HAL_PWREx_EnablePVM3();HAL_PWREx_EnablePVM4(); \ + } while(0) #define __HAL_PWR_SRAM2CONTENT_PRESERVE_DISABLE HAL_PWREx_DisableSRAM2ContentRetention #define __HAL_PWR_SRAM2CONTENT_PRESERVE_ENABLE HAL_PWREx_EnableSRAM2ContentRetention #define __HAL_PWR_VDDIO2_DISABLE HAL_PWREx_DisableVddIO2 @@ -2454,8 +2670,8 @@ extern "C" { #define RCC_StopWakeUpClock_HSI RCC_STOP_WAKEUPCLOCK_HSI #define HAL_RCC_CCSCallback HAL_RCC_CSSCallback -#define HAL_RC48_EnableBuffer_Cmd(cmd) (((cmd\ - )==ENABLE) ? HAL_RCCEx_EnableHSI48_VREFINT() : HAL_RCCEx_DisableHSI48_VREFINT()) +#define HAL_RC48_EnableBuffer_Cmd(cmd) (((cmd)==ENABLE) ? \ + HAL_RCCEx_EnableHSI48_VREFINT() : HAL_RCCEx_DisableHSI48_VREFINT()) #define __ADC_CLK_DISABLE __HAL_RCC_ADC_CLK_DISABLE #define __ADC_CLK_ENABLE __HAL_RCC_ADC_CLK_ENABLE @@ -2959,6 +3175,11 @@ extern "C" { #define __HAL_RCC_WWDG_IS_CLK_ENABLED __HAL_RCC_WWDG1_IS_CLK_ENABLED #define __HAL_RCC_WWDG_IS_CLK_DISABLED __HAL_RCC_WWDG1_IS_CLK_DISABLED +#define RCC_SPI4CLKSOURCE_D2PCLK1 RCC_SPI4CLKSOURCE_D2PCLK2 +#define RCC_SPI5CLKSOURCE_D2PCLK1 RCC_SPI5CLKSOURCE_D2PCLK2 +#define RCC_SPI45CLKSOURCE_D2PCLK1 RCC_SPI45CLKSOURCE_D2PCLK2 +#define RCC_SPI45CLKSOURCE_CDPCLK1 RCC_SPI45CLKSOURCE_CDPCLK2 +#define RCC_SPI45CLKSOURCE_PCLK1 RCC_SPI45CLKSOURCE_PCLK2 #endif #define __WWDG_CLK_DISABLE __HAL_RCC_WWDG_CLK_DISABLE @@ -3423,7 +3644,8 @@ extern "C" { #define RCC_MCOSOURCE_PLLCLK_NODIV RCC_MCO1SOURCE_PLLCLK #define RCC_MCOSOURCE_PLLCLK_DIV2 RCC_MCO1SOURCE_PLLCLK_DIV2 -#if defined(STM32L4) || defined(STM32WB) || defined(STM32G0) || defined(STM32G4) || defined(STM32L5) || defined(STM32WL) || defined(STM32C0) +#if defined(STM32L4) || defined(STM32WB) || defined(STM32G0) || defined(STM32G4) || defined(STM32L5) || \ + defined(STM32WL) || defined(STM32C0) #define RCC_RTCCLKSOURCE_NO_CLK RCC_RTCCLKSOURCE_NONE #else #define RCC_RTCCLKSOURCE_NONE RCC_RTCCLKSOURCE_NO_CLK @@ -3568,6 +3790,92 @@ extern "C" { #define IS_RCC_PLLFRACN_VALUE IS_RCC_PLL_FRACN_VALUE #endif /* STM32U5 */ +#if defined(STM32H5) +#define __HAL_RCC_PLLFRACN_ENABLE __HAL_RCC_PLL_FRACN_ENABLE +#define __HAL_RCC_PLLFRACN_DISABLE __HAL_RCC_PLL_FRACN_DISABLE +#define __HAL_RCC_PLLFRACN_CONFIG __HAL_RCC_PLL_FRACN_CONFIG +#define IS_RCC_PLLFRACN_VALUE IS_RCC_PLL_FRACN_VALUE + +#define RCC_PLLSOURCE_NONE RCC_PLL1_SOURCE_NONE +#define RCC_PLLSOURCE_HSI RCC_PLL1_SOURCE_HSI +#define RCC_PLLSOURCE_CSI RCC_PLL1_SOURCE_CSI +#define RCC_PLLSOURCE_HSE RCC_PLL1_SOURCE_HSE +#define RCC_PLLVCIRANGE_0 RCC_PLL1_VCIRANGE_0 +#define RCC_PLLVCIRANGE_1 RCC_PLL1_VCIRANGE_1 +#define RCC_PLLVCIRANGE_2 RCC_PLL1_VCIRANGE_2 +#define RCC_PLLVCIRANGE_3 RCC_PLL1_VCIRANGE_3 +#define RCC_PLL1VCOWIDE RCC_PLL1_VCORANGE_WIDE +#define RCC_PLL1VCOMEDIUM RCC_PLL1_VCORANGE_MEDIUM + +#define IS_RCC_PLLSOURCE IS_RCC_PLL1_SOURCE +#define IS_RCC_PLLRGE_VALUE IS_RCC_PLL1_VCIRGE_VALUE +#define IS_RCC_PLLVCORGE_VALUE IS_RCC_PLL1_VCORGE_VALUE +#define IS_RCC_PLLCLOCKOUT_VALUE IS_RCC_PLL1_CLOCKOUT_VALUE +#define IS_RCC_PLL_FRACN_VALUE IS_RCC_PLL1_FRACN_VALUE +#define IS_RCC_PLLM_VALUE IS_RCC_PLL1_DIVM_VALUE +#define IS_RCC_PLLN_VALUE IS_RCC_PLL1_MULN_VALUE +#define IS_RCC_PLLP_VALUE IS_RCC_PLL1_DIVP_VALUE +#define IS_RCC_PLLQ_VALUE IS_RCC_PLL1_DIVQ_VALUE +#define IS_RCC_PLLR_VALUE IS_RCC_PLL1_DIVR_VALUE + +#define __HAL_RCC_PLL_ENABLE __HAL_RCC_PLL1_ENABLE +#define __HAL_RCC_PLL_DISABLE __HAL_RCC_PLL1_DISABLE +#define __HAL_RCC_PLL_FRACN_ENABLE __HAL_RCC_PLL1_FRACN_ENABLE +#define __HAL_RCC_PLL_FRACN_DISABLE __HAL_RCC_PLL1_FRACN_DISABLE +#define __HAL_RCC_PLL_CONFIG __HAL_RCC_PLL1_CONFIG +#define __HAL_RCC_PLL_PLLSOURCE_CONFIG __HAL_RCC_PLL1_PLLSOURCE_CONFIG +#define __HAL_RCC_PLL_DIVM_CONFIG __HAL_RCC_PLL1_DIVM_CONFIG +#define __HAL_RCC_PLL_FRACN_CONFIG __HAL_RCC_PLL1_FRACN_CONFIG +#define __HAL_RCC_PLL_VCIRANGE __HAL_RCC_PLL1_VCIRANGE +#define __HAL_RCC_PLL_VCORANGE __HAL_RCC_PLL1_VCORANGE +#define __HAL_RCC_GET_PLL_OSCSOURCE __HAL_RCC_GET_PLL1_OSCSOURCE +#define __HAL_RCC_PLLCLKOUT_ENABLE __HAL_RCC_PLL1_CLKOUT_ENABLE +#define __HAL_RCC_PLLCLKOUT_DISABLE __HAL_RCC_PLL1_CLKOUT_DISABLE +#define __HAL_RCC_GET_PLLCLKOUT_CONFIG __HAL_RCC_GET_PLL1_CLKOUT_CONFIG + +#define __HAL_RCC_PLL2FRACN_ENABLE __HAL_RCC_PLL2_FRACN_ENABLE +#define __HAL_RCC_PLL2FRACN_DISABLE __HAL_RCC_PLL2_FRACN_DISABLE +#define __HAL_RCC_PLL2CLKOUT_ENABLE __HAL_RCC_PLL2_CLKOUT_ENABLE +#define __HAL_RCC_PLL2CLKOUT_DISABLE __HAL_RCC_PLL2_CLKOUT_DISABLE +#define __HAL_RCC_PLL2FRACN_CONFIG __HAL_RCC_PLL2_FRACN_CONFIG +#define __HAL_RCC_GET_PLL2CLKOUT_CONFIG __HAL_RCC_GET_PLL2_CLKOUT_CONFIG + +#define __HAL_RCC_PLL3FRACN_ENABLE __HAL_RCC_PLL3_FRACN_ENABLE +#define __HAL_RCC_PLL3FRACN_DISABLE __HAL_RCC_PLL3_FRACN_DISABLE +#define __HAL_RCC_PLL3CLKOUT_ENABLE __HAL_RCC_PLL3_CLKOUT_ENABLE +#define __HAL_RCC_PLL3CLKOUT_DISABLE __HAL_RCC_PLL3_CLKOUT_DISABLE +#define __HAL_RCC_PLL3FRACN_CONFIG __HAL_RCC_PLL3_FRACN_CONFIG +#define __HAL_RCC_GET_PLL3CLKOUT_CONFIG __HAL_RCC_GET_PLL3_CLKOUT_CONFIG + +#define RCC_PLL2VCIRANGE_0 RCC_PLL2_VCIRANGE_0 +#define RCC_PLL2VCIRANGE_1 RCC_PLL2_VCIRANGE_1 +#define RCC_PLL2VCIRANGE_2 RCC_PLL2_VCIRANGE_2 +#define RCC_PLL2VCIRANGE_3 RCC_PLL2_VCIRANGE_3 + +#define RCC_PLL2VCOWIDE RCC_PLL2_VCORANGE_WIDE +#define RCC_PLL2VCOMEDIUM RCC_PLL2_VCORANGE_MEDIUM + +#define RCC_PLL2SOURCE_NONE RCC_PLL2_SOURCE_NONE +#define RCC_PLL2SOURCE_HSI RCC_PLL2_SOURCE_HSI +#define RCC_PLL2SOURCE_CSI RCC_PLL2_SOURCE_CSI +#define RCC_PLL2SOURCE_HSE RCC_PLL2_SOURCE_HSE + +#define RCC_PLL3VCIRANGE_0 RCC_PLL3_VCIRANGE_0 +#define RCC_PLL3VCIRANGE_1 RCC_PLL3_VCIRANGE_1 +#define RCC_PLL3VCIRANGE_2 RCC_PLL3_VCIRANGE_2 +#define RCC_PLL3VCIRANGE_3 RCC_PLL3_VCIRANGE_3 + +#define RCC_PLL3VCOWIDE RCC_PLL3_VCORANGE_WIDE +#define RCC_PLL3VCOMEDIUM RCC_PLL3_VCORANGE_MEDIUM + +#define RCC_PLL3SOURCE_NONE RCC_PLL3_SOURCE_NONE +#define RCC_PLL3SOURCE_HSI RCC_PLL3_SOURCE_HSI +#define RCC_PLL3SOURCE_CSI RCC_PLL3_SOURCE_CSI +#define RCC_PLL3SOURCE_HSE RCC_PLL3_SOURCE_HSE + + +#endif /* STM32H5 */ + /** * @} */ @@ -3584,9 +3892,9 @@ extern "C" { /** @defgroup HAL_RTC_Aliased_Macros HAL RTC Aliased Macros maintained for legacy purpose * @{ */ -#if defined (STM32G0) || defined (STM32L5) || defined (STM32L412xx) || defined (STM32L422xx) || defined (STM32L4P5xx)|| \ - defined (STM32L4Q5xx) || defined (STM32G4) || defined (STM32WL) || defined (STM32U5) || \ - defined (STM32C0) +#if defined (STM32G0) || defined (STM32L5) || defined (STM32L412xx) || defined (STM32L422xx) || \ + defined (STM32L4P5xx)|| defined (STM32L4Q5xx) || defined (STM32G4) || defined (STM32WL) || defined (STM32U5) || \ + defined (STM32WBA) || defined (STM32H5) || defined (STM32C0) #else #define __HAL_RTC_CLEAR_FLAG __HAL_RTC_EXTI_CLEAR_FLAG #endif @@ -3621,6 +3929,13 @@ extern "C" { __HAL_RTC_TAMPER_TIMESTAMP_EXTI_GENERATE_SWIT())) #endif /* STM32F1 */ +#if defined (STM32F0) || defined (STM32F2) || defined (STM32F3) || defined (STM32F4) || defined (STM32F7) || \ + defined (STM32H7) || \ + defined (STM32L0) || defined (STM32L1) || \ + defined (STM32WB) +#define __HAL_RTC_TAMPER_GET_IT __HAL_RTC_TAMPER_GET_FLAG +#endif + #define IS_ALARM IS_RTC_ALARM #define IS_ALARM_MASK IS_RTC_ALARM_MASK #define IS_TAMPER IS_RTC_TAMPER @@ -3639,6 +3954,11 @@ extern "C" { #define __RTC_WRITEPROTECTION_ENABLE __HAL_RTC_WRITEPROTECTION_ENABLE #define __RTC_WRITEPROTECTION_DISABLE __HAL_RTC_WRITEPROTECTION_DISABLE +#if defined (STM32H5) +#define __HAL_RCC_RTCAPB_CLK_ENABLE __HAL_RCC_RTC_CLK_ENABLE +#define __HAL_RCC_RTCAPB_CLK_DISABLE __HAL_RCC_RTC_CLK_DISABLE +#endif /* STM32H5 */ + /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_adc.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_adc.h index 072ea91f..c3a94a6e 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_adc.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_adc.h @@ -21,7 +21,7 @@ #define __STM32F4xx_ADC_H #ifdef __cplusplus - extern "C" { +extern "C" { #endif /* Includes ------------------------------------------------------------------*/ @@ -101,11 +101,11 @@ typedef struct If trigger is set to ADC_SOFTWARE_START, this parameter is discarded. This parameter can be a value of @ref ADC_External_trigger_edge_Regular */ FunctionalState DMAContinuousRequests; /*!< Specifies whether the DMA requests are performed in one shot mode (DMA transfer stop when number of conversions is reached) - or in Continuous mode (DMA transfer unlimited, whatever number of conversions). - Note: In continuous mode, DMA must be configured in circular mode. Otherwise an overrun will be triggered when DMA buffer maximum pointer is reached. - Note: This parameter must be modified when no conversion is on going on both regular and injected groups (ADC disabled, or ADC enabled without continuous mode or external trigger that could launch a conversion). - This parameter can be set to ENABLE or DISABLE. */ -}ADC_InitTypeDef; + or in Continuous mode (DMA transfer unlimited, whatever number of conversions). + Note: In continuous mode, DMA must be configured in circular mode. Otherwise an overrun will be triggered when DMA buffer maximum pointer is reached. + Note: This parameter must be modified when no conversion is on going on both regular and injected groups (ADC disabled, or ADC enabled without continuous mode or external trigger that could launch a conversion). + This parameter can be set to ENABLE or DISABLE. */ +} ADC_InitTypeDef; @@ -130,7 +130,7 @@ typedef struct sampling time constraints must be respected (sampling time can be adjusted in function of ADC clock frequency and sampling time setting) Refer to device datasheet for timings values, parameters TS_vrefint, TS_temp (values rough order: 4us min). */ uint32_t Offset; /*!< Reserved for future use, can be set to 0 */ -}ADC_ChannelConfTypeDef; +} ADC_ChannelConfTypeDef; /** * @brief ADC Configuration multi-mode structure definition @@ -150,7 +150,7 @@ typedef struct is interrupt mode or in polling mode. This parameter can be set to ENABLE or DISABLE */ uint32_t WatchdogNumber; /*!< Reserved for future use, can be set to 0 */ -}ADC_AnalogWDGConfTypeDef; +} ADC_AnalogWDGConfTypeDef; /** * @brief HAL ADC state machine: ADC states definition (bitfields) @@ -217,7 +217,7 @@ typedef struct void (* MspInitCallback)(struct __ADC_HandleTypeDef *hadc); /*!< ADC Msp Init callback */ void (* MspDeInitCallback)(struct __ADC_HandleTypeDef *hadc); /*!< ADC Msp DeInit callback */ #endif /* USE_HAL_ADC_REGISTER_CALLBACKS */ -}ADC_HandleTypeDef; +} ADC_HandleTypeDef; #if (USE_HAL_ADC_REGISTER_CALLBACKS == 1) /** @@ -401,7 +401,7 @@ typedef void (*pADC_CallbackTypeDef)(ADC_HandleTypeDef *hadc); /*!< pointer to * @} */ - /** @defgroup ADC_EOCSelection ADC EOC Selection +/** @defgroup ADC_EOCSelection ADC EOC Selection * @{ */ #define ADC_EOC_SEQ_CONV 0x00000000U @@ -562,10 +562,10 @@ typedef void (*pADC_CallbackTypeDef)(ADC_HandleTypeDef *hadc); /*!< pointer to * @{ */ /* Initialization/de-initialization functions ***********************************/ -HAL_StatusTypeDef HAL_ADC_Init(ADC_HandleTypeDef* hadc); +HAL_StatusTypeDef HAL_ADC_Init(ADC_HandleTypeDef *hadc); HAL_StatusTypeDef HAL_ADC_DeInit(ADC_HandleTypeDef *hadc); -void HAL_ADC_MspInit(ADC_HandleTypeDef* hadc); -void HAL_ADC_MspDeInit(ADC_HandleTypeDef* hadc); +void HAL_ADC_MspInit(ADC_HandleTypeDef *hadc); +void HAL_ADC_MspDeInit(ADC_HandleTypeDef *hadc); #if (USE_HAL_ADC_REGISTER_CALLBACKS == 1) /* Callbacks Register/UnRegister functions ***********************************/ @@ -580,25 +580,25 @@ HAL_StatusTypeDef HAL_ADC_UnRegisterCallback(ADC_HandleTypeDef *hadc, HAL_ADC_Ca * @{ */ /* I/O operation functions ******************************************************/ -HAL_StatusTypeDef HAL_ADC_Start(ADC_HandleTypeDef* hadc); -HAL_StatusTypeDef HAL_ADC_Stop(ADC_HandleTypeDef* hadc); -HAL_StatusTypeDef HAL_ADC_PollForConversion(ADC_HandleTypeDef* hadc, uint32_t Timeout); +HAL_StatusTypeDef HAL_ADC_Start(ADC_HandleTypeDef *hadc); +HAL_StatusTypeDef HAL_ADC_Stop(ADC_HandleTypeDef *hadc); +HAL_StatusTypeDef HAL_ADC_PollForConversion(ADC_HandleTypeDef *hadc, uint32_t Timeout); -HAL_StatusTypeDef HAL_ADC_PollForEvent(ADC_HandleTypeDef* hadc, uint32_t EventType, uint32_t Timeout); +HAL_StatusTypeDef HAL_ADC_PollForEvent(ADC_HandleTypeDef *hadc, uint32_t EventType, uint32_t Timeout); -HAL_StatusTypeDef HAL_ADC_Start_IT(ADC_HandleTypeDef* hadc); -HAL_StatusTypeDef HAL_ADC_Stop_IT(ADC_HandleTypeDef* hadc); +HAL_StatusTypeDef HAL_ADC_Start_IT(ADC_HandleTypeDef *hadc); +HAL_StatusTypeDef HAL_ADC_Stop_IT(ADC_HandleTypeDef *hadc); -void HAL_ADC_IRQHandler(ADC_HandleTypeDef* hadc); +void HAL_ADC_IRQHandler(ADC_HandleTypeDef *hadc); -HAL_StatusTypeDef HAL_ADC_Start_DMA(ADC_HandleTypeDef* hadc, uint32_t* pData, uint32_t Length); -HAL_StatusTypeDef HAL_ADC_Stop_DMA(ADC_HandleTypeDef* hadc); +HAL_StatusTypeDef HAL_ADC_Start_DMA(ADC_HandleTypeDef *hadc, uint32_t *pData, uint32_t Length); +HAL_StatusTypeDef HAL_ADC_Stop_DMA(ADC_HandleTypeDef *hadc); -uint32_t HAL_ADC_GetValue(ADC_HandleTypeDef* hadc); +uint32_t HAL_ADC_GetValue(ADC_HandleTypeDef *hadc); -void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc); -void HAL_ADC_ConvHalfCpltCallback(ADC_HandleTypeDef* hadc); -void HAL_ADC_LevelOutOfWindowCallback(ADC_HandleTypeDef* hadc); +void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef *hadc); +void HAL_ADC_ConvHalfCpltCallback(ADC_HandleTypeDef *hadc); +void HAL_ADC_LevelOutOfWindowCallback(ADC_HandleTypeDef *hadc); void HAL_ADC_ErrorCallback(ADC_HandleTypeDef *hadc); /** * @} @@ -608,8 +608,8 @@ void HAL_ADC_ErrorCallback(ADC_HandleTypeDef *hadc); * @{ */ /* Peripheral Control functions *************************************************/ -HAL_StatusTypeDef HAL_ADC_ConfigChannel(ADC_HandleTypeDef* hadc, ADC_ChannelConfTypeDef* sConfig); -HAL_StatusTypeDef HAL_ADC_AnalogWDGConfig(ADC_HandleTypeDef* hadc, ADC_AnalogWDGConfTypeDef* AnalogWDGConfig); +HAL_StatusTypeDef HAL_ADC_ConfigChannel(ADC_HandleTypeDef *hadc, ADC_ChannelConfTypeDef *sConfig); +HAL_StatusTypeDef HAL_ADC_AnalogWDGConfig(ADC_HandleTypeDef *hadc, ADC_AnalogWDGConfTypeDef *AnalogWDGConfig); /** * @} */ @@ -618,7 +618,7 @@ HAL_StatusTypeDef HAL_ADC_AnalogWDGConfig(ADC_HandleTypeDef* hadc, ADC_AnalogWDG * @{ */ /* Peripheral State functions ***************************************************/ -uint32_t HAL_ADC_GetState(ADC_HandleTypeDef* hadc); +uint32_t HAL_ADC_GetState(ADC_HandleTypeDef *hadc); uint32_t HAL_ADC_GetError(ADC_HandleTypeDef *hadc); /** * @} diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_adc_ex.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_adc_ex.h index b0a4eb72..8ce8484d 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_adc_ex.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_adc_ex.h @@ -21,7 +21,7 @@ #define __STM32F4xx_ADC_EX_H #ifdef __cplusplus - extern "C" { +extern "C" { #endif /* Includes ------------------------------------------------------------------*/ @@ -106,7 +106,7 @@ typedef struct If trigger is set to ADC_INJECTED_SOFTWARE_START, this parameter is discarded. Caution: this setting impacts the entire injected group. Therefore, call of HAL_ADCEx_InjectedConfigChannel() to configure a channel on injected group can impact the configuration of other channels previously set. */ -}ADC_InjectionConfTypeDef; +} ADC_InjectionConfTypeDef; /** * @brief ADC Configuration multi-mode structure definition @@ -119,7 +119,7 @@ typedef struct This parameter can be a value of @ref ADCEx_Direct_memory_access_mode_for_multi_mode */ uint32_t TwoSamplingDelay; /*!< Configures the Delay between 2 sampling phases. This parameter can be a value of @ref ADC_delay_between_2_sampling_phases */ -}ADC_MultiModeTypeDef; +} ADC_MultiModeTypeDef; /** * @} @@ -264,20 +264,20 @@ typedef struct */ /* I/O operation functions ******************************************************/ -HAL_StatusTypeDef HAL_ADCEx_InjectedStart(ADC_HandleTypeDef* hadc); -HAL_StatusTypeDef HAL_ADCEx_InjectedStop(ADC_HandleTypeDef* hadc); -HAL_StatusTypeDef HAL_ADCEx_InjectedPollForConversion(ADC_HandleTypeDef* hadc, uint32_t Timeout); -HAL_StatusTypeDef HAL_ADCEx_InjectedStart_IT(ADC_HandleTypeDef* hadc); -HAL_StatusTypeDef HAL_ADCEx_InjectedStop_IT(ADC_HandleTypeDef* hadc); -uint32_t HAL_ADCEx_InjectedGetValue(ADC_HandleTypeDef* hadc, uint32_t InjectedRank); -HAL_StatusTypeDef HAL_ADCEx_MultiModeStart_DMA(ADC_HandleTypeDef* hadc, uint32_t* pData, uint32_t Length); -HAL_StatusTypeDef HAL_ADCEx_MultiModeStop_DMA(ADC_HandleTypeDef* hadc); -uint32_t HAL_ADCEx_MultiModeGetValue(ADC_HandleTypeDef* hadc); -void HAL_ADCEx_InjectedConvCpltCallback(ADC_HandleTypeDef* hadc); +HAL_StatusTypeDef HAL_ADCEx_InjectedStart(ADC_HandleTypeDef *hadc); +HAL_StatusTypeDef HAL_ADCEx_InjectedStop(ADC_HandleTypeDef *hadc); +HAL_StatusTypeDef HAL_ADCEx_InjectedPollForConversion(ADC_HandleTypeDef *hadc, uint32_t Timeout); +HAL_StatusTypeDef HAL_ADCEx_InjectedStart_IT(ADC_HandleTypeDef *hadc); +HAL_StatusTypeDef HAL_ADCEx_InjectedStop_IT(ADC_HandleTypeDef *hadc); +uint32_t HAL_ADCEx_InjectedGetValue(ADC_HandleTypeDef *hadc, uint32_t InjectedRank); +HAL_StatusTypeDef HAL_ADCEx_MultiModeStart_DMA(ADC_HandleTypeDef *hadc, uint32_t *pData, uint32_t Length); +HAL_StatusTypeDef HAL_ADCEx_MultiModeStop_DMA(ADC_HandleTypeDef *hadc); +uint32_t HAL_ADCEx_MultiModeGetValue(ADC_HandleTypeDef *hadc); +void HAL_ADCEx_InjectedConvCpltCallback(ADC_HandleTypeDef *hadc); /* Peripheral Control functions *************************************************/ -HAL_StatusTypeDef HAL_ADCEx_InjectedConfigChannel(ADC_HandleTypeDef* hadc,ADC_InjectionConfTypeDef* sConfigInjected); -HAL_StatusTypeDef HAL_ADCEx_MultiModeConfigChannel(ADC_HandleTypeDef* hadc, ADC_MultiModeTypeDef* multimode); +HAL_StatusTypeDef HAL_ADCEx_InjectedConfigChannel(ADC_HandleTypeDef *hadc, ADC_InjectionConfTypeDef *sConfigInjected); +HAL_StatusTypeDef HAL_ADCEx_MultiModeConfigChannel(ADC_HandleTypeDef *hadc, ADC_MultiModeTypeDef *multimode); /** * @} diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_can.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_can.h index aa4a40d0..b4c229b2 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_can.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_can.h @@ -102,21 +102,25 @@ typedef struct { uint32_t FilterIdHigh; /*!< Specifies the filter identification number (MSBs for a 32-bit configuration, first one for a 16-bit configuration). - This parameter must be a number between Min_Data = 0x0000 and Max_Data = 0xFFFF. */ + This parameter must be a number between + Min_Data = 0x0000 and Max_Data = 0xFFFF. */ uint32_t FilterIdLow; /*!< Specifies the filter identification number (LSBs for a 32-bit configuration, second one for a 16-bit configuration). - This parameter must be a number between Min_Data = 0x0000 and Max_Data = 0xFFFF. */ + This parameter must be a number between + Min_Data = 0x0000 and Max_Data = 0xFFFF. */ uint32_t FilterMaskIdHigh; /*!< Specifies the filter mask number or identification number, according to the mode (MSBs for a 32-bit configuration, first one for a 16-bit configuration). - This parameter must be a number between Min_Data = 0x0000 and Max_Data = 0xFFFF. */ + This parameter must be a number between + Min_Data = 0x0000 and Max_Data = 0xFFFF. */ uint32_t FilterMaskIdLow; /*!< Specifies the filter mask number or identification number, according to the mode (LSBs for a 32-bit configuration, second one for a 16-bit configuration). - This parameter must be a number between Min_Data = 0x0000 and Max_Data = 0xFFFF. */ + This parameter must be a number between + Min_Data = 0x0000 and Max_Data = 0xFFFF. */ uint32_t FilterFIFOAssignment; /*!< Specifies the FIFO (0 or 1U) which will be assigned to the filter. This parameter can be a value of @ref CAN_filter_FIFO */ @@ -205,7 +209,11 @@ typedef struct /** * @brief CAN handle Structure definition */ +#if USE_HAL_CAN_REGISTER_CALLBACKS == 1 typedef struct __CAN_HandleTypeDef +#else +typedef struct +#endif /* USE_HAL_CAN_REGISTER_CALLBACKS */ { CAN_TypeDef *Instance; /*!< Register base address */ @@ -294,11 +302,11 @@ typedef void (*pCAN_CallbackTypeDef)(CAN_HandleTypeDef *hcan); /*!< pointer to #define HAL_CAN_ERROR_RX_FOV0 (0x00000200U) /*!< Rx FIFO0 overrun error */ #define HAL_CAN_ERROR_RX_FOV1 (0x00000400U) /*!< Rx FIFO1 overrun error */ #define HAL_CAN_ERROR_TX_ALST0 (0x00000800U) /*!< TxMailbox 0 transmit failure due to arbitration lost */ -#define HAL_CAN_ERROR_TX_TERR0 (0x00001000U) /*!< TxMailbox 0 transmit failure due to transmit error */ +#define HAL_CAN_ERROR_TX_TERR0 (0x00001000U) /*!< TxMailbox 0 transmit failure due to transmit error */ #define HAL_CAN_ERROR_TX_ALST1 (0x00002000U) /*!< TxMailbox 1 transmit failure due to arbitration lost */ -#define HAL_CAN_ERROR_TX_TERR1 (0x00004000U) /*!< TxMailbox 1 transmit failure due to transmit error */ +#define HAL_CAN_ERROR_TX_TERR1 (0x00004000U) /*!< TxMailbox 1 transmit failure due to transmit error */ #define HAL_CAN_ERROR_TX_ALST2 (0x00008000U) /*!< TxMailbox 2 transmit failure due to arbitration lost */ -#define HAL_CAN_ERROR_TX_TERR2 (0x00010000U) /*!< TxMailbox 2 transmit failure due to transmit error */ +#define HAL_CAN_ERROR_TX_TERR2 (0x00010000U) /*!< TxMailbox 2 transmit failure due to transmit error */ #define HAL_CAN_ERROR_TIMEOUT (0x00020000U) /*!< Timeout error */ #define HAL_CAN_ERROR_NOT_INITIALIZED (0x00040000U) /*!< Peripheral not initialized */ #define HAL_CAN_ERROR_NOT_READY (0x00080000U) /*!< Peripheral not ready */ @@ -329,7 +337,8 @@ typedef void (*pCAN_CallbackTypeDef)(CAN_HandleTypeDef *hcan); /*!< pointer to #define CAN_MODE_NORMAL (0x00000000U) /*!< Normal mode */ #define CAN_MODE_LOOPBACK ((uint32_t)CAN_BTR_LBKM) /*!< Loopback mode */ #define CAN_MODE_SILENT ((uint32_t)CAN_BTR_SILM) /*!< Silent mode */ -#define CAN_MODE_SILENT_LOOPBACK ((uint32_t)(CAN_BTR_LBKM | CAN_BTR_SILM)) /*!< Loopback combined with silent mode */ +#define CAN_MODE_SILENT_LOOPBACK ((uint32_t)(CAN_BTR_LBKM | CAN_BTR_SILM)) /*!< Loopback combined with + silent mode */ /** * @} */ @@ -644,7 +653,8 @@ void HAL_CAN_MspDeInit(CAN_HandleTypeDef *hcan); #if USE_HAL_CAN_REGISTER_CALLBACKS == 1 /* Callbacks Register/UnRegister functions ***********************************/ -HAL_StatusTypeDef HAL_CAN_RegisterCallback(CAN_HandleTypeDef *hcan, HAL_CAN_CallbackIDTypeDef CallbackID, void (* pCallback)(CAN_HandleTypeDef *_hcan)); +HAL_StatusTypeDef HAL_CAN_RegisterCallback(CAN_HandleTypeDef *hcan, HAL_CAN_CallbackIDTypeDef CallbackID, + void (* pCallback)(CAN_HandleTypeDef *_hcan)); HAL_StatusTypeDef HAL_CAN_UnRegisterCallback(CAN_HandleTypeDef *hcan, HAL_CAN_CallbackIDTypeDef CallbackID); #endif /* (USE_HAL_CAN_REGISTER_CALLBACKS) */ @@ -658,7 +668,7 @@ HAL_StatusTypeDef HAL_CAN_UnRegisterCallback(CAN_HandleTypeDef *hcan, HAL_CAN_Ca */ /* Configuration functions ****************************************************/ -HAL_StatusTypeDef HAL_CAN_ConfigFilter(CAN_HandleTypeDef *hcan, CAN_FilterTypeDef *sFilterConfig); +HAL_StatusTypeDef HAL_CAN_ConfigFilter(CAN_HandleTypeDef *hcan, const CAN_FilterTypeDef *sFilterConfig); /** * @} @@ -674,14 +684,16 @@ HAL_StatusTypeDef HAL_CAN_Start(CAN_HandleTypeDef *hcan); HAL_StatusTypeDef HAL_CAN_Stop(CAN_HandleTypeDef *hcan); HAL_StatusTypeDef HAL_CAN_RequestSleep(CAN_HandleTypeDef *hcan); HAL_StatusTypeDef HAL_CAN_WakeUp(CAN_HandleTypeDef *hcan); -uint32_t HAL_CAN_IsSleepActive(CAN_HandleTypeDef *hcan); -HAL_StatusTypeDef HAL_CAN_AddTxMessage(CAN_HandleTypeDef *hcan, CAN_TxHeaderTypeDef *pHeader, uint8_t aData[], uint32_t *pTxMailbox); +uint32_t HAL_CAN_IsSleepActive(const CAN_HandleTypeDef *hcan); +HAL_StatusTypeDef HAL_CAN_AddTxMessage(CAN_HandleTypeDef *hcan, const CAN_TxHeaderTypeDef *pHeader, + const uint8_t aData[], uint32_t *pTxMailbox); HAL_StatusTypeDef HAL_CAN_AbortTxRequest(CAN_HandleTypeDef *hcan, uint32_t TxMailboxes); -uint32_t HAL_CAN_GetTxMailboxesFreeLevel(CAN_HandleTypeDef *hcan); -uint32_t HAL_CAN_IsTxMessagePending(CAN_HandleTypeDef *hcan, uint32_t TxMailboxes); -uint32_t HAL_CAN_GetTxTimestamp(CAN_HandleTypeDef *hcan, uint32_t TxMailbox); -HAL_StatusTypeDef HAL_CAN_GetRxMessage(CAN_HandleTypeDef *hcan, uint32_t RxFifo, CAN_RxHeaderTypeDef *pHeader, uint8_t aData[]); -uint32_t HAL_CAN_GetRxFifoFillLevel(CAN_HandleTypeDef *hcan, uint32_t RxFifo); +uint32_t HAL_CAN_GetTxMailboxesFreeLevel(const CAN_HandleTypeDef *hcan); +uint32_t HAL_CAN_IsTxMessagePending(const CAN_HandleTypeDef *hcan, uint32_t TxMailboxes); +uint32_t HAL_CAN_GetTxTimestamp(const CAN_HandleTypeDef *hcan, uint32_t TxMailbox); +HAL_StatusTypeDef HAL_CAN_GetRxMessage(CAN_HandleTypeDef *hcan, uint32_t RxFifo, + CAN_RxHeaderTypeDef *pHeader, uint8_t aData[]); +uint32_t HAL_CAN_GetRxFifoFillLevel(const CAN_HandleTypeDef *hcan, uint32_t RxFifo); /** * @} @@ -729,8 +741,8 @@ void HAL_CAN_ErrorCallback(CAN_HandleTypeDef *hcan); * @{ */ /* Peripheral State and Error functions ***************************************/ -HAL_CAN_StateTypeDef HAL_CAN_GetState(CAN_HandleTypeDef *hcan); -uint32_t HAL_CAN_GetError(CAN_HandleTypeDef *hcan); +HAL_CAN_StateTypeDef HAL_CAN_GetState(const CAN_HandleTypeDef *hcan); +uint32_t HAL_CAN_GetError(const CAN_HandleTypeDef *hcan); HAL_StatusTypeDef HAL_CAN_ResetError(CAN_HandleTypeDef *hcan); /** @@ -806,7 +818,8 @@ HAL_StatusTypeDef HAL_CAN_ResetError(CAN_HandleTypeDef *hcan); #define IS_CAN_TX_MAILBOX(TRANSMITMAILBOX) (((TRANSMITMAILBOX) == CAN_TX_MAILBOX0 ) || \ ((TRANSMITMAILBOX) == CAN_TX_MAILBOX1 ) || \ ((TRANSMITMAILBOX) == CAN_TX_MAILBOX2 )) -#define IS_CAN_TX_MAILBOX_LIST(TRANSMITMAILBOX) ((TRANSMITMAILBOX) <= (CAN_TX_MAILBOX0 | CAN_TX_MAILBOX1 | CAN_TX_MAILBOX2)) +#define IS_CAN_TX_MAILBOX_LIST(TRANSMITMAILBOX) ((TRANSMITMAILBOX) <= (CAN_TX_MAILBOX0 | CAN_TX_MAILBOX1 | \ + CAN_TX_MAILBOX2)) #define IS_CAN_STDID(STDID) ((STDID) <= 0x7FFU) #define IS_CAN_EXTID(EXTID) ((EXTID) <= 0x1FFFFFFFU) #define IS_CAN_DLC(DLC) ((DLC) <= 8U) diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_cec.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_cec.h index 9d6c226a..2abdc3b5 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_cec.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_cec.h @@ -48,61 +48,70 @@ extern "C" { typedef struct { uint32_t SignalFreeTime; /*!< Set SFT field, specifies the Signal Free Time. - It can be one of @ref CEC_Signal_Free_Time + It can be one of CEC_Signal_Free_Time and belongs to the set {0,...,7} where 0x0 is the default configuration else means 0.5 + (SignalFreeTime - 1) nominal data bit periods */ uint32_t Tolerance; /*!< Set RXTOL bit, specifies the tolerance accepted on the received waveforms, - it can be a value of @ref CEC_Tolerance : it is either CEC_STANDARD_TOLERANCE - or CEC_EXTENDED_TOLERANCE */ + it can be a value of CEC_Tolerance : + it is either CEC_STANDARD_TOLERANCE or CEC_EXTENDED_TOLERANCE */ - uint32_t BRERxStop; /*!< Set BRESTP bit @ref CEC_BRERxStop : specifies whether or not a Bit Rising Error stops the reception. + uint32_t BRERxStop; /*!< Set BRESTP bit CEC_BRERxStop : specifies whether or not a Bit Rising + Error stops the reception. CEC_NO_RX_STOP_ON_BRE: reception is not stopped. CEC_RX_STOP_ON_BRE: reception is stopped. */ - uint32_t BREErrorBitGen; /*!< Set BREGEN bit @ref CEC_BREErrorBitGen : specifies whether or not an Error-Bit is generated on the + uint32_t BREErrorBitGen; /*!< Set BREGEN bit CEC_BREErrorBitGen : specifies whether or not an + Error-Bit is generated on the CEC line upon Bit Rising Error detection. CEC_BRE_ERRORBIT_NO_GENERATION: no error-bit generation. CEC_BRE_ERRORBIT_GENERATION: error-bit generation if BRESTP is set. */ - uint32_t LBPEErrorBitGen; /*!< Set LBPEGEN bit @ref CEC_LBPEErrorBitGen : specifies whether or not an Error-Bit is generated on the + uint32_t LBPEErrorBitGen; /*!< Set LBPEGEN bit CEC_LBPEErrorBitGen : specifies whether or not an + Error-Bit is generated on the CEC line upon Long Bit Period Error detection. CEC_LBPE_ERRORBIT_NO_GENERATION: no error-bit generation. CEC_LBPE_ERRORBIT_GENERATION: error-bit generation. */ - uint32_t BroadcastMsgNoErrorBitGen; /*!< Set BRDNOGEN bit @ref CEC_BroadCastMsgErrorBitGen : allows to avoid an Error-Bit generation on the CEC line + uint32_t BroadcastMsgNoErrorBitGen; /*!< Set BRDNOGEN bit CEC_BroadCastMsgErrorBitGen : allows to avoid an + Error-Bit generation on the CEC line upon an error detected on a broadcast message. - It supersedes BREGEN and LBPEGEN bits for a broadcast message error handling. It can take two values: + It supersedes BREGEN and LBPEGEN bits for a broadcast message error + handling. It can take two values: 1) CEC_BROADCASTERROR_ERRORBIT_GENERATION. - a) BRE detection: error-bit generation on the CEC line if BRESTP=CEC_RX_STOP_ON_BRE - and BREGEN=CEC_BRE_ERRORBIT_NO_GENERATION. + a) BRE detection: error-bit generation on the CEC line if + BRESTP=CEC_RX_STOP_ON_BRE and BREGEN=CEC_BRE_ERRORBIT_NO_GENERATION. b) LBPE detection: error-bit generation on the CEC line if LBPGEN=CEC_LBPE_ERRORBIT_NO_GENERATION. 2) CEC_BROADCASTERROR_NO_ERRORBIT_GENERATION. - no error-bit generation in case neither a) nor b) are satisfied. Additionally, - there is no error-bit generation in case of Short Bit Period Error detection in - a broadcast message while LSTN bit is set. */ + no error-bit generation in case neither a) nor b) are satisfied. + Additionally, there is no error-bit generation in case of Short Bit + Period Error detection in a broadcast message while LSTN bit is set. */ - uint32_t SignalFreeTimeOption; /*!< Set SFTOP bit @ref CEC_SFT_Option : specifies when SFT timer starts. + uint32_t SignalFreeTimeOption; /*!< Set SFTOP bit CEC_SFT_Option : specifies when SFT timer starts. CEC_SFT_START_ON_TXSOM SFT: timer starts when TXSOM is set by software. - CEC_SFT_START_ON_TX_RX_END: SFT timer starts automatically at the end of message transmission/reception. */ + CEC_SFT_START_ON_TX_RX_END: SFT timer starts automatically at the end + of message transmission/reception. */ - uint32_t ListenMode; /*!< Set LSTN bit @ref CEC_Listening_Mode : specifies device listening mode. It can take two values: + uint32_t ListenMode; /*!< Set LSTN bit CEC_Listening_Mode : specifies device listening mode. + It can take two values: - CEC_REDUCED_LISTENING_MODE: CEC peripheral receives only message addressed to its - own address (OAR). Messages addressed to different destination are ignored. + CEC_REDUCED_LISTENING_MODE: CEC peripheral receives only message addressed + to its own address (OAR). Messages addressed to different destination + are ignored. Broadcast messages are always received. - CEC_FULL_LISTENING_MODE: CEC peripheral receives messages addressed to its own - address (OAR) with positive acknowledge. Messages addressed to different destination - are received, but without interfering with the CEC bus: no acknowledge sent. */ + CEC_FULL_LISTENING_MODE: CEC peripheral receives messages addressed to its + own address (OAR) with positive acknowledge. Messages addressed to + different destination are received, but without interfering with the + CEC bus: no acknowledge sent. */ - uint16_t OwnAddress; /*!< Own addresses configuration - This parameter can be a value of @ref CEC_OWN_ADDRESS */ + uint16_t OwnAddress; /*!< Own addresses configuration + This parameter can be a value of CEC_OWN_ADDRESS */ uint8_t *RxBuffer; /*!< CEC Rx buffer pointer */ @@ -111,7 +120,8 @@ typedef struct /** * @brief HAL CEC State definition - * @note HAL CEC State value is a combination of 2 different substates: gState and RxState (see @ref CEC_State_Definition). + * @note HAL CEC State value is a combination of 2 different substates: gState and RxState + (see CEC_State_Definition). * - gState contains CEC state information related to global Handle management * and also information related to Tx operations. * gState value coding follow below described bitmap : @@ -159,37 +169,37 @@ typedef struct __CEC_HandleTypeDef typedef struct #endif /* USE_HAL_CEC_REGISTER_CALLBACKS */ { - CEC_TypeDef *Instance; /*!< CEC registers base address */ + CEC_TypeDef *Instance; /*!< CEC registers base address */ - CEC_InitTypeDef Init; /*!< CEC communication parameters */ + CEC_InitTypeDef Init; /*!< CEC communication parameters */ - uint8_t *pTxBuffPtr; /*!< Pointer to CEC Tx transfer Buffer */ + const uint8_t *pTxBuffPtr; /*!< Pointer to CEC Tx transfer Buffer */ - uint16_t TxXferCount; /*!< CEC Tx Transfer Counter */ + uint16_t TxXferCount; /*!< CEC Tx Transfer Counter */ - uint16_t RxXferSize; /*!< CEC Rx Transfer size, 0: header received only */ + uint16_t RxXferSize; /*!< CEC Rx Transfer size, 0: header received only */ - HAL_LockTypeDef Lock; /*!< Locking object */ + HAL_LockTypeDef Lock; /*!< Locking object */ HAL_CEC_StateTypeDef gState; /*!< CEC state information related to global Handle management and also related to Tx operations. - This parameter can be a value of @ref HAL_CEC_StateTypeDef */ + This parameter can be a value of HAL_CEC_StateTypeDef */ HAL_CEC_StateTypeDef RxState; /*!< CEC state information related to Rx operations. - This parameter can be a value of @ref HAL_CEC_StateTypeDef */ + This parameter can be a value of HAL_CEC_StateTypeDef */ uint32_t ErrorCode; /*!< For errors handling purposes, copy of ISR register - in case error is reported */ + in case error is reported */ #if (USE_HAL_CEC_REGISTER_CALLBACKS == 1) void (* TxCpltCallback)(struct __CEC_HandleTypeDef - *hcec); /*!< CEC Tx Transfer completed callback */ + *hcec); /*!< CEC Tx Transfer completed callback */ void (* RxCpltCallback)(struct __CEC_HandleTypeDef *hcec, - uint32_t RxFrameSize); /*!< CEC Rx Transfer completed callback */ - void (* ErrorCallback)(struct __CEC_HandleTypeDef *hcec); /*!< CEC error callback */ + uint32_t RxFrameSize); /*!< CEC Rx Transfer completed callback */ + void (* ErrorCallback)(struct __CEC_HandleTypeDef *hcec); /*!< CEC error callback */ - void (* MspInitCallback)(struct __CEC_HandleTypeDef *hcec); /*!< CEC Msp Init callback */ - void (* MspDeInitCallback)(struct __CEC_HandleTypeDef *hcec); /*!< CEC Msp DeInit callback */ + void (* MspInitCallback)(struct __CEC_HandleTypeDef *hcec); /*!< CEC Msp Init callback */ + void (* MspDeInitCallback)(struct __CEC_HandleTypeDef *hcec); /*!< CEC Msp DeInit callback */ #endif /* (USE_HAL_CEC_REGISTER_CALLBACKS) */ } CEC_HandleTypeDef; @@ -202,7 +212,7 @@ typedef enum { HAL_CEC_TX_CPLT_CB_ID = 0x00U, /*!< CEC Tx Transfer completed callback ID */ HAL_CEC_RX_CPLT_CB_ID = 0x01U, /*!< CEC Rx Transfer completed callback ID */ - HAL_CEC_ERROR_CB_ID = 0x02U, /*!< CEC error callback ID */ + HAL_CEC_ERROR_CB_ID = 0x02U, /*!< CEC error callback ID */ HAL_CEC_MSPINIT_CB_ID = 0x03U, /*!< CEC Msp Init callback ID */ HAL_CEC_MSPDEINIT_CB_ID = 0x04U /*!< CEC Msp DeInit callback ID */ } HAL_CEC_CallbackIDTypeDef; @@ -212,7 +222,8 @@ typedef enum */ typedef void (*pCEC_CallbackTypeDef)(CEC_HandleTypeDef *hcec); /*!< pointer to an CEC callback function */ typedef void (*pCEC_RxCallbackTypeDef)(CEC_HandleTypeDef *hcec, - uint32_t RxFrameSize); /*!< pointer to an Rx Transfer completed callback function */ + uint32_t RxFrameSize); /*!< pointer to an Rx Transfer completed + callback function */ #endif /* USE_HAL_CEC_REGISTER_CALLBACKS */ /** * @} @@ -358,16 +369,16 @@ typedef void (*pCEC_RxCallbackTypeDef)(CEC_HandleTypeDef *hcec, /** @defgroup CEC_OWN_ADDRESS CEC Own Address * @{ */ -#define CEC_OWN_ADDRESS_NONE ((uint16_t) 0x0000U) /* Reset value */ -#define CEC_OWN_ADDRESS_0 ((uint16_t) 0x0001U) /* Logical Address 0 */ -#define CEC_OWN_ADDRESS_1 ((uint16_t) 0x0002U) /* Logical Address 1 */ -#define CEC_OWN_ADDRESS_2 ((uint16_t) 0x0004U) /* Logical Address 2 */ -#define CEC_OWN_ADDRESS_3 ((uint16_t) 0x0008U) /* Logical Address 3 */ -#define CEC_OWN_ADDRESS_4 ((uint16_t) 0x0010U) /* Logical Address 4 */ -#define CEC_OWN_ADDRESS_5 ((uint16_t) 0x0020U) /* Logical Address 5 */ -#define CEC_OWN_ADDRESS_6 ((uint16_t) 0x0040U) /* Logical Address 6 */ -#define CEC_OWN_ADDRESS_7 ((uint16_t) 0x0080U) /* Logical Address 7 */ -#define CEC_OWN_ADDRESS_8 ((uint16_t) 0x0100U) /* Logical Address 9 */ +#define CEC_OWN_ADDRESS_NONE ((uint16_t) 0x0000U) /* Reset value */ +#define CEC_OWN_ADDRESS_0 ((uint16_t) 0x0001U) /* Logical Address 0 */ +#define CEC_OWN_ADDRESS_1 ((uint16_t) 0x0002U) /* Logical Address 1 */ +#define CEC_OWN_ADDRESS_2 ((uint16_t) 0x0004U) /* Logical Address 2 */ +#define CEC_OWN_ADDRESS_3 ((uint16_t) 0x0008U) /* Logical Address 3 */ +#define CEC_OWN_ADDRESS_4 ((uint16_t) 0x0010U) /* Logical Address 4 */ +#define CEC_OWN_ADDRESS_5 ((uint16_t) 0x0020U) /* Logical Address 5 */ +#define CEC_OWN_ADDRESS_6 ((uint16_t) 0x0040U) /* Logical Address 6 */ +#define CEC_OWN_ADDRESS_7 ((uint16_t) 0x0080U) /* Logical Address 7 */ +#define CEC_OWN_ADDRESS_8 ((uint16_t) 0x0100U) /* Logical Address 9 */ #define CEC_OWN_ADDRESS_9 ((uint16_t) 0x0200U) /* Logical Address 10 */ #define CEC_OWN_ADDRESS_10 ((uint16_t) 0x0400U) /* Logical Address 11 */ #define CEC_OWN_ADDRESS_11 ((uint16_t) 0x0800U) /* Logical Address 12 */ @@ -421,8 +432,8 @@ typedef void (*pCEC_RxCallbackTypeDef)(CEC_HandleTypeDef *hcec, /** @defgroup CEC_ALL_ERROR CEC all RX or TX errors flags * @{ */ -#define CEC_ISR_ALL_ERROR ((uint32_t)CEC_ISR_RXOVR|CEC_ISR_BRE|CEC_ISR_SBPE|CEC_ISR_LBPE|CEC_ISR_RXACKE|\ - CEC_ISR_ARBLST|CEC_ISR_TXUDR|CEC_ISR_TXERR|CEC_ISR_TXACKE) +#define CEC_ISR_ALL_ERROR ((uint32_t)CEC_ISR_RXOVR|CEC_ISR_BRE|CEC_ISR_SBPE|CEC_ISR_LBPE|CEC_ISR_RXACKE|\ + CEC_ISR_ARBLST|CEC_ISR_TXUDR|CEC_ISR_TXERR|CEC_ISR_TXACKE) /** * @} */ @@ -430,7 +441,7 @@ typedef void (*pCEC_RxCallbackTypeDef)(CEC_HandleTypeDef *hcec, /** @defgroup CEC_IER_ALL_RX CEC all RX errors interrupts enabling flag * @{ */ -#define CEC_IER_RX_ALL_ERR ((uint32_t)CEC_IER_RXACKEIE|CEC_IER_LBPEIE|CEC_IER_SBPEIE|CEC_IER_BREIE|CEC_IER_RXOVRIE) +#define CEC_IER_RX_ALL_ERR ((uint32_t)CEC_IER_RXACKEIE|CEC_IER_LBPEIE|CEC_IER_SBPEIE|CEC_IER_BREIE|CEC_IER_RXOVRIE) /** * @} */ @@ -438,7 +449,7 @@ typedef void (*pCEC_RxCallbackTypeDef)(CEC_HandleTypeDef *hcec, /** @defgroup CEC_IER_ALL_TX CEC all TX errors interrupts enabling flag * @{ */ -#define CEC_IER_TX_ALL_ERR ((uint32_t)CEC_IER_TXACKEIE|CEC_IER_TXERRIE|CEC_IER_TXUDRIE|CEC_IER_ARBLSTIE) +#define CEC_IER_TX_ALL_ERR ((uint32_t)CEC_IER_TXACKEIE|CEC_IER_TXERRIE|CEC_IER_TXUDRIE|CEC_IER_ARBLSTIE) /** * @} */ @@ -622,7 +633,8 @@ typedef void (*pCEC_RxCallbackTypeDef)(CEC_HandleTypeDef *hcec, * @param __ADDRESS__ Own Address value (CEC logical address is identified by bit position) * @retval none */ -#define __HAL_CEC_SET_OAR(__HANDLE__,__ADDRESS__) SET_BIT((__HANDLE__)->Instance->CFGR, (__ADDRESS__)<< CEC_CFGR_OAR_LSB_POS) +#define __HAL_CEC_SET_OAR(__HANDLE__,__ADDRESS__) SET_BIT((__HANDLE__)->Instance->CFGR, \ + (__ADDRESS__)<< CEC_CFGR_OAR_LSB_POS) /** * @} @@ -660,8 +672,8 @@ HAL_StatusTypeDef HAL_CEC_UnRegisterRxCpltCallback(CEC_HandleTypeDef *hcec); */ /* I/O operation functions ***************************************************/ HAL_StatusTypeDef HAL_CEC_Transmit_IT(CEC_HandleTypeDef *hcec, uint8_t InitiatorAddress, uint8_t DestinationAddress, - uint8_t *pData, uint32_t Size); -uint32_t HAL_CEC_GetLastReceivedFrameSize(CEC_HandleTypeDef *hcec); + const uint8_t *pData, uint32_t Size); +uint32_t HAL_CEC_GetLastReceivedFrameSize(const CEC_HandleTypeDef *hcec); void HAL_CEC_ChangeRxBuffer(CEC_HandleTypeDef *hcec, uint8_t *Rxbuffer); void HAL_CEC_IRQHandler(CEC_HandleTypeDef *hcec); void HAL_CEC_TxCpltCallback(CEC_HandleTypeDef *hcec); @@ -675,8 +687,8 @@ void HAL_CEC_ErrorCallback(CEC_HandleTypeDef *hcec); * @{ */ /* Peripheral State functions ************************************************/ -HAL_CEC_StateTypeDef HAL_CEC_GetState(CEC_HandleTypeDef *hcec); -uint32_t HAL_CEC_GetError(CEC_HandleTypeDef *hcec); +HAL_CEC_StateTypeDef HAL_CEC_GetState(const CEC_HandleTypeDef *hcec); +uint32_t HAL_CEC_GetError(const CEC_HandleTypeDef *hcec); /** * @} */ @@ -731,8 +743,9 @@ uint32_t HAL_CEC_GetError(CEC_HandleTypeDef *hcec); #define IS_CEC_LBPEERRORBITGEN(__ERRORBITGEN__) (((__ERRORBITGEN__) == CEC_LBPE_ERRORBIT_NO_GENERATION) || \ ((__ERRORBITGEN__) == CEC_LBPE_ERRORBIT_GENERATION)) -#define IS_CEC_BROADCASTERROR_NO_ERRORBIT_GENERATION(__ERRORBITGEN__) (((__ERRORBITGEN__) == CEC_BROADCASTERROR_ERRORBIT_GENERATION) || \ - ((__ERRORBITGEN__) == CEC_BROADCASTERROR_NO_ERRORBIT_GENERATION)) +#define IS_CEC_BROADCASTERROR_NO_ERRORBIT_GENERATION(__ERRORBITGEN__) \ + (((__ERRORBITGEN__) == CEC_BROADCASTERROR_ERRORBIT_GENERATION) || \ + ((__ERRORBITGEN__) == CEC_BROADCASTERROR_NO_ERRORBIT_GENERATION)) #define IS_CEC_SFTOP(__SFTOP__) (((__SFTOP__) == CEC_SFT_START_ON_TXSOM) || \ ((__SFTOP__) == CEC_SFT_START_ON_TX_RX_END)) @@ -789,4 +802,3 @@ uint32_t HAL_CEC_GetError(CEC_HandleTypeDef *hcec); #endif #endif /* STM32F4xxHAL_CEC_H */ - diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_cortex.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_cortex.h index 8643779a..51ab3678 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_cortex.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_cortex.h @@ -286,6 +286,7 @@ void HAL_MPU_Enable(uint32_t MPU_Control); void HAL_MPU_Disable(void); void HAL_MPU_ConfigRegion(MPU_Region_InitTypeDef *MPU_Init); #endif /* __MPU_PRESENT */ +void HAL_CORTEX_ClearEvent(void); /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_crc.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_crc.h index 41edbe38..ac36ed8c 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_crc.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_crc.h @@ -157,7 +157,7 @@ uint32_t HAL_CRC_Calculate(CRC_HandleTypeDef *hcrc, uint32_t pBuffer[], uint32_t /** @defgroup CRC_Exported_Functions_Group3 Peripheral State functions * @{ */ -HAL_CRC_StateTypeDef HAL_CRC_GetState(CRC_HandleTypeDef *hcrc); +HAL_CRC_StateTypeDef HAL_CRC_GetState(const CRC_HandleTypeDef *hcrc); /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dac.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dac.h index a79ca73a..3ee217af 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dac.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dac.h @@ -78,19 +78,19 @@ typedef struct __IO uint32_t ErrorCode; /*!< DAC Error code */ #if (USE_HAL_DAC_REGISTER_CALLBACKS == 1) - void (* ConvCpltCallbackCh1) (struct __DAC_HandleTypeDef *hdac); - void (* ConvHalfCpltCallbackCh1) (struct __DAC_HandleTypeDef *hdac); - void (* ErrorCallbackCh1) (struct __DAC_HandleTypeDef *hdac); - void (* DMAUnderrunCallbackCh1) (struct __DAC_HandleTypeDef *hdac); + void (* ConvCpltCallbackCh1)(struct __DAC_HandleTypeDef *hdac); + void (* ConvHalfCpltCallbackCh1)(struct __DAC_HandleTypeDef *hdac); + void (* ErrorCallbackCh1)(struct __DAC_HandleTypeDef *hdac); + void (* DMAUnderrunCallbackCh1)(struct __DAC_HandleTypeDef *hdac); #if defined(DAC_CHANNEL2_SUPPORT) - void (* ConvCpltCallbackCh2) (struct __DAC_HandleTypeDef *hdac); - void (* ConvHalfCpltCallbackCh2) (struct __DAC_HandleTypeDef *hdac); - void (* ErrorCallbackCh2) (struct __DAC_HandleTypeDef *hdac); - void (* DMAUnderrunCallbackCh2) (struct __DAC_HandleTypeDef *hdac); + void (* ConvCpltCallbackCh2)(struct __DAC_HandleTypeDef *hdac); + void (* ConvHalfCpltCallbackCh2)(struct __DAC_HandleTypeDef *hdac); + void (* ErrorCallbackCh2)(struct __DAC_HandleTypeDef *hdac); + void (* DMAUnderrunCallbackCh2)(struct __DAC_HandleTypeDef *hdac); #endif /* DAC_CHANNEL2_SUPPORT */ - void (* MspInitCallback) (struct __DAC_HandleTypeDef *hdac); - void (* MspDeInitCallback) (struct __DAC_HandleTypeDef *hdac); + void (* MspInitCallback)(struct __DAC_HandleTypeDef *hdac); + void (* MspDeInitCallback)(struct __DAC_HandleTypeDef *hdac); #endif /* USE_HAL_DAC_REGISTER_CALLBACKS */ } DAC_HandleTypeDef; @@ -404,7 +404,7 @@ void HAL_DAC_MspDeInit(DAC_HandleTypeDef *hdac); /* IO operation functions *****************************************************/ HAL_StatusTypeDef HAL_DAC_Start(DAC_HandleTypeDef *hdac, uint32_t Channel); HAL_StatusTypeDef HAL_DAC_Stop(DAC_HandleTypeDef *hdac, uint32_t Channel); -HAL_StatusTypeDef HAL_DAC_Start_DMA(DAC_HandleTypeDef *hdac, uint32_t Channel, uint32_t *pData, uint32_t Length, +HAL_StatusTypeDef HAL_DAC_Start_DMA(DAC_HandleTypeDef *hdac, uint32_t Channel, const uint32_t *pData, uint32_t Length, uint32_t Alignment); HAL_StatusTypeDef HAL_DAC_Stop_DMA(DAC_HandleTypeDef *hdac, uint32_t Channel); void HAL_DAC_IRQHandler(DAC_HandleTypeDef *hdac); @@ -430,8 +430,9 @@ HAL_StatusTypeDef HAL_DAC_UnRegisterCallback(DAC_HandleTypeDef *hdac, HAL_DA * @{ */ /* Peripheral Control functions ***********************************************/ -uint32_t HAL_DAC_GetValue(DAC_HandleTypeDef *hdac, uint32_t Channel); -HAL_StatusTypeDef HAL_DAC_ConfigChannel(DAC_HandleTypeDef *hdac, DAC_ChannelConfTypeDef *sConfig, uint32_t Channel); +uint32_t HAL_DAC_GetValue(const DAC_HandleTypeDef *hdac, uint32_t Channel); +HAL_StatusTypeDef HAL_DAC_ConfigChannel(DAC_HandleTypeDef *hdac, + const DAC_ChannelConfTypeDef *sConfig, uint32_t Channel); /** * @} */ @@ -440,8 +441,8 @@ HAL_StatusTypeDef HAL_DAC_ConfigChannel(DAC_HandleTypeDef *hdac, DAC_ChannelConf * @{ */ /* Peripheral State and Error functions ***************************************/ -HAL_DAC_StateTypeDef HAL_DAC_GetState(DAC_HandleTypeDef *hdac); -uint32_t HAL_DAC_GetError(DAC_HandleTypeDef *hdac); +HAL_DAC_StateTypeDef HAL_DAC_GetState(const DAC_HandleTypeDef *hdac); +uint32_t HAL_DAC_GetError(const DAC_HandleTypeDef *hdac); /** * @} @@ -477,4 +478,3 @@ void DAC_DMAHalfConvCpltCh1(DMA_HandleTypeDef *hdma); #endif /* STM32F4xx_HAL_DAC_H */ - diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dac_ex.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dac_ex.h index db109902..1bb5ce45 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dac_ex.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dac_ex.h @@ -81,6 +81,7 @@ extern "C" { * @} */ + /** * @} */ @@ -147,19 +148,18 @@ HAL_StatusTypeDef HAL_DACEx_TriangleWaveGenerate(DAC_HandleTypeDef *hdac, uint32 HAL_StatusTypeDef HAL_DACEx_NoiseWaveGenerate(DAC_HandleTypeDef *hdac, uint32_t Channel, uint32_t Amplitude); #if defined(DAC_CHANNEL2_SUPPORT) -#endif HAL_StatusTypeDef HAL_DACEx_DualStart(DAC_HandleTypeDef *hdac); HAL_StatusTypeDef HAL_DACEx_DualStop(DAC_HandleTypeDef *hdac); HAL_StatusTypeDef HAL_DACEx_DualSetValue(DAC_HandleTypeDef *hdac, uint32_t Alignment, uint32_t Data1, uint32_t Data2); -uint32_t HAL_DACEx_DualGetValue(DAC_HandleTypeDef *hdac); +uint32_t HAL_DACEx_DualGetValue(const DAC_HandleTypeDef *hdac); +#endif /* DAC_CHANNEL2_SUPPORT */ #if defined(DAC_CHANNEL2_SUPPORT) -#endif void HAL_DACEx_ConvCpltCallbackCh2(DAC_HandleTypeDef *hdac); void HAL_DACEx_ConvHalfCpltCallbackCh2(DAC_HandleTypeDef *hdac); void HAL_DACEx_ErrorCallbackCh2(DAC_HandleTypeDef *hdac); void HAL_DACEx_DMAUnderrunCallbackCh2(DAC_HandleTypeDef *hdac); - +#endif /* DAC_CHANNEL2_SUPPORT */ /** * @} @@ -202,4 +202,3 @@ void DAC_DMAHalfConvCpltCh2(DMA_HandleTypeDef *hdma); #endif #endif /* STM32F4xx_HAL_DAC_EX_H */ - diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_def.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_def.h index 59720256..7c2f96fb 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_def.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_def.h @@ -54,7 +54,9 @@ typedef enum /* Exported macro ------------------------------------------------------------*/ +#if !defined(UNUSED) #define UNUSED(X) (void)X /* To avoid gcc/g++ warnings */ +#endif /* UNUSED */ #define HAL_MAX_DELAY 0xFFFFFFFFU diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dfsdm.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dfsdm.h index ad39ff6c..2b0f1937 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dfsdm.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dfsdm.h @@ -811,11 +811,11 @@ HAL_StatusTypeDef HAL_DFSDM_ChannelScdStart_IT(DFSDM_Channel_HandleTypeDef *hdfs HAL_StatusTypeDef HAL_DFSDM_ChannelScdStop(DFSDM_Channel_HandleTypeDef *hdfsdm_channel); HAL_StatusTypeDef HAL_DFSDM_ChannelScdStop_IT(DFSDM_Channel_HandleTypeDef *hdfsdm_channel); -int16_t HAL_DFSDM_ChannelGetAwdValue(DFSDM_Channel_HandleTypeDef *hdfsdm_channel); +int16_t HAL_DFSDM_ChannelGetAwdValue(const DFSDM_Channel_HandleTypeDef *hdfsdm_channel); HAL_StatusTypeDef HAL_DFSDM_ChannelModifyOffset(DFSDM_Channel_HandleTypeDef *hdfsdm_channel, int32_t Offset); -HAL_StatusTypeDef HAL_DFSDM_ChannelPollForCkab(DFSDM_Channel_HandleTypeDef *hdfsdm_channel, uint32_t Timeout); -HAL_StatusTypeDef HAL_DFSDM_ChannelPollForScd(DFSDM_Channel_HandleTypeDef *hdfsdm_channel, uint32_t Timeout); +HAL_StatusTypeDef HAL_DFSDM_ChannelPollForCkab(const DFSDM_Channel_HandleTypeDef *hdfsdm_channel, uint32_t Timeout); +HAL_StatusTypeDef HAL_DFSDM_ChannelPollForScd(const DFSDM_Channel_HandleTypeDef *hdfsdm_channel, uint32_t Timeout); void HAL_DFSDM_ChannelCkabCallback(DFSDM_Channel_HandleTypeDef *hdfsdm_channel); void HAL_DFSDM_ChannelScdCallback(DFSDM_Channel_HandleTypeDef *hdfsdm_channel); @@ -827,7 +827,7 @@ void HAL_DFSDM_ChannelScdCallback(DFSDM_Channel_HandleTypeDef *hdfsdm_channel); * @{ */ /* Channel state function *****************************************************/ -HAL_DFSDM_Channel_StateTypeDef HAL_DFSDM_ChannelGetState(DFSDM_Channel_HandleTypeDef *hdfsdm_channel); +HAL_DFSDM_Channel_StateTypeDef HAL_DFSDM_ChannelGetState(const DFSDM_Channel_HandleTypeDef *hdfsdm_channel); /** * @} */ @@ -887,16 +887,16 @@ HAL_StatusTypeDef HAL_DFSDM_FilterInjectedStop(DFSDM_Filter_HandleTypeDef *hdfsd HAL_StatusTypeDef HAL_DFSDM_FilterInjectedStop_IT(DFSDM_Filter_HandleTypeDef *hdfsdm_filter); HAL_StatusTypeDef HAL_DFSDM_FilterInjectedStop_DMA(DFSDM_Filter_HandleTypeDef *hdfsdm_filter); HAL_StatusTypeDef HAL_DFSDM_FilterAwdStart_IT(DFSDM_Filter_HandleTypeDef *hdfsdm_filter, - DFSDM_Filter_AwdParamTypeDef* awdParam); + const DFSDM_Filter_AwdParamTypeDef* awdParam); HAL_StatusTypeDef HAL_DFSDM_FilterAwdStop_IT(DFSDM_Filter_HandleTypeDef *hdfsdm_filter); HAL_StatusTypeDef HAL_DFSDM_FilterExdStart(DFSDM_Filter_HandleTypeDef *hdfsdm_filter, uint32_t Channel); HAL_StatusTypeDef HAL_DFSDM_FilterExdStop(DFSDM_Filter_HandleTypeDef *hdfsdm_filter); -int32_t HAL_DFSDM_FilterGetRegularValue(DFSDM_Filter_HandleTypeDef *hdfsdm_filter, uint32_t* Channel); -int32_t HAL_DFSDM_FilterGetInjectedValue(DFSDM_Filter_HandleTypeDef *hdfsdm_filter, uint32_t* Channel); -int32_t HAL_DFSDM_FilterGetExdMaxValue(DFSDM_Filter_HandleTypeDef *hdfsdm_filter, uint32_t* Channel); -int32_t HAL_DFSDM_FilterGetExdMinValue(DFSDM_Filter_HandleTypeDef *hdfsdm_filter, uint32_t* Channel); -uint32_t HAL_DFSDM_FilterGetConvTimeValue(DFSDM_Filter_HandleTypeDef *hdfsdm_filter); +int32_t HAL_DFSDM_FilterGetRegularValue(const DFSDM_Filter_HandleTypeDef *hdfsdm_filter, uint32_t* Channel); +int32_t HAL_DFSDM_FilterGetInjectedValue(const DFSDM_Filter_HandleTypeDef *hdfsdm_filter, uint32_t* Channel); +int32_t HAL_DFSDM_FilterGetExdMaxValue(const DFSDM_Filter_HandleTypeDef *hdfsdm_filter, uint32_t* Channel); +int32_t HAL_DFSDM_FilterGetExdMinValue(const DFSDM_Filter_HandleTypeDef *hdfsdm_filter, uint32_t* Channel); +uint32_t HAL_DFSDM_FilterGetConvTimeValue(const DFSDM_Filter_HandleTypeDef *hdfsdm_filter); void HAL_DFSDM_IRQHandler(DFSDM_Filter_HandleTypeDef *hdfsdm_filter); @@ -917,8 +917,8 @@ void HAL_DFSDM_FilterErrorCallback(DFSDM_Filter_HandleTypeDef *hdfsdm_filter); * @{ */ /* Filter state functions *****************************************************/ -HAL_DFSDM_Filter_StateTypeDef HAL_DFSDM_FilterGetState(DFSDM_Filter_HandleTypeDef *hdfsdm_filter); -uint32_t HAL_DFSDM_FilterGetError(DFSDM_Filter_HandleTypeDef *hdfsdm_filter); +HAL_DFSDM_Filter_StateTypeDef HAL_DFSDM_FilterGetState(const DFSDM_Filter_HandleTypeDef *hdfsdm_filter); +uint32_t HAL_DFSDM_FilterGetError(const DFSDM_Filter_HandleTypeDef *hdfsdm_filter); /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dsi.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dsi.h index 6da96681..6b51ecb2 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dsi.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_dsi.h @@ -976,7 +976,7 @@ typedef void (*pDSI_CallbackTypeDef)(DSI_HandleTypeDef *hdsi); /*!< pointer to #define __HAL_DSI_WRAPPER_ENABLE(__HANDLE__) do { \ __IO uint32_t tmpreg = 0x00U; \ SET_BIT((__HANDLE__)->Instance->WCR, DSI_WCR_DSIEN);\ - /* Delay after an DSI warpper enabling */ \ + /* Delay after an DSI wrapper enabling */ \ tmpreg = READ_BIT((__HANDLE__)->Instance->WCR, DSI_WCR_DSIEN);\ UNUSED(tmpreg); \ } while(0U) @@ -989,7 +989,7 @@ typedef void (*pDSI_CallbackTypeDef)(DSI_HandleTypeDef *hdsi); /*!< pointer to #define __HAL_DSI_WRAPPER_DISABLE(__HANDLE__) do { \ __IO uint32_t tmpreg = 0x00U; \ CLEAR_BIT((__HANDLE__)->Instance->WCR, DSI_WCR_DSIEN);\ - /* Delay after an DSI warpper disabling*/ \ + /* Delay after an DSI wrapper disabling*/ \ tmpreg = READ_BIT((__HANDLE__)->Instance->WCR, DSI_WCR_DSIEN);\ UNUSED(tmpreg); \ } while(0U) @@ -1271,10 +1271,10 @@ HAL_DSI_StateTypeDef HAL_DSI_GetState(DSI_HandleTypeDef *hdsi); || ((LooselyPacked) == DSI_LOOSELY_PACKED_DISABLE)) #define IS_DSI_DE_POLARITY(DataEnable) (((DataEnable) == DSI_DATA_ENABLE_ACTIVE_HIGH)\ || ((DataEnable) == DSI_DATA_ENABLE_ACTIVE_LOW)) -#define IS_DSI_VSYNC_POLARITY(VSYNC) (((VSYNC) == DSI_VSYNC_ACTIVE_HIGH)\ - || ((VSYNC) == DSI_VSYNC_ACTIVE_LOW)) -#define IS_DSI_HSYNC_POLARITY(HSYNC) (((HSYNC) == DSI_HSYNC_ACTIVE_HIGH)\ - || ((HSYNC) == DSI_HSYNC_ACTIVE_LOW)) +#define IS_DSI_VSYNC_POLARITY(Vsync) (((Vsync) == DSI_VSYNC_ACTIVE_HIGH)\ + || ((Vsync) == DSI_VSYNC_ACTIVE_LOW)) +#define IS_DSI_HSYNC_POLARITY(Hsync) (((Hsync) == DSI_HSYNC_ACTIVE_HIGH)\ + || ((Hsync) == DSI_HSYNC_ACTIVE_LOW)) #define IS_DSI_VIDEO_MODE_TYPE(VideoModeType) (((VideoModeType) == DSI_VID_MODE_NB_PULSES) || \ ((VideoModeType) == DSI_VID_MODE_NB_EVENTS) || \ ((VideoModeType) == DSI_VID_MODE_BURST)) diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_eth.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_eth.h index ba5a09bb..91fdb62b 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_eth.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_eth.h @@ -171,6 +171,7 @@ typedef struct * */ +#ifdef HAL_ETH_USE_PTP /** * @brief ETH Timeupdate structure definition */ @@ -182,6 +183,7 @@ typedef struct /** * */ +#endif /* HAL_ETH_USE_PTP */ /** * @brief DMA Receive Descriptors Wrapper structure definition @@ -347,7 +349,6 @@ typedef struct uint32_t BurstMode; /*!< Sets the AHB Master interface burst transfers. This parameter can be a value of @ref ETH_Burst_Mode */ - FunctionalState DropTCPIPChecksumErrorFrame; /*!< Selects or not the Dropping of TCP/IP Checksum Error Frames */ FunctionalState ReceiveStoreForward; /*!< Enables or disables the Receive store and forward mode */ @@ -407,6 +408,7 @@ typedef enum * */ +#ifdef HAL_ETH_USE_PTP /** * @brief HAL ETH PTP Update type enum definition */ @@ -418,13 +420,13 @@ typedef enum /** * */ +#endif /* HAL_ETH_USE_PTP */ /** * @brief ETH Init Structure definition */ typedef struct { - uint8_t *MACAddr; /*!< MAC Address of used Hardware: must be pointer on an array of 6 bytes */ @@ -443,6 +445,7 @@ typedef struct * */ +#ifdef HAL_ETH_USE_PTP /** * @brief ETH PTP Init Structure definition */ @@ -470,6 +473,7 @@ typedef struct /** * */ +#endif /* HAL_ETH_USE_PTP */ /** * @brief HAL State structures definition @@ -538,7 +542,7 @@ typedef struct __IO HAL_ETH_StateTypeDef gState; /*!< ETH state information related to global Handle management and also related to Tx operations. This parameter can - be a value of @ref HAL_ETH_StateTypeDef */ + be a value of @ref ETH_State_Codes */ __IO uint32_t ErrorCode; /*!< Holds the global Error code of the ETH HAL status machine This parameter can be a value of @ref ETH_Error_Code.*/ @@ -595,14 +599,12 @@ typedef enum { HAL_ETH_MSPINIT_CB_ID = 0x00U, /*!< ETH MspInit callback ID */ HAL_ETH_MSPDEINIT_CB_ID = 0x01U, /*!< ETH MspDeInit callback ID */ - HAL_ETH_TX_COMPLETE_CB_ID = 0x02U, /*!< ETH Tx Complete Callback ID */ HAL_ETH_RX_COMPLETE_CB_ID = 0x03U, /*!< ETH Rx Complete Callback ID */ HAL_ETH_ERROR_CB_ID = 0x04U, /*!< ETH Error Callback ID */ HAL_ETH_PMT_CB_ID = 0x06U, /*!< ETH Power Management Callback ID */ HAL_ETH_WAKEUP_CB_ID = 0x08U /*!< ETH Wake UP Callback ID */ - } HAL_ETH_CallbackIDTypeDef; /** @@ -1298,7 +1300,7 @@ TDES7 | Transmit Time Stamp High [31:0] * @} */ -/** @defgroup HAL_ETH_StateTypeDef ETH States +/** @defgroup ETH_State_Codes ETH States * @{ */ #define HAL_ETH_STATE_RESET 0x00000000U /*!< Peripheral not yet Initialized or disabled */ @@ -1902,6 +1904,7 @@ TDES7 | Transmit Time Stamp High [31:0] * enabled @ref ETH_MAC_Interrupts * @retval None */ + #define __HAL_ETH_MAC_ENABLE_IT(__HANDLE__, __INTERRUPT__) ((__HANDLE__)->Instance->MACIER \ |= (__INTERRUPT__)) @@ -1921,8 +1924,8 @@ TDES7 | Transmit Time Stamp High [31:0] * @param __INTERRUPT__: specifies the flag to check. @ref ETH_MAC_Interrupts * @retval The state of ETH MAC IT (SET or RESET). */ -#define __HAL_ETH_MAC_GET_IT(__HANDLE__, __INTERRUPT__) (((__HANDLE__)->Instance->MACSR &\ - ( __INTERRUPT__)) == ( __INTERRUPT__)) +#define __HAL_ETH_MAC_GET_IT(__HANDLE__, __INTERRUPT__) (((__HANDLE__)->Instance->MACSR &\ + ( __INTERRUPT__)) == ( __INTERRUPT__)) /*!< External interrupt line 19 Connected to the ETH wakeup EXTI Line */ #define ETH_WAKEUP_EXTI_LINE 0x00080000U @@ -1991,6 +1994,7 @@ TDES7 | Transmit Time Stamp High [31:0] (__FLAG__)) == (__FLAG__)) ? SET : RESET) #define __HAL_ETH_SET_PTP_CONTROL(__HANDLE__, __FLAG__) ((__HANDLE__)->Instance->PTPTSCR |= (__FLAG__)) + /** * @} */ @@ -2059,7 +2063,7 @@ HAL_StatusTypeDef HAL_ETH_UnRegisterTxPtpCallback(ETH_HandleTypeDef *heth); HAL_StatusTypeDef HAL_ETH_Transmit(ETH_HandleTypeDef *heth, ETH_TxPacketConfig *pTxConfig, uint32_t Timeout); HAL_StatusTypeDef HAL_ETH_Transmit_IT(ETH_HandleTypeDef *heth, ETH_TxPacketConfig *pTxConfig); -HAL_StatusTypeDef HAL_ETH_WritePHYRegister(ETH_HandleTypeDef *heth, uint32_t PHYAddr, uint32_t PHYReg, +HAL_StatusTypeDef HAL_ETH_WritePHYRegister(const ETH_HandleTypeDef *heth, uint32_t PHYAddr, uint32_t PHYReg, uint32_t RegValue); HAL_StatusTypeDef HAL_ETH_ReadPHYRegister(ETH_HandleTypeDef *heth, uint32_t PHYAddr, uint32_t PHYReg, uint32_t *pRegValue); @@ -2095,12 +2099,14 @@ void HAL_ETH_SetRxVLANIdentifier(ETH_HandleTypeDef *heth, uint32_t /* MAC L2 Packet Filtering APIs **********************************************/ HAL_StatusTypeDef HAL_ETH_GetMACFilterConfig(ETH_HandleTypeDef *heth, ETH_MACFilterConfigTypeDef *pFilterConfig); -HAL_StatusTypeDef HAL_ETH_SetMACFilterConfig(ETH_HandleTypeDef *heth, ETH_MACFilterConfigTypeDef *pFilterConfig); +HAL_StatusTypeDef HAL_ETH_SetMACFilterConfig(ETH_HandleTypeDef *heth, const ETH_MACFilterConfigTypeDef *pFilterConfig); HAL_StatusTypeDef HAL_ETH_SetHashTable(ETH_HandleTypeDef *heth, uint32_t *pHashTable); -HAL_StatusTypeDef HAL_ETH_SetSourceMACAddrMatch(ETH_HandleTypeDef *heth, uint32_t AddrNbr, uint8_t *pMACAddr); +HAL_StatusTypeDef HAL_ETH_SetSourceMACAddrMatch(const ETH_HandleTypeDef *heth, uint32_t AddrNbr, + const uint8_t *pMACAddr); /* MAC Power Down APIs *****************************************************/ -void HAL_ETH_EnterPowerDownMode(ETH_HandleTypeDef *heth, ETH_PowerDownConfigTypeDef *pPowerDownConfig); +void HAL_ETH_EnterPowerDownMode(ETH_HandleTypeDef *heth, + const ETH_PowerDownConfigTypeDef *pPowerDownConfig); void HAL_ETH_ExitPowerDownMode(ETH_HandleTypeDef *heth); HAL_StatusTypeDef HAL_ETH_SetWakeUpFilter(ETH_HandleTypeDef *heth, uint32_t *pFilter, uint32_t Count); @@ -2112,11 +2118,11 @@ HAL_StatusTypeDef HAL_ETH_SetWakeUpFilter(ETH_HandleTypeDef *heth, uint32_t *pFi * @{ */ /* Peripheral State functions **************************************************/ -HAL_ETH_StateTypeDef HAL_ETH_GetState(ETH_HandleTypeDef *heth); -uint32_t HAL_ETH_GetError(ETH_HandleTypeDef *heth); -uint32_t HAL_ETH_GetDMAError(ETH_HandleTypeDef *heth); -uint32_t HAL_ETH_GetMACError(ETH_HandleTypeDef *heth); -uint32_t HAL_ETH_GetMACWakeUpSource(ETH_HandleTypeDef *heth); +HAL_ETH_StateTypeDef HAL_ETH_GetState(const ETH_HandleTypeDef *heth); +uint32_t HAL_ETH_GetError(const ETH_HandleTypeDef *heth); +uint32_t HAL_ETH_GetDMAError(const ETH_HandleTypeDef *heth); +uint32_t HAL_ETH_GetMACError(const ETH_HandleTypeDef *heth); +uint32_t HAL_ETH_GetMACWakeUpSource(const ETH_HandleTypeDef *heth); /** * @} */ @@ -2140,5 +2146,3 @@ uint32_t HAL_ETH_GetMACWakeUpSource(ETH_HandleTypeDef *heth); #endif #endif /* STM32F4xx_HAL_ETH_H */ - - diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpi2c.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpi2c.h index 400b923e..60ed94b8 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpi2c.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpi2c.h @@ -119,8 +119,6 @@ typedef enum HAL_FMPI2C_STATE_BUSY_RX_LISTEN = 0x2AU, /*!< Address Listen Mode and Data Reception process is ongoing */ HAL_FMPI2C_STATE_ABORT = 0x60U, /*!< Abort user request ongoing */ - HAL_FMPI2C_STATE_TIMEOUT = 0xA0U, /*!< Timeout state */ - HAL_FMPI2C_STATE_ERROR = 0xE0U /*!< Error */ } HAL_FMPI2C_StateTypeDef; @@ -208,6 +206,7 @@ typedef struct __FMPI2C_HandleTypeDef DMA_HandleTypeDef *hdmarx; /*!< FMPI2C Rx DMA handle parameters */ + HAL_LockTypeDef Lock; /*!< FMPI2C locking object */ __IO HAL_FMPI2C_StateTypeDef State; /*!< FMPI2C communication state */ @@ -218,6 +217,10 @@ typedef struct __FMPI2C_HandleTypeDef __IO uint32_t AddrEventCount; /*!< FMPI2C Address Event counter */ + __IO uint32_t Devaddress; /*!< FMPI2C Target device address */ + + __IO uint32_t Memaddress; /*!< FMPI2C Target memory address */ + #if (USE_HAL_FMPI2C_REGISTER_CALLBACKS == 1) void (* MasterTxCpltCallback)(struct __FMPI2C_HandleTypeDef *hfmpi2c); /*!< FMPI2C Master Tx Transfer completed callback */ @@ -276,7 +279,7 @@ typedef enum typedef void (*pFMPI2C_CallbackTypeDef)(FMPI2C_HandleTypeDef *hfmpi2c); /*!< pointer to an FMPI2C callback function */ typedef void (*pFMPI2C_AddrCallbackTypeDef)(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t TransferDirection, - uint16_t AddrMatchCode); + uint16_t AddrMatchCode); /*!< pointer to an FMPI2C Address Match callback function */ #endif /* USE_HAL_FMPI2C_REGISTER_CALLBACKS */ @@ -458,10 +461,10 @@ typedef void (*pFMPI2C_AddrCallbackTypeDef)(FMPI2C_HandleTypeDef *hfmpi2c, uint */ #if (USE_HAL_FMPI2C_REGISTER_CALLBACKS == 1) #define __HAL_FMPI2C_RESET_HANDLE_STATE(__HANDLE__) do{ \ - (__HANDLE__)->State = HAL_FMPI2C_STATE_RESET; \ - (__HANDLE__)->MspInitCallback = NULL; \ - (__HANDLE__)->MspDeInitCallback = NULL; \ - } while(0) + (__HANDLE__)->State = HAL_FMPI2C_STATE_RESET; \ + (__HANDLE__)->MspInitCallback = NULL; \ + (__HANDLE__)->MspDeInitCallback = NULL; \ + } while(0) #else #define __HAL_FMPI2C_RESET_HANDLE_STATE(__HANDLE__) ((__HANDLE__)->State = HAL_FMPI2C_STATE_RESET) #endif /* USE_HAL_FMPI2C_REGISTER_CALLBACKS */ @@ -513,7 +516,7 @@ typedef void (*pFMPI2C_AddrCallbackTypeDef)(FMPI2C_HandleTypeDef *hfmpi2c, uint * @retval The new state of __INTERRUPT__ (SET or RESET). */ #define __HAL_FMPI2C_GET_IT_SOURCE(__HANDLE__, __INTERRUPT__) ((((__HANDLE__)->Instance->CR1 & \ - (__INTERRUPT__)) == (__INTERRUPT__)) ? SET : RESET) + (__INTERRUPT__)) == (__INTERRUPT__)) ? SET : RESET) /** @brief Check whether the specified FMPI2C flag is set or not. * @param __HANDLE__ specifies the FMPI2C Handle. @@ -540,7 +543,7 @@ typedef void (*pFMPI2C_AddrCallbackTypeDef)(FMPI2C_HandleTypeDef *hfmpi2c, uint */ #define FMPI2C_FLAG_MASK (0x0001FFFFU) #define __HAL_FMPI2C_GET_FLAG(__HANDLE__, __FLAG__) (((((__HANDLE__)->Instance->ISR) & \ - (__FLAG__)) == (__FLAG__)) ? SET : RESET) + (__FLAG__)) == (__FLAG__)) ? SET : RESET) /** @brief Clear the FMPI2C pending flags which are cleared by writing 1 in a specific bit. * @param __HANDLE__ specifies the FMPI2C Handle. @@ -560,8 +563,8 @@ typedef void (*pFMPI2C_AddrCallbackTypeDef)(FMPI2C_HandleTypeDef *hfmpi2c, uint * @retval None */ #define __HAL_FMPI2C_CLEAR_FLAG(__HANDLE__, __FLAG__) (((__FLAG__) == FMPI2C_FLAG_TXE) ? \ - ((__HANDLE__)->Instance->ISR |= (__FLAG__)) : \ - ((__HANDLE__)->Instance->ICR = (__FLAG__))) + ((__HANDLE__)->Instance->ISR |= (__FLAG__)) : \ + ((__HANDLE__)->Instance->ICR = (__FLAG__))) /** @brief Enable the specified FMPI2C peripheral. * @param __HANDLE__ specifies the FMPI2C Handle. @@ -604,7 +607,7 @@ void HAL_FMPI2C_MspDeInit(FMPI2C_HandleTypeDef *hfmpi2c); /* Callbacks Register/UnRegister functions ***********************************/ #if (USE_HAL_FMPI2C_REGISTER_CALLBACKS == 1) HAL_StatusTypeDef HAL_FMPI2C_RegisterCallback(FMPI2C_HandleTypeDef *hfmpi2c, HAL_FMPI2C_CallbackIDTypeDef CallbackID, - pFMPI2C_CallbackTypeDef pCallback); + pFMPI2C_CallbackTypeDef pCallback); HAL_StatusTypeDef HAL_FMPI2C_UnRegisterCallback(FMPI2C_HandleTypeDef *hfmpi2c, HAL_FMPI2C_CallbackIDTypeDef CallbackID); HAL_StatusTypeDef HAL_FMPI2C_RegisterAddrCallback(FMPI2C_HandleTypeDef *hfmpi2c, pFMPI2C_AddrCallbackTypeDef pCallback); @@ -620,64 +623,64 @@ HAL_StatusTypeDef HAL_FMPI2C_UnRegisterAddrCallback(FMPI2C_HandleTypeDef *hfmpi2 /* IO operation functions ****************************************************/ /******* Blocking mode: Polling */ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size, uint32_t Timeout); + uint16_t Size, uint32_t Timeout); HAL_StatusTypeDef HAL_FMPI2C_Master_Receive(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size, uint32_t Timeout); + uint16_t Size, uint32_t Timeout); HAL_StatusTypeDef HAL_FMPI2C_Slave_Transmit(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size, - uint32_t Timeout); + uint32_t Timeout); HAL_StatusTypeDef HAL_FMPI2C_Slave_Receive(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size, - uint32_t Timeout); + uint32_t Timeout); HAL_StatusTypeDef HAL_FMPI2C_Mem_Write(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint16_t MemAddress, - uint16_t MemAddSize, uint8_t *pData, uint16_t Size, uint32_t Timeout); + uint16_t MemAddSize, uint8_t *pData, uint16_t Size, uint32_t Timeout); HAL_StatusTypeDef HAL_FMPI2C_Mem_Read(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint16_t MemAddress, - uint16_t MemAddSize, uint8_t *pData, uint16_t Size, uint32_t Timeout); + uint16_t MemAddSize, uint8_t *pData, uint16_t Size, uint32_t Timeout); HAL_StatusTypeDef HAL_FMPI2C_IsDeviceReady(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint32_t Trials, - uint32_t Timeout); + uint32_t Timeout); /******* Non-Blocking mode: Interrupt */ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size); + uint16_t Size); HAL_StatusTypeDef HAL_FMPI2C_Master_Receive_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size); + uint16_t Size); HAL_StatusTypeDef HAL_FMPI2C_Slave_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size); HAL_StatusTypeDef HAL_FMPI2C_Slave_Receive_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size); HAL_StatusTypeDef HAL_FMPI2C_Mem_Write_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint16_t MemAddress, - uint16_t MemAddSize, uint8_t *pData, uint16_t Size); + uint16_t MemAddSize, uint8_t *pData, uint16_t Size); HAL_StatusTypeDef HAL_FMPI2C_Mem_Read_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint16_t MemAddress, - uint16_t MemAddSize, uint8_t *pData, uint16_t Size); + uint16_t MemAddSize, uint8_t *pData, uint16_t Size); HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size, uint32_t XferOptions); + uint16_t Size, uint32_t XferOptions); HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Receive_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size, uint32_t XferOptions); + uint16_t Size, uint32_t XferOptions); HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size, - uint32_t XferOptions); + uint32_t XferOptions); HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Receive_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size, - uint32_t XferOptions); + uint32_t XferOptions); HAL_StatusTypeDef HAL_FMPI2C_EnableListen_IT(FMPI2C_HandleTypeDef *hfmpi2c); HAL_StatusTypeDef HAL_FMPI2C_DisableListen_IT(FMPI2C_HandleTypeDef *hfmpi2c); HAL_StatusTypeDef HAL_FMPI2C_Master_Abort_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress); /******* Non-Blocking mode: DMA */ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size); + uint16_t Size); HAL_StatusTypeDef HAL_FMPI2C_Master_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size); + uint16_t Size); HAL_StatusTypeDef HAL_FMPI2C_Slave_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size); HAL_StatusTypeDef HAL_FMPI2C_Slave_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size); HAL_StatusTypeDef HAL_FMPI2C_Mem_Write_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint16_t MemAddress, - uint16_t MemAddSize, uint8_t *pData, uint16_t Size); + uint16_t MemAddSize, uint8_t *pData, uint16_t Size); HAL_StatusTypeDef HAL_FMPI2C_Mem_Read_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint16_t MemAddress, - uint16_t MemAddSize, uint8_t *pData, uint16_t Size); + uint16_t MemAddSize, uint8_t *pData, uint16_t Size); HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size, uint32_t XferOptions); + uint16_t Size, uint32_t XferOptions); HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size, uint32_t XferOptions); + uint16_t Size, uint32_t XferOptions); HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size, - uint32_t XferOptions); + uint32_t XferOptions); HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size, - uint32_t XferOptions); + uint32_t XferOptions); /** * @} */ @@ -706,9 +709,9 @@ void HAL_FMPI2C_AbortCpltCallback(FMPI2C_HandleTypeDef *hfmpi2c); * @{ */ /* Peripheral State, Mode and Error functions *********************************/ -HAL_FMPI2C_StateTypeDef HAL_FMPI2C_GetState(FMPI2C_HandleTypeDef *hfmpi2c); -HAL_FMPI2C_ModeTypeDef HAL_FMPI2C_GetMode(FMPI2C_HandleTypeDef *hfmpi2c); -uint32_t HAL_FMPI2C_GetError(FMPI2C_HandleTypeDef *hfmpi2c); +HAL_FMPI2C_StateTypeDef HAL_FMPI2C_GetState(const FMPI2C_HandleTypeDef *hfmpi2c); +HAL_FMPI2C_ModeTypeDef HAL_FMPI2C_GetMode(const FMPI2C_HandleTypeDef *hfmpi2c); +uint32_t HAL_FMPI2C_GetError(const FMPI2C_HandleTypeDef *hfmpi2c); /** * @} @@ -733,28 +736,28 @@ uint32_t HAL_FMPI2C_GetError(FMPI2C_HandleTypeDef *hfmpi2c); */ #define IS_FMPI2C_ADDRESSING_MODE(MODE) (((MODE) == FMPI2C_ADDRESSINGMODE_7BIT) || \ - ((MODE) == FMPI2C_ADDRESSINGMODE_10BIT)) + ((MODE) == FMPI2C_ADDRESSINGMODE_10BIT)) #define IS_FMPI2C_DUAL_ADDRESS(ADDRESS) (((ADDRESS) == FMPI2C_DUALADDRESS_DISABLE) || \ - ((ADDRESS) == FMPI2C_DUALADDRESS_ENABLE)) + ((ADDRESS) == FMPI2C_DUALADDRESS_ENABLE)) #define IS_FMPI2C_OWN_ADDRESS2_MASK(MASK) (((MASK) == FMPI2C_OA2_NOMASK) || \ - ((MASK) == FMPI2C_OA2_MASK01) || \ - ((MASK) == FMPI2C_OA2_MASK02) || \ - ((MASK) == FMPI2C_OA2_MASK03) || \ - ((MASK) == FMPI2C_OA2_MASK04) || \ - ((MASK) == FMPI2C_OA2_MASK05) || \ - ((MASK) == FMPI2C_OA2_MASK06) || \ - ((MASK) == FMPI2C_OA2_MASK07)) + ((MASK) == FMPI2C_OA2_MASK01) || \ + ((MASK) == FMPI2C_OA2_MASK02) || \ + ((MASK) == FMPI2C_OA2_MASK03) || \ + ((MASK) == FMPI2C_OA2_MASK04) || \ + ((MASK) == FMPI2C_OA2_MASK05) || \ + ((MASK) == FMPI2C_OA2_MASK06) || \ + ((MASK) == FMPI2C_OA2_MASK07)) #define IS_FMPI2C_GENERAL_CALL(CALL) (((CALL) == FMPI2C_GENERALCALL_DISABLE) || \ - ((CALL) == FMPI2C_GENERALCALL_ENABLE)) + ((CALL) == FMPI2C_GENERALCALL_ENABLE)) #define IS_FMPI2C_NO_STRETCH(STRETCH) (((STRETCH) == FMPI2C_NOSTRETCH_DISABLE) || \ - ((STRETCH) == FMPI2C_NOSTRETCH_ENABLE)) + ((STRETCH) == FMPI2C_NOSTRETCH_ENABLE)) #define IS_FMPI2C_MEMADD_SIZE(SIZE) (((SIZE) == FMPI2C_MEMADD_SIZE_8BIT) || \ - ((SIZE) == FMPI2C_MEMADD_SIZE_16BIT)) + ((SIZE) == FMPI2C_MEMADD_SIZE_16BIT)) #define IS_TRANSFER_MODE(MODE) (((MODE) == FMPI2C_RELOAD_MODE) || \ ((MODE) == FMPI2C_AUTOEND_MODE) || \ @@ -766,25 +769,25 @@ uint32_t HAL_FMPI2C_GetError(FMPI2C_HandleTypeDef *hfmpi2c); ((REQUEST) == FMPI2C_NO_STARTSTOP)) #define IS_FMPI2C_TRANSFER_OPTIONS_REQUEST(REQUEST) (((REQUEST) == FMPI2C_FIRST_FRAME) || \ - ((REQUEST) == FMPI2C_FIRST_AND_NEXT_FRAME) || \ - ((REQUEST) == FMPI2C_NEXT_FRAME) || \ - ((REQUEST) == FMPI2C_FIRST_AND_LAST_FRAME) || \ - ((REQUEST) == FMPI2C_LAST_FRAME) || \ - ((REQUEST) == FMPI2C_LAST_FRAME_NO_STOP) || \ - IS_FMPI2C_TRANSFER_OTHER_OPTIONS_REQUEST(REQUEST)) + ((REQUEST) == FMPI2C_FIRST_AND_NEXT_FRAME) || \ + ((REQUEST) == FMPI2C_NEXT_FRAME) || \ + ((REQUEST) == FMPI2C_FIRST_AND_LAST_FRAME) || \ + ((REQUEST) == FMPI2C_LAST_FRAME) || \ + ((REQUEST) == FMPI2C_LAST_FRAME_NO_STOP) || \ + IS_FMPI2C_TRANSFER_OTHER_OPTIONS_REQUEST(REQUEST)) #define IS_FMPI2C_TRANSFER_OTHER_OPTIONS_REQUEST(REQUEST) (((REQUEST) == FMPI2C_OTHER_FRAME) || \ - ((REQUEST) == FMPI2C_OTHER_AND_LAST_FRAME)) + ((REQUEST) == FMPI2C_OTHER_AND_LAST_FRAME)) #define FMPI2C_RESET_CR2(__HANDLE__) ((__HANDLE__)->Instance->CR2 &= \ - (uint32_t)~((uint32_t)(FMPI2C_CR2_SADD | FMPI2C_CR2_HEAD10R | \ - FMPI2C_CR2_NBYTES | FMPI2C_CR2_RELOAD | \ - FMPI2C_CR2_RD_WRN))) + (uint32_t)~((uint32_t)(FMPI2C_CR2_SADD | FMPI2C_CR2_HEAD10R | \ + FMPI2C_CR2_NBYTES | FMPI2C_CR2_RELOAD | \ + FMPI2C_CR2_RD_WRN))) #define FMPI2C_GET_ADDR_MATCH(__HANDLE__) ((uint16_t)(((__HANDLE__)->Instance->ISR & FMPI2C_ISR_ADDCODE) \ - >> 16U)) + >> 16U)) #define FMPI2C_GET_DIR(__HANDLE__) ((uint8_t)(((__HANDLE__)->Instance->ISR & FMPI2C_ISR_DIR) \ - >> 16U)) + >> 16U)) #define FMPI2C_GET_STOP_MODE(__HANDLE__) ((__HANDLE__)->Instance->CR2 & FMPI2C_CR2_AUTOEND) #define FMPI2C_GET_OWN_ADDRESS1(__HANDLE__) ((uint16_t)((__HANDLE__)->Instance->OAR1 & FMPI2C_OAR1_OA1)) #define FMPI2C_GET_OWN_ADDRESS2(__HANDLE__) ((uint16_t)((__HANDLE__)->Instance->OAR2 & FMPI2C_OAR2_OA2)) @@ -793,19 +796,19 @@ uint32_t HAL_FMPI2C_GetError(FMPI2C_HandleTypeDef *hfmpi2c); #define IS_FMPI2C_OWN_ADDRESS2(ADDRESS2) ((ADDRESS2) <= (uint16_t)0x00FFU) #define FMPI2C_MEM_ADD_MSB(__ADDRESS__) ((uint8_t)((uint16_t)(((uint16_t)((__ADDRESS__) & \ - (uint16_t)(0xFF00U))) >> 8U))) + (uint16_t)(0xFF00U))) >> 8U))) #define FMPI2C_MEM_ADD_LSB(__ADDRESS__) ((uint8_t)((uint16_t)((__ADDRESS__) & (uint16_t)(0x00FFU)))) #define FMPI2C_GENERATE_START(__ADDMODE__,__ADDRESS__) (((__ADDMODE__) == FMPI2C_ADDRESSINGMODE_7BIT) ? \ - (uint32_t)((((uint32_t)(__ADDRESS__) & (FMPI2C_CR2_SADD)) | \ - (FMPI2C_CR2_START) | (FMPI2C_CR2_AUTOEND)) & \ - (~FMPI2C_CR2_RD_WRN)) : \ - (uint32_t)((((uint32_t)(__ADDRESS__) & (FMPI2C_CR2_SADD)) | \ - (FMPI2C_CR2_ADD10) | (FMPI2C_CR2_START)) & \ - (~FMPI2C_CR2_RD_WRN))) + (uint32_t)((((uint32_t)(__ADDRESS__) & (FMPI2C_CR2_SADD)) | \ + (FMPI2C_CR2_START) | (FMPI2C_CR2_AUTOEND)) & \ + (~FMPI2C_CR2_RD_WRN)) : \ + (uint32_t)((((uint32_t)(__ADDRESS__) & (FMPI2C_CR2_SADD)) | \ + (FMPI2C_CR2_ADD10) | (FMPI2C_CR2_START) | \ + (FMPI2C_CR2_AUTOEND)) & (~FMPI2C_CR2_RD_WRN))) #define FMPI2C_CHECK_FLAG(__ISR__, __FLAG__) ((((__ISR__) & ((__FLAG__) & FMPI2C_FLAG_MASK)) == \ - ((__FLAG__) & FMPI2C_FLAG_MASK)) ? SET : RESET) + ((__FLAG__) & FMPI2C_FLAG_MASK)) ? SET : RESET) #define FMPI2C_CHECK_IT_SOURCE(__CR1__, __IT__) ((((__CR1__) & (__IT__)) == (__IT__)) ? SET : RESET) /** * @} diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpi2c_ex.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpi2c_ex.h index 8b90f407..995b5d3f 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpi2c_ex.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpi2c_ex.h @@ -115,12 +115,12 @@ void HAL_FMPI2CEx_DisableFastModePlus(uint32_t ConfigFastModePlus); * @{ */ #define IS_FMPI2C_ANALOG_FILTER(FILTER) (((FILTER) == FMPI2C_ANALOGFILTER_ENABLE) || \ - ((FILTER) == FMPI2C_ANALOGFILTER_DISABLE)) + ((FILTER) == FMPI2C_ANALOGFILTER_DISABLE)) #define IS_FMPI2C_DIGITAL_FILTER(FILTER) ((FILTER) <= 0x0000000FU) #define IS_FMPI2C_FASTMODEPLUS(__CONFIG__) ((((__CONFIG__) & (FMPI2C_FASTMODEPLUS_SCL)) == FMPI2C_FASTMODEPLUS_SCL) || \ - (((__CONFIG__) & (FMPI2C_FASTMODEPLUS_SDA)) == FMPI2C_FASTMODEPLUS_SDA)) + (((__CONFIG__) & (FMPI2C_FASTMODEPLUS_SDA)) == FMPI2C_FASTMODEPLUS_SDA)) /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpsmbus.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpsmbus.h index 6d1ff4b1..84c54002 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpsmbus.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpsmbus.h @@ -101,8 +101,6 @@ typedef struct #define HAL_FMPSMBUS_STATE_MASTER_BUSY_RX (0x00000022U) /*!< Master Data Reception process is ongoing */ #define HAL_FMPSMBUS_STATE_SLAVE_BUSY_TX (0x00000032U) /*!< Slave Data Transmission process is ongoing */ #define HAL_FMPSMBUS_STATE_SLAVE_BUSY_RX (0x00000042U) /*!< Slave Data Reception process is ongoing */ -#define HAL_FMPSMBUS_STATE_TIMEOUT (0x00000003U) /*!< Timeout state */ -#define HAL_FMPSMBUS_STATE_ERROR (0x00000004U) /*!< Reception process is ongoing */ #define HAL_FMPSMBUS_STATE_LISTEN (0x00000008U) /*!< Address Listen Mode is ongoing */ /** * @} @@ -208,7 +206,7 @@ typedef enum typedef void (*pFMPSMBUS_CallbackTypeDef)(FMPSMBUS_HandleTypeDef *hfmpsmbus); /*!< pointer to an FMPSMBUS callback function */ typedef void (*pFMPSMBUS_AddrCallbackTypeDef)(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint8_t TransferDirection, - uint16_t AddrMatchCode); + uint16_t AddrMatchCode); /*!< pointer to an FMPSMBUS Address Match callback function */ #endif /* USE_HAL_FMPSMBUS_REGISTER_CALLBACKS */ @@ -372,9 +370,9 @@ typedef void (*pFMPSMBUS_AddrCallbackTypeDef)(FMPSMBUS_HandleTypeDef *hfmpsmbus #define FMPSMBUS_IT_RXI FMPI2C_CR1_RXIE #define FMPSMBUS_IT_TXI FMPI2C_CR1_TXIE #define FMPSMBUS_IT_TX (FMPSMBUS_IT_ERRI | FMPSMBUS_IT_TCI | FMPSMBUS_IT_STOPI | \ - FMPSMBUS_IT_NACKI | FMPSMBUS_IT_TXI) + FMPSMBUS_IT_NACKI | FMPSMBUS_IT_TXI) #define FMPSMBUS_IT_RX (FMPSMBUS_IT_ERRI | FMPSMBUS_IT_TCI | FMPSMBUS_IT_NACKI | \ - FMPSMBUS_IT_RXI) + FMPSMBUS_IT_RXI) #define FMPSMBUS_IT_ALERT (FMPSMBUS_IT_ERRI) #define FMPSMBUS_IT_ADDR (FMPSMBUS_IT_ADDRI | FMPSMBUS_IT_STOPI | FMPSMBUS_IT_NACKI) /** @@ -423,10 +421,10 @@ typedef void (*pFMPSMBUS_AddrCallbackTypeDef)(FMPSMBUS_HandleTypeDef *hfmpsmbus */ #if (USE_HAL_FMPSMBUS_REGISTER_CALLBACKS == 1) #define __HAL_FMPSMBUS_RESET_HANDLE_STATE(__HANDLE__) do{ \ - (__HANDLE__)->State = HAL_FMPSMBUS_STATE_RESET; \ - (__HANDLE__)->MspInitCallback = NULL; \ - (__HANDLE__)->MspDeInitCallback = NULL; \ - } while(0) + (__HANDLE__)->State = HAL_FMPSMBUS_STATE_RESET; \ + (__HANDLE__)->MspInitCallback = NULL; \ + (__HANDLE__)->MspDeInitCallback = NULL; \ + } while(0) #else #define __HAL_FMPSMBUS_RESET_HANDLE_STATE(__HANDLE__) ((__HANDLE__)->State = HAL_FMPSMBUS_STATE_RESET) #endif /* USE_HAL_FMPSMBUS_REGISTER_CALLBACKS */ @@ -512,6 +510,7 @@ typedef void (*pFMPSMBUS_AddrCallbackTypeDef)(FMPSMBUS_HandleTypeDef *hfmpsmbus * @param __HANDLE__ specifies the FMPSMBUS Handle. * @param __FLAG__ specifies the flag to clear. * This parameter can be any combination of the following values: + * @arg @ref FMPSMBUS_FLAG_TXE Transmit data register empty * @arg @ref FMPSMBUS_FLAG_ADDR Address matched (slave mode) * @arg @ref FMPSMBUS_FLAG_AF NACK received flag * @arg @ref FMPSMBUS_FLAG_STOPF STOP detection flag @@ -524,7 +523,9 @@ typedef void (*pFMPSMBUS_AddrCallbackTypeDef)(FMPSMBUS_HandleTypeDef *hfmpsmbus * * @retval None */ -#define __HAL_FMPSMBUS_CLEAR_FLAG(__HANDLE__, __FLAG__) ((__HANDLE__)->Instance->ICR = (__FLAG__)) +#define __HAL_FMPSMBUS_CLEAR_FLAG(__HANDLE__, __FLAG__) (((__FLAG__) == FMPSMBUS_FLAG_TXE) ? \ + ((__HANDLE__)->Instance->ISR |= (__FLAG__)) : \ + ((__HANDLE__)->Instance->ICR = (__FLAG__))) /** @brief Enable the specified FMPSMBUS peripheral. * @param __HANDLE__ specifies the FMPSMBUS Handle. @@ -557,84 +558,84 @@ typedef void (*pFMPSMBUS_AddrCallbackTypeDef)(FMPSMBUS_HandleTypeDef *hfmpsmbus */ #define IS_FMPSMBUS_ANALOG_FILTER(FILTER) (((FILTER) == FMPSMBUS_ANALOGFILTER_ENABLE) || \ - ((FILTER) == FMPSMBUS_ANALOGFILTER_DISABLE)) + ((FILTER) == FMPSMBUS_ANALOGFILTER_DISABLE)) #define IS_FMPSMBUS_DIGITAL_FILTER(FILTER) ((FILTER) <= 0x0000000FU) #define IS_FMPSMBUS_ADDRESSING_MODE(MODE) (((MODE) == FMPSMBUS_ADDRESSINGMODE_7BIT) || \ - ((MODE) == FMPSMBUS_ADDRESSINGMODE_10BIT)) + ((MODE) == FMPSMBUS_ADDRESSINGMODE_10BIT)) #define IS_FMPSMBUS_DUAL_ADDRESS(ADDRESS) (((ADDRESS) == FMPSMBUS_DUALADDRESS_DISABLE) || \ - ((ADDRESS) == FMPSMBUS_DUALADDRESS_ENABLE)) + ((ADDRESS) == FMPSMBUS_DUALADDRESS_ENABLE)) #define IS_FMPSMBUS_OWN_ADDRESS2_MASK(MASK) (((MASK) == FMPSMBUS_OA2_NOMASK) || \ - ((MASK) == FMPSMBUS_OA2_MASK01) || \ - ((MASK) == FMPSMBUS_OA2_MASK02) || \ - ((MASK) == FMPSMBUS_OA2_MASK03) || \ - ((MASK) == FMPSMBUS_OA2_MASK04) || \ - ((MASK) == FMPSMBUS_OA2_MASK05) || \ - ((MASK) == FMPSMBUS_OA2_MASK06) || \ - ((MASK) == FMPSMBUS_OA2_MASK07)) + ((MASK) == FMPSMBUS_OA2_MASK01) || \ + ((MASK) == FMPSMBUS_OA2_MASK02) || \ + ((MASK) == FMPSMBUS_OA2_MASK03) || \ + ((MASK) == FMPSMBUS_OA2_MASK04) || \ + ((MASK) == FMPSMBUS_OA2_MASK05) || \ + ((MASK) == FMPSMBUS_OA2_MASK06) || \ + ((MASK) == FMPSMBUS_OA2_MASK07)) #define IS_FMPSMBUS_GENERAL_CALL(CALL) (((CALL) == FMPSMBUS_GENERALCALL_DISABLE) || \ - ((CALL) == FMPSMBUS_GENERALCALL_ENABLE)) + ((CALL) == FMPSMBUS_GENERALCALL_ENABLE)) #define IS_FMPSMBUS_NO_STRETCH(STRETCH) (((STRETCH) == FMPSMBUS_NOSTRETCH_DISABLE) || \ - ((STRETCH) == FMPSMBUS_NOSTRETCH_ENABLE)) + ((STRETCH) == FMPSMBUS_NOSTRETCH_ENABLE)) #define IS_FMPSMBUS_PEC(PEC) (((PEC) == FMPSMBUS_PEC_DISABLE) || \ - ((PEC) == FMPSMBUS_PEC_ENABLE)) + ((PEC) == FMPSMBUS_PEC_ENABLE)) #define IS_FMPSMBUS_PERIPHERAL_MODE(MODE) (((MODE) == FMPSMBUS_PERIPHERAL_MODE_FMPSMBUS_HOST) || \ - ((MODE) == FMPSMBUS_PERIPHERAL_MODE_FMPSMBUS_SLAVE) || \ - ((MODE) == FMPSMBUS_PERIPHERAL_MODE_FMPSMBUS_SLAVE_ARP)) + ((MODE) == FMPSMBUS_PERIPHERAL_MODE_FMPSMBUS_SLAVE) || \ + ((MODE) == FMPSMBUS_PERIPHERAL_MODE_FMPSMBUS_SLAVE_ARP)) #define IS_FMPSMBUS_TRANSFER_MODE(MODE) (((MODE) == FMPSMBUS_RELOAD_MODE) || \ - ((MODE) == FMPSMBUS_AUTOEND_MODE) || \ - ((MODE) == FMPSMBUS_SOFTEND_MODE) || \ - ((MODE) == FMPSMBUS_SENDPEC_MODE) || \ - ((MODE) == (FMPSMBUS_RELOAD_MODE | FMPSMBUS_SENDPEC_MODE)) || \ - ((MODE) == (FMPSMBUS_AUTOEND_MODE | FMPSMBUS_SENDPEC_MODE)) || \ - ((MODE) == (FMPSMBUS_AUTOEND_MODE | FMPSMBUS_RELOAD_MODE)) || \ - ((MODE) == (FMPSMBUS_AUTOEND_MODE | FMPSMBUS_SENDPEC_MODE | \ - FMPSMBUS_RELOAD_MODE ))) + ((MODE) == FMPSMBUS_AUTOEND_MODE) || \ + ((MODE) == FMPSMBUS_SOFTEND_MODE) || \ + ((MODE) == FMPSMBUS_SENDPEC_MODE) || \ + ((MODE) == (FMPSMBUS_RELOAD_MODE | FMPSMBUS_SENDPEC_MODE)) || \ + ((MODE) == (FMPSMBUS_AUTOEND_MODE | FMPSMBUS_SENDPEC_MODE)) || \ + ((MODE) == (FMPSMBUS_AUTOEND_MODE | FMPSMBUS_RELOAD_MODE)) || \ + ((MODE) == (FMPSMBUS_AUTOEND_MODE | FMPSMBUS_SENDPEC_MODE | \ + FMPSMBUS_RELOAD_MODE ))) #define IS_FMPSMBUS_TRANSFER_REQUEST(REQUEST) (((REQUEST) == FMPSMBUS_GENERATE_STOP) || \ - ((REQUEST) == FMPSMBUS_GENERATE_START_READ) || \ - ((REQUEST) == FMPSMBUS_GENERATE_START_WRITE) || \ - ((REQUEST) == FMPSMBUS_NO_STARTSTOP)) + ((REQUEST) == FMPSMBUS_GENERATE_START_READ) || \ + ((REQUEST) == FMPSMBUS_GENERATE_START_WRITE) || \ + ((REQUEST) == FMPSMBUS_NO_STARTSTOP)) #define IS_FMPSMBUS_TRANSFER_OPTIONS_REQUEST(REQUEST) (IS_FMPSMBUS_TRANSFER_OTHER_OPTIONS_REQUEST(REQUEST) || \ - ((REQUEST) == FMPSMBUS_FIRST_FRAME) || \ - ((REQUEST) == FMPSMBUS_NEXT_FRAME) || \ - ((REQUEST) == FMPSMBUS_FIRST_AND_LAST_FRAME_NO_PEC) || \ - ((REQUEST) == FMPSMBUS_LAST_FRAME_NO_PEC) || \ - ((REQUEST) == FMPSMBUS_FIRST_FRAME_WITH_PEC) || \ - ((REQUEST) == FMPSMBUS_FIRST_AND_LAST_FRAME_WITH_PEC) || \ - ((REQUEST) == FMPSMBUS_LAST_FRAME_WITH_PEC)) + ((REQUEST) == FMPSMBUS_FIRST_FRAME) || \ + ((REQUEST) == FMPSMBUS_NEXT_FRAME) || \ + ((REQUEST) == FMPSMBUS_FIRST_AND_LAST_FRAME_NO_PEC) || \ + ((REQUEST) == FMPSMBUS_LAST_FRAME_NO_PEC) || \ + ((REQUEST) == FMPSMBUS_FIRST_FRAME_WITH_PEC) || \ + ((REQUEST) == FMPSMBUS_FIRST_AND_LAST_FRAME_WITH_PEC) || \ + ((REQUEST) == FMPSMBUS_LAST_FRAME_WITH_PEC)) #define IS_FMPSMBUS_TRANSFER_OTHER_OPTIONS_REQUEST(REQUEST) (((REQUEST) == FMPSMBUS_OTHER_FRAME_NO_PEC) || \ - ((REQUEST) == FMPSMBUS_OTHER_AND_LAST_FRAME_NO_PEC) || \ - ((REQUEST) == FMPSMBUS_OTHER_FRAME_WITH_PEC) || \ - ((REQUEST) == FMPSMBUS_OTHER_AND_LAST_FRAME_WITH_PEC)) + ((REQUEST) == FMPSMBUS_OTHER_AND_LAST_FRAME_NO_PEC) || \ + ((REQUEST) == FMPSMBUS_OTHER_FRAME_WITH_PEC) || \ + ((REQUEST) == FMPSMBUS_OTHER_AND_LAST_FRAME_WITH_PEC)) #define FMPSMBUS_RESET_CR1(__HANDLE__) ((__HANDLE__)->Instance->CR1 &= \ - (uint32_t)~((uint32_t)(FMPI2C_CR1_SMBHEN | FMPI2C_CR1_SMBDEN | \ - FMPI2C_CR1_PECEN))) + (uint32_t)~((uint32_t)(FMPI2C_CR1_SMBHEN | FMPI2C_CR1_SMBDEN | \ + FMPI2C_CR1_PECEN))) #define FMPSMBUS_RESET_CR2(__HANDLE__) ((__HANDLE__)->Instance->CR2 &= \ - (uint32_t)~((uint32_t)(FMPI2C_CR2_SADD | FMPI2C_CR2_HEAD10R | \ - FMPI2C_CR2_NBYTES | FMPI2C_CR2_RELOAD | \ - FMPI2C_CR2_RD_WRN))) + (uint32_t)~((uint32_t)(FMPI2C_CR2_SADD | FMPI2C_CR2_HEAD10R | \ + FMPI2C_CR2_NBYTES | FMPI2C_CR2_RELOAD | \ + FMPI2C_CR2_RD_WRN))) #define FMPSMBUS_GENERATE_START(__ADDMODE__,__ADDRESS__) (((__ADDMODE__) == FMPSMBUS_ADDRESSINGMODE_7BIT) ? \ - (uint32_t)((((uint32_t)(__ADDRESS__) & (FMPI2C_CR2_SADD)) | \ - (FMPI2C_CR2_START) | (FMPI2C_CR2_AUTOEND)) & \ - (~FMPI2C_CR2_RD_WRN)) : \ - (uint32_t)((((uint32_t)(__ADDRESS__) & \ - (FMPI2C_CR2_SADD)) | (FMPI2C_CR2_ADD10) | \ - (FMPI2C_CR2_START)) & (~FMPI2C_CR2_RD_WRN))) + (uint32_t)((((uint32_t)(__ADDRESS__) & (FMPI2C_CR2_SADD)) | \ + (FMPI2C_CR2_START) | (FMPI2C_CR2_AUTOEND)) & \ + (~FMPI2C_CR2_RD_WRN)) : \ + (uint32_t)((((uint32_t)(__ADDRESS__) & \ + (FMPI2C_CR2_SADD)) | (FMPI2C_CR2_ADD10) | \ + (FMPI2C_CR2_START)) & (~FMPI2C_CR2_RD_WRN))) #define FMPSMBUS_GET_ADDR_MATCH(__HANDLE__) (((__HANDLE__)->Instance->ISR & FMPI2C_ISR_ADDCODE) >> 17U) #define FMPSMBUS_GET_DIR(__HANDLE__) (((__HANDLE__)->Instance->ISR & FMPI2C_ISR_DIR) >> 16U) @@ -643,7 +644,7 @@ typedef void (*pFMPSMBUS_AddrCallbackTypeDef)(FMPSMBUS_HandleTypeDef *hfmpsmbus #define FMPSMBUS_GET_ALERT_ENABLED(__HANDLE__) ((__HANDLE__)->Instance->CR1 & FMPI2C_CR1_ALERTEN) #define FMPSMBUS_CHECK_FLAG(__ISR__, __FLAG__) ((((__ISR__) & ((__FLAG__) & FMPSMBUS_FLAG_MASK)) == \ - ((__FLAG__) & FMPSMBUS_FLAG_MASK)) ? SET : RESET) + ((__FLAG__) & FMPSMBUS_FLAG_MASK)) ? SET : RESET) #define FMPSMBUS_CHECK_IT_SOURCE(__CR1__, __IT__) ((((__CR1__) & (__IT__)) == (__IT__)) ? SET : RESET) #define IS_FMPSMBUS_OWN_ADDRESS1(ADDRESS1) ((ADDRESS1) <= 0x000003FFU) @@ -676,13 +677,13 @@ HAL_StatusTypeDef HAL_FMPSMBUS_ConfigDigitalFilter(FMPSMBUS_HandleTypeDef *hfmps /* Callbacks Register/UnRegister functions ***********************************/ #if (USE_HAL_FMPSMBUS_REGISTER_CALLBACKS == 1) HAL_StatusTypeDef HAL_FMPSMBUS_RegisterCallback(FMPSMBUS_HandleTypeDef *hfmpsmbus, - HAL_FMPSMBUS_CallbackIDTypeDef CallbackID, - pFMPSMBUS_CallbackTypeDef pCallback); + HAL_FMPSMBUS_CallbackIDTypeDef CallbackID, + pFMPSMBUS_CallbackTypeDef pCallback); HAL_StatusTypeDef HAL_FMPSMBUS_UnRegisterCallback(FMPSMBUS_HandleTypeDef *hfmpsmbus, - HAL_FMPSMBUS_CallbackIDTypeDef CallbackID); + HAL_FMPSMBUS_CallbackIDTypeDef CallbackID); HAL_StatusTypeDef HAL_FMPSMBUS_RegisterAddrCallback(FMPSMBUS_HandleTypeDef *hfmpsmbus, - pFMPSMBUS_AddrCallbackTypeDef pCallback); + pFMPSMBUS_AddrCallbackTypeDef pCallback); HAL_StatusTypeDef HAL_FMPSMBUS_UnRegisterAddrCallback(FMPSMBUS_HandleTypeDef *hfmpsmbus); #endif /* USE_HAL_FMPSMBUS_REGISTER_CALLBACKS */ /** @@ -699,7 +700,7 @@ HAL_StatusTypeDef HAL_FMPSMBUS_UnRegisterAddrCallback(FMPSMBUS_HandleTypeDef *hf */ /******* Blocking mode: Polling */ HAL_StatusTypeDef HAL_FMPSMBUS_IsDeviceReady(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint16_t DevAddress, uint32_t Trials, - uint32_t Timeout); + uint32_t Timeout); /** * @} */ @@ -709,14 +710,14 @@ HAL_StatusTypeDef HAL_FMPSMBUS_IsDeviceReady(FMPSMBUS_HandleTypeDef *hfmpsmbus, */ /******* Non-Blocking mode: Interrupt */ HAL_StatusTypeDef HAL_FMPSMBUS_Master_Transmit_IT(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint16_t DevAddress, - uint8_t *pData, uint16_t Size, uint32_t XferOptions); + uint8_t *pData, uint16_t Size, uint32_t XferOptions); HAL_StatusTypeDef HAL_FMPSMBUS_Master_Receive_IT(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint16_t DevAddress, - uint8_t *pData, uint16_t Size, uint32_t XferOptions); + uint8_t *pData, uint16_t Size, uint32_t XferOptions); HAL_StatusTypeDef HAL_FMPSMBUS_Master_Abort_IT(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint16_t DevAddress); HAL_StatusTypeDef HAL_FMPSMBUS_Slave_Transmit_IT(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint8_t *pData, uint16_t Size, - uint32_t XferOptions); + uint32_t XferOptions); HAL_StatusTypeDef HAL_FMPSMBUS_Slave_Receive_IT(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint8_t *pData, uint16_t Size, - uint32_t XferOptions); + uint32_t XferOptions); HAL_StatusTypeDef HAL_FMPSMBUS_EnableAlert_IT(FMPSMBUS_HandleTypeDef *hfmpsmbus); HAL_StatusTypeDef HAL_FMPSMBUS_DisableAlert_IT(FMPSMBUS_HandleTypeDef *hfmpsmbus); @@ -749,8 +750,8 @@ void HAL_FMPSMBUS_ErrorCallback(FMPSMBUS_HandleTypeDef *hfmpsmbus); */ /* Peripheral State and Errors functions **************************************************/ -uint32_t HAL_FMPSMBUS_GetState(FMPSMBUS_HandleTypeDef *hfmpsmbus); -uint32_t HAL_FMPSMBUS_GetError(FMPSMBUS_HandleTypeDef *hfmpsmbus); +uint32_t HAL_FMPSMBUS_GetState(const FMPSMBUS_HandleTypeDef *hfmpsmbus); +uint32_t HAL_FMPSMBUS_GetError(const FMPSMBUS_HandleTypeDef *hfmpsmbus); /** * @} diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpsmbus_ex.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpsmbus_ex.h index 001dc547..e83a598d 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpsmbus_ex.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_fmpsmbus_ex.h @@ -104,9 +104,9 @@ void HAL_FMPSMBUSEx_DisableFastModePlus(uint32_t ConfigFastModePlus); * @{ */ #define IS_FMPSMBUS_FASTMODEPLUS(__CONFIG__) ((((__CONFIG__) & (FMPSMBUS_FASTMODEPLUS_SCL)) == \ - FMPSMBUS_FASTMODEPLUS_SCL) || \ - (((__CONFIG__) & (FMPSMBUS_FASTMODEPLUS_SDA)) == \ - FMPSMBUS_FASTMODEPLUS_SDA)) + FMPSMBUS_FASTMODEPLUS_SCL) || \ + (((__CONFIG__) & (FMPSMBUS_FASTMODEPLUS_SDA)) == \ + FMPSMBUS_FASTMODEPLUS_SDA)) /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_hcd.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_hcd.h index 9cd14862..a6fee35e 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_hcd.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_hcd.h @@ -159,6 +159,10 @@ typedef struct #define __HAL_HCD_GET_FLAG(__HANDLE__, __INTERRUPT__) ((USB_ReadInterrupts((__HANDLE__)->Instance)\ & (__INTERRUPT__)) == (__INTERRUPT__)) + +#define __HAL_HCD_GET_CH_FLAG(__HANDLE__, __chnum__, __INTERRUPT__) \ + ((USB_ReadChInterrupts((__HANDLE__)->Instance, (__chnum__)) & (__INTERRUPT__)) == (__INTERRUPT__)) + #define __HAL_HCD_CLEAR_FLAG(__HANDLE__, __INTERRUPT__) (((__HANDLE__)->Instance->GINTSTS) = (__INTERRUPT__)) #define __HAL_HCD_IS_INVALID_INTERRUPT(__HANDLE__) (USB_ReadInterrupts((__HANDLE__)->Instance) == 0U) @@ -167,6 +171,9 @@ typedef struct #define __HAL_HCD_UNMASK_HALT_HC_INT(chnum) (USBx_HC(chnum)->HCINTMSK |= USB_OTG_HCINTMSK_CHHM) #define __HAL_HCD_MASK_ACK_HC_INT(chnum) (USBx_HC(chnum)->HCINTMSK &= ~USB_OTG_HCINTMSK_ACKM) #define __HAL_HCD_UNMASK_ACK_HC_INT(chnum) (USBx_HC(chnum)->HCINTMSK |= USB_OTG_HCINTMSK_ACKM) +#define __HAL_HCD_SET_HC_CSPLT(chnum) (USBx_HC(chnum)->HCSPLT |= USB_OTG_HCSPLT_COMPLSPLT) +#define __HAL_HCD_CLEAR_HC_CSPLT(chnum) (USBx_HC(chnum)->HCSPLT &= ~USB_OTG_HCSPLT_COMPLSPLT) +#define __HAL_HCD_CLEAR_HC_SSPLT(chnum) (USBx_HC(chnum)->HCSPLT &= ~USB_OTG_HCSPLT_SPLITEN) /** * @} */ @@ -248,6 +255,11 @@ HAL_StatusTypeDef HAL_HCD_HC_SubmitRequest(HCD_HandleTypeDef *hhcd, uint8_t ch_n uint8_t token, uint8_t *pbuff, uint16_t length, uint8_t do_ping); +HAL_StatusTypeDef HAL_HCD_HC_SetHubInfo(HCD_HandleTypeDef *hhcd, uint8_t ch_num, + uint8_t addr, uint8_t PortNbr); + +HAL_StatusTypeDef HAL_HCD_HC_ClearHubInfo(HCD_HandleTypeDef *hhcd, uint8_t ch_num); + /* Non-Blocking mode: Interrupt */ void HAL_HCD_IRQHandler(HCD_HandleTypeDef *hhcd); void HAL_HCD_WKUP_IRQHandler(HCD_HandleTypeDef *hhcd); @@ -277,16 +289,13 @@ HAL_StatusTypeDef HAL_HCD_Stop(HCD_HandleTypeDef *hhcd); /** @addtogroup HCD_Exported_Functions_Group4 Peripheral State functions * @{ */ -HCD_StateTypeDef HAL_HCD_GetState(HCD_HandleTypeDef *hhcd); -HCD_URBStateTypeDef HAL_HCD_HC_GetURBState(HCD_HandleTypeDef *hhcd, uint8_t chnum); -HCD_HCStateTypeDef HAL_HCD_HC_GetState(HCD_HandleTypeDef *hhcd, uint8_t chnum); -uint32_t HAL_HCD_HC_GetXferCount(HCD_HandleTypeDef *hhcd, uint8_t chnum); +HCD_StateTypeDef HAL_HCD_GetState(HCD_HandleTypeDef const *hhcd); +HCD_URBStateTypeDef HAL_HCD_HC_GetURBState(HCD_HandleTypeDef const *hhcd, uint8_t chnum); +HCD_HCStateTypeDef HAL_HCD_HC_GetState(HCD_HandleTypeDef const *hhcd, uint8_t chnum); +uint32_t HAL_HCD_HC_GetXferCount(HCD_HandleTypeDef const *hhcd, uint8_t chnum); uint32_t HAL_HCD_GetCurrentFrame(HCD_HandleTypeDef *hhcd); uint32_t HAL_HCD_GetCurrentSpeed(HCD_HandleTypeDef *hhcd); -/** - * @} - */ /** * @} @@ -307,6 +316,9 @@ uint32_t HAL_HCD_GetCurrentSpeed(HCD_HandleTypeDef *hhcd); /** * @} */ +/** + * @} + */ #endif /* defined (USB_OTG_FS) || defined (USB_OTG_HS) */ #ifdef __cplusplus diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_irda.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_irda.h index 0d1c5f15..5735215d 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_irda.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_irda.h @@ -593,8 +593,8 @@ void HAL_IRDA_AbortReceiveCpltCallback(IRDA_HandleTypeDef *hirda); * @{ */ /* Peripheral State functions **************************************************/ -HAL_IRDA_StateTypeDef HAL_IRDA_GetState(IRDA_HandleTypeDef *hirda); -uint32_t HAL_IRDA_GetError(IRDA_HandleTypeDef *hirda); +HAL_IRDA_StateTypeDef HAL_IRDA_GetState(const IRDA_HandleTypeDef *hirda); +uint32_t HAL_IRDA_GetError(const IRDA_HandleTypeDef *hirda); /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_lptim.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_lptim.h index 7a41ee9d..fba2d52c 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_lptim.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_lptim.h @@ -689,9 +689,9 @@ HAL_StatusTypeDef HAL_LPTIM_Counter_Stop_IT(LPTIM_HandleTypeDef *hlptim); * @{ */ /* Reading operation functions ************************************************/ -uint32_t HAL_LPTIM_ReadCounter(LPTIM_HandleTypeDef *hlptim); -uint32_t HAL_LPTIM_ReadAutoReload(LPTIM_HandleTypeDef *hlptim); -uint32_t HAL_LPTIM_ReadCompare(LPTIM_HandleTypeDef *hlptim); +uint32_t HAL_LPTIM_ReadCounter(const LPTIM_HandleTypeDef *hlptim); +uint32_t HAL_LPTIM_ReadAutoReload(const LPTIM_HandleTypeDef *hlptim); +uint32_t HAL_LPTIM_ReadCompare(const LPTIM_HandleTypeDef *hlptim); /** * @} */ @@ -727,7 +727,7 @@ HAL_StatusTypeDef HAL_LPTIM_UnRegisterCallback(LPTIM_HandleTypeDef *lphtim, HAL_ * @{ */ /* Peripheral State functions ************************************************/ -HAL_LPTIM_StateTypeDef HAL_LPTIM_GetState(LPTIM_HandleTypeDef *hlptim); +HAL_LPTIM_StateTypeDef HAL_LPTIM_GetState(const LPTIM_HandleTypeDef *hlptim); /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_ltdc.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_ltdc.h index ab951d3e..78349ebb 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_ltdc.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_ltdc.h @@ -338,14 +338,14 @@ typedef void (*pLTDC_CallbackTypeDef)(LTDC_HandleTypeDef *hltdc); /*!< pointer /** @defgroup LTDC_Pixelformat LTDC Pixel format * @{ */ -#define LTDC_PIXEL_FORMAT_ARGB8888 0x00000000U /*!< ARGB8888 LTDC pixel format */ -#define LTDC_PIXEL_FORMAT_RGB888 0x00000001U /*!< RGB888 LTDC pixel format */ -#define LTDC_PIXEL_FORMAT_RGB565 0x00000002U /*!< RGB565 LTDC pixel format */ -#define LTDC_PIXEL_FORMAT_ARGB1555 0x00000003U /*!< ARGB1555 LTDC pixel format */ -#define LTDC_PIXEL_FORMAT_ARGB4444 0x00000004U /*!< ARGB4444 LTDC pixel format */ -#define LTDC_PIXEL_FORMAT_L8 0x00000005U /*!< L8 LTDC pixel format */ -#define LTDC_PIXEL_FORMAT_AL44 0x00000006U /*!< AL44 LTDC pixel format */ -#define LTDC_PIXEL_FORMAT_AL88 0x00000007U /*!< AL88 LTDC pixel format */ +#define LTDC_PIXEL_FORMAT_ARGB8888 0x00000000U /*!< ARGB8888 LTDC pixel format */ +#define LTDC_PIXEL_FORMAT_RGB888 0x00000001U /*!< RGB888 LTDC pixel format */ +#define LTDC_PIXEL_FORMAT_RGB565 0x00000002U /*!< RGB565 LTDC pixel format */ +#define LTDC_PIXEL_FORMAT_ARGB1555 0x00000003U /*!< ARGB1555 LTDC pixel format */ +#define LTDC_PIXEL_FORMAT_ARGB4444 0x00000004U /*!< ARGB4444 LTDC pixel format */ +#define LTDC_PIXEL_FORMAT_L8 0x00000005U /*!< L8 LTDC pixel format */ +#define LTDC_PIXEL_FORMAT_AL44 0x00000006U /*!< AL44 LTDC pixel format */ +#define LTDC_PIXEL_FORMAT_AL88 0x00000007U /*!< AL88 LTDC pixel format */ /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_nand.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_nand.h index 26f65025..9fc12a01 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_nand.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_nand.h @@ -109,9 +109,8 @@ typedef struct FunctionalState ExtraCommandEnable; /*!< NAND extra command needed for Page reading mode. This parameter is mandatory for some NAND parts after the read command (NAND_CMD_AREA_TRUE1) and before DATA reading sequence. - Example: Toshiba THTH58BYG3S0HBAI6. This parameter could be ENABLE or DISABLE - Please check the Read Mode sequnece in the NAND device datasheet */ + Please check the Read Mode sequence in the NAND device datasheet */ } NAND_DeviceConfigTypeDef; /** @@ -131,7 +130,7 @@ typedef struct __IO HAL_NAND_StateTypeDef State; /*!< NAND device access state */ - NAND_DeviceConfigTypeDef Config; /*!< NAND phusical characteristic information structure */ + NAND_DeviceConfigTypeDef Config; /*!< NAND physical characteristic information structure */ #if (USE_HAL_NAND_REGISTER_CALLBACKS == 1) void (* MspInitCallback)(struct __NAND_HandleTypeDef *hnand); /*!< NAND Msp Init callback */ @@ -219,27 +218,27 @@ void HAL_NAND_ITCallback(NAND_HandleTypeDef *hnand); /* IO operation functions ****************************************************/ HAL_StatusTypeDef HAL_NAND_Reset(NAND_HandleTypeDef *hnand); -HAL_StatusTypeDef HAL_NAND_Read_Page_8b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, uint8_t *pBuffer, - uint32_t NumPageToRead); -HAL_StatusTypeDef HAL_NAND_Write_Page_8b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, uint8_t *pBuffer, - uint32_t NumPageToWrite); -HAL_StatusTypeDef HAL_NAND_Read_SpareArea_8b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, +HAL_StatusTypeDef HAL_NAND_Read_Page_8b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, + uint8_t *pBuffer, uint32_t NumPageToRead); +HAL_StatusTypeDef HAL_NAND_Write_Page_8b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, + const uint8_t *pBuffer, uint32_t NumPageToWrite); +HAL_StatusTypeDef HAL_NAND_Read_SpareArea_8b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, uint8_t *pBuffer, uint32_t NumSpareAreaToRead); -HAL_StatusTypeDef HAL_NAND_Write_SpareArea_8b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, - uint8_t *pBuffer, uint32_t NumSpareAreaTowrite); - -HAL_StatusTypeDef HAL_NAND_Read_Page_16b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, uint16_t *pBuffer, - uint32_t NumPageToRead); -HAL_StatusTypeDef HAL_NAND_Write_Page_16b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, uint16_t *pBuffer, - uint32_t NumPageToWrite); -HAL_StatusTypeDef HAL_NAND_Read_SpareArea_16b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, +HAL_StatusTypeDef HAL_NAND_Write_SpareArea_8b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, + const uint8_t *pBuffer, uint32_t NumSpareAreaTowrite); + +HAL_StatusTypeDef HAL_NAND_Read_Page_16b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, + uint16_t *pBuffer, uint32_t NumPageToRead); +HAL_StatusTypeDef HAL_NAND_Write_Page_16b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, + const uint16_t *pBuffer, uint32_t NumPageToWrite); +HAL_StatusTypeDef HAL_NAND_Read_SpareArea_16b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, uint16_t *pBuffer, uint32_t NumSpareAreaToRead); -HAL_StatusTypeDef HAL_NAND_Write_SpareArea_16b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, - uint16_t *pBuffer, uint32_t NumSpareAreaTowrite); +HAL_StatusTypeDef HAL_NAND_Write_SpareArea_16b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, + const uint16_t *pBuffer, uint32_t NumSpareAreaTowrite); -HAL_StatusTypeDef HAL_NAND_Erase_Block(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress); +HAL_StatusTypeDef HAL_NAND_Erase_Block(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress); -uint32_t HAL_NAND_Address_Inc(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress); +uint32_t HAL_NAND_Address_Inc(const NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress); #if (USE_HAL_NAND_REGISTER_CALLBACKS == 1) /* NAND callback registering/unregistering */ @@ -269,8 +268,8 @@ HAL_StatusTypeDef HAL_NAND_GetECC(NAND_HandleTypeDef *hnand, uint32_t *ECCval, * @{ */ /* NAND State functions *******************************************************/ -HAL_NAND_StateTypeDef HAL_NAND_GetState(NAND_HandleTypeDef *hnand); -uint32_t HAL_NAND_Read_Status(NAND_HandleTypeDef *hnand); +HAL_NAND_StateTypeDef HAL_NAND_GetState(const NAND_HandleTypeDef *hnand); +uint32_t HAL_NAND_Read_Status(const NAND_HandleTypeDef *hnand); /** * @} */ @@ -290,7 +289,7 @@ uint32_t HAL_NAND_Read_Status(NAND_HandleTypeDef *hnand); #define NAND_DEVICE2 0x80000000UL #else #define NAND_DEVICE 0x80000000UL -#endif +#endif /* NAND_SECOND_BANK */ #define NAND_WRITE_TIMEOUT 0x01000000UL #define CMD_AREA (1UL<<16U) /* A16 = CLE high */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_nor.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_nor.h index cc7f1b5e..427c2ccd 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_nor.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_nor.h @@ -238,7 +238,7 @@ HAL_StatusTypeDef HAL_NOR_WriteOperation_Disable(NOR_HandleTypeDef *hnor); */ /* NOR State functions ********************************************************/ -HAL_NOR_StateTypeDef HAL_NOR_GetState(NOR_HandleTypeDef *hnor); +HAL_NOR_StateTypeDef HAL_NOR_GetState(const NOR_HandleTypeDef *hnor); HAL_NOR_StatusTypeDef HAL_NOR_GetStatus(NOR_HandleTypeDef *hnor, uint32_t Address, uint32_t Timeout); /** * @} diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_pcd.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_pcd.h index 92488b2f..de1ec241 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_pcd.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_pcd.h @@ -190,14 +190,14 @@ typedef struct * @brief macros to handle interrupts and specific clock configurations * @{ */ -#if defined (USB_OTG_FS) || defined (USB_OTG_HS) #define __HAL_PCD_ENABLE(__HANDLE__) (void)USB_EnableGlobalInt ((__HANDLE__)->Instance) #define __HAL_PCD_DISABLE(__HANDLE__) (void)USB_DisableGlobalInt ((__HANDLE__)->Instance) #define __HAL_PCD_GET_FLAG(__HANDLE__, __INTERRUPT__) \ ((USB_ReadInterrupts((__HANDLE__)->Instance) & (__INTERRUPT__)) == (__INTERRUPT__)) -#define __HAL_PCD_CLEAR_FLAG(__HANDLE__, __INTERRUPT__) (((__HANDLE__)->Instance->GINTSTS) &= (__INTERRUPT__)) +#if defined (USB_OTG_FS) || defined (USB_OTG_HS) +#define __HAL_PCD_CLEAR_FLAG(__HANDLE__, __INTERRUPT__) (((__HANDLE__)->Instance->GINTSTS) &= (__INTERRUPT__)) #define __HAL_PCD_IS_INVALID_INTERRUPT(__HANDLE__) (USB_ReadInterrupts((__HANDLE__)->Instance) == 0U) #define __HAL_PCD_UNGATE_PHYCLOCK(__HANDLE__) \ @@ -368,9 +368,11 @@ HAL_StatusTypeDef HAL_PCD_EP_Flush(PCD_HandleTypeDef *hpcd, uint8_t ep_addr); HAL_StatusTypeDef HAL_PCD_EP_Abort(PCD_HandleTypeDef *hpcd, uint8_t ep_addr); HAL_StatusTypeDef HAL_PCD_ActivateRemoteWakeup(PCD_HandleTypeDef *hpcd); HAL_StatusTypeDef HAL_PCD_DeActivateRemoteWakeup(PCD_HandleTypeDef *hpcd); -HAL_StatusTypeDef HAL_PCD_SetTestMode(PCD_HandleTypeDef *hpcd, uint8_t testmode); +#if defined (USB_OTG_FS) || defined (USB_OTG_HS) +HAL_StatusTypeDef HAL_PCD_SetTestMode(const PCD_HandleTypeDef *hpcd, uint8_t testmode); +#endif /* defined (USB_OTG_FS) || defined (USB_OTG_HS) */ -uint32_t HAL_PCD_EP_GetRxCount(PCD_HandleTypeDef *hpcd, uint8_t ep_addr); +uint32_t HAL_PCD_EP_GetRxCount(PCD_HandleTypeDef const *hpcd, uint8_t ep_addr); /** * @} */ @@ -379,7 +381,7 @@ uint32_t HAL_PCD_EP_GetRxCount(PCD_HandleTypeDef *hpcd, uint8_t ep_addr /** @addtogroup PCD_Exported_Functions_Group4 Peripheral State functions * @{ */ -PCD_StateTypeDef HAL_PCD_GetState(PCD_HandleTypeDef *hpcd); +PCD_StateTypeDef HAL_PCD_GetState(PCD_HandleTypeDef const *hpcd); /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_pcd_ex.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_pcd_ex.h index 72ded2b6..0c6f2e01 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_pcd_ex.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_pcd_ex.h @@ -51,15 +51,21 @@ HAL_StatusTypeDef HAL_PCDEx_SetTxFiFo(PCD_HandleTypeDef *hpcd, uint8_t fifo, uin HAL_StatusTypeDef HAL_PCDEx_SetRxFiFo(PCD_HandleTypeDef *hpcd, uint16_t size); #endif /* defined (USB_OTG_FS) || defined (USB_OTG_HS) */ -#if defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) +#if defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) \ + || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) \ + || defined(STM32F423xx) HAL_StatusTypeDef HAL_PCDEx_ActivateLPM(PCD_HandleTypeDef *hpcd); HAL_StatusTypeDef HAL_PCDEx_DeActivateLPM(PCD_HandleTypeDef *hpcd); -#endif /* defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) */ -#if defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) +#endif /* defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || + defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || + defined(STM32F423xx) */ +#if defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) \ + || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) HAL_StatusTypeDef HAL_PCDEx_ActivateBCD(PCD_HandleTypeDef *hpcd); HAL_StatusTypeDef HAL_PCDEx_DeActivateBCD(PCD_HandleTypeDef *hpcd); void HAL_PCDEx_BCD_VBUSDetect(PCD_HandleTypeDef *hpcd); -#endif /* defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) */ +#endif /* defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || + defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) */ void HAL_PCDEx_LPM_Callback(PCD_HandleTypeDef *hpcd, PCD_LPM_MsgTypeDef msg); void HAL_PCDEx_BCD_Callback(PCD_HandleTypeDef *hpcd, PCD_BCD_MsgTypeDef msg); diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_pwr.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_pwr.h index 3b9ad5a3..c391153b 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_pwr.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_pwr.h @@ -114,6 +114,8 @@ typedef struct */ #define PWR_SLEEPENTRY_WFI ((uint8_t)0x01) #define PWR_SLEEPENTRY_WFE ((uint8_t)0x02) +#define PWR_SLEEPENTRY_WFE_NO_EVT_CLEAR ((uint8_t)0x03) + /** * @} */ @@ -123,6 +125,7 @@ typedef struct */ #define PWR_STOPENTRY_WFI ((uint8_t)0x01) #define PWR_STOPENTRY_WFE ((uint8_t)0x02) +#define PWR_STOPENTRY_WFE_NO_EVT_CLEAR ((uint8_t)0x03) /** * @} */ @@ -401,8 +404,14 @@ void HAL_PWR_DisableSEVOnPend(void); ((MODE) == PWR_PVD_MODE_NORMAL)) #define IS_PWR_REGULATOR(REGULATOR) (((REGULATOR) == PWR_MAINREGULATOR_ON) || \ ((REGULATOR) == PWR_LOWPOWERREGULATOR_ON)) -#define IS_PWR_SLEEP_ENTRY(ENTRY) (((ENTRY) == PWR_SLEEPENTRY_WFI) || ((ENTRY) == PWR_SLEEPENTRY_WFE)) -#define IS_PWR_STOP_ENTRY(ENTRY) (((ENTRY) == PWR_STOPENTRY_WFI) || ((ENTRY) == PWR_STOPENTRY_WFE)) + +#define IS_PWR_SLEEP_ENTRY(ENTRY) (((ENTRY) == PWR_SLEEPENTRY_WFI) || \ + ((ENTRY) == PWR_SLEEPENTRY_WFE) || \ + ((ENTRY) == PWR_SLEEPENTRY_WFE_NO_EVT_CLEAR)) + +#define IS_PWR_STOP_ENTRY(ENTRY) (((ENTRY) == PWR_STOPENTRY_WFI) || \ + ((ENTRY) == PWR_STOPENTRY_WFE) || \ + ((ENTRY) == PWR_STOPENTRY_WFE_NO_EVT_CLEAR)) /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rcc.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rcc.h index dcf58144..c348e5ef 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rcc.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rcc.h @@ -660,7 +660,6 @@ typedef struct * @brief Force or release AHB1 peripheral reset. * @{ */ -#define __HAL_RCC_AHB1_FORCE_RESET() (RCC->AHB1RSTR = 0xFFFFFFFFU) #define __HAL_RCC_GPIOA_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIOARST)) #define __HAL_RCC_GPIOB_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIOBRST)) #define __HAL_RCC_GPIOC_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIOCRST)) @@ -683,7 +682,6 @@ typedef struct * @brief Force or release APB1 peripheral reset. * @{ */ -#define __HAL_RCC_APB1_FORCE_RESET() (RCC->APB1RSTR = 0xFFFFFFFFU) #define __HAL_RCC_TIM5_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM5RST)) #define __HAL_RCC_WWDG_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_WWDGRST)) #define __HAL_RCC_SPI2_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_SPI2RST)) @@ -708,7 +706,6 @@ typedef struct * @brief Force or release APB2 peripheral reset. * @{ */ -#define __HAL_RCC_APB2_FORCE_RESET() (RCC->APB2RSTR = 0xFFFFFFFFU) #define __HAL_RCC_TIM1_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_TIM1RST)) #define __HAL_RCC_USART1_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_USART1RST)) #define __HAL_RCC_USART6_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_USART6RST)) @@ -1407,7 +1404,7 @@ void HAL_RCC_CSSCallback(void); ((__SOURCE__) == RCC_RTCCLKSOURCE_HSE_DIV30) || \ ((__SOURCE__) == RCC_RTCCLKSOURCE_HSE_DIV31)) -#define IS_RCC_PLLM_VALUE(VALUE) ((VALUE) <= 63U) +#define IS_RCC_PLLM_VALUE(VALUE) ((2U <= (VALUE)) && ((VALUE) <= 63U)) #define IS_RCC_PLLP_VALUE(VALUE) (((VALUE) == 2U) || ((VALUE) == 4U) || ((VALUE) == 6U) || ((VALUE) == 8U)) diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rcc_ex.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rcc_ex.h index dbcc98cd..735c8f7c 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rcc_ex.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rcc_ex.h @@ -526,7 +526,7 @@ typedef struct * @{ */ #define RCC_I2SCLKSOURCE_PLLI2S 0x00000000U -#define RCC_I2SCLKSOURCE_EXT 0x00000001U +#define RCC_I2SCLKSOURCE_EXT RCC_CFGR_I2SSRC /** * @} */ @@ -1665,6 +1665,7 @@ typedef struct * @brief Force or release AHB1 peripheral reset. * @{ */ +#define __HAL_RCC_AHB1_FORCE_RESET() (RCC->AHB1RSTR = 0x22E017FFU) #define __HAL_RCC_GPIOD_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIODRST)) #define __HAL_RCC_GPIOE_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIOERST)) #define __HAL_RCC_GPIOF_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIOFRST)) @@ -1696,7 +1697,12 @@ typedef struct * @brief Force or release AHB2 peripheral reset. * @{ */ -#define __HAL_RCC_AHB2_FORCE_RESET() (RCC->AHB2RSTR = 0xFFFFFFFFU) +#if defined(STM32F427xx) || defined(STM32F429xx) || defined(STM32F469xx) +#define __HAL_RCC_AHB2_FORCE_RESET() (RCC->AHB2RSTR = 0x000000C1U) +#endif /* STM32F427xx || STM32F429xx || STM32F469xx */ +#if defined(STM32F437xx) || defined(STM32F439xx) || defined(STM32F479xx) +#define __HAL_RCC_AHB2_FORCE_RESET() (RCC->AHB2RSTR = 0x000000F1U) +#endif /* STM32F437xx || STM32F439xx || STM32F479xx */ #define __HAL_RCC_USB_OTG_FS_FORCE_RESET() (RCC->AHB2RSTR |= (RCC_AHB2RSTR_OTGFSRST)) #define __HAL_RCC_RNG_FORCE_RESET() (RCC->AHB2RSTR |= (RCC_AHB2RSTR_RNGRST)) #define __HAL_RCC_DCMI_FORCE_RESET() (RCC->AHB2RSTR |= (RCC_AHB2RSTR_DCMIRST)) @@ -1721,7 +1727,12 @@ typedef struct * @brief Force or release AHB3 peripheral reset. * @{ */ -#define __HAL_RCC_AHB3_FORCE_RESET() (RCC->AHB3RSTR = 0xFFFFFFFFU) +#if defined(STM32F427xx) || defined(STM32F429xx) || defined(STM32F437xx) || defined(STM32F439xx) +#define __HAL_RCC_AHB3_FORCE_RESET() (RCC->AHB3RSTR = 0x00000001U) +#endif /* STM32F427xx || STM32F429xx || STM32F437xx || STM32F439xx */ +#if defined(STM32F469xx) || defined(STM32F479xx) +#define __HAL_RCC_AHB3_FORCE_RESET() (RCC->AHB3RSTR = 0x00000003U) +#endif /* STM32F469xx || STM32F479xx */ #define __HAL_RCC_AHB3_RELEASE_RESET() (RCC->AHB3RSTR = 0x00U) #define __HAL_RCC_FMC_FORCE_RESET() (RCC->AHB3RSTR |= (RCC_AHB3RSTR_FMCRST)) #define __HAL_RCC_FMC_RELEASE_RESET() (RCC->AHB3RSTR &= ~(RCC_AHB3RSTR_FMCRST)) @@ -1738,6 +1749,7 @@ typedef struct * @brief Force or release APB1 peripheral reset. * @{ */ +#define __HAL_RCC_APB1_FORCE_RESET() (RCC->APB1RSTR = 0xF6FEC9FFU) #define __HAL_RCC_TIM6_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM6RST)) #define __HAL_RCC_TIM7_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM7RST)) #define __HAL_RCC_TIM12_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM12RST)) @@ -1783,6 +1795,15 @@ typedef struct * @brief Force or release APB2 peripheral reset. * @{ */ +#if defined(STM32F469xx) || defined(STM32F479xx) +#define __HAL_RCC_APB2_FORCE_RESET() (RCC->APB2RSTR = 0x0C777933U) +#endif /* STM32F469xx || STM32F479xx */ +#if defined(STM32F429xx) || defined(STM32F439xx) +#define __HAL_RCC_APB2_FORCE_RESET() (RCC->APB2RSTR = 0x04777933U) +#endif /* STM32F429xx || STM32F439xx */ +#if defined(STM32F427xx) || defined(STM32F437xx) +#define __HAL_RCC_APB2_FORCE_RESET() (RCC->APB2RSTR = 0x00777933U) +#endif /* STM32F427xx || STM32F437xx */ #define __HAL_RCC_TIM8_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_TIM8RST)) #define __HAL_RCC_SPI5_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_SPI5RST)) #define __HAL_RCC_SPI6_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_SPI6RST)) @@ -2590,6 +2611,12 @@ typedef struct * @brief Force or release AHB1 peripheral reset. * @{ */ +#if defined (STM32F405xx) || defined (STM32F415xx) +#define __HAL_RCC_AHB1_FORCE_RESET() (RCC->AHB1RSTR = 0x206011FFU) +#endif /* STM32F405xx || STM32F415xx */ +#if defined (STM32F407xx) || defined (STM32F417xx) +#define __HAL_RCC_AHB1_FORCE_RESET() (RCC->AHB1RSTR = 0x226011FFU) +#endif /* STM32F407xx || STM32F417xx */ #define __HAL_RCC_GPIOD_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIODRST)) #define __HAL_RCC_GPIOE_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIOERST)) #define __HAL_RCC_GPIOF_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIOFRST)) @@ -2615,7 +2642,12 @@ typedef struct * @brief Force or release AHB2 peripheral reset. * @{ */ -#define __HAL_RCC_AHB2_FORCE_RESET() (RCC->AHB2RSTR = 0xFFFFFFFFU) +#if defined (STM32F415xx) || defined (STM32F417xx) +#define __HAL_RCC_AHB2_FORCE_RESET() (RCC->AHB2RSTR = 0x000000F1U) +#endif /* STM32F415xx || STM32F417xx */ +#if defined (STM32F405xx) || defined (STM32F407xx) +#define __HAL_RCC_AHB2_FORCE_RESET() (RCC->AHB2RSTR = 0x000000C1U) +#endif /* STM32F405xx || STM32F407xx */ #define __HAL_RCC_AHB2_RELEASE_RESET() (RCC->AHB2RSTR = 0x00U) #if defined(STM32F407xx)|| defined(STM32F417xx) @@ -2644,7 +2676,7 @@ typedef struct * @brief Force or release AHB3 peripheral reset. * @{ */ -#define __HAL_RCC_AHB3_FORCE_RESET() (RCC->AHB3RSTR = 0xFFFFFFFFU) +#define __HAL_RCC_AHB3_FORCE_RESET() (RCC->AHB3RSTR = 0x00000001U) #define __HAL_RCC_AHB3_RELEASE_RESET() (RCC->AHB3RSTR = 0x00U) #define __HAL_RCC_FSMC_FORCE_RESET() (RCC->AHB3RSTR |= (RCC_AHB3RSTR_FSMCRST)) @@ -2657,6 +2689,7 @@ typedef struct * @brief Force or release APB1 peripheral reset. * @{ */ +#define __HAL_RCC_APB1_FORCE_RESET() (RCC->APB1RSTR = 0xF6FEC9FFU) #define __HAL_RCC_TIM6_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM6RST)) #define __HAL_RCC_TIM7_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM7RST)) #define __HAL_RCC_TIM12_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM12RST)) @@ -2698,6 +2731,7 @@ typedef struct * @brief Force or release APB2 peripheral reset. * @{ */ +#define __HAL_RCC_APB2_FORCE_RESET() (RCC->APB2RSTR = 0x04777933U) #define __HAL_RCC_TIM8_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_TIM8RST)) #define __HAL_RCC_SDIO_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_SDIORST)) #define __HAL_RCC_SPI4_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_SPI4RST)) @@ -3097,7 +3131,7 @@ typedef struct * @brief Force or release AHB1 peripheral reset. * @{ */ -#define __HAL_RCC_AHB1_FORCE_RESET() (RCC->AHB1RSTR = 0xFFFFFFFFU) +#define __HAL_RCC_AHB1_FORCE_RESET() (RCC->AHB1RSTR = 0x0060109FU) #define __HAL_RCC_GPIOD_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIODRST)) #define __HAL_RCC_GPIOE_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIOERST)) #define __HAL_RCC_CRC_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_CRCRST)) @@ -3114,7 +3148,7 @@ typedef struct * @brief Force or release AHB2 peripheral reset. * @{ */ -#define __HAL_RCC_AHB2_FORCE_RESET() (RCC->AHB2RSTR = 0xFFFFFFFFU) +#define __HAL_RCC_AHB2_FORCE_RESET() (RCC->AHB2RSTR = 0x00000080U) #define __HAL_RCC_USB_OTG_FS_FORCE_RESET() (RCC->AHB2RSTR |= (RCC_AHB2RSTR_OTGFSRST)) #define __HAL_RCC_AHB2_RELEASE_RESET() (RCC->AHB2RSTR = 0x00U) @@ -3127,7 +3161,7 @@ typedef struct * @brief Force or release APB1 peripheral reset. * @{ */ -#define __HAL_RCC_APB1_FORCE_RESET() (RCC->APB1RSTR = 0xFFFFFFFFU) +#define __HAL_RCC_APB1_FORCE_RESET() (RCC->APB1RSTR = 0x10E2C80FU) #define __HAL_RCC_TIM2_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM2RST)) #define __HAL_RCC_TIM3_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM3RST)) #define __HAL_RCC_TIM4_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM4RST)) @@ -3148,7 +3182,7 @@ typedef struct * @brief Force or release APB2 peripheral reset. * @{ */ -#define __HAL_RCC_APB2_FORCE_RESET() (RCC->APB2RSTR = 0xFFFFFFFFU) +#define __HAL_RCC_APB2_FORCE_RESET() (RCC->APB2RSTR = 0x00077931U) #define __HAL_RCC_SDIO_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_SDIORST)) #define __HAL_RCC_SPI4_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_SPI4RST)) #define __HAL_RCC_TIM10_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_TIM10RST)) @@ -3413,6 +3447,7 @@ typedef struct * @brief Force or release AHB1 peripheral reset. * @{ */ +#define __HAL_RCC_AHB1_FORCE_RESET() (RCC->AHB1RSTR = 0x80601087U) #define __HAL_RCC_CRC_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_CRCRST)) #define __HAL_RCC_RNG_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_RNGRST)) #define __HAL_RCC_CRC_RELEASE_RESET() (RCC->AHB1RSTR &= ~(RCC_AHB1RSTR_CRCRST)) @@ -3445,6 +3480,12 @@ typedef struct * @brief Force or release APB1 peripheral reset. * @{ */ +#if defined (STM32F410Rx) || defined (STM32F410Cx) +#define __HAL_RCC_APB1_FORCE_RESET() (RCC->APB1RSTR = 0x31624A18U) +#endif /* STM32F410Rx || STM32F410Cx */ +#if defined (STM32F410Tx) +#define __HAL_RCC_APB1_FORCE_RESET() (RCC->APB1RSTR = 0x31620A18U) +#endif /* STM32F410Tx */ #define __HAL_RCC_TIM6_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM6RST)) #define __HAL_RCC_LPTIM1_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_LPTIM1RST)) #define __HAL_RCC_FMPI2C1_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_FMPI2C1RST)) @@ -3462,6 +3503,12 @@ typedef struct * @brief Force or release APB2 peripheral reset. * @{ */ +#if defined (STM32F410Rx) || defined (STM32F410Cx) +#define __HAL_RCC_APB2_FORCE_RESET() (RCC->APB2RSTR = 0x00155131U) +#endif /* STM32F410Rx || STM32F410Cx */ +#if defined (STM32F410Tx) +#define __HAL_RCC_APB2_FORCE_RESET() (RCC->APB2RSTR = 0x00055111U) +#endif /* STM32F410Tx */ #define __HAL_RCC_SPI5_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_SPI5RST)) #define __HAL_RCC_SPI5_RELEASE_RESET() (RCC->APB2RSTR &= ~(RCC_APB2RSTR_SPI5RST)) /** @@ -3754,6 +3801,7 @@ typedef struct * @brief Force or release AHB1 peripheral reset. * @{ */ +#define __HAL_RCC_AHB1_FORCE_RESET() (RCC->AHB1RSTR = 0x0060109FU) #define __HAL_RCC_GPIOD_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIODRST)) #define __HAL_RCC_GPIOE_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIOERST)) #define __HAL_RCC_CRC_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_CRCRST)) @@ -3769,7 +3817,7 @@ typedef struct * @brief Force or release AHB2 peripheral reset. * @{ */ -#define __HAL_RCC_AHB2_FORCE_RESET() (RCC->AHB2RSTR = 0xFFFFFFFFU) +#define __HAL_RCC_AHB2_FORCE_RESET() (RCC->AHB2RSTR = 0x00000080U) #define __HAL_RCC_USB_OTG_FS_FORCE_RESET() (RCC->AHB2RSTR |= (RCC_AHB2RSTR_OTGFSRST)) #define __HAL_RCC_AHB2_RELEASE_RESET() (RCC->AHB2RSTR = 0x00U) @@ -3792,6 +3840,7 @@ typedef struct * @brief Force or release APB1 peripheral reset. * @{ */ +#define __HAL_RCC_APB1_FORCE_RESET() (RCC->APB1RSTR = 0x10E2C80FU) #define __HAL_RCC_TIM2_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM2RST)) #define __HAL_RCC_TIM3_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM3RST)) #define __HAL_RCC_TIM4_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM4RST)) @@ -3811,6 +3860,7 @@ typedef struct * @brief Force or release APB2 peripheral reset. * @{ */ +#define __HAL_RCC_APB2_FORCE_RESET() (RCC->APB2RSTR = 0x00177931U) #define __HAL_RCC_SPI5_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_SPI5RST)) #define __HAL_RCC_SDIO_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_SDIORST)) #define __HAL_RCC_SPI4_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_SPI4RST)) @@ -4430,6 +4480,7 @@ typedef struct * @brief Force or release AHB1 peripheral reset. * @{ */ +#define __HAL_RCC_AHB1_FORCE_RESET() (RCC->AHB1RSTR = 0x206010FFU) #define __HAL_RCC_GPIOD_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIODRST)) #define __HAL_RCC_GPIOE_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIOERST)) #define __HAL_RCC_GPIOF_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIOFRST)) @@ -4451,7 +4502,7 @@ typedef struct * @brief Force or release AHB2 peripheral reset. * @{ */ -#define __HAL_RCC_AHB2_FORCE_RESET() (RCC->AHB2RSTR = 0xFFFFFFFFU) +#define __HAL_RCC_AHB2_FORCE_RESET() (RCC->AHB2RSTR = 0x00000081U) #define __HAL_RCC_USB_OTG_FS_FORCE_RESET() (RCC->AHB2RSTR |= (RCC_AHB2RSTR_OTGFSRST)) #define __HAL_RCC_RNG_FORCE_RESET() (RCC->AHB2RSTR |= (RCC_AHB2RSTR_RNGRST)) #define __HAL_RCC_DCMI_FORCE_RESET() (RCC->AHB2RSTR |= (RCC_AHB2RSTR_DCMIRST)) @@ -4468,7 +4519,7 @@ typedef struct * @brief Force or release AHB3 peripheral reset. * @{ */ -#define __HAL_RCC_AHB3_FORCE_RESET() (RCC->AHB3RSTR = 0xFFFFFFFFU) +#define __HAL_RCC_AHB3_FORCE_RESET() (RCC->AHB3RSTR = 0x00000003U) #define __HAL_RCC_AHB3_RELEASE_RESET() (RCC->AHB3RSTR = 0x00U) #define __HAL_RCC_FMC_FORCE_RESET() (RCC->AHB3RSTR |= (RCC_AHB3RSTR_FMCRST)) @@ -4484,6 +4535,7 @@ typedef struct * @brief Force or release APB1 peripheral reset. * @{ */ +#define __HAL_RCC_APB1_FORCE_RESET() (RCC->APB1RSTR = 0x3FFFC9FFU) #define __HAL_RCC_TIM6_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM6RST)) #define __HAL_RCC_TIM7_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM7RST)) #define __HAL_RCC_TIM12_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM12RST)) @@ -4531,6 +4583,7 @@ typedef struct * @brief Force or release APB2 peripheral reset. * @{ */ +#define __HAL_RCC_APB2_FORCE_RESET() (RCC->APB2RSTR = 0x00C77933U) #define __HAL_RCC_TIM8_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_TIM8RST)) #define __HAL_RCC_SAI1_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_SAI1RST)) #define __HAL_RCC_SAI2_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_SAI2RST)) @@ -5343,6 +5396,18 @@ typedef struct * @brief Force or release AHB1 peripheral reset. * @{ */ +#if defined (STM32F412Zx) || defined(STM32F413xx) || defined (STM32F423xx) +#define __HAL_RCC_AHB1_FORCE_RESET() (RCC->AHB1RSTR = 0x006010FFU) +#endif /* STM32F412Zx || STM32F413xx || STM32F423xx */ +#if defined (STM32F412Cx) +#define __HAL_RCC_AHB1_FORCE_RESET() (RCC->AHB1RSTR = 0x00601087U) +#endif /* STM32F412Cx */ +#if defined (STM32F412Vx) +#define __HAL_RCC_AHB1_FORCE_RESET() (RCC->AHB1RSTR = 0x0060109FU) +#endif /* STM32F412Vx */ +#if defined (STM32F412Rx) +#define __HAL_RCC_AHB1_FORCE_RESET() (RCC->AHB1RSTR = 0x0060108FU) +#endif /* STM32F412Rx */ #if defined(STM32F412Rx) || defined(STM32F412Vx) || defined(STM32F412Zx) || defined(STM32F413xx) || defined(STM32F423xx) #define __HAL_RCC_GPIOD_FORCE_RESET() (RCC->AHB1RSTR |= (RCC_AHB1RSTR_GPIODRST)) #endif /* STM32F412Rx || STM32F412Vx || STM32F412Zx || STM32F413xx || STM32F423xx */ @@ -5374,10 +5439,11 @@ typedef struct * @brief Force or release AHB2 peripheral reset. * @{ */ -#define __HAL_RCC_AHB2_FORCE_RESET() (RCC->AHB2RSTR = 0xFFFFFFFFU) +#define __HAL_RCC_AHB2_FORCE_RESET() (RCC->AHB2RSTR = 0x000000C0U) #define __HAL_RCC_AHB2_RELEASE_RESET() (RCC->AHB2RSTR = 0x00U) #if defined(STM32F423xx) +#define __HAL_RCC_AHB2_FORCE_RESET() (RCC->AHB2RSTR = 0x000000D0U) #define __HAL_RCC_AES_FORCE_RESET() (RCC->AHB2RSTR |= (RCC_AHB2RSTR_AESRST)) #define __HAL_RCC_AES_RELEASE_RESET() (RCC->AHB2RSTR &= ~(RCC_AHB2RSTR_AESRST)) #endif /* STM32F423xx */ @@ -5396,7 +5462,7 @@ typedef struct * @{ */ #if defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F413xx) || defined(STM32F423xx) -#define __HAL_RCC_AHB3_FORCE_RESET() (RCC->AHB3RSTR = 0xFFFFFFFFU) +#define __HAL_RCC_AHB3_FORCE_RESET() (RCC->AHB3RSTR = 0x00000003U) #define __HAL_RCC_AHB3_RELEASE_RESET() (RCC->AHB3RSTR = 0x00U) #define __HAL_RCC_FSMC_FORCE_RESET() (RCC->AHB3RSTR |= (RCC_AHB3RSTR_FSMCRST)) @@ -5423,6 +5489,12 @@ typedef struct * @brief Force or release APB1 peripheral reset. * @{ */ +#if defined(STM32F413xx) || defined(STM32F423xx) +#define __HAL_RCC_APB1_FORCE_RESET() (RCC->APB1RSTR = 0xFFFECBFFU) +#endif /* STM32F413xx || STM32F423xx */ +#if defined (STM32F412Zx) || defined (STM32F412Vx) || defined (STM32F412Rx) || defined (STM32F412Cx) +#define __HAL_RCC_APB1_FORCE_RESET() (RCC->APB1RSTR = 0x17E6C9FFU) +#endif /* STM32F412Zx || STM32F412Vx || STM32F412Rx || STM32F412Cx */ #define __HAL_RCC_TIM2_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM2RST)) #define __HAL_RCC_TIM3_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM3RST)) #define __HAL_RCC_TIM4_FORCE_RESET() (RCC->APB1RSTR |= (RCC_APB1RSTR_TIM4RST)) @@ -5486,6 +5558,12 @@ typedef struct * @brief Force or release APB2 peripheral reset. * @{ */ +#if defined(STM32F413xx)|| defined(STM32F423xx) +#define __HAL_RCC_APB2_FORCE_RESET() (RCC->APB2RSTR = 0x035779F3U) +#endif /* STM32F413xx || STM32F423xx */ +#if defined (STM32F412Zx) || defined (STM32F412Vx) || defined (STM32F412Rx) || defined (STM32F412Cx) +#define __HAL_RCC_APB2_FORCE_RESET() (RCC->APB2RSTR = 0x01177933U) +#endif /* STM32F412Zx || STM32F412Vx || STM32F412Rx || STM32F412Cx */ #define __HAL_RCC_TIM8_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_TIM8RST)) #if defined(STM32F413xx) || defined(STM32F423xx) #define __HAL_RCC_UART9_FORCE_RESET() (RCC->APB2RSTR |= (RCC_APB2RSTR_UART9RST)) @@ -6101,7 +6179,7 @@ typedef struct * @arg RCC_I2SCLKSOURCE_EXT: External clock mapped on the I2S_CKIN pin * used as I2S clock source. */ -#define __HAL_RCC_I2S_CONFIG(__SOURCE__) (*(__IO uint32_t *) RCC_CFGR_I2SSRC_BB = (__SOURCE__)) +#define __HAL_RCC_I2S_CONFIG(__SOURCE__) (MODIFY_REG(RCC->CFGR, RCC_CFGR_I2SSRC, (__SOURCE__))) /** @brief Macro to get the I2S clock source (I2SCLK). diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rng.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rng.h index 7d4dca1e..87279ef7 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rng.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rng.h @@ -304,7 +304,7 @@ uint32_t HAL_RNG_GetRandomNumber_IT(RNG_HandleTypeDef *hrng); /* Obsolete, use HAL_RNG_GenerateRandomNumber_IT() instead */ HAL_StatusTypeDef HAL_RNG_GenerateRandomNumber(RNG_HandleTypeDef *hrng, uint32_t *random32bit); HAL_StatusTypeDef HAL_RNG_GenerateRandomNumber_IT(RNG_HandleTypeDef *hrng); -uint32_t HAL_RNG_ReadLastRandomNumber(RNG_HandleTypeDef *hrng); +uint32_t HAL_RNG_ReadLastRandomNumber(const RNG_HandleTypeDef *hrng); void HAL_RNG_IRQHandler(RNG_HandleTypeDef *hrng); void HAL_RNG_ErrorCallback(RNG_HandleTypeDef *hrng); @@ -317,8 +317,8 @@ void HAL_RNG_ReadyDataCallback(RNG_HandleTypeDef *hrng, uint32_t random32bit); /** @defgroup RNG_Exported_Functions_Group3 Peripheral State functions * @{ */ -HAL_RNG_StateTypeDef HAL_RNG_GetState(RNG_HandleTypeDef *hrng); -uint32_t HAL_RNG_GetError(RNG_HandleTypeDef *hrng); +HAL_RNG_StateTypeDef HAL_RNG_GetState(const RNG_HandleTypeDef *hrng); +uint32_t HAL_RNG_GetError(const RNG_HandleTypeDef *hrng); /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rtc.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rtc.h index 24affc54..9d58161f 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rtc.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rtc.h @@ -6,7 +6,7 @@ ****************************************************************************** * @attention * - * Copyright (c) 2017 STMicroelectronics. + * Copyright (c) 2016 STMicroelectronics. * All rights reserved. * * This software is licensed under terms that can be found in the LICENSE file @@ -489,6 +489,12 @@ typedef void (*pRTC_CallbackTypeDef)(RTC_HandleTypeDef *hrtc); /*!< pointer to (__HANDLE__)->Instance->WPR = 0xFFU; \ } while(0U) +/** + * @brief Check whether the RTC Calendar is initialized. + * @param __HANDLE__ specifies the RTC handle. + * @retval None + */ +#define __HAL_RTC_IS_CALENDAR_INITIALIZED(__HANDLE__) (((((__HANDLE__)->Instance->ISR) & (RTC_FLAG_INITS)) == RTC_FLAG_INITS) ? 1U : 0U) /** * @brief Enable the RTC ALARMA peripheral. @@ -772,6 +778,7 @@ HAL_RTCStateTypeDef HAL_RTC_GetState(RTC_HandleTypeDef *hrtc); RTC_DR_MT | RTC_DR_MU | \ RTC_DR_DT | RTC_DR_DU | \ RTC_DR_WDU)) +#define RTC_ISR_RESERVED_MASK ((uint32_t)(RTC_FLAGS_MASK | RTC_ISR_INIT)) #define RTC_INIT_MASK 0xFFFFFFFFU #define RTC_RSF_MASK ((uint32_t)~(RTC_ISR_INIT | RTC_ISR_RSF)) #define RTC_FLAGS_MASK ((uint32_t)(RTC_FLAG_INITF | RTC_FLAG_INITS | \ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rtc_ex.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rtc_ex.h index ff5f14d4..bee4e320 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rtc_ex.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_rtc_ex.h @@ -6,7 +6,7 @@ ****************************************************************************** * @attention * - * Copyright (c) 2017 STMicroelectronics. + * Copyright (c) 2016 STMicroelectronics. * All rights reserved. * * This software is licensed under terms that can be found in the LICENSE file @@ -162,7 +162,7 @@ typedef struct * @{ */ #define RTC_TAMPERTRIGGER_RISINGEDGE 0x00000000U -#define RTC_TAMPERTRIGGER_FALLINGEDGE RTC_TAFCR_TAMP1TRG +#define RTC_TAMPERTRIGGER_FALLINGEDGE 0x00000002U #define RTC_TAMPERTRIGGER_LOWLEVEL RTC_TAMPERTRIGGER_RISINGEDGE #define RTC_TAMPERTRIGGER_HIGHLEVEL RTC_TAMPERTRIGGER_FALLINGEDGE /** @@ -635,18 +635,6 @@ typedef struct */ #define __HAL_RTC_TAMPER_DISABLE_IT(__HANDLE__, __INTERRUPT__) ((__HANDLE__)->Instance->TAFCR &= ~(__INTERRUPT__)) -/** - * @brief Check whether the specified RTC Tamper interrupt has occurred or not. - * @param __HANDLE__ specifies the RTC handle. - * @param __INTERRUPT__ specifies the RTC Tamper interrupt to check. - * This parameter can be: - * @arg RTC_IT_TAMP1: Tamper 1 interrupt - * @arg RTC_IT_TAMP2: Tamper 2 interrupt - * @note RTC_IT_TAMP2 is not applicable to all devices. - * @retval None - */ -#define __HAL_RTC_TAMPER_GET_IT(__HANDLE__, __INTERRUPT__) (((((__HANDLE__)->Instance->ISR) & ((__INTERRUPT__) >> 4U)) != 0U) ? 1U : 0U) - /** * @brief Check whether the specified RTC Tamper interrupt has been enabled or not. * @param __HANDLE__ specifies the RTC handle. diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_sai.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_sai.h index 6c23f93c..368316d2 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_sai.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_sai.h @@ -748,8 +748,8 @@ void HAL_SAI_ErrorCallback(SAI_HandleTypeDef *hsai); * @{ */ /* Peripheral State functions ************************************************/ -HAL_SAI_StateTypeDef HAL_SAI_GetState(SAI_HandleTypeDef *hsai); -uint32_t HAL_SAI_GetError(SAI_HandleTypeDef *hsai); +HAL_SAI_StateTypeDef HAL_SAI_GetState(const SAI_HandleTypeDef *hsai); +uint32_t HAL_SAI_GetError(const SAI_HandleTypeDef *hsai); /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_sai_ex.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_sai_ex.h index c6fdd566..75ba6d20 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_sai_ex.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_sai_ex.h @@ -71,8 +71,8 @@ extern "C" { */ /* Extended features functions ************************************************/ -void SAI_BlockSynchroConfig(SAI_HandleTypeDef *hsai); -uint32_t SAI_GetInputClock(SAI_HandleTypeDef *hsai); +void SAI_BlockSynchroConfig(const SAI_HandleTypeDef *hsai); +uint32_t SAI_GetInputClock(const SAI_HandleTypeDef *hsai); /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_smartcard.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_smartcard.h index 135f4026..7aa4ce34 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_smartcard.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_smartcard.h @@ -671,8 +671,8 @@ void HAL_SMARTCARD_AbortReceiveCpltCallback(SMARTCARD_HandleTypeDef *hsc); * @{ */ /* Peripheral State functions **************************************************/ -HAL_SMARTCARD_StateTypeDef HAL_SMARTCARD_GetState(SMARTCARD_HandleTypeDef *hsc); -uint32_t HAL_SMARTCARD_GetError(SMARTCARD_HandleTypeDef *hsc); +HAL_SMARTCARD_StateTypeDef HAL_SMARTCARD_GetState(const SMARTCARD_HandleTypeDef *hsc); +uint32_t HAL_SMARTCARD_GetError(const SMARTCARD_HandleTypeDef *hsc); /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_smbus.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_smbus.h index ea92afa3..d0ffed83 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_smbus.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_smbus.h @@ -150,35 +150,35 @@ typedef enum */ typedef struct __SMBUS_HandleTypeDef { - I2C_TypeDef *Instance; /*!< SMBUS registers base address */ + I2C_TypeDef *Instance; /*!< SMBUS registers base address */ - SMBUS_InitTypeDef Init; /*!< SMBUS communication parameters */ + SMBUS_InitTypeDef Init; /*!< SMBUS communication parameters */ - uint8_t *pBuffPtr; /*!< Pointer to SMBUS transfer buffer */ + uint8_t *pBuffPtr; /*!< Pointer to SMBUS transfer buffer */ - uint16_t XferSize; /*!< SMBUS transfer size */ + uint16_t XferSize; /*!< SMBUS transfer size */ - __IO uint16_t XferCount; /*!< SMBUS transfer counter */ + __IO uint16_t XferCount; /*!< SMBUS transfer counter */ __IO uint32_t XferOptions; /*!< SMBUS transfer options this parameter can - be a value of @ref SMBUS_OPTIONS */ + be a value of @ref SMBUS_XferOptions_definition */ __IO uint32_t PreviousState; /*!< SMBUS communication Previous state and mode - context for internal usage */ + context for internal usage */ - HAL_LockTypeDef Lock; /*!< SMBUS locking object */ + HAL_LockTypeDef Lock; /*!< SMBUS locking object */ - __IO HAL_SMBUS_StateTypeDef State; /*!< SMBUS communication state */ + __IO HAL_SMBUS_StateTypeDef State; /*!< SMBUS communication state */ - __IO HAL_SMBUS_ModeTypeDef Mode; /*!< SMBUS communication mode */ + __IO HAL_SMBUS_ModeTypeDef Mode; /*!< SMBUS communication mode */ - __IO uint32_t ErrorCode; /*!< SMBUS Error code */ + __IO uint32_t ErrorCode; /*!< SMBUS Error code */ - __IO uint32_t Devaddress; /*!< SMBUS Target device address */ + __IO uint32_t Devaddress; /*!< SMBUS Target device address */ - __IO uint32_t EventCount; /*!< SMBUS Event counter */ + __IO uint32_t EventCount; /*!< SMBUS Event counter */ - uint8_t XferPEC; /*!< SMBUS PEC data in reception mode */ + uint8_t XferPEC; /*!< SMBUS PEC data in reception mode */ #if (USE_HAL_SMBUS_REGISTER_CALLBACKS == 1) void (* MasterTxCpltCallback)(struct __SMBUS_HandleTypeDef *hsmbus); /*!< SMBUS Master Tx Transfer completed callback */ @@ -604,6 +604,10 @@ void HAL_SMBUS_ListenCpltCallback(SMBUS_HandleTypeDef *hsmbus); void HAL_SMBUS_ErrorCallback(SMBUS_HandleTypeDef *hsmbus); void HAL_SMBUS_AbortCpltCallback(SMBUS_HandleTypeDef *hsmbus); +/** + * @} + */ + /** * @} */ @@ -718,10 +722,6 @@ uint32_t HAL_SMBUS_GetError(SMBUS_HandleTypeDef *hsmbus); * @} */ -/** -* @} -*/ - #ifdef __cplusplus } #endif diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_spdifrx.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_spdifrx.h index 7ea04aa9..386903bc 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_spdifrx.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_spdifrx.h @@ -56,7 +56,8 @@ typedef struct uint32_t WaitForActivity; /*!< Specifies the wait for activity on SPDIF selected input. This parameter can be a value of @ref SPDIFRX_Wait_For_Activity. */ - uint32_t ChannelSelection; /*!< Specifies whether the control flow will take the channel status from channel A or B. + uint32_t ChannelSelection; /*!< Specifies whether the control flow will take the channel status + from channel A or B. This parameter can be a value of @ref SPDIFRX_Channel_Selection */ uint32_t DataFormat; /*!< Specifies the Data samples format (LSB, MSB, ...). @@ -65,16 +66,19 @@ typedef struct uint32_t StereoMode; /*!< Specifies whether the peripheral is in stereo or mono mode. This parameter can be a value of @ref SPDIFRX_Stereo_Mode */ - uint32_t PreambleTypeMask; /*!< Specifies whether The preamble type bits are copied or not into the received frame. - This parameter can be a value of @ref SPDIFRX_PT_Mask */ + uint32_t PreambleTypeMask; /*!< Specifies whether The preamble type bits are copied or not + into the received frame. + This parameter can be a value of @ref SPDIFRX_PT_Mask */ - uint32_t ChannelStatusMask; /*!< Specifies whether the channel status and user bits are copied or not into the received frame. + uint32_t ChannelStatusMask; /*!< Specifies whether the channel status and user bits are copied or not + into the received frame. This parameter can be a value of @ref SPDIFRX_ChannelStatus_Mask */ uint32_t ValidityBitMask; /*!< Specifies whether the validity bit is copied or not into the received frame. This parameter can be a value of @ref SPDIFRX_V_Mask */ - uint32_t ParityErrorMask; /*!< Specifies whether the parity error bit is copied or not into the received frame. + uint32_t ParityErrorMask; /*!< Specifies whether the parity error bit is copied or not + into the received frame. This parameter can be a value of @ref SPDIFRX_PE_Mask */ } SPDIFRX_InitTypeDef; @@ -89,17 +93,20 @@ typedef struct uint32_t StereoMode; /*!< Specifies whether the peripheral is in stereo or mono mode. This parameter can be a value of @ref SPDIFRX_Stereo_Mode */ - uint32_t PreambleTypeMask; /*!< Specifies whether The preamble type bits are copied or not into the received frame. - This parameter can be a value of @ref SPDIFRX_PT_Mask */ + uint32_t PreambleTypeMask; /*!< Specifies whether The preamble type bits are copied or not + into the received frame. + This parameter can be a value of @ref SPDIFRX_PT_Mask */ - uint32_t ChannelStatusMask; /*!< Specifies whether the channel status and user bits are copied or not into the received frame. - This parameter can be a value of @ref SPDIFRX_ChannelStatus_Mask */ + uint32_t ChannelStatusMask; /*!< Specifies whether the channel status and user bits are copied or not + into the received frame. + This parameter can be a value of @ref SPDIFRX_ChannelStatus_Mask */ uint32_t ValidityBitMask; /*!< Specifies whether the validity bit is copied or not into the received frame. - This parameter can be a value of @ref SPDIFRX_V_Mask */ + This parameter can be a value of @ref SPDIFRX_V_Mask */ - uint32_t ParityErrorMask; /*!< Specifies whether the parity error bit is copied or not into the received frame. - This parameter can be a value of @ref SPDIFRX_PE_Mask */ + uint32_t ParityErrorMask; /*!< Specifies whether the parity error bit is copied or not + into the received frame. + This parameter can be a value of @ref SPDIFRX_PE_Mask */ } SPDIFRX_SetDataFormatTypeDef; @@ -151,7 +158,8 @@ typedef struct decremented when a sample is received. NbSamplesReceived = RxBufferSize-RxBufferCount) */ - DMA_HandleTypeDef *hdmaCsRx; /* SPDIFRX EC60958_channel_status and user_information DMA handle parameters */ + DMA_HandleTypeDef *hdmaCsRx; /* SPDIFRX EC60958_channel_status and user_information + DMA handle parameters */ DMA_HandleTypeDef *hdmaDrRx; /* SPDIFRX Rx DMA handle parameters */ @@ -162,9 +170,11 @@ typedef struct __IO uint32_t ErrorCode; /* SPDIFRX Error code */ #if (USE_HAL_SPDIFRX_REGISTER_CALLBACKS == 1) - void (*RxHalfCpltCallback)(struct __SPDIFRX_HandleTypeDef *hspdif); /*!< SPDIFRX Data flow half completed callback */ + void (*RxHalfCpltCallback)(struct __SPDIFRX_HandleTypeDef *hspdif); /*!< SPDIFRX Data flow half completed + callback */ void (*RxCpltCallback)(struct __SPDIFRX_HandleTypeDef *hspdif); /*!< SPDIFRX Data flow completed callback */ - void (*CxHalfCpltCallback)(struct __SPDIFRX_HandleTypeDef *hspdif); /*!< SPDIFRX Control flow half completed callback */ + void (*CxHalfCpltCallback)(struct __SPDIFRX_HandleTypeDef *hspdif); /*!< SPDIFRX Control flow half completed + callback */ void (*CxCpltCallback)(struct __SPDIFRX_HandleTypeDef *hspdif); /*!< SPDIFRX Control flow completed callback */ void (*ErrorCallback)(struct __SPDIFRX_HandleTypeDef *hspdif); /*!< SPDIFRX error callback */ void (* MspInitCallback)(struct __SPDIFRX_HandleTypeDef *hspdif); /*!< SPDIFRX Msp Init callback */ @@ -172,9 +182,6 @@ typedef struct #endif /* USE_HAL_SPDIFRX_REGISTER_CALLBACKS */ } SPDIFRX_HandleTypeDef; -/** - * @} - */ #if (USE_HAL_SPDIFRX_REGISTER_CALLBACKS == 1) /** @@ -194,9 +201,12 @@ typedef enum /** * @brief HAL SPDIFRX Callback pointer definition */ -typedef void (*pSPDIFRX_CallbackTypeDef)(SPDIFRX_HandleTypeDef *hspdif); /*!< pointer to an SPDIFRX callback function */ +typedef void (*pSPDIFRX_CallbackTypeDef)(SPDIFRX_HandleTypeDef *hspdif); /*!< pointer to an SPDIFRX callback + function */ #endif /* USE_HAL_SPDIFRX_REGISTER_CALLBACKS */ - +/** + * @} + */ /* Exported constants --------------------------------------------------------*/ /** @defgroup SPDIFRX_Exported_Constants SPDIFRX Exported Constants * @{ @@ -260,8 +270,10 @@ typedef void (*pSPDIFRX_CallbackTypeDef)(SPDIFRX_HandleTypeDef *hspdif); /*!< /** @defgroup SPDIFRX_ChannelStatus_Mask SPDIFRX Channel Status Mask * @{ */ -#define SPDIFRX_CHANNELSTATUS_OFF ((uint32_t)0x00000000U) /* The channel status and user bits are copied into the SPDIF_DR */ -#define SPDIFRX_CHANNELSTATUS_ON ((uint32_t)SPDIFRX_CR_CUMSK) /* The channel status and user bits are not copied into the SPDIF_DR, zeros are written instead*/ +#define SPDIFRX_CHANNELSTATUS_OFF ((uint32_t)0x00000000U) /* The channel status and user bits are copied + into the SPDIF_DR */ +#define SPDIFRX_CHANNELSTATUS_ON ((uint32_t)SPDIFRX_CR_CUMSK) /* The channel status and user bits are not copied + into the SPDIF_DR, zeros are written instead*/ /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_sram.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_sram.h index e135aa13..a6e61110 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_sram.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_sram.h @@ -209,7 +209,7 @@ HAL_StatusTypeDef HAL_SRAM_WriteOperation_Disable(SRAM_HandleTypeDef *hsram); */ /* SRAM State functions ******************************************************/ -HAL_SRAM_StateTypeDef HAL_SRAM_GetState(SRAM_HandleTypeDef *hsram); +HAL_SRAM_StateTypeDef HAL_SRAM_GetState(const SRAM_HandleTypeDef *hsram); /** * @} diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_tim.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_tim.h index 8c81414f..1a8357cd 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_tim.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_tim.h @@ -385,29 +385,28 @@ typedef struct */ typedef enum { - HAL_TIM_BASE_MSPINIT_CB_ID = 0x00U /*!< TIM Base MspInit Callback ID */ - , HAL_TIM_BASE_MSPDEINIT_CB_ID = 0x01U /*!< TIM Base MspDeInit Callback ID */ - , HAL_TIM_IC_MSPINIT_CB_ID = 0x02U /*!< TIM IC MspInit Callback ID */ - , HAL_TIM_IC_MSPDEINIT_CB_ID = 0x03U /*!< TIM IC MspDeInit Callback ID */ - , HAL_TIM_OC_MSPINIT_CB_ID = 0x04U /*!< TIM OC MspInit Callback ID */ - , HAL_TIM_OC_MSPDEINIT_CB_ID = 0x05U /*!< TIM OC MspDeInit Callback ID */ - , HAL_TIM_PWM_MSPINIT_CB_ID = 0x06U /*!< TIM PWM MspInit Callback ID */ - , HAL_TIM_PWM_MSPDEINIT_CB_ID = 0x07U /*!< TIM PWM MspDeInit Callback ID */ - , HAL_TIM_ONE_PULSE_MSPINIT_CB_ID = 0x08U /*!< TIM One Pulse MspInit Callback ID */ - , HAL_TIM_ONE_PULSE_MSPDEINIT_CB_ID = 0x09U /*!< TIM One Pulse MspDeInit Callback ID */ - , HAL_TIM_ENCODER_MSPINIT_CB_ID = 0x0AU /*!< TIM Encoder MspInit Callback ID */ - , HAL_TIM_ENCODER_MSPDEINIT_CB_ID = 0x0BU /*!< TIM Encoder MspDeInit Callback ID */ - , HAL_TIM_HALL_SENSOR_MSPINIT_CB_ID = 0x0CU /*!< TIM Hall Sensor MspDeInit Callback ID */ - , HAL_TIM_HALL_SENSOR_MSPDEINIT_CB_ID = 0x0DU /*!< TIM Hall Sensor MspDeInit Callback ID */ + HAL_TIM_BASE_MSPINIT_CB_ID = 0x00U /*!< TIM Base MspInit Callback ID */ + , HAL_TIM_BASE_MSPDEINIT_CB_ID = 0x01U /*!< TIM Base MspDeInit Callback ID */ + , HAL_TIM_IC_MSPINIT_CB_ID = 0x02U /*!< TIM IC MspInit Callback ID */ + , HAL_TIM_IC_MSPDEINIT_CB_ID = 0x03U /*!< TIM IC MspDeInit Callback ID */ + , HAL_TIM_OC_MSPINIT_CB_ID = 0x04U /*!< TIM OC MspInit Callback ID */ + , HAL_TIM_OC_MSPDEINIT_CB_ID = 0x05U /*!< TIM OC MspDeInit Callback ID */ + , HAL_TIM_PWM_MSPINIT_CB_ID = 0x06U /*!< TIM PWM MspInit Callback ID */ + , HAL_TIM_PWM_MSPDEINIT_CB_ID = 0x07U /*!< TIM PWM MspDeInit Callback ID */ + , HAL_TIM_ONE_PULSE_MSPINIT_CB_ID = 0x08U /*!< TIM One Pulse MspInit Callback ID */ + , HAL_TIM_ONE_PULSE_MSPDEINIT_CB_ID = 0x09U /*!< TIM One Pulse MspDeInit Callback ID */ + , HAL_TIM_ENCODER_MSPINIT_CB_ID = 0x0AU /*!< TIM Encoder MspInit Callback ID */ + , HAL_TIM_ENCODER_MSPDEINIT_CB_ID = 0x0BU /*!< TIM Encoder MspDeInit Callback ID */ + , HAL_TIM_HALL_SENSOR_MSPINIT_CB_ID = 0x0CU /*!< TIM Hall Sensor MspDeInit Callback ID */ + , HAL_TIM_HALL_SENSOR_MSPDEINIT_CB_ID = 0x0DU /*!< TIM Hall Sensor MspDeInit Callback ID */ , HAL_TIM_PERIOD_ELAPSED_CB_ID = 0x0EU /*!< TIM Period Elapsed Callback ID */ , HAL_TIM_PERIOD_ELAPSED_HALF_CB_ID = 0x0FU /*!< TIM Period Elapsed half complete Callback ID */ , HAL_TIM_TRIGGER_CB_ID = 0x10U /*!< TIM Trigger Callback ID */ , HAL_TIM_TRIGGER_HALF_CB_ID = 0x11U /*!< TIM Trigger half complete Callback ID */ - , HAL_TIM_IC_CAPTURE_CB_ID = 0x12U /*!< TIM Input Capture Callback ID */ , HAL_TIM_IC_CAPTURE_HALF_CB_ID = 0x13U /*!< TIM Input Capture half complete Callback ID */ , HAL_TIM_OC_DELAY_ELAPSED_CB_ID = 0x14U /*!< TIM Output Compare Delay Elapsed Callback ID */ - , HAL_TIM_PWM_PULSE_FINISHED_CB_ID = 0x15U /*!< TIM PWM Pulse Finished Callback ID */ + , HAL_TIM_PWM_PULSE_FINISHED_CB_ID = 0x15U /*!< TIM PWM Pulse Finished Callback ID */ , HAL_TIM_PWM_PULSE_FINISHED_HALF_CB_ID = 0x16U /*!< TIM PWM Pulse Finished half complete Callback ID */ , HAL_TIM_ERROR_CB_ID = 0x17U /*!< TIM Error Callback ID */ , HAL_TIM_COMMUTATION_CB_ID = 0x18U /*!< TIM Commutation Callback ID */ @@ -1654,6 +1653,10 @@ typedef void (*pTIM_CallbackTypeDef)(TIM_HandleTypeDef *htim); /*!< pointer to #define IS_TIM_OPM_CHANNELS(__CHANNEL__) (((__CHANNEL__) == TIM_CHANNEL_1) || \ ((__CHANNEL__) == TIM_CHANNEL_2)) +#define IS_TIM_PERIOD(__HANDLE__, __PERIOD__) ((IS_TIM_32B_COUNTER_INSTANCE(((__HANDLE__)->Instance)) == 0U) ? \ + (((__PERIOD__) > 0U) && ((__PERIOD__) <= 0x0000FFFFU)) : \ + ((__PERIOD__) > 0U)) + #define IS_TIM_COMPLEMENTARY_CHANNELS(__CHANNEL__) (((__CHANNEL__) == TIM_CHANNEL_1) || \ ((__CHANNEL__) == TIM_CHANNEL_2) || \ ((__CHANNEL__) == TIM_CHANNEL_3)) @@ -1705,7 +1708,6 @@ typedef void (*pTIM_CallbackTypeDef)(TIM_HandleTypeDef *htim); /*!< pointer to #define IS_TIM_BREAK_FILTER(__BRKFILTER__) ((__BRKFILTER__) <= 0xFUL) - #define IS_TIM_BREAK_STATE(__STATE__) (((__STATE__) == TIM_BREAK_ENABLE) || \ ((__STATE__) == TIM_BREAK_DISABLE)) @@ -1896,7 +1898,7 @@ HAL_StatusTypeDef HAL_TIM_Base_Stop(TIM_HandleTypeDef *htim); HAL_StatusTypeDef HAL_TIM_Base_Start_IT(TIM_HandleTypeDef *htim); HAL_StatusTypeDef HAL_TIM_Base_Stop_IT(TIM_HandleTypeDef *htim); /* Non-Blocking mode: DMA */ -HAL_StatusTypeDef HAL_TIM_Base_Start_DMA(TIM_HandleTypeDef *htim, uint32_t *pData, uint16_t Length); +HAL_StatusTypeDef HAL_TIM_Base_Start_DMA(TIM_HandleTypeDef *htim, const uint32_t *pData, uint16_t Length); HAL_StatusTypeDef HAL_TIM_Base_Stop_DMA(TIM_HandleTypeDef *htim); /** * @} @@ -1918,7 +1920,8 @@ HAL_StatusTypeDef HAL_TIM_OC_Stop(TIM_HandleTypeDef *htim, uint32_t Channel); HAL_StatusTypeDef HAL_TIM_OC_Start_IT(TIM_HandleTypeDef *htim, uint32_t Channel); HAL_StatusTypeDef HAL_TIM_OC_Stop_IT(TIM_HandleTypeDef *htim, uint32_t Channel); /* Non-Blocking mode: DMA */ -HAL_StatusTypeDef HAL_TIM_OC_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, uint32_t *pData, uint16_t Length); +HAL_StatusTypeDef HAL_TIM_OC_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, const uint32_t *pData, + uint16_t Length); HAL_StatusTypeDef HAL_TIM_OC_Stop_DMA(TIM_HandleTypeDef *htim, uint32_t Channel); /** * @} @@ -1940,7 +1943,8 @@ HAL_StatusTypeDef HAL_TIM_PWM_Stop(TIM_HandleTypeDef *htim, uint32_t Channel); HAL_StatusTypeDef HAL_TIM_PWM_Start_IT(TIM_HandleTypeDef *htim, uint32_t Channel); HAL_StatusTypeDef HAL_TIM_PWM_Stop_IT(TIM_HandleTypeDef *htim, uint32_t Channel); /* Non-Blocking mode: DMA */ -HAL_StatusTypeDef HAL_TIM_PWM_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, uint32_t *pData, uint16_t Length); +HAL_StatusTypeDef HAL_TIM_PWM_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, const uint32_t *pData, + uint16_t Length); HAL_StatusTypeDef HAL_TIM_PWM_Stop_DMA(TIM_HandleTypeDef *htim, uint32_t Channel); /** * @} @@ -1992,7 +1996,7 @@ HAL_StatusTypeDef HAL_TIM_OnePulse_Stop_IT(TIM_HandleTypeDef *htim, uint32_t Out * @{ */ /* Timer Encoder functions ****************************************************/ -HAL_StatusTypeDef HAL_TIM_Encoder_Init(TIM_HandleTypeDef *htim, TIM_Encoder_InitTypeDef *sConfig); +HAL_StatusTypeDef HAL_TIM_Encoder_Init(TIM_HandleTypeDef *htim, const TIM_Encoder_InitTypeDef *sConfig); HAL_StatusTypeDef HAL_TIM_Encoder_DeInit(TIM_HandleTypeDef *htim); void HAL_TIM_Encoder_MspInit(TIM_HandleTypeDef *htim); void HAL_TIM_Encoder_MspDeInit(TIM_HandleTypeDef *htim); @@ -2025,21 +2029,26 @@ void HAL_TIM_IRQHandler(TIM_HandleTypeDef *htim); * @{ */ /* Control functions *********************************************************/ -HAL_StatusTypeDef HAL_TIM_OC_ConfigChannel(TIM_HandleTypeDef *htim, TIM_OC_InitTypeDef *sConfig, uint32_t Channel); -HAL_StatusTypeDef HAL_TIM_PWM_ConfigChannel(TIM_HandleTypeDef *htim, TIM_OC_InitTypeDef *sConfig, uint32_t Channel); -HAL_StatusTypeDef HAL_TIM_IC_ConfigChannel(TIM_HandleTypeDef *htim, TIM_IC_InitTypeDef *sConfig, uint32_t Channel); +HAL_StatusTypeDef HAL_TIM_OC_ConfigChannel(TIM_HandleTypeDef *htim, const TIM_OC_InitTypeDef *sConfig, + uint32_t Channel); +HAL_StatusTypeDef HAL_TIM_PWM_ConfigChannel(TIM_HandleTypeDef *htim, const TIM_OC_InitTypeDef *sConfig, + uint32_t Channel); +HAL_StatusTypeDef HAL_TIM_IC_ConfigChannel(TIM_HandleTypeDef *htim, const TIM_IC_InitTypeDef *sConfig, + uint32_t Channel); HAL_StatusTypeDef HAL_TIM_OnePulse_ConfigChannel(TIM_HandleTypeDef *htim, TIM_OnePulse_InitTypeDef *sConfig, uint32_t OutputChannel, uint32_t InputChannel); -HAL_StatusTypeDef HAL_TIM_ConfigOCrefClear(TIM_HandleTypeDef *htim, TIM_ClearInputConfigTypeDef *sClearInputConfig, +HAL_StatusTypeDef HAL_TIM_ConfigOCrefClear(TIM_HandleTypeDef *htim, + const TIM_ClearInputConfigTypeDef *sClearInputConfig, uint32_t Channel); -HAL_StatusTypeDef HAL_TIM_ConfigClockSource(TIM_HandleTypeDef *htim, TIM_ClockConfigTypeDef *sClockSourceConfig); +HAL_StatusTypeDef HAL_TIM_ConfigClockSource(TIM_HandleTypeDef *htim, const TIM_ClockConfigTypeDef *sClockSourceConfig); HAL_StatusTypeDef HAL_TIM_ConfigTI1Input(TIM_HandleTypeDef *htim, uint32_t TI1_Selection); -HAL_StatusTypeDef HAL_TIM_SlaveConfigSynchro(TIM_HandleTypeDef *htim, TIM_SlaveConfigTypeDef *sSlaveConfig); -HAL_StatusTypeDef HAL_TIM_SlaveConfigSynchro_IT(TIM_HandleTypeDef *htim, TIM_SlaveConfigTypeDef *sSlaveConfig); +HAL_StatusTypeDef HAL_TIM_SlaveConfigSynchro(TIM_HandleTypeDef *htim, const TIM_SlaveConfigTypeDef *sSlaveConfig); +HAL_StatusTypeDef HAL_TIM_SlaveConfigSynchro_IT(TIM_HandleTypeDef *htim, const TIM_SlaveConfigTypeDef *sSlaveConfig); HAL_StatusTypeDef HAL_TIM_DMABurst_WriteStart(TIM_HandleTypeDef *htim, uint32_t BurstBaseAddress, - uint32_t BurstRequestSrc, uint32_t *BurstBuffer, uint32_t BurstLength); + uint32_t BurstRequestSrc, const uint32_t *BurstBuffer, + uint32_t BurstLength); HAL_StatusTypeDef HAL_TIM_DMABurst_MultiWriteStart(TIM_HandleTypeDef *htim, uint32_t BurstBaseAddress, - uint32_t BurstRequestSrc, uint32_t *BurstBuffer, + uint32_t BurstRequestSrc, const uint32_t *BurstBuffer, uint32_t BurstLength, uint32_t DataLength); HAL_StatusTypeDef HAL_TIM_DMABurst_WriteStop(TIM_HandleTypeDef *htim, uint32_t BurstRequestSrc); HAL_StatusTypeDef HAL_TIM_DMABurst_ReadStart(TIM_HandleTypeDef *htim, uint32_t BurstBaseAddress, @@ -2049,7 +2058,7 @@ HAL_StatusTypeDef HAL_TIM_DMABurst_MultiReadStart(TIM_HandleTypeDef *htim, uint3 uint32_t BurstLength, uint32_t DataLength); HAL_StatusTypeDef HAL_TIM_DMABurst_ReadStop(TIM_HandleTypeDef *htim, uint32_t BurstRequestSrc); HAL_StatusTypeDef HAL_TIM_GenerateEvent(TIM_HandleTypeDef *htim, uint32_t EventSource); -uint32_t HAL_TIM_ReadCapturedValue(TIM_HandleTypeDef *htim, uint32_t Channel); +uint32_t HAL_TIM_ReadCapturedValue(const TIM_HandleTypeDef *htim, uint32_t Channel); /** * @} */ @@ -2086,17 +2095,17 @@ HAL_StatusTypeDef HAL_TIM_UnRegisterCallback(TIM_HandleTypeDef *htim, HAL_TIM_Ca * @{ */ /* Peripheral State functions ************************************************/ -HAL_TIM_StateTypeDef HAL_TIM_Base_GetState(TIM_HandleTypeDef *htim); -HAL_TIM_StateTypeDef HAL_TIM_OC_GetState(TIM_HandleTypeDef *htim); -HAL_TIM_StateTypeDef HAL_TIM_PWM_GetState(TIM_HandleTypeDef *htim); -HAL_TIM_StateTypeDef HAL_TIM_IC_GetState(TIM_HandleTypeDef *htim); -HAL_TIM_StateTypeDef HAL_TIM_OnePulse_GetState(TIM_HandleTypeDef *htim); -HAL_TIM_StateTypeDef HAL_TIM_Encoder_GetState(TIM_HandleTypeDef *htim); +HAL_TIM_StateTypeDef HAL_TIM_Base_GetState(const TIM_HandleTypeDef *htim); +HAL_TIM_StateTypeDef HAL_TIM_OC_GetState(const TIM_HandleTypeDef *htim); +HAL_TIM_StateTypeDef HAL_TIM_PWM_GetState(const TIM_HandleTypeDef *htim); +HAL_TIM_StateTypeDef HAL_TIM_IC_GetState(const TIM_HandleTypeDef *htim); +HAL_TIM_StateTypeDef HAL_TIM_OnePulse_GetState(const TIM_HandleTypeDef *htim); +HAL_TIM_StateTypeDef HAL_TIM_Encoder_GetState(const TIM_HandleTypeDef *htim); /* Peripheral Channel state functions ************************************************/ -HAL_TIM_ActiveChannel HAL_TIM_GetActiveChannel(TIM_HandleTypeDef *htim); -HAL_TIM_ChannelStateTypeDef HAL_TIM_GetChannelState(TIM_HandleTypeDef *htim, uint32_t Channel); -HAL_TIM_DMABurstStateTypeDef HAL_TIM_DMABurstState(TIM_HandleTypeDef *htim); +HAL_TIM_ActiveChannel HAL_TIM_GetActiveChannel(const TIM_HandleTypeDef *htim); +HAL_TIM_ChannelStateTypeDef HAL_TIM_GetChannelState(const TIM_HandleTypeDef *htim, uint32_t Channel); +HAL_TIM_DMABurstStateTypeDef HAL_TIM_DMABurstState(const TIM_HandleTypeDef *htim); /** * @} */ @@ -2110,9 +2119,9 @@ HAL_TIM_DMABurstStateTypeDef HAL_TIM_DMABurstState(TIM_HandleTypeDef *htim); /** @defgroup TIM_Private_Functions TIM Private Functions * @{ */ -void TIM_Base_SetConfig(TIM_TypeDef *TIMx, TIM_Base_InitTypeDef *Structure); +void TIM_Base_SetConfig(TIM_TypeDef *TIMx, const TIM_Base_InitTypeDef *Structure); void TIM_TI1_SetConfig(TIM_TypeDef *TIMx, uint32_t TIM_ICPolarity, uint32_t TIM_ICSelection, uint32_t TIM_ICFilter); -void TIM_OC2_SetConfig(TIM_TypeDef *TIMx, TIM_OC_InitTypeDef *OC_Config); +void TIM_OC2_SetConfig(TIM_TypeDef *TIMx, const TIM_OC_InitTypeDef *OC_Config); void TIM_ETR_SetConfig(TIM_TypeDef *TIMx, uint32_t TIM_ExtTRGPrescaler, uint32_t TIM_ExtTRGPolarity, uint32_t ExtTRGFilter); diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_tim_ex.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_tim_ex.h index 39fb500f..561e9bbe 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_tim_ex.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_tim_ex.h @@ -74,9 +74,8 @@ typedef struct #if defined (TIM2) #if defined(TIM8) #define TIM_TIM2_TIM8_TRGO 0x00000000U /*!< TIM2 ITR1 is connected to TIM8 TRGO */ -#else -#define TIM_TIM2_ETH_PTP TIM_OR_ITR1_RMP_0 /*!< TIM2 ITR1 is connected to PTP trigger output */ #endif /* TIM8 */ +#define TIM_TIM2_ETH_PTP TIM_OR_ITR1_RMP_0 /*!< TIM2 ITR1 is connected to PTP trigger output */ #define TIM_TIM2_USBFS_SOF TIM_OR_ITR1_RMP_1 /*!< TIM2 ITR1 is connected to OTG FS SOF */ #define TIM_TIM2_USBHS_SOF (TIM_OR_ITR1_RMP_1 | TIM_OR_ITR1_RMP_0) /*!< TIM2 ITR1 is connected to OTG HS SOF */ #endif /* TIM2 */ @@ -205,7 +204,7 @@ typedef struct * @{ */ /* Timer Hall Sensor functions **********************************************/ -HAL_StatusTypeDef HAL_TIMEx_HallSensor_Init(TIM_HandleTypeDef *htim, TIM_HallSensor_InitTypeDef *sConfig); +HAL_StatusTypeDef HAL_TIMEx_HallSensor_Init(TIM_HandleTypeDef *htim, const TIM_HallSensor_InitTypeDef *sConfig); HAL_StatusTypeDef HAL_TIMEx_HallSensor_DeInit(TIM_HandleTypeDef *htim); void HAL_TIMEx_HallSensor_MspInit(TIM_HandleTypeDef *htim); @@ -238,7 +237,8 @@ HAL_StatusTypeDef HAL_TIMEx_OCN_Start_IT(TIM_HandleTypeDef *htim, uint32_t Chann HAL_StatusTypeDef HAL_TIMEx_OCN_Stop_IT(TIM_HandleTypeDef *htim, uint32_t Channel); /* Non-Blocking mode: DMA */ -HAL_StatusTypeDef HAL_TIMEx_OCN_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, uint32_t *pData, uint16_t Length); +HAL_StatusTypeDef HAL_TIMEx_OCN_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, const uint32_t *pData, + uint16_t Length); HAL_StatusTypeDef HAL_TIMEx_OCN_Stop_DMA(TIM_HandleTypeDef *htim, uint32_t Channel); /** * @} @@ -257,7 +257,8 @@ HAL_StatusTypeDef HAL_TIMEx_PWMN_Stop(TIM_HandleTypeDef *htim, uint32_t Channel) HAL_StatusTypeDef HAL_TIMEx_PWMN_Start_IT(TIM_HandleTypeDef *htim, uint32_t Channel); HAL_StatusTypeDef HAL_TIMEx_PWMN_Stop_IT(TIM_HandleTypeDef *htim, uint32_t Channel); /* Non-Blocking mode: DMA */ -HAL_StatusTypeDef HAL_TIMEx_PWMN_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, uint32_t *pData, uint16_t Length); +HAL_StatusTypeDef HAL_TIMEx_PWMN_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, const uint32_t *pData, + uint16_t Length); HAL_StatusTypeDef HAL_TIMEx_PWMN_Stop_DMA(TIM_HandleTypeDef *htim, uint32_t Channel); /** * @} @@ -291,9 +292,9 @@ HAL_StatusTypeDef HAL_TIMEx_ConfigCommutEvent_IT(TIM_HandleTypeDef *htim, uint32 HAL_StatusTypeDef HAL_TIMEx_ConfigCommutEvent_DMA(TIM_HandleTypeDef *htim, uint32_t InputTrigger, uint32_t CommutationSource); HAL_StatusTypeDef HAL_TIMEx_MasterConfigSynchronization(TIM_HandleTypeDef *htim, - TIM_MasterConfigTypeDef *sMasterConfig); + const TIM_MasterConfigTypeDef *sMasterConfig); HAL_StatusTypeDef HAL_TIMEx_ConfigBreakDeadTime(TIM_HandleTypeDef *htim, - TIM_BreakDeadTimeConfigTypeDef *sBreakDeadTimeConfig); + const TIM_BreakDeadTimeConfigTypeDef *sBreakDeadTimeConfig); HAL_StatusTypeDef HAL_TIMEx_RemapConfig(TIM_HandleTypeDef *htim, uint32_t Remap); /** * @} @@ -316,8 +317,8 @@ void HAL_TIMEx_BreakCallback(TIM_HandleTypeDef *htim); * @{ */ /* Extended Peripheral State functions ***************************************/ -HAL_TIM_StateTypeDef HAL_TIMEx_HallSensor_GetState(TIM_HandleTypeDef *htim); -HAL_TIM_ChannelStateTypeDef HAL_TIMEx_GetChannelNState(TIM_HandleTypeDef *htim, uint32_t ChannelN); +HAL_TIM_StateTypeDef HAL_TIMEx_HallSensor_GetState(const TIM_HandleTypeDef *htim); +HAL_TIM_ChannelStateTypeDef HAL_TIMEx_GetChannelNState(const TIM_HandleTypeDef *htim, uint32_t ChannelN); /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_uart.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_uart.h index c5f5d3e3..e6ce82fc 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_uart.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_uart.h @@ -137,12 +137,23 @@ typedef enum /** * @brief HAL UART Reception type definition * @note HAL UART Reception type value aims to identify which type of Reception is ongoing. - * It is expected to admit following values : + * This parameter can be a value of @ref UART_Reception_Type_Values : * HAL_UART_RECEPTION_STANDARD = 0x00U, * HAL_UART_RECEPTION_TOIDLE = 0x01U, */ typedef uint32_t HAL_UART_RxTypeTypeDef; +/** + * @brief HAL UART Rx Event type definition + * @note HAL UART Rx Event type value aims to identify which type of Event has occurred + * leading to call of the RxEvent callback. + * This parameter can be a value of @ref UART_RxEvent_Type_Values : + * HAL_UART_RXEVENT_TC = 0x00U, + * HAL_UART_RXEVENT_HT = 0x01U, + * HAL_UART_RXEVENT_IDLE = 0x02U, + */ +typedef uint32_t HAL_UART_RxEventTypeTypeDef; + /** * @brief UART handle Structure definition */ @@ -166,6 +177,8 @@ typedef struct __UART_HandleTypeDef __IO HAL_UART_RxTypeTypeDef ReceptionType; /*!< Type of ongoing reception */ + __IO HAL_UART_RxEventTypeTypeDef RxEventType; /*!< Type of Rx Event */ + DMA_HandleTypeDef *hdmatx; /*!< UART Tx DMA Handle parameters */ DMA_HandleTypeDef *hdmarx; /*!< UART Rx DMA Handle parameters */ @@ -381,7 +394,7 @@ typedef void (*pUART_RxEventCallbackTypeDef)(struct __UART_HandleTypeDef *huart * @} */ -/** @defgroup UART_RECEPTION_TYPE_Values UART Reception type values +/** @defgroup UART_Reception_Type_Values UART Reception type values * @{ */ #define HAL_UART_RECEPTION_STANDARD (0x00000000U) /*!< Standard reception */ @@ -390,6 +403,16 @@ typedef void (*pUART_RxEventCallbackTypeDef)(struct __UART_HandleTypeDef *huart * @} */ +/** @defgroup UART_RxEvent_Type_Values UART RxEvent type values + * @{ + */ +#define HAL_UART_RXEVENT_TC (0x00000000U) /*!< RxEvent linked to Transfer Complete event */ +#define HAL_UART_RXEVENT_HT (0x00000001U) /*!< RxEvent linked to Half Transfer event */ +#define HAL_UART_RXEVENT_IDLE (0x00000002U) +/** + * @} + */ + /** * @} */ @@ -734,6 +757,8 @@ HAL_StatusTypeDef HAL_UARTEx_ReceiveToIdle(UART_HandleTypeDef *huart, uint8_t *p HAL_StatusTypeDef HAL_UARTEx_ReceiveToIdle_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size); HAL_StatusTypeDef HAL_UARTEx_ReceiveToIdle_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size); +HAL_UART_RxEventTypeTypeDef HAL_UARTEx_GetRxEventType(UART_HandleTypeDef *huart); + /* Transfer Abort functions */ HAL_StatusTypeDef HAL_UART_Abort(UART_HandleTypeDef *huart); HAL_StatusTypeDef HAL_UART_AbortTransmit(UART_HandleTypeDef *huart); @@ -775,8 +800,8 @@ HAL_StatusTypeDef HAL_HalfDuplex_EnableReceiver(UART_HandleTypeDef *huart); * @{ */ /* Peripheral State functions **************************************************/ -HAL_UART_StateTypeDef HAL_UART_GetState(UART_HandleTypeDef *huart); -uint32_t HAL_UART_GetError(UART_HandleTypeDef *huart); +HAL_UART_StateTypeDef HAL_UART_GetState(const UART_HandleTypeDef *huart); +uint32_t HAL_UART_GetError(const UART_HandleTypeDef *huart); /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_usart.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_usart.h index b64e95dc..655f8482 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_usart.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_hal_usart.h @@ -544,8 +544,8 @@ void HAL_USART_AbortCpltCallback(USART_HandleTypeDef *husart); * @{ */ /* Peripheral State functions ************************************************/ -HAL_USART_StateTypeDef HAL_USART_GetState(USART_HandleTypeDef *husart); -uint32_t HAL_USART_GetError(USART_HandleTypeDef *husart); +HAL_USART_StateTypeDef HAL_USART_GetState(const USART_HandleTypeDef *husart); +uint32_t HAL_USART_GetError(const USART_HandleTypeDef *husart); /** * @} */ @@ -624,7 +624,7 @@ uint32_t HAL_USART_GetError(USART_HandleTypeDef *husart); */ /* Private functions ---------------------------------------------------------*/ -/** @defgroup USART_Private_Functions USART Private Functions +/** @addtogroup USART_Private_Functions * @{ */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_adc.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_adc.h index 44e8af28..43b4a1ed 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_adc.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_adc.h @@ -684,8 +684,8 @@ typedef struct */ /** @defgroup ADC_LL_EC_REG_CONTINUOUS_MODE ADC group regular - Continuous mode -* @{ -*/ + * @{ + */ #define LL_ADC_REG_CONV_SINGLE 0x00000000UL /*!< ADC conversions are performed in single mode: one conversion per trigger */ #define LL_ADC_REG_CONV_CONTINUOUS (ADC_CR2_CONT) /*!< ADC conversions are performed in continuous mode: after the first trigger, following conversions launched successively automatically */ /** @@ -808,8 +808,8 @@ typedef struct */ /** @defgroup ADC_LL_EC_INJ_TRIG_AUTO ADC group injected - Automatic trigger mode -* @{ -*/ + * @{ + */ #define LL_ADC_INJ_TRIG_INDEPENDENT 0x00000000UL /*!< ADC group injected conversion trigger independent. Setting mandatory if ADC group injected injected trigger source is set to an external trigger. */ #define LL_ADC_INJ_TRIG_FROM_GRP_REGULAR (ADC_CR1_JAUTO) /*!< ADC group injected conversion trigger from ADC group regular. Setting compliant only with group injected trigger source set to SW start, without any further action on ADC group injected conversion start or stop: in this case, ADC group injected is controlled only from ADC group regular. */ /** @@ -1866,12 +1866,12 @@ __STATIC_INLINE uint32_t LL_ADC_DMA_GetRegAddr(ADC_TypeDef *ADCx, uint32_t Regis if (Register == LL_ADC_DMA_REG_REGULAR_DATA) { /* Retrieve address of register DR */ - data_reg_addr = (uint32_t)&(ADCx->DR); + data_reg_addr = (uint32_t) & (ADCx->DR); } else /* (Register == LL_ADC_DMA_REG_REGULAR_DATA_MULTI) */ { /* Retrieve address of register CDR */ - data_reg_addr = (uint32_t)&((__LL_ADC_COMMON_INSTANCE(ADCx))->CDR); + data_reg_addr = (uint32_t) & ((__LL_ADC_COMMON_INSTANCE(ADCx))->CDR); } return data_reg_addr; @@ -1879,8 +1879,11 @@ __STATIC_INLINE uint32_t LL_ADC_DMA_GetRegAddr(ADC_TypeDef *ADCx, uint32_t Regis #else __STATIC_INLINE uint32_t LL_ADC_DMA_GetRegAddr(ADC_TypeDef *ADCx, uint32_t Register) { + /* Prevent unused argument compilation warning */ + (void)Register; + /* Retrieve address of register DR */ - return (uint32_t)&(ADCx->DR); + return (uint32_t) & (ADCx->DR); } #endif @@ -2145,11 +2148,11 @@ __STATIC_INLINE uint32_t LL_ADC_GetSequencersScanMode(ADC_TypeDef *ADCx) */ __STATIC_INLINE void LL_ADC_REG_SetTriggerSource(ADC_TypeDef *ADCx, uint32_t TriggerSource) { -/* Note: On this STM32 series, ADC group regular external trigger edge */ -/* is used to perform a ADC conversion start. */ -/* This function does not set external trigger edge. */ -/* This feature is set using function */ -/* @ref LL_ADC_REG_StartConversionExtTrig(). */ + /* Note: On this STM32 series, ADC group regular external trigger edge */ + /* is used to perform a ADC conversion start. */ + /* This function does not set external trigger edge. */ + /* This feature is set using function */ + /* @ref LL_ADC_REG_StartConversionExtTrig(). */ MODIFY_REG(ADCx->CR2, ADC_CR2_EXTSEL, (TriggerSource & ADC_CR2_EXTSEL)); } @@ -2588,10 +2591,10 @@ __STATIC_INLINE uint32_t LL_ADC_REG_GetSequencerRanks(ADC_TypeDef *ADCx, uint32_ { __IO uint32_t *preg = __ADC_PTR_REG_OFFSET(ADCx->SQR1, __ADC_MASK_SHIFT(Rank, ADC_REG_SQRX_REGOFFSET_MASK)); - return (uint32_t) (READ_BIT(*preg, - ADC_CHANNEL_ID_NUMBER_MASK << (Rank & ADC_REG_RANK_ID_SQRX_MASK)) - >> (Rank & ADC_REG_RANK_ID_SQRX_MASK) - ); + return (uint32_t)(READ_BIT(*preg, + ADC_CHANNEL_ID_NUMBER_MASK << (Rank & ADC_REG_RANK_ID_SQRX_MASK)) + >> (Rank & ADC_REG_RANK_ID_SQRX_MASK) + ); } /** @@ -2780,11 +2783,11 @@ __STATIC_INLINE uint32_t LL_ADC_REG_GetFlagEndOfConversion(ADC_TypeDef *ADCx) */ __STATIC_INLINE void LL_ADC_INJ_SetTriggerSource(ADC_TypeDef *ADCx, uint32_t TriggerSource) { -/* Note: On this STM32 series, ADC group injected external trigger edge */ -/* is used to perform a ADC conversion start. */ -/* This function does not set external trigger edge. */ -/* This feature is set using function */ -/* @ref LL_ADC_INJ_StartConversionExtTrig(). */ + /* Note: On this STM32 series, ADC group injected external trigger edge */ + /* is used to perform a ADC conversion start. */ + /* This function does not set external trigger edge. */ + /* This feature is set using function */ + /* @ref LL_ADC_INJ_StartConversionExtTrig(). */ MODIFY_REG(ADCx->CR2, ADC_CR2_JEXTSEL, (TriggerSource & ADC_CR2_JEXTSEL)); } @@ -4015,7 +4018,7 @@ __STATIC_INLINE uint16_t LL_ADC_REG_ReadConversionData10(ADC_TypeDef *ADCx) */ __STATIC_INLINE uint8_t LL_ADC_REG_ReadConversionData8(ADC_TypeDef *ADCx) { - return (uint16_t)(READ_BIT(ADCx->DR, ADC_DR_DATA)); + return (uint8_t)(READ_BIT(ADCx->DR, ADC_DR_DATA)); } /** @@ -4030,7 +4033,7 @@ __STATIC_INLINE uint8_t LL_ADC_REG_ReadConversionData8(ADC_TypeDef *ADCx) */ __STATIC_INLINE uint8_t LL_ADC_REG_ReadConversionData6(ADC_TypeDef *ADCx) { - return (uint16_t)(READ_BIT(ADCx->DR, ADC_DR_DATA)); + return (uint8_t)(READ_BIT(ADCx->DR, ADC_DR_DATA)); } #if defined(ADC_MULTIMODE_SUPPORT) @@ -4546,7 +4549,7 @@ __STATIC_INLINE uint32_t LL_ADC_IsActiveFlag_SLV1_AWD1(ADC_Common_TypeDef *ADCxy */ __STATIC_INLINE uint32_t LL_ADC_IsActiveFlag_SLV2_AWD1(ADC_Common_TypeDef *ADCxy_COMMON) { - return (READ_BIT(ADCxy_COMMON->CSR, LL_ADC_FLAG_AWD1_SLV2) == (LL_ADC_FLAG_AWD1_SLV2)); + return (READ_BIT(ADCxy_COMMON->CSR, LL_ADC_FLAG_AWD1_SLV2) == (LL_ADC_FLAG_AWD1_SLV2)); } #endif /* ADC_MULTIMODE_SUPPORT */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_cortex.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_cortex.h index d478e130..3861f55f 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_cortex.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_cortex.h @@ -371,6 +371,16 @@ __STATIC_INLINE void LL_LPM_DisableEventOnPend(void) CLEAR_BIT(SCB->SCR, ((uint32_t)SCB_SCR_SEVONPEND_Msk)); } +/** + * @brief Clear pending events. + * @retval None + */ +__STATIC_INLINE void LL_LPM_ClearEvent(void) +{ + __SEV(); + __WFE(); +} + /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_crc.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_crc.h index 0a280062..8ada06d5 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_crc.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_crc.h @@ -135,7 +135,7 @@ __STATIC_INLINE void LL_CRC_FeedData32(CRC_TypeDef *CRCx, uint32_t InData) * @param CRCx CRC Instance * @retval Current CRC calculation result as stored in CRC_DR register (32 bits). */ -__STATIC_INLINE uint32_t LL_CRC_ReadData32(CRC_TypeDef *CRCx) +__STATIC_INLINE uint32_t LL_CRC_ReadData32(const CRC_TypeDef *CRCx) { return (uint32_t)(READ_REG(CRCx->DR)); } @@ -173,7 +173,7 @@ __STATIC_INLINE void LL_CRC_Write_IDR(CRC_TypeDef *CRCx, uint32_t InData) * @{ */ -ErrorStatus LL_CRC_DeInit(CRC_TypeDef *CRCx); +ErrorStatus LL_CRC_DeInit(const CRC_TypeDef *CRCx); /** * @} diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_dac.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_dac.h index ea1500ba..4bff015f 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_dac.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_dac.h @@ -118,10 +118,10 @@ extern "C" { /* DAC registers bits positions */ #if defined(DAC_CHANNEL2_SUPPORT) -#endif #define DAC_DHR12RD_DACC2DHR_BITOFFSET_POS DAC_DHR12RD_DACC2DHR_Pos #define DAC_DHR12LD_DACC2DHR_BITOFFSET_POS DAC_DHR12LD_DACC2DHR_Pos #define DAC_DHR8RD_DACC2DHR_BITOFFSET_POS DAC_DHR8RD_DACC2DHR_Pos +#endif /* DAC_CHANNEL2_SUPPORT */ /* Miscellaneous data */ #define DAC_DIGITAL_SCALE_12BITS 4095UL /* Full-scale digital value with a resolution of 12 @@ -429,7 +429,7 @@ typedef struct * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval 1...2 (value "2" depending on DAC channel 2 availability) */ @@ -449,36 +449,26 @@ typedef struct * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. */ #if defined(DAC_CHANNEL2_SUPPORT) -#define __LL_DAC_DECIMAL_NB_TO_CHANNEL(__DECIMAL_NB__) \ - (((__DECIMAL_NB__) == 1UL) \ - ? ( \ - LL_DAC_CHANNEL_1 \ - ) \ - : \ - (((__DECIMAL_NB__) == 2UL) \ - ? ( \ - LL_DAC_CHANNEL_2 \ - ) \ - : \ - ( \ - 0UL \ - ) \ - ) \ +#define __LL_DAC_DECIMAL_NB_TO_CHANNEL(__DECIMAL_NB__) \ + (((__DECIMAL_NB__) == 1UL) \ + ? (LL_DAC_CHANNEL_1) \ + : \ + (((__DECIMAL_NB__) == 2UL) \ + ? (LL_DAC_CHANNEL_2) \ + : \ + (0UL) \ + ) \ ) #else -#define __LL_DAC_DECIMAL_NB_TO_CHANNEL(__DECIMAL_NB__) \ - (((__DECIMAL_NB__) == 1UL) \ - ? ( \ - LL_DAC_CHANNEL_1 \ - ) \ - : \ - ( \ - 0UL \ - ) \ +#define __LL_DAC_DECIMAL_NB_TO_CHANNEL(__DECIMAL_NB__) \ + (((__DECIMAL_NB__) == 1UL) \ + ? (LL_DAC_CHANNEL_1) \ + : \ + (0UL) \ ) #endif /* DAC_CHANNEL2_SUPPORT */ @@ -514,12 +504,10 @@ typedef struct * @arg @ref LL_DAC_RESOLUTION_8B * @retval DAC conversion data (unit: digital value) */ -#define __LL_DAC_CALC_VOLTAGE_TO_DATA(__VREFANALOG_VOLTAGE__,\ - __DAC_VOLTAGE__,\ - __DAC_RESOLUTION__) \ -((__DAC_VOLTAGE__) * __LL_DAC_DIGITAL_SCALE(__DAC_RESOLUTION__) \ - / (__VREFANALOG_VOLTAGE__) \ -) +#define __LL_DAC_CALC_VOLTAGE_TO_DATA(__VREFANALOG_VOLTAGE__, __DAC_VOLTAGE__, __DAC_RESOLUTION__) \ + ((__DAC_VOLTAGE__) * __LL_DAC_DIGITAL_SCALE(__DAC_RESOLUTION__) \ + / (__VREFANALOG_VOLTAGE__) \ + ) /** * @} @@ -534,6 +522,7 @@ typedef struct /** @defgroup DAC_LL_Exported_Functions DAC Exported Functions * @{ */ + /** * @brief Set the conversion trigger source for the selected DAC channel. * @note For conversion trigger source to be effective, DAC trigger @@ -549,7 +538,7 @@ typedef struct * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @param TriggerSource This parameter can be one of the following values: * @arg @ref LL_DAC_TRIG_SOFTWARE @@ -582,7 +571,7 @@ __STATIC_INLINE void LL_DAC_SetTriggerSource(DAC_TypeDef *DACx, uint32_t DAC_Cha * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval Returned value can be one of the following values: * @arg @ref LL_DAC_TRIG_SOFTWARE @@ -594,7 +583,7 @@ __STATIC_INLINE void LL_DAC_SetTriggerSource(DAC_TypeDef *DACx, uint32_t DAC_Cha * @arg @ref LL_DAC_TRIG_EXT_TIM2_TRGO * @arg @ref LL_DAC_TRIG_EXT_EXTI_LINE9 */ -__STATIC_INLINE uint32_t LL_DAC_GetTriggerSource(DAC_TypeDef *DACx, uint32_t DAC_Channel) +__STATIC_INLINE uint32_t LL_DAC_GetTriggerSource(const DAC_TypeDef *DACx, uint32_t DAC_Channel) { return (uint32_t)(READ_BIT(DACx->CR, DAC_CR_TSEL1 << (DAC_Channel & DAC_CR_CHX_BITOFFSET_MASK)) >> (DAC_Channel & DAC_CR_CHX_BITOFFSET_MASK) @@ -611,7 +600,7 @@ __STATIC_INLINE uint32_t LL_DAC_GetTriggerSource(DAC_TypeDef *DACx, uint32_t DAC * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @param WaveAutoGeneration This parameter can be one of the following values: * @arg @ref LL_DAC_WAVE_AUTO_GENERATION_NONE @@ -636,14 +625,14 @@ __STATIC_INLINE void LL_DAC_SetWaveAutoGeneration(DAC_TypeDef *DACx, uint32_t DA * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval Returned value can be one of the following values: * @arg @ref LL_DAC_WAVE_AUTO_GENERATION_NONE * @arg @ref LL_DAC_WAVE_AUTO_GENERATION_NOISE * @arg @ref LL_DAC_WAVE_AUTO_GENERATION_TRIANGLE */ -__STATIC_INLINE uint32_t LL_DAC_GetWaveAutoGeneration(DAC_TypeDef *DACx, uint32_t DAC_Channel) +__STATIC_INLINE uint32_t LL_DAC_GetWaveAutoGeneration(const DAC_TypeDef *DACx, uint32_t DAC_Channel) { return (uint32_t)(READ_BIT(DACx->CR, DAC_CR_WAVE1 << (DAC_Channel & DAC_CR_CHX_BITOFFSET_MASK)) >> (DAC_Channel & DAC_CR_CHX_BITOFFSET_MASK) @@ -665,7 +654,7 @@ __STATIC_INLINE uint32_t LL_DAC_GetWaveAutoGeneration(DAC_TypeDef *DACx, uint32_ * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @param NoiseLFSRMask This parameter can be one of the following values: * @arg @ref LL_DAC_NOISE_LFSR_UNMASK_BIT0 @@ -699,7 +688,7 @@ __STATIC_INLINE void LL_DAC_SetWaveNoiseLFSR(DAC_TypeDef *DACx, uint32_t DAC_Cha * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval Returned value can be one of the following values: * @arg @ref LL_DAC_NOISE_LFSR_UNMASK_BIT0 @@ -715,7 +704,7 @@ __STATIC_INLINE void LL_DAC_SetWaveNoiseLFSR(DAC_TypeDef *DACx, uint32_t DAC_Cha * @arg @ref LL_DAC_NOISE_LFSR_UNMASK_BITS10_0 * @arg @ref LL_DAC_NOISE_LFSR_UNMASK_BITS11_0 */ -__STATIC_INLINE uint32_t LL_DAC_GetWaveNoiseLFSR(DAC_TypeDef *DACx, uint32_t DAC_Channel) +__STATIC_INLINE uint32_t LL_DAC_GetWaveNoiseLFSR(const DAC_TypeDef *DACx, uint32_t DAC_Channel) { return (uint32_t)(READ_BIT(DACx->CR, DAC_CR_MAMP1 << (DAC_Channel & DAC_CR_CHX_BITOFFSET_MASK)) >> (DAC_Channel & DAC_CR_CHX_BITOFFSET_MASK) @@ -737,7 +726,7 @@ __STATIC_INLINE uint32_t LL_DAC_GetWaveNoiseLFSR(DAC_TypeDef *DACx, uint32_t DAC * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @param TriangleAmplitude This parameter can be one of the following values: * @arg @ref LL_DAC_TRIANGLE_AMPLITUDE_1 @@ -772,7 +761,7 @@ __STATIC_INLINE void LL_DAC_SetWaveTriangleAmplitude(DAC_TypeDef *DACx, uint32_t * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval Returned value can be one of the following values: * @arg @ref LL_DAC_TRIANGLE_AMPLITUDE_1 @@ -788,7 +777,7 @@ __STATIC_INLINE void LL_DAC_SetWaveTriangleAmplitude(DAC_TypeDef *DACx, uint32_t * @arg @ref LL_DAC_TRIANGLE_AMPLITUDE_2047 * @arg @ref LL_DAC_TRIANGLE_AMPLITUDE_4095 */ -__STATIC_INLINE uint32_t LL_DAC_GetWaveTriangleAmplitude(DAC_TypeDef *DACx, uint32_t DAC_Channel) +__STATIC_INLINE uint32_t LL_DAC_GetWaveTriangleAmplitude(const DAC_TypeDef *DACx, uint32_t DAC_Channel) { return (uint32_t)(READ_BIT(DACx->CR, DAC_CR_MAMP1 << (DAC_Channel & DAC_CR_CHX_BITOFFSET_MASK)) >> (DAC_Channel & DAC_CR_CHX_BITOFFSET_MASK) @@ -804,7 +793,7 @@ __STATIC_INLINE uint32_t LL_DAC_GetWaveTriangleAmplitude(DAC_TypeDef *DACx, uint * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @param OutputBuffer This parameter can be one of the following values: * @arg @ref LL_DAC_OUTPUT_BUFFER_ENABLE @@ -827,13 +816,13 @@ __STATIC_INLINE void LL_DAC_SetOutputBuffer(DAC_TypeDef *DACx, uint32_t DAC_Chan * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval Returned value can be one of the following values: * @arg @ref LL_DAC_OUTPUT_BUFFER_ENABLE * @arg @ref LL_DAC_OUTPUT_BUFFER_DISABLE */ -__STATIC_INLINE uint32_t LL_DAC_GetOutputBuffer(DAC_TypeDef *DACx, uint32_t DAC_Channel) +__STATIC_INLINE uint32_t LL_DAC_GetOutputBuffer(const DAC_TypeDef *DACx, uint32_t DAC_Channel) { return (uint32_t)(READ_BIT(DACx->CR, DAC_CR_BOFF1 << (DAC_Channel & DAC_CR_CHX_BITOFFSET_MASK)) >> (DAC_Channel & DAC_CR_CHX_BITOFFSET_MASK) @@ -859,7 +848,7 @@ __STATIC_INLINE uint32_t LL_DAC_GetOutputBuffer(DAC_TypeDef *DACx, uint32_t DAC_ * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval None */ @@ -880,7 +869,7 @@ __STATIC_INLINE void LL_DAC_EnableDMAReq(DAC_TypeDef *DACx, uint32_t DAC_Channel * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval None */ @@ -900,11 +889,11 @@ __STATIC_INLINE void LL_DAC_DisableDMAReq(DAC_TypeDef *DACx, uint32_t DAC_Channe * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_DAC_IsDMAReqEnabled(DAC_TypeDef *DACx, uint32_t DAC_Channel) +__STATIC_INLINE uint32_t LL_DAC_IsDMAReqEnabled(const DAC_TypeDef *DACx, uint32_t DAC_Channel) { return ((READ_BIT(DACx->CR, DAC_CR_DMAEN1 << (DAC_Channel & DAC_CR_CHX_BITOFFSET_MASK)) @@ -938,7 +927,7 @@ __STATIC_INLINE uint32_t LL_DAC_IsDMAReqEnabled(DAC_TypeDef *DACx, uint32_t DAC_ * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @param Register This parameter can be one of the following values: * @arg @ref LL_DAC_DMA_REG_DATA_12BITS_RIGHT_ALIGNED @@ -946,7 +935,7 @@ __STATIC_INLINE uint32_t LL_DAC_IsDMAReqEnabled(DAC_TypeDef *DACx, uint32_t DAC_ * @arg @ref LL_DAC_DMA_REG_DATA_8BITS_RIGHT_ALIGNED * @retval DAC register address */ -__STATIC_INLINE uint32_t LL_DAC_DMA_GetRegAddr(DAC_TypeDef *DACx, uint32_t DAC_Channel, uint32_t Register) +__STATIC_INLINE uint32_t LL_DAC_DMA_GetRegAddr(const DAC_TypeDef *DACx, uint32_t DAC_Channel, uint32_t Register) { /* Retrieve address of register DHR12Rx, DHR12Lx or DHR8Rx depending on */ /* DAC channel selected. */ @@ -973,7 +962,7 @@ __STATIC_INLINE uint32_t LL_DAC_DMA_GetRegAddr(DAC_TypeDef *DACx, uint32_t DAC_C * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval None */ @@ -992,7 +981,7 @@ __STATIC_INLINE void LL_DAC_Enable(DAC_TypeDef *DACx, uint32_t DAC_Channel) * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval None */ @@ -1012,11 +1001,11 @@ __STATIC_INLINE void LL_DAC_Disable(DAC_TypeDef *DACx, uint32_t DAC_Channel) * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_DAC_IsEnabled(DAC_TypeDef *DACx, uint32_t DAC_Channel) +__STATIC_INLINE uint32_t LL_DAC_IsEnabled(const DAC_TypeDef *DACx, uint32_t DAC_Channel) { return ((READ_BIT(DACx->CR, DAC_CR_EN1 << (DAC_Channel & DAC_CR_CHX_BITOFFSET_MASK)) @@ -1040,7 +1029,7 @@ __STATIC_INLINE uint32_t LL_DAC_IsEnabled(DAC_TypeDef *DACx, uint32_t DAC_Channe * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval None */ @@ -1059,7 +1048,7 @@ __STATIC_INLINE void LL_DAC_EnableTrigger(DAC_TypeDef *DACx, uint32_t DAC_Channe * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval None */ @@ -1079,11 +1068,11 @@ __STATIC_INLINE void LL_DAC_DisableTrigger(DAC_TypeDef *DACx, uint32_t DAC_Chann * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_DAC_IsTriggerEnabled(DAC_TypeDef *DACx, uint32_t DAC_Channel) +__STATIC_INLINE uint32_t LL_DAC_IsTriggerEnabled(const DAC_TypeDef *DACx, uint32_t DAC_Channel) { return ((READ_BIT(DACx->CR, DAC_CR_TEN1 << (DAC_Channel & DAC_CR_CHX_BITOFFSET_MASK)) @@ -1110,7 +1099,7 @@ __STATIC_INLINE uint32_t LL_DAC_IsTriggerEnabled(DAC_TypeDef *DACx, uint32_t DAC * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval None */ @@ -1131,7 +1120,7 @@ __STATIC_INLINE void LL_DAC_TrigSWConversion(DAC_TypeDef *DACx, uint32_t DAC_Cha * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @param Data Value between Min_Data=0x000 and Max_Data=0xFFF * @retval None @@ -1155,7 +1144,7 @@ __STATIC_INLINE void LL_DAC_ConvertData12RightAligned(DAC_TypeDef *DACx, uint32_ * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @param Data Value between Min_Data=0x000 and Max_Data=0xFFF * @retval None @@ -1179,7 +1168,7 @@ __STATIC_INLINE void LL_DAC_ConvertData12LeftAligned(DAC_TypeDef *DACx, uint32_t * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @param Data Value between Min_Data=0x00 and Max_Data=0xFF * @retval None @@ -1267,11 +1256,11 @@ __STATIC_INLINE void LL_DAC_ConvertDualData8RightAligned(DAC_TypeDef *DACx, uint * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @retval Value between Min_Data=0x000 and Max_Data=0xFFF */ -__STATIC_INLINE uint32_t LL_DAC_RetrieveOutputData(DAC_TypeDef *DACx, uint32_t DAC_Channel) +__STATIC_INLINE uint32_t LL_DAC_RetrieveOutputData(const DAC_TypeDef *DACx, uint32_t DAC_Channel) { __IO uint32_t const *preg = __DAC_PTR_REG_OFFSET(DACx->DOR1, (DAC_Channel >> DAC_REG_DORX_REGOFFSET_BITOFFSET_POS) & DAC_REG_DORX_REGOFFSET_MASK_POSBIT0); @@ -1293,7 +1282,7 @@ __STATIC_INLINE uint32_t LL_DAC_RetrieveOutputData(DAC_TypeDef *DACx, uint32_t D * @param DACx DAC instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_DAC_IsActiveFlag_DMAUDR1(DAC_TypeDef *DACx) +__STATIC_INLINE uint32_t LL_DAC_IsActiveFlag_DMAUDR1(const DAC_TypeDef *DACx) { return ((READ_BIT(DACx->SR, LL_DAC_FLAG_DMAUDR1) == (LL_DAC_FLAG_DMAUDR1)) ? 1UL : 0UL); } @@ -1305,7 +1294,7 @@ __STATIC_INLINE uint32_t LL_DAC_IsActiveFlag_DMAUDR1(DAC_TypeDef *DACx) * @param DACx DAC instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_DAC_IsActiveFlag_DMAUDR2(DAC_TypeDef *DACx) +__STATIC_INLINE uint32_t LL_DAC_IsActiveFlag_DMAUDR2(const DAC_TypeDef *DACx) { return ((READ_BIT(DACx->SR, LL_DAC_FLAG_DMAUDR2) == (LL_DAC_FLAG_DMAUDR2)) ? 1UL : 0UL); } @@ -1397,7 +1386,7 @@ __STATIC_INLINE void LL_DAC_DisableIT_DMAUDR2(DAC_TypeDef *DACx) * @param DACx DAC instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_DAC_IsEnabledIT_DMAUDR1(DAC_TypeDef *DACx) +__STATIC_INLINE uint32_t LL_DAC_IsEnabledIT_DMAUDR1(const DAC_TypeDef *DACx) { return ((READ_BIT(DACx->CR, LL_DAC_IT_DMAUDRIE1) == (LL_DAC_IT_DMAUDRIE1)) ? 1UL : 0UL); } @@ -1409,7 +1398,7 @@ __STATIC_INLINE uint32_t LL_DAC_IsEnabledIT_DMAUDR1(DAC_TypeDef *DACx) * @param DACx DAC instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_DAC_IsEnabledIT_DMAUDR2(DAC_TypeDef *DACx) +__STATIC_INLINE uint32_t LL_DAC_IsEnabledIT_DMAUDR2(const DAC_TypeDef *DACx) { return ((READ_BIT(DACx->CR, LL_DAC_IT_DMAUDRIE2) == (LL_DAC_IT_DMAUDRIE2)) ? 1UL : 0UL); } @@ -1424,8 +1413,8 @@ __STATIC_INLINE uint32_t LL_DAC_IsEnabledIT_DMAUDR2(DAC_TypeDef *DACx) * @{ */ -ErrorStatus LL_DAC_DeInit(DAC_TypeDef *DACx); -ErrorStatus LL_DAC_Init(DAC_TypeDef *DACx, uint32_t DAC_Channel, LL_DAC_InitTypeDef *DAC_InitStruct); +ErrorStatus LL_DAC_DeInit(const DAC_TypeDef *DACx); +ErrorStatus LL_DAC_Init(DAC_TypeDef *DACx, uint32_t DAC_Channel, const LL_DAC_InitTypeDef *DAC_InitStruct); void LL_DAC_StructInit(LL_DAC_InitTypeDef *DAC_InitStruct); /** @@ -1452,4 +1441,3 @@ void LL_DAC_StructInit(LL_DAC_InitTypeDef *DAC_InitStruct); #endif #endif /* STM32F4xx_LL_DAC_H */ - diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_fmc.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_fmc.h index 3dc863d4..dda9d895 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_fmc.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_fmc.h @@ -369,7 +369,7 @@ typedef struct delay between ALE low and RE low. This parameter can be a number between Min_Data = 0 and Max_Data = 255 */ } FMC_NAND_InitTypeDef; -#endif +#endif /* FMC_Bank3 || FMC_Bank2_3 */ #if defined(FMC_Bank3) || defined(FMC_Bank2_3) || defined(FMC_Bank4) /** @@ -1388,7 +1388,7 @@ HAL_StatusTypeDef FMC_SDRAM_SendCommand(FMC_SDRAM_TypeDef *Device, HAL_StatusTypeDef FMC_SDRAM_ProgramRefreshRate(FMC_SDRAM_TypeDef *Device, uint32_t RefreshRate); HAL_StatusTypeDef FMC_SDRAM_SetAutoRefreshNumber(FMC_SDRAM_TypeDef *Device, uint32_t AutoRefreshNumber); -uint32_t FMC_SDRAM_GetModeStatus(FMC_SDRAM_TypeDef *Device, uint32_t Bank); +uint32_t FMC_SDRAM_GetModeStatus(const FMC_SDRAM_TypeDef *Device, uint32_t Bank); /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_fmpi2c.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_fmpi2c.h index dcc00160..a3c14b6e 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_fmpi2c.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_fmpi2c.h @@ -295,7 +295,7 @@ typedef struct #define LL_FMPI2C_GENERATE_RESTART_7BIT_WRITE (uint32_t)(0x80000000U | FMPI2C_CR2_START) /*!< Generate Restart for write request, slave 7Bit address. */ #define LL_FMPI2C_GENERATE_RESTART_10BIT_READ (uint32_t)(0x80000000U | FMPI2C_CR2_START | \ - FMPI2C_CR2_RD_WRN | FMPI2C_CR2_HEAD10R) + FMPI2C_CR2_RD_WRN | FMPI2C_CR2_HEAD10R) /*!< Generate Restart for read request, slave 10Bit address. */ #define LL_FMPI2C_GENERATE_RESTART_10BIT_WRITE (uint32_t)(0x80000000U | FMPI2C_CR2_START) /*!< Generate Restart for write request, slave 10Bit address.*/ @@ -343,7 +343,7 @@ typedef struct #define LL_FMPI2C_FMPSMBUS_TIMEOUTB FMPI2C_TIMEOUTR_TEXTEN /*!< TimeoutB (extended clock) enable bit */ #define LL_FMPI2C_FMPSMBUS_ALL_TIMEOUT (uint32_t)(FMPI2C_TIMEOUTR_TIMOUTEN | \ - FMPI2C_TIMEOUTR_TEXTEN) /*!< TimeoutA and TimeoutB + FMPI2C_TIMEOUTR_TEXTEN) /*!< TimeoutA and TimeoutB (extended clock) enable bits */ /** * @} @@ -452,7 +452,7 @@ __STATIC_INLINE void LL_FMPI2C_Disable(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabled(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabled(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_PE) == (FMPI2C_CR1_PE)) ? 1UL : 0UL); } @@ -501,7 +501,7 @@ __STATIC_INLINE void LL_FMPI2C_SetDigitalFilter(FMPI2C_TypeDef *FMPI2Cx, uint32_ * @param FMPI2Cx FMPI2C Instance. * @retval Value between Min_Data=0x0 and Max_Data=0xF */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetDigitalFilter(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetDigitalFilter(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_DNF) >> FMPI2C_CR1_DNF_Pos); } @@ -536,7 +536,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableAnalogFilter(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledAnalogFilter(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledAnalogFilter(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_ANFOFF) != (FMPI2C_CR1_ANFOFF)) ? 1UL : 0UL); } @@ -569,7 +569,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableDMAReq_TX(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledDMAReq_TX(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledDMAReq_TX(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_TXDMAEN) == (FMPI2C_CR1_TXDMAEN)) ? 1UL : 0UL); } @@ -602,7 +602,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableDMAReq_RX(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledDMAReq_RX(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledDMAReq_RX(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_RXDMAEN) == (FMPI2C_CR1_RXDMAEN)) ? 1UL : 0UL); } @@ -617,7 +617,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledDMAReq_RX(FMPI2C_TypeDef *FMPI2Cx) * @arg @ref LL_FMPI2C_DMA_REG_DATA_RECEIVE * @retval Address of data register */ -__STATIC_INLINE uint32_t LL_FMPI2C_DMA_GetRegAddr(FMPI2C_TypeDef *FMPI2Cx, uint32_t Direction) +__STATIC_INLINE uint32_t LL_FMPI2C_DMA_GetRegAddr(const FMPI2C_TypeDef *FMPI2Cx, uint32_t Direction) { uint32_t data_reg_addr; @@ -665,7 +665,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableClockStretching(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledClockStretching(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledClockStretching(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_NOSTRETCH) != (FMPI2C_CR1_NOSTRETCH)) ? 1UL : 0UL); } @@ -698,7 +698,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableSlaveByteControl(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledSlaveByteControl(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledSlaveByteControl(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_SBC) == (FMPI2C_CR1_SBC)) ? 1UL : 0UL); } @@ -733,7 +733,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableGeneralCall(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledGeneralCall(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledGeneralCall(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_GCEN) == (FMPI2C_CR1_GCEN)) ? 1UL : 0UL); } @@ -761,7 +761,7 @@ __STATIC_INLINE void LL_FMPI2C_SetMasterAddressingMode(FMPI2C_TypeDef *FMPI2Cx, * @arg @ref LL_FMPI2C_ADDRESSING_MODE_7BIT * @arg @ref LL_FMPI2C_ADDRESSING_MODE_10BIT */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetMasterAddressingMode(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetMasterAddressingMode(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->CR2, FMPI2C_CR2_ADD10)); } @@ -810,7 +810,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableOwnAddress1(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledOwnAddress1(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledOwnAddress1(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->OAR1, FMPI2C_OAR1_OA1EN) == (FMPI2C_OAR1_OA1EN)) ? 1UL : 0UL); } @@ -866,7 +866,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableOwnAddress2(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledOwnAddress2(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledOwnAddress2(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->OAR2, FMPI2C_OAR2_OA2EN) == (FMPI2C_OAR2_OA2EN)) ? 1UL : 0UL); } @@ -891,7 +891,7 @@ __STATIC_INLINE void LL_FMPI2C_SetTiming(FMPI2C_TypeDef *FMPI2Cx, uint32_t Timin * @param FMPI2Cx FMPI2C Instance. * @retval Value between Min_Data=0x0 and Max_Data=0xF */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetTimingPrescaler(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetTimingPrescaler(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->TIMINGR, FMPI2C_TIMINGR_PRESC) >> FMPI2C_TIMINGR_PRESC_Pos); } @@ -902,7 +902,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_GetTimingPrescaler(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval Value between Min_Data=0x00 and Max_Data=0xFF */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetClockLowPeriod(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetClockLowPeriod(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->TIMINGR, FMPI2C_TIMINGR_SCLL) >> FMPI2C_TIMINGR_SCLL_Pos); } @@ -913,7 +913,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_GetClockLowPeriod(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval Value between Min_Data=0x00 and Max_Data=0xFF */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetClockHighPeriod(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetClockHighPeriod(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->TIMINGR, FMPI2C_TIMINGR_SCLH) >> FMPI2C_TIMINGR_SCLH_Pos); } @@ -924,7 +924,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_GetClockHighPeriod(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval Value between Min_Data=0x0 and Max_Data=0xF */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetDataHoldTime(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetDataHoldTime(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->TIMINGR, FMPI2C_TIMINGR_SDADEL) >> FMPI2C_TIMINGR_SDADEL_Pos); } @@ -935,7 +935,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_GetDataHoldTime(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval Value between Min_Data=0x0 and Max_Data=0xF */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetDataSetupTime(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetDataSetupTime(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->TIMINGR, FMPI2C_TIMINGR_SCLDEL) >> FMPI2C_TIMINGR_SCLDEL_Pos); } @@ -972,7 +972,7 @@ __STATIC_INLINE void LL_FMPI2C_SetMode(FMPI2C_TypeDef *FMPI2Cx, uint32_t Periphe * @arg @ref LL_FMPI2C_MODE_SMBUS_DEVICE * @arg @ref LL_FMPI2C_MODE_SMBUS_DEVICE_ARP */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetMode(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetMode(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_SMBHEN | FMPI2C_CR1_SMBDEN)); } @@ -1021,7 +1021,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableSMBusAlert(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledSMBusAlert(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledSMBusAlert(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_ALERTEN) == (FMPI2C_CR1_ALERTEN)) ? 1UL : 0UL); } @@ -1060,7 +1060,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableSMBusPEC(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledSMBusPEC(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledSMBusPEC(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_PECEN) == (FMPI2C_CR1_PECEN)) ? 1UL : 0UL); } @@ -1082,7 +1082,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledSMBusPEC(FMPI2C_TypeDef *FMPI2Cx) * @retval None */ __STATIC_INLINE void LL_FMPI2C_ConfigSMBusTimeout(FMPI2C_TypeDef *FMPI2Cx, uint32_t TimeoutA, uint32_t TimeoutAMode, - uint32_t TimeoutB) + uint32_t TimeoutB) { MODIFY_REG(FMPI2Cx->TIMEOUTR, FMPI2C_TIMEOUTR_TIMEOUTA | FMPI2C_TIMEOUTR_TIDLE | FMPI2C_TIMEOUTR_TIMEOUTB, TimeoutA | TimeoutAMode | (TimeoutB << FMPI2C_TIMEOUTR_TIMEOUTB_Pos)); @@ -1111,7 +1111,7 @@ __STATIC_INLINE void LL_FMPI2C_SetSMBusTimeoutA(FMPI2C_TypeDef *FMPI2Cx, uint32_ * @param FMPI2Cx FMPI2C Instance. * @retval Value between Min_Data=0 and Max_Data=0xFFF */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetSMBusTimeoutA(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetSMBusTimeoutA(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->TIMEOUTR, FMPI2C_TIMEOUTR_TIMEOUTA)); } @@ -1143,7 +1143,7 @@ __STATIC_INLINE void LL_FMPI2C_SetSMBusTimeoutAMode(FMPI2C_TypeDef *FMPI2Cx, uin * @arg @ref LL_FMPI2C_FMPSMBUS_TIMEOUTA_MODE_SCL_LOW * @arg @ref LL_FMPI2C_FMPSMBUS_TIMEOUTA_MODE_SDA_SCL_HIGH */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetSMBusTimeoutAMode(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetSMBusTimeoutAMode(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->TIMEOUTR, FMPI2C_TIMEOUTR_TIDLE)); } @@ -1171,7 +1171,7 @@ __STATIC_INLINE void LL_FMPI2C_SetSMBusTimeoutB(FMPI2C_TypeDef *FMPI2Cx, uint32_ * @param FMPI2Cx FMPI2C Instance. * @retval Value between Min_Data=0 and Max_Data=0xFFF */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetSMBusTimeoutB(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetSMBusTimeoutB(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->TIMEOUTR, FMPI2C_TIMEOUTR_TIMEOUTB) >> FMPI2C_TIMEOUTR_TIMEOUTB_Pos); } @@ -1225,7 +1225,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableSMBusTimeout(FMPI2C_TypeDef *FMPI2Cx, uint * @arg @ref LL_FMPI2C_FMPSMBUS_ALL_TIMEOUT * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledSMBusTimeout(FMPI2C_TypeDef *FMPI2Cx, uint32_t ClockTimeout) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledSMBusTimeout(const FMPI2C_TypeDef *FMPI2Cx, uint32_t ClockTimeout) { return ((READ_BIT(FMPI2Cx->TIMEOUTR, (FMPI2C_TIMEOUTR_TIMOUTEN | FMPI2C_TIMEOUTR_TEXTEN)) == \ (ClockTimeout)) ? 1UL : 0UL); @@ -1267,7 +1267,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableIT_TX(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledIT_TX(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledIT_TX(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_TXIE) == (FMPI2C_CR1_TXIE)) ? 1UL : 0UL); } @@ -1300,7 +1300,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableIT_RX(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledIT_RX(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledIT_RX(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_RXIE) == (FMPI2C_CR1_RXIE)) ? 1UL : 0UL); } @@ -1333,7 +1333,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableIT_ADDR(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledIT_ADDR(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledIT_ADDR(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_ADDRIE) == (FMPI2C_CR1_ADDRIE)) ? 1UL : 0UL); } @@ -1366,7 +1366,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableIT_NACK(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledIT_NACK(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledIT_NACK(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_NACKIE) == (FMPI2C_CR1_NACKIE)) ? 1UL : 0UL); } @@ -1399,7 +1399,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableIT_STOP(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledIT_STOP(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledIT_STOP(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_STOPIE) == (FMPI2C_CR1_STOPIE)) ? 1UL : 0UL); } @@ -1438,7 +1438,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableIT_TC(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledIT_TC(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledIT_TC(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_TCIE) == (FMPI2C_CR1_TCIE)) ? 1UL : 0UL); } @@ -1489,7 +1489,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableIT_ERR(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledIT_ERR(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledIT_ERR(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR1, FMPI2C_CR1_ERRIE) == (FMPI2C_CR1_ERRIE)) ? 1UL : 0UL); } @@ -1510,7 +1510,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledIT_ERR(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_TXE(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_TXE(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_TXE) == (FMPI2C_ISR_TXE)) ? 1UL : 0UL); } @@ -1523,7 +1523,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_TXE(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_TXIS(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_TXIS(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_TXIS) == (FMPI2C_ISR_TXIS)) ? 1UL : 0UL); } @@ -1536,7 +1536,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_TXIS(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_RXNE(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_RXNE(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_RXNE) == (FMPI2C_ISR_RXNE)) ? 1UL : 0UL); } @@ -1549,7 +1549,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_RXNE(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_ADDR(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_ADDR(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_ADDR) == (FMPI2C_ISR_ADDR)) ? 1UL : 0UL); } @@ -1562,7 +1562,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_ADDR(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_NACK(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_NACK(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_NACKF) == (FMPI2C_ISR_NACKF)) ? 1UL : 0UL); } @@ -1575,7 +1575,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_NACK(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_STOP(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_STOP(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_STOPF) == (FMPI2C_ISR_STOPF)) ? 1UL : 0UL); } @@ -1588,7 +1588,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_STOP(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_TC(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_TC(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_TC) == (FMPI2C_ISR_TC)) ? 1UL : 0UL); } @@ -1601,7 +1601,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_TC(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_TCR(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_TCR(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_TCR) == (FMPI2C_ISR_TCR)) ? 1UL : 0UL); } @@ -1614,7 +1614,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_TCR(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_BERR(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_BERR(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_BERR) == (FMPI2C_ISR_BERR)) ? 1UL : 0UL); } @@ -1627,7 +1627,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_BERR(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_ARLO(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_ARLO(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_ARLO) == (FMPI2C_ISR_ARLO)) ? 1UL : 0UL); } @@ -1640,7 +1640,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_ARLO(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_OVR(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_OVR(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_OVR) == (FMPI2C_ISR_OVR)) ? 1UL : 0UL); } @@ -1655,7 +1655,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_OVR(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveSMBusFlag_PECERR(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveSMBusFlag_PECERR(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_PECERR) == (FMPI2C_ISR_PECERR)) ? 1UL : 0UL); } @@ -1670,7 +1670,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsActiveSMBusFlag_PECERR(FMPI2C_TypeDef *FMPI * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveSMBusFlag_TIMEOUT(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveSMBusFlag_TIMEOUT(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_TIMEOUT) == (FMPI2C_ISR_TIMEOUT)) ? 1UL : 0UL); } @@ -1686,7 +1686,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsActiveSMBusFlag_TIMEOUT(FMPI2C_TypeDef *FMP * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveSMBusFlag_ALERT(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveSMBusFlag_ALERT(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_ALERT) == (FMPI2C_ISR_ALERT)) ? 1UL : 0UL); } @@ -1699,7 +1699,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsActiveSMBusFlag_ALERT(FMPI2C_TypeDef *FMPI2 * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_BUSY(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsActiveFlag_BUSY(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_BUSY) == (FMPI2C_ISR_BUSY)) ? 1UL : 0UL); } @@ -1860,7 +1860,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableAutoEndMode(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledAutoEndMode(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledAutoEndMode(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR2, FMPI2C_CR2_AUTOEND) == (FMPI2C_CR2_AUTOEND)) ? 1UL : 0UL); } @@ -1895,7 +1895,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableReloadMode(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledReloadMode(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledReloadMode(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR2, FMPI2C_CR2_RELOAD) == (FMPI2C_CR2_RELOAD)) ? 1UL : 0UL); } @@ -1919,7 +1919,7 @@ __STATIC_INLINE void LL_FMPI2C_SetTransferSize(FMPI2C_TypeDef *FMPI2Cx, uint32_t * @param FMPI2Cx FMPI2C Instance. * @retval Value between Min_Data=0x0 and Max_Data=0xFF */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetTransferSize(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetTransferSize(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->CR2, FMPI2C_CR2_NBYTES) >> FMPI2C_CR2_NBYTES_Pos); } @@ -1996,7 +1996,7 @@ __STATIC_INLINE void LL_FMPI2C_DisableAuto10BitRead(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledAuto10BitRead(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledAuto10BitRead(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR2, FMPI2C_CR2_HEAD10R) != (FMPI2C_CR2_HEAD10R)) ? 1UL : 0UL); } @@ -2024,7 +2024,7 @@ __STATIC_INLINE void LL_FMPI2C_SetTransferRequest(FMPI2C_TypeDef *FMPI2Cx, uint3 * @arg @ref LL_FMPI2C_REQUEST_WRITE * @arg @ref LL_FMPI2C_REQUEST_READ */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetTransferRequest(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetTransferRequest(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->CR2, FMPI2C_CR2_RD_WRN)); } @@ -2048,7 +2048,7 @@ __STATIC_INLINE void LL_FMPI2C_SetSlaveAddr(FMPI2C_TypeDef *FMPI2Cx, uint32_t Sl * @param FMPI2Cx FMPI2C Instance. * @retval Value between Min_Data=0x0 and Max_Data=0x3F */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetSlaveAddr(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetSlaveAddr(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->CR2, FMPI2C_CR2_SADD)); } @@ -2092,13 +2092,20 @@ __STATIC_INLINE uint32_t LL_FMPI2C_GetSlaveAddr(FMPI2C_TypeDef *FMPI2Cx) * @retval None */ __STATIC_INLINE void LL_FMPI2C_HandleTransfer(FMPI2C_TypeDef *FMPI2Cx, uint32_t SlaveAddr, uint32_t SlaveAddrSize, - uint32_t TransferSize, uint32_t EndMode, uint32_t Request) + uint32_t TransferSize, uint32_t EndMode, uint32_t Request) { + /* Declaration of tmp to prevent undefined behavior of volatile usage */ + uint32_t tmp = ((uint32_t)(((uint32_t)SlaveAddr & FMPI2C_CR2_SADD) | \ + ((uint32_t)SlaveAddrSize & FMPI2C_CR2_ADD10) | \ + (((uint32_t)TransferSize << FMPI2C_CR2_NBYTES_Pos) & FMPI2C_CR2_NBYTES) | \ + (uint32_t)EndMode | (uint32_t)Request) & (~0x80000000U)); + + /* update CR2 register */ MODIFY_REG(FMPI2Cx->CR2, FMPI2C_CR2_SADD | FMPI2C_CR2_ADD10 | (FMPI2C_CR2_RD_WRN & (uint32_t)(Request >> (31U - FMPI2C_CR2_RD_WRN_Pos))) | FMPI2C_CR2_START | FMPI2C_CR2_STOP | FMPI2C_CR2_RELOAD | FMPI2C_CR2_NBYTES | FMPI2C_CR2_AUTOEND | FMPI2C_CR2_HEAD10R, - SlaveAddr | SlaveAddrSize | (TransferSize << FMPI2C_CR2_NBYTES_Pos) | EndMode | Request); + tmp); } /** @@ -2111,7 +2118,7 @@ __STATIC_INLINE void LL_FMPI2C_HandleTransfer(FMPI2C_TypeDef *FMPI2Cx, uint32_t * @arg @ref LL_FMPI2C_DIRECTION_WRITE * @arg @ref LL_FMPI2C_DIRECTION_READ */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetTransferDirection(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetTransferDirection(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_DIR)); } @@ -2122,7 +2129,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_GetTransferDirection(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval Value between Min_Data=0x00 and Max_Data=0x3F */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetAddressMatchCode(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetAddressMatchCode(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->ISR, FMPI2C_ISR_ADDCODE) >> FMPI2C_ISR_ADDCODE_Pos << 1); } @@ -2152,7 +2159,7 @@ __STATIC_INLINE void LL_FMPI2C_EnableSMBusPECCompare(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledSMBusPECCompare(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledSMBusPECCompare(const FMPI2C_TypeDef *FMPI2Cx) { return ((READ_BIT(FMPI2Cx->CR2, FMPI2C_CR2_PECBYTE) == (FMPI2C_CR2_PECBYTE)) ? 1UL : 0UL); } @@ -2165,7 +2172,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_IsEnabledSMBusPECCompare(FMPI2C_TypeDef *FMPI * @param FMPI2Cx FMPI2C Instance. * @retval Value between Min_Data=0x00 and Max_Data=0xFF */ -__STATIC_INLINE uint32_t LL_FMPI2C_GetSMBusPEC(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint32_t LL_FMPI2C_GetSMBusPEC(const FMPI2C_TypeDef *FMPI2Cx) { return (uint32_t)(READ_BIT(FMPI2Cx->PECR, FMPI2C_PECR_PEC)); } @@ -2176,7 +2183,7 @@ __STATIC_INLINE uint32_t LL_FMPI2C_GetSMBusPEC(FMPI2C_TypeDef *FMPI2Cx) * @param FMPI2Cx FMPI2C Instance. * @retval Value between Min_Data=0x00 and Max_Data=0xFF */ -__STATIC_INLINE uint8_t LL_FMPI2C_ReceiveData8(FMPI2C_TypeDef *FMPI2Cx) +__STATIC_INLINE uint8_t LL_FMPI2C_ReceiveData8(const FMPI2C_TypeDef *FMPI2Cx) { return (uint8_t)(READ_BIT(FMPI2Cx->RXDR, FMPI2C_RXDR_RXDATA)); } @@ -2202,8 +2209,8 @@ __STATIC_INLINE void LL_FMPI2C_TransmitData8(FMPI2C_TypeDef *FMPI2Cx, uint8_t Da * @{ */ -ErrorStatus LL_FMPI2C_Init(FMPI2C_TypeDef *FMPI2Cx, LL_FMPI2C_InitTypeDef *FMPI2C_InitStruct); -ErrorStatus LL_FMPI2C_DeInit(FMPI2C_TypeDef *FMPI2Cx); +ErrorStatus LL_FMPI2C_Init(FMPI2C_TypeDef *FMPI2Cx, const LL_FMPI2C_InitTypeDef *FMPI2C_InitStruct); +ErrorStatus LL_FMPI2C_DeInit(const FMPI2C_TypeDef *FMPI2Cx); void LL_FMPI2C_StructInit(LL_FMPI2C_InitTypeDef *FMPI2C_InitStruct); diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_fsmc.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_fsmc.h index 8b0ceb75..5fb1c4f2 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_fsmc.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_fsmc.h @@ -304,7 +304,7 @@ typedef struct delay between ALE low and RE low. This parameter can be a number between Min_Data = 0 and Max_Data = 255 */ } FSMC_NAND_InitTypeDef; -#endif +#endif /* FSMC_Bank2_3 */ #if defined(FSMC_Bank2_3) || defined(FSMC_Bank4) /** diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_i2c.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_i2c.h index babba6bf..92d4a74a 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_i2c.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_i2c.h @@ -263,7 +263,7 @@ typedef struct * @} */ -/** @defgroup I2C_LL_EM_Exported_Macros_Helper Exported_Macros_Helper +/** @defgroup I2C_LL_EM_Exported_Macros_Helper Exported Macros Helper * @{ */ @@ -869,7 +869,7 @@ __STATIC_INLINE void LL_I2C_ConfigSpeed(I2C_TypeDef *I2Cx, uint32_t PeriphClock, /** * @brief Configure peripheral mode. - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @rmtoll CR1 SMBUS LL_I2C_SetMode\n * CR1 SMBTYPE LL_I2C_SetMode\n @@ -889,7 +889,7 @@ __STATIC_INLINE void LL_I2C_SetMode(I2C_TypeDef *I2Cx, uint32_t PeripheralMode) /** * @brief Get peripheral mode. - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @rmtoll CR1 SMBUS LL_I2C_GetMode\n * CR1 SMBTYPE LL_I2C_GetMode\n @@ -908,7 +908,7 @@ __STATIC_INLINE uint32_t LL_I2C_GetMode(I2C_TypeDef *I2Cx) /** * @brief Enable SMBus alert (Host or Device mode) - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @note SMBus Device mode: * - SMBus Alert pin is drived low and @@ -926,7 +926,7 @@ __STATIC_INLINE void LL_I2C_EnableSMBusAlert(I2C_TypeDef *I2Cx) /** * @brief Disable SMBus alert (Host or Device mode) - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @note SMBus Device mode: * - SMBus Alert pin is not drived (can be used as a standard GPIO) and @@ -944,7 +944,7 @@ __STATIC_INLINE void LL_I2C_DisableSMBusAlert(I2C_TypeDef *I2Cx) /** * @brief Check if SMBus alert (Host or Device mode) is enabled or disabled. - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @rmtoll CR1 ALERT LL_I2C_IsEnabledSMBusAlert * @param I2Cx I2C Instance. @@ -957,7 +957,7 @@ __STATIC_INLINE uint32_t LL_I2C_IsEnabledSMBusAlert(I2C_TypeDef *I2Cx) /** * @brief Enable SMBus Packet Error Calculation (PEC). - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @rmtoll CR1 ENPEC LL_I2C_EnableSMBusPEC * @param I2Cx I2C Instance. @@ -970,7 +970,7 @@ __STATIC_INLINE void LL_I2C_EnableSMBusPEC(I2C_TypeDef *I2Cx) /** * @brief Disable SMBus Packet Error Calculation (PEC). - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @rmtoll CR1 ENPEC LL_I2C_DisableSMBusPEC * @param I2Cx I2C Instance. @@ -983,7 +983,7 @@ __STATIC_INLINE void LL_I2C_DisableSMBusPEC(I2C_TypeDef *I2Cx) /** * @brief Check if SMBus Packet Error Calculation (PEC) is enabled or disabled. - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @rmtoll CR1 ENPEC LL_I2C_IsEnabledSMBusPEC * @param I2Cx I2C Instance. @@ -1166,7 +1166,7 @@ __STATIC_INLINE uint32_t LL_I2C_IsEnabledIT_BUF(I2C_TypeDef *I2Cx) /** * @brief Enable Error interrupts. - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @note Any of these errors will generate interrupt : * Bus Error detection (BERR) @@ -1187,7 +1187,7 @@ __STATIC_INLINE void LL_I2C_EnableIT_ERR(I2C_TypeDef *I2Cx) /** * @brief Disable Error interrupts. - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @note Any of these errors will generate interrupt : * Bus Error detection (BERR) @@ -1370,7 +1370,7 @@ __STATIC_INLINE uint32_t LL_I2C_IsActiveFlag_OVR(I2C_TypeDef *I2Cx) /** * @brief Indicate the status of SMBus PEC error flag in reception. - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @rmtoll SR1 PECERR LL_I2C_IsActiveSMBusFlag_PECERR * @param I2Cx I2C Instance. @@ -1383,7 +1383,7 @@ __STATIC_INLINE uint32_t LL_I2C_IsActiveSMBusFlag_PECERR(I2C_TypeDef *I2Cx) /** * @brief Indicate the status of SMBus Timeout detection flag. - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @rmtoll SR1 TIMEOUT LL_I2C_IsActiveSMBusFlag_TIMEOUT * @param I2Cx I2C Instance. @@ -1396,7 +1396,7 @@ __STATIC_INLINE uint32_t LL_I2C_IsActiveSMBusFlag_TIMEOUT(I2C_TypeDef *I2Cx) /** * @brief Indicate the status of SMBus alert flag. - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @rmtoll SR1 SMBALERT LL_I2C_IsActiveSMBusFlag_ALERT * @param I2Cx I2C Instance. @@ -1435,7 +1435,7 @@ __STATIC_INLINE uint32_t LL_I2C_IsActiveFlag_DUAL(I2C_TypeDef *I2Cx) /** * @brief Indicate the status of SMBus Host address reception (Slave mode). - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @note RESET: No SMBus Host address * SET: SMBus Host address received. @@ -1451,7 +1451,7 @@ __STATIC_INLINE uint32_t LL_I2C_IsActiveSMBusFlag_SMBHOST(I2C_TypeDef *I2Cx) /** * @brief Indicate the status of SMBus Device default address reception (Slave mode). - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @note RESET: No SMBus Device default address * SET: SMBus Device default address received. @@ -1583,7 +1583,7 @@ __STATIC_INLINE void LL_I2C_ClearSMBusFlag_PECERR(I2C_TypeDef *I2Cx) /** * @brief Clear SMBus Timeout detection flag. - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @rmtoll SR1 TIMEOUT LL_I2C_ClearSMBusFlag_TIMEOUT * @param I2Cx I2C Instance. @@ -1596,7 +1596,7 @@ __STATIC_INLINE void LL_I2C_ClearSMBusFlag_TIMEOUT(I2C_TypeDef *I2Cx) /** * @brief Clear SMBus Alert flag. - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @rmtoll SR1 SMBALERT LL_I2C_ClearSMBusFlag_ALERT * @param I2Cx I2C Instance. @@ -1774,7 +1774,7 @@ __STATIC_INLINE uint32_t LL_I2C_IsEnabledLastDMA(I2C_TypeDef *I2Cx) /** * @brief Enable transfer or internal comparison of the SMBus Packet Error byte (transmission or reception mode). - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @note This feature is cleared by hardware when the PEC byte is transferred or compared, * or by a START or STOP condition, it is also cleared by software. @@ -1789,7 +1789,7 @@ __STATIC_INLINE void LL_I2C_EnableSMBusPECCompare(I2C_TypeDef *I2Cx) /** * @brief Disable transfer or internal comparison of the SMBus Packet Error byte (transmission or reception mode). - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @rmtoll CR1 PEC LL_I2C_DisableSMBusPECCompare * @param I2Cx I2C Instance. @@ -1802,7 +1802,7 @@ __STATIC_INLINE void LL_I2C_DisableSMBusPECCompare(I2C_TypeDef *I2Cx) /** * @brief Check if the SMBus Packet Error byte transfer or internal comparison is requested or not. - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @rmtoll CR1 PEC LL_I2C_IsEnabledSMBusPECCompare * @param I2Cx I2C Instance. @@ -1815,7 +1815,7 @@ __STATIC_INLINE uint32_t LL_I2C_IsEnabledSMBusPECCompare(I2C_TypeDef *I2Cx) /** * @brief Get the SMBus Packet Error byte calculated. - * @note Macro @ref IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not + * @note Macro IS_SMBUS_ALL_INSTANCE(I2Cx) can be used to check whether or not * SMBus feature is supported by the I2Cx Instance. * @rmtoll SR2 PEC LL_I2C_GetSMBusPEC * @param I2Cx I2C Instance. diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_lptim.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_lptim.h index 9495e019..273d89dc 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_lptim.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_lptim.h @@ -315,14 +315,27 @@ typedef struct * @{ */ +/** Legacy definitions for compatibility purpose +@cond 0 + */ +#define LL_LPTIM_ClearFLAG_CMPM LL_LPTIM_ClearFlag_CMPM +#define LL_LPTIM_ClearFLAG_CC1 LL_LPTIM_ClearFlag_CC1 +#define LL_LPTIM_ClearFLAG_CC2 LL_LPTIM_ClearFlag_CC2 +#define LL_LPTIM_ClearFLAG_CC1O LL_LPTIM_ClearFlag_CC1O +#define LL_LPTIM_ClearFLAG_CC2O LL_LPTIM_ClearFlag_CC2O +#define LL_LPTIM_ClearFLAG_ARRM LL_LPTIM_ClearFlag_ARRM +/** +@endcond + */ + #if defined(USE_FULL_LL_DRIVER) /** @defgroup LPTIM_LL_EF_Init Initialisation and deinitialisation functions * @{ */ -ErrorStatus LL_LPTIM_DeInit(LPTIM_TypeDef *LPTIMx); +ErrorStatus LL_LPTIM_DeInit(const LPTIM_TypeDef *LPTIMx); void LL_LPTIM_StructInit(LL_LPTIM_InitTypeDef *LPTIM_InitStruct); -ErrorStatus LL_LPTIM_Init(LPTIM_TypeDef *LPTIMx, LL_LPTIM_InitTypeDef *LPTIM_InitStruct); +ErrorStatus LL_LPTIM_Init(LPTIM_TypeDef *LPTIMx, const LL_LPTIM_InitTypeDef *LPTIM_InitStruct); void LL_LPTIM_Disable(LPTIM_TypeDef *LPTIMx); /** * @} @@ -352,7 +365,7 @@ __STATIC_INLINE void LL_LPTIM_Enable(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsEnabled(LPTIM_TypeDef *const LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsEnabled(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->CR, LPTIM_CR_ENABLE) == LPTIM_CR_ENABLE) ? 1UL : 0UL)); } @@ -398,7 +411,7 @@ __STATIC_INLINE void LL_LPTIM_SetUpdateMode(LPTIM_TypeDef *LPTIMx, uint32_t Upda * @arg @ref LL_LPTIM_UPDATE_MODE_IMMEDIATE * @arg @ref LL_LPTIM_UPDATE_MODE_ENDOFPERIOD */ -__STATIC_INLINE uint32_t LL_LPTIM_GetUpdateMode(LPTIM_TypeDef *const LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_GetUpdateMode(const LPTIM_TypeDef *LPTIMx) { return (uint32_t)(READ_BIT(LPTIMx->CFGR, LPTIM_CFGR_PRELOAD)); } @@ -427,7 +440,7 @@ __STATIC_INLINE void LL_LPTIM_SetAutoReload(LPTIM_TypeDef *LPTIMx, uint32_t Auto * @param LPTIMx Low-Power Timer instance * @retval AutoReload Value between Min_Data=0x0001 and Max_Data=0xFFFF */ -__STATIC_INLINE uint32_t LL_LPTIM_GetAutoReload(LPTIM_TypeDef *const LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_GetAutoReload(const LPTIM_TypeDef *LPTIMx) { return (uint32_t)(READ_BIT(LPTIMx->ARR, LPTIM_ARR_ARR)); } @@ -454,7 +467,7 @@ __STATIC_INLINE void LL_LPTIM_SetCompare(LPTIM_TypeDef *LPTIMx, uint32_t Compare * @param LPTIMx Low-Power Timer instance * @retval CompareValue Value between Min_Data=0x00 and Max_Data=0xFFFF */ -__STATIC_INLINE uint32_t LL_LPTIM_GetCompare(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_GetCompare(const LPTIM_TypeDef *LPTIMx) { return (uint32_t)(READ_BIT(LPTIMx->CMP, LPTIM_CMP_CMP)); } @@ -469,7 +482,7 @@ __STATIC_INLINE uint32_t LL_LPTIM_GetCompare(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval Counter value */ -__STATIC_INLINE uint32_t LL_LPTIM_GetCounter(LPTIM_TypeDef *const LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_GetCounter(const LPTIM_TypeDef *LPTIMx) { return (uint32_t)(READ_BIT(LPTIMx->CNT, LPTIM_CNT_CNT)); } @@ -497,7 +510,7 @@ __STATIC_INLINE void LL_LPTIM_SetCounterMode(LPTIM_TypeDef *LPTIMx, uint32_t Cou * @arg @ref LL_LPTIM_COUNTER_MODE_INTERNAL * @arg @ref LL_LPTIM_COUNTER_MODE_EXTERNAL */ -__STATIC_INLINE uint32_t LL_LPTIM_GetCounterMode(LPTIM_TypeDef *const LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_GetCounterMode(const LPTIM_TypeDef *LPTIMx) { return (uint32_t)(READ_BIT(LPTIMx->CFGR, LPTIM_CFGR_COUNTMODE)); } @@ -546,7 +559,7 @@ __STATIC_INLINE void LL_LPTIM_SetWaveform(LPTIM_TypeDef *LPTIMx, uint32_t Wavefo * @arg @ref LL_LPTIM_OUTPUT_WAVEFORM_PWM * @arg @ref LL_LPTIM_OUTPUT_WAVEFORM_SETONCE */ -__STATIC_INLINE uint32_t LL_LPTIM_GetWaveform(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_GetWaveform(const LPTIM_TypeDef *LPTIMx) { return (uint32_t)(READ_BIT(LPTIMx->CFGR, LPTIM_CFGR_WAVE)); } @@ -573,7 +586,7 @@ __STATIC_INLINE void LL_LPTIM_SetPolarity(LPTIM_TypeDef *LPTIMx, uint32_t Polari * @arg @ref LL_LPTIM_OUTPUT_POLARITY_REGULAR * @arg @ref LL_LPTIM_OUTPUT_POLARITY_INVERSE */ -__STATIC_INLINE uint32_t LL_LPTIM_GetPolarity(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_GetPolarity(const LPTIM_TypeDef *LPTIMx) { return (uint32_t)(READ_BIT(LPTIMx->CFGR, LPTIM_CFGR_WAVPOL)); } @@ -617,7 +630,7 @@ __STATIC_INLINE void LL_LPTIM_SetPrescaler(LPTIM_TypeDef *LPTIMx, uint32_t Presc * @arg @ref LL_LPTIM_PRESCALER_DIV64 * @arg @ref LL_LPTIM_PRESCALER_DIV128 */ -__STATIC_INLINE uint32_t LL_LPTIM_GetPrescaler(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_GetPrescaler(const LPTIM_TypeDef *LPTIMx) { return (uint32_t)(READ_BIT(LPTIMx->CFGR, LPTIM_CFGR_PRESC)); } @@ -683,7 +696,7 @@ __STATIC_INLINE void LL_LPTIM_DisableTimeout(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledTimeout(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledTimeout(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->CFGR, LPTIM_CFGR_TIMOUT) == LPTIM_CFGR_TIMOUT) ? 1UL : 0UL)); } @@ -744,7 +757,7 @@ __STATIC_INLINE void LL_LPTIM_ConfigTrigger(LPTIM_TypeDef *LPTIMx, uint32_t Sour * @arg @ref LL_LPTIM_TRIG_SOURCE_TIM1_TRGO * @arg @ref LL_LPTIM_TRIG_SOURCE_TIM5_TRGO */ -__STATIC_INLINE uint32_t LL_LPTIM_GetTriggerSource(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_GetTriggerSource(const LPTIM_TypeDef *LPTIMx) { return (uint32_t)(READ_BIT(LPTIMx->CFGR, LPTIM_CFGR_TRIGSEL)); } @@ -759,7 +772,7 @@ __STATIC_INLINE uint32_t LL_LPTIM_GetTriggerSource(LPTIM_TypeDef *LPTIMx) * @arg @ref LL_LPTIM_TRIG_FILTER_4 * @arg @ref LL_LPTIM_TRIG_FILTER_8 */ -__STATIC_INLINE uint32_t LL_LPTIM_GetTriggerFilter(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_GetTriggerFilter(const LPTIM_TypeDef *LPTIMx) { return (uint32_t)(READ_BIT(LPTIMx->CFGR, LPTIM_CFGR_TRGFLT)); } @@ -773,7 +786,7 @@ __STATIC_INLINE uint32_t LL_LPTIM_GetTriggerFilter(LPTIM_TypeDef *LPTIMx) * @arg @ref LL_LPTIM_TRIG_POLARITY_FALLING * @arg @ref LL_LPTIM_TRIG_POLARITY_RISING_FALLING */ -__STATIC_INLINE uint32_t LL_LPTIM_GetTriggerPolarity(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_GetTriggerPolarity(const LPTIM_TypeDef *LPTIMx) { return (uint32_t)(READ_BIT(LPTIMx->CFGR, LPTIM_CFGR_TRIGEN)); } @@ -809,7 +822,7 @@ __STATIC_INLINE void LL_LPTIM_SetClockSource(LPTIM_TypeDef *LPTIMx, uint32_t Clo * @arg @ref LL_LPTIM_CLK_SOURCE_INTERNAL * @arg @ref LL_LPTIM_CLK_SOURCE_EXTERNAL */ -__STATIC_INLINE uint32_t LL_LPTIM_GetClockSource(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_GetClockSource(const LPTIM_TypeDef *LPTIMx) { return (uint32_t)(READ_BIT(LPTIMx->CFGR, LPTIM_CFGR_CKSEL)); } @@ -851,7 +864,7 @@ __STATIC_INLINE void LL_LPTIM_ConfigClock(LPTIM_TypeDef *LPTIMx, uint32_t ClockF * @arg @ref LL_LPTIM_CLK_POLARITY_FALLING * @arg @ref LL_LPTIM_CLK_POLARITY_RISING_FALLING */ -__STATIC_INLINE uint32_t LL_LPTIM_GetClockPolarity(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_GetClockPolarity(const LPTIM_TypeDef *LPTIMx) { return (uint32_t)(READ_BIT(LPTIMx->CFGR, LPTIM_CFGR_CKPOL)); } @@ -866,7 +879,7 @@ __STATIC_INLINE uint32_t LL_LPTIM_GetClockPolarity(LPTIM_TypeDef *LPTIMx) * @arg @ref LL_LPTIM_CLK_FILTER_4 * @arg @ref LL_LPTIM_CLK_FILTER_8 */ -__STATIC_INLINE uint32_t LL_LPTIM_GetClockFilter(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_GetClockFilter(const LPTIM_TypeDef *LPTIMx) { return (uint32_t)(READ_BIT(LPTIMx->CFGR, LPTIM_CFGR_CKFLT)); } @@ -904,7 +917,7 @@ __STATIC_INLINE void LL_LPTIM_SetEncoderMode(LPTIM_TypeDef *LPTIMx, uint32_t Enc * @arg @ref LL_LPTIM_ENCODER_MODE_FALLING * @arg @ref LL_LPTIM_ENCODER_MODE_RISING_FALLING */ -__STATIC_INLINE uint32_t LL_LPTIM_GetEncoderMode(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_GetEncoderMode(const LPTIM_TypeDef *LPTIMx) { return (uint32_t)(READ_BIT(LPTIMx->CFGR, LPTIM_CFGR_CKPOL)); } @@ -943,7 +956,7 @@ __STATIC_INLINE void LL_LPTIM_DisableEncoderMode(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledEncoderMode(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledEncoderMode(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->CFGR, LPTIM_CFGR_ENC) == LPTIM_CFGR_ENC) ? 1UL : 0UL)); } @@ -956,13 +969,14 @@ __STATIC_INLINE uint32_t LL_LPTIM_IsEnabledEncoderMode(LPTIM_TypeDef *LPTIMx) * @{ */ + /** * @brief Clear the compare match flag (CMPMCF) - * @rmtoll ICR CMPMCF LL_LPTIM_ClearFLAG_CMPM + * @rmtoll ICR CMPMCF LL_LPTIM_ClearFlag_CMPM * @param LPTIMx Low-Power Timer instance * @retval None */ -__STATIC_INLINE void LL_LPTIM_ClearFLAG_CMPM(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE void LL_LPTIM_ClearFlag_CMPM(LPTIM_TypeDef *LPTIMx) { SET_BIT(LPTIMx->ICR, LPTIM_ICR_CMPMCF); } @@ -973,18 +987,18 @@ __STATIC_INLINE void LL_LPTIM_ClearFLAG_CMPM(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsActiveFlag_CMPM(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsActiveFlag_CMPM(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->ISR, LPTIM_ISR_CMPM) == LPTIM_ISR_CMPM) ? 1UL : 0UL)); } /** * @brief Clear the autoreload match flag (ARRMCF) - * @rmtoll ICR ARRMCF LL_LPTIM_ClearFLAG_ARRM + * @rmtoll ICR ARRMCF LL_LPTIM_ClearFlag_ARRM * @param LPTIMx Low-Power Timer instance * @retval None */ -__STATIC_INLINE void LL_LPTIM_ClearFLAG_ARRM(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE void LL_LPTIM_ClearFlag_ARRM(LPTIM_TypeDef *LPTIMx) { SET_BIT(LPTIMx->ICR, LPTIM_ICR_ARRMCF); } @@ -995,7 +1009,7 @@ __STATIC_INLINE void LL_LPTIM_ClearFLAG_ARRM(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsActiveFlag_ARRM(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsActiveFlag_ARRM(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->ISR, LPTIM_ISR_ARRM) == LPTIM_ISR_ARRM) ? 1UL : 0UL)); } @@ -1017,7 +1031,7 @@ __STATIC_INLINE void LL_LPTIM_ClearFlag_EXTTRIG(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsActiveFlag_EXTTRIG(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsActiveFlag_EXTTRIG(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->ISR, LPTIM_ISR_EXTTRIG) == LPTIM_ISR_EXTTRIG) ? 1UL : 0UL)); } @@ -1040,7 +1054,7 @@ __STATIC_INLINE void LL_LPTIM_ClearFlag_CMPOK(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsActiveFlag_CMPOK(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsActiveFlag_CMPOK(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->ISR, LPTIM_ISR_CMPOK) == LPTIM_ISR_CMPOK) ? 1UL : 0UL)); } @@ -1063,7 +1077,7 @@ __STATIC_INLINE void LL_LPTIM_ClearFlag_ARROK(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsActiveFlag_ARROK(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsActiveFlag_ARROK(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->ISR, LPTIM_ISR_ARROK) == LPTIM_ISR_ARROK) ? 1UL : 0UL)); } @@ -1086,7 +1100,7 @@ __STATIC_INLINE void LL_LPTIM_ClearFlag_UP(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsActiveFlag_UP(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsActiveFlag_UP(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->ISR, LPTIM_ISR_UP) == LPTIM_ISR_UP) ? 1UL : 0UL)); } @@ -1109,7 +1123,7 @@ __STATIC_INLINE void LL_LPTIM_ClearFlag_DOWN(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsActiveFlag_DOWN(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsActiveFlag_DOWN(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->ISR, LPTIM_ISR_DOWN) == LPTIM_ISR_DOWN) ? 1UL : 0UL)); } @@ -1150,7 +1164,7 @@ __STATIC_INLINE void LL_LPTIM_DisableIT_CMPM(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledIT_CMPM(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledIT_CMPM(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->IER, LPTIM_IER_CMPMIE) == LPTIM_IER_CMPMIE) ? 1UL : 0UL)); } @@ -1183,7 +1197,7 @@ __STATIC_INLINE void LL_LPTIM_DisableIT_ARRM(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledIT_ARRM(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledIT_ARRM(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->IER, LPTIM_IER_ARRMIE) == LPTIM_IER_ARRMIE) ? 1UL : 0UL)); } @@ -1216,7 +1230,7 @@ __STATIC_INLINE void LL_LPTIM_DisableIT_EXTTRIG(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledIT_EXTTRIG(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledIT_EXTTRIG(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->IER, LPTIM_IER_EXTTRIGIE) == LPTIM_IER_EXTTRIGIE) ? 1UL : 0UL)); } @@ -1249,7 +1263,7 @@ __STATIC_INLINE void LL_LPTIM_DisableIT_CMPOK(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledIT_CMPOK(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledIT_CMPOK(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->IER, LPTIM_IER_CMPOKIE) == LPTIM_IER_CMPOKIE) ? 1UL : 0UL)); } @@ -1282,7 +1296,7 @@ __STATIC_INLINE void LL_LPTIM_DisableIT_ARROK(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit(1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledIT_ARROK(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledIT_ARROK(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->IER, LPTIM_IER_ARROKIE) == LPTIM_IER_ARROKIE) ? 1UL : 0UL)); } @@ -1315,7 +1329,7 @@ __STATIC_INLINE void LL_LPTIM_DisableIT_UP(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit(1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledIT_UP(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledIT_UP(const LPTIM_TypeDef *LPTIMx) { return (((READ_BIT(LPTIMx->IER, LPTIM_IER_UPIE) == LPTIM_IER_UPIE) ? 1UL : 0UL)); } @@ -1348,7 +1362,7 @@ __STATIC_INLINE void LL_LPTIM_DisableIT_DOWN(LPTIM_TypeDef *LPTIMx) * @param LPTIMx Low-Power Timer instance * @retval State of bit(1 or 0). */ -__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledIT_DOWN(LPTIM_TypeDef *LPTIMx) +__STATIC_INLINE uint32_t LL_LPTIM_IsEnabledIT_DOWN(const LPTIM_TypeDef *LPTIMx) { return ((READ_BIT(LPTIMx->IER, LPTIM_IER_DOWNIE) == LPTIM_IER_DOWNIE) ? 1UL : 0UL); } diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_rng.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_rng.h index 151cb4a7..21c7330c 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_rng.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_rng.h @@ -38,6 +38,7 @@ extern "C" { */ /* Private types -------------------------------------------------------------*/ +/* Private defines -----------------------------------------------------------*/ /* Private variables ---------------------------------------------------------*/ /* Private constants ---------------------------------------------------------*/ /* Private macros ------------------------------------------------------------*/ @@ -145,7 +146,7 @@ __STATIC_INLINE void LL_RNG_Disable(RNG_TypeDef *RNGx) * @param RNGx RNG Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_RNG_IsEnabled(RNG_TypeDef *RNGx) +__STATIC_INLINE uint32_t LL_RNG_IsEnabled(const RNG_TypeDef *RNGx) { return ((READ_BIT(RNGx->CR, RNG_CR_RNGEN) == (RNG_CR_RNGEN)) ? 1UL : 0UL); } @@ -164,7 +165,7 @@ __STATIC_INLINE uint32_t LL_RNG_IsEnabled(RNG_TypeDef *RNGx) * @param RNGx RNG Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_RNG_IsActiveFlag_DRDY(RNG_TypeDef *RNGx) +__STATIC_INLINE uint32_t LL_RNG_IsActiveFlag_DRDY(const RNG_TypeDef *RNGx) { return ((READ_BIT(RNGx->SR, RNG_SR_DRDY) == (RNG_SR_DRDY)) ? 1UL : 0UL); } @@ -175,7 +176,7 @@ __STATIC_INLINE uint32_t LL_RNG_IsActiveFlag_DRDY(RNG_TypeDef *RNGx) * @param RNGx RNG Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_RNG_IsActiveFlag_CECS(RNG_TypeDef *RNGx) +__STATIC_INLINE uint32_t LL_RNG_IsActiveFlag_CECS(const RNG_TypeDef *RNGx) { return ((READ_BIT(RNGx->SR, RNG_SR_CECS) == (RNG_SR_CECS)) ? 1UL : 0UL); } @@ -186,7 +187,7 @@ __STATIC_INLINE uint32_t LL_RNG_IsActiveFlag_CECS(RNG_TypeDef *RNGx) * @param RNGx RNG Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_RNG_IsActiveFlag_SECS(RNG_TypeDef *RNGx) +__STATIC_INLINE uint32_t LL_RNG_IsActiveFlag_SECS(const RNG_TypeDef *RNGx) { return ((READ_BIT(RNGx->SR, RNG_SR_SECS) == (RNG_SR_SECS)) ? 1UL : 0UL); } @@ -197,7 +198,7 @@ __STATIC_INLINE uint32_t LL_RNG_IsActiveFlag_SECS(RNG_TypeDef *RNGx) * @param RNGx RNG Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_RNG_IsActiveFlag_CEIS(RNG_TypeDef *RNGx) +__STATIC_INLINE uint32_t LL_RNG_IsActiveFlag_CEIS(const RNG_TypeDef *RNGx) { return ((READ_BIT(RNGx->SR, RNG_SR_CEIS) == (RNG_SR_CEIS)) ? 1UL : 0UL); } @@ -208,7 +209,7 @@ __STATIC_INLINE uint32_t LL_RNG_IsActiveFlag_CEIS(RNG_TypeDef *RNGx) * @param RNGx RNG Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_RNG_IsActiveFlag_SEIS(RNG_TypeDef *RNGx) +__STATIC_INLINE uint32_t LL_RNG_IsActiveFlag_SEIS(const RNG_TypeDef *RNGx) { return ((READ_BIT(RNGx->SR, RNG_SR_SEIS) == (RNG_SR_SEIS)) ? 1UL : 0UL); } @@ -274,7 +275,7 @@ __STATIC_INLINE void LL_RNG_DisableIT(RNG_TypeDef *RNGx) * @param RNGx RNG Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_RNG_IsEnabledIT(RNG_TypeDef *RNGx) +__STATIC_INLINE uint32_t LL_RNG_IsEnabledIT(const RNG_TypeDef *RNGx) { return ((READ_BIT(RNGx->CR, RNG_CR_IE) == (RNG_CR_IE)) ? 1UL : 0UL); } @@ -293,7 +294,7 @@ __STATIC_INLINE uint32_t LL_RNG_IsEnabledIT(RNG_TypeDef *RNGx) * @param RNGx RNG Instance * @retval Generated 32-bit random value */ -__STATIC_INLINE uint32_t LL_RNG_ReadRandData32(RNG_TypeDef *RNGx) +__STATIC_INLINE uint32_t LL_RNG_ReadRandData32(const RNG_TypeDef *RNGx) { return (uint32_t)(READ_REG(RNGx->DR)); } @@ -306,7 +307,7 @@ __STATIC_INLINE uint32_t LL_RNG_ReadRandData32(RNG_TypeDef *RNGx) /** @defgroup RNG_LL_EF_Init Initialization and de-initialization functions * @{ */ -ErrorStatus LL_RNG_DeInit(RNG_TypeDef *RNGx); +ErrorStatus LL_RNG_DeInit(const RNG_TypeDef *RNGx); /** * @} diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_rtc.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_rtc.h index 180cc15f..74a0aee0 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_rtc.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_rtc.h @@ -6,7 +6,7 @@ ****************************************************************************** * @attention * - * Copyright (c) 2017 STMicroelectronics. + * Copyright (c) 2016 STMicroelectronics. * All rights reserved. * * This software is licensed under terms that can be found in the LICENSE file @@ -1030,7 +1030,7 @@ __STATIC_INLINE void LL_RTC_TIME_SetFormat(RTC_TypeDef *RTCx, uint32_t TimeForma /** * @brief Get time format (AM or PM notation) - * @note if shadow mode is disabled (BYPSHAD=0), need to check if RSF flag is set + * @note if RTC shadow registers are not bypassed (BYPSHAD=0), need to check if RSF flag is set * before reading this bit * @note Read either RTC_SSR or RTC_TR locks the values in the higher-order calendar * shadow registers until RTC_DR is read (LL_RTC_ReadReg(RTC, DR)). @@ -1064,7 +1064,7 @@ __STATIC_INLINE void LL_RTC_TIME_SetHour(RTC_TypeDef *RTCx, uint32_t Hours) /** * @brief Get Hours in BCD format - * @note if shadow mode is disabled (BYPSHAD=0), need to check if RSF flag is set + * @note if RTC shadow registers are not bypassed (BYPSHAD=0), need to check if RSF flag is set * before reading this bit * @note Read either RTC_SSR or RTC_TR locks the values in the higher-order calendar * shadow registers until RTC_DR is read (LL_RTC_ReadReg(RTC, DR)). @@ -1099,7 +1099,7 @@ __STATIC_INLINE void LL_RTC_TIME_SetMinute(RTC_TypeDef *RTCx, uint32_t Minutes) /** * @brief Get Minutes in BCD format - * @note if shadow mode is disabled (BYPSHAD=0), need to check if RSF flag is set + * @note if RTC shadow registers are not bypassed (BYPSHAD=0), need to check if RSF flag is set * before reading this bit * @note Read either RTC_SSR or RTC_TR locks the values in the higher-order calendar * shadow registers until RTC_DR is read (LL_RTC_ReadReg(RTC, DR)). @@ -1134,7 +1134,7 @@ __STATIC_INLINE void LL_RTC_TIME_SetSecond(RTC_TypeDef *RTCx, uint32_t Seconds) /** * @brief Get Seconds in BCD format - * @note if shadow mode is disabled (BYPSHAD=0), need to check if RSF flag is set + * @note if RTC shadow registers are not bypassed (BYPSHAD=0), need to check if RSF flag is set * before reading this bit * @note Read either RTC_SSR or RTC_TR locks the values in the higher-order calendar * shadow registers until RTC_DR is read (LL_RTC_ReadReg(RTC, DR)). @@ -1184,7 +1184,7 @@ __STATIC_INLINE void LL_RTC_TIME_Config(RTC_TypeDef *RTCx, uint32_t Format12_24, /** * @brief Get time (hour, minute and second) in BCD format - * @note if shadow mode is disabled (BYPSHAD=0), need to check if RSF flag is set + * @note if RTC shadow registers are not bypassed (BYPSHAD=0), need to check if RSF flag is set * before reading this bit * @note Read either RTC_SSR or RTC_TR locks the values in the higher-order calendar * shadow registers until RTC_DR is read (LL_RTC_ReadReg(RTC, DR)). @@ -1326,7 +1326,7 @@ __STATIC_INLINE void LL_RTC_DATE_SetYear(RTC_TypeDef *RTCx, uint32_t Year) /** * @brief Get Year in BCD format - * @note if shadow mode is disabled (BYPSHAD=0), need to check if RSF flag is set + * @note if RTC shadow registers are not bypassed (BYPSHAD=0), need to check if RSF flag is set * before reading this bit * @note helper macro __LL_RTC_CONVERT_BCD2BIN is available to convert Year from BCD to Binary format * @rmtoll DR YT LL_RTC_DATE_GetYear\n @@ -1360,7 +1360,7 @@ __STATIC_INLINE void LL_RTC_DATE_SetWeekDay(RTC_TypeDef *RTCx, uint32_t WeekDay) /** * @brief Get Week day - * @note if shadow mode is disabled (BYPSHAD=0), need to check if RSF flag is set + * @note if RTC shadow registers are not bypassed (BYPSHAD=0), need to check if RSF flag is set * before reading this bit * @rmtoll DR WDU LL_RTC_DATE_GetWeekDay * @param RTCx RTC Instance @@ -1407,7 +1407,7 @@ __STATIC_INLINE void LL_RTC_DATE_SetMonth(RTC_TypeDef *RTCx, uint32_t Month) /** * @brief Get Month in BCD format - * @note if shadow mode is disabled (BYPSHAD=0), need to check if RSF flag is set + * @note if RTC shadow registers are not bypassed (BYPSHAD=0), need to check if RSF flag is set * before reading this bit * @note helper macro __LL_RTC_CONVERT_BCD2BIN is available to convert Month from BCD to Binary format * @rmtoll DR MT LL_RTC_DATE_GetMonth\n @@ -1449,7 +1449,7 @@ __STATIC_INLINE void LL_RTC_DATE_SetDay(RTC_TypeDef *RTCx, uint32_t Day) /** * @brief Get Day in BCD format - * @note if shadow mode is disabled (BYPSHAD=0), need to check if RSF flag is set + * @note if RTC shadow registers are not bypassed (BYPSHAD=0), need to check if RSF flag is set * before reading this bit * @note helper macro __LL_RTC_CONVERT_BCD2BIN is available to convert Day from BCD to Binary format * @rmtoll DR DT LL_RTC_DATE_GetDay\n @@ -1511,7 +1511,7 @@ __STATIC_INLINE void LL_RTC_DATE_Config(RTC_TypeDef *RTCx, uint32_t WeekDay, uin /** * @brief Get date (WeekDay, Day, Month and Year) in BCD format - * @note if shadow mode is disabled (BYPSHAD=0), need to check if RSF flag is set + * @note if RTC shadow registers are not bypassed (BYPSHAD=0), need to check if RSF flag is set * before reading this bit * @note helper macros __LL_RTC_GET_WEEKDAY, __LL_RTC_GET_YEAR, __LL_RTC_GET_MONTH, * and __LL_RTC_GET_DAY are available to get independently each parameter. diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_sdmmc.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_sdmmc.h index 35c47e3c..8bd32f41 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_sdmmc.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_sdmmc.h @@ -300,10 +300,14 @@ typedef struct #define SDMMC_SINGLE_BUS_SUPPORT 0x00010000U #define SDMMC_CARD_LOCKED 0x02000000U -#ifndef SDMMC_DATATIMEOUT -#define SDMMC_DATATIMEOUT 0xFFFFFFFFU +#ifndef SDMMC_DATATIMEOUT /*Hardware Data Timeout (ms) */ +#define SDMMC_DATATIMEOUT ((uint32_t)0xFFFFFFFFU) #endif /* SDMMC_DATATIMEOUT */ +#ifndef SDMMC_SWDATATIMEOUT /*Software Data Timeout (ms) */ +#define SDMMC_SWDATATIMEOUT SDMMC_DATATIMEOUT +#endif /* SDMMC_SWDATATIMEOUT */ + #define SDMMC_0TO7BITS 0x000000FFU #define SDMMC_8TO15BITS 0x0000FF00U #define SDMMC_16TO23BITS 0x00FF0000U diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_tim.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_tim.h index 61148e4e..a11f5619 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_tim.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_tim.h @@ -562,10 +562,10 @@ typedef struct /** @defgroup TIM_LL_EC_COUNTERMODE Counter Mode * @{ */ -#define LL_TIM_COUNTERMODE_UP 0x00000000U /*!> 16U) >> TIM_CCMR1_IC1PSC_Pos))) -/** - * @} - */ - - /** * @} */ @@ -1143,7 +1135,7 @@ __STATIC_INLINE void LL_TIM_DisableCounter(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledCounter(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledCounter(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->CR1, TIM_CR1_CEN) == (TIM_CR1_CEN)) ? 1UL : 0UL); } @@ -1176,7 +1168,7 @@ __STATIC_INLINE void LL_TIM_DisableUpdateEvent(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval Inverted state of bit (0 or 1). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledUpdateEvent(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledUpdateEvent(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->CR1, TIM_CR1_UDIS) == (uint32_t)RESET) ? 1UL : 0UL); } @@ -1210,7 +1202,7 @@ __STATIC_INLINE void LL_TIM_SetUpdateSource(TIM_TypeDef *TIMx, uint32_t UpdateSo * @arg @ref LL_TIM_UPDATESOURCE_REGULAR * @arg @ref LL_TIM_UPDATESOURCE_COUNTER */ -__STATIC_INLINE uint32_t LL_TIM_GetUpdateSource(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_GetUpdateSource(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_BIT(TIMx->CR1, TIM_CR1_URS)); } @@ -1237,7 +1229,7 @@ __STATIC_INLINE void LL_TIM_SetOnePulseMode(TIM_TypeDef *TIMx, uint32_t OnePulse * @arg @ref LL_TIM_ONEPULSEMODE_SINGLE * @arg @ref LL_TIM_ONEPULSEMODE_REPETITIVE */ -__STATIC_INLINE uint32_t LL_TIM_GetOnePulseMode(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_GetOnePulseMode(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_BIT(TIMx->CR1, TIM_CR1_OPM)); } @@ -1281,7 +1273,7 @@ __STATIC_INLINE void LL_TIM_SetCounterMode(TIM_TypeDef *TIMx, uint32_t CounterMo * @arg @ref LL_TIM_COUNTERMODE_CENTER_DOWN * @arg @ref LL_TIM_COUNTERMODE_CENTER_UP_DOWN */ -__STATIC_INLINE uint32_t LL_TIM_GetCounterMode(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_GetCounterMode(const TIM_TypeDef *TIMx) { uint32_t counter_mode; @@ -1323,7 +1315,7 @@ __STATIC_INLINE void LL_TIM_DisableARRPreload(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledARRPreload(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledARRPreload(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->CR1, TIM_CR1_ARPE) == (TIM_CR1_ARPE)) ? 1UL : 0UL); } @@ -1360,7 +1352,7 @@ __STATIC_INLINE void LL_TIM_SetClockDivision(TIM_TypeDef *TIMx, uint32_t ClockDi * @arg @ref LL_TIM_CLOCKDIVISION_DIV2 * @arg @ref LL_TIM_CLOCKDIVISION_DIV4 */ -__STATIC_INLINE uint32_t LL_TIM_GetClockDivision(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_GetClockDivision(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_BIT(TIMx->CR1, TIM_CR1_CKD)); } @@ -1387,7 +1379,7 @@ __STATIC_INLINE void LL_TIM_SetCounter(TIM_TypeDef *TIMx, uint32_t Counter) * @param TIMx Timer instance * @retval Counter value (between Min_Data=0 and Max_Data=0xFFFF or 0xFFFFFFFF) */ -__STATIC_INLINE uint32_t LL_TIM_GetCounter(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_GetCounter(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_REG(TIMx->CNT)); } @@ -1400,7 +1392,7 @@ __STATIC_INLINE uint32_t LL_TIM_GetCounter(TIM_TypeDef *TIMx) * @arg @ref LL_TIM_COUNTERDIRECTION_UP * @arg @ref LL_TIM_COUNTERDIRECTION_DOWN */ -__STATIC_INLINE uint32_t LL_TIM_GetDirection(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_GetDirection(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_BIT(TIMx->CR1, TIM_CR1_DIR)); } @@ -1427,7 +1419,7 @@ __STATIC_INLINE void LL_TIM_SetPrescaler(TIM_TypeDef *TIMx, uint32_t Prescaler) * @param TIMx Timer instance * @retval Prescaler value between Min_Data=0 and Max_Data=65535 */ -__STATIC_INLINE uint32_t LL_TIM_GetPrescaler(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_GetPrescaler(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_REG(TIMx->PSC)); } @@ -1456,7 +1448,7 @@ __STATIC_INLINE void LL_TIM_SetAutoReload(TIM_TypeDef *TIMx, uint32_t AutoReload * @param TIMx Timer instance * @retval Auto-reload value */ -__STATIC_INLINE uint32_t LL_TIM_GetAutoReload(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_GetAutoReload(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_REG(TIMx->ARR)); } @@ -1483,7 +1475,7 @@ __STATIC_INLINE void LL_TIM_SetRepetitionCounter(TIM_TypeDef *TIMx, uint32_t Rep * @param TIMx Timer instance * @retval Repetition counter value */ -__STATIC_INLINE uint32_t LL_TIM_GetRepetitionCounter(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_GetRepetitionCounter(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_REG(TIMx->RCR)); } @@ -1524,6 +1516,17 @@ __STATIC_INLINE void LL_TIM_CC_DisablePreload(TIM_TypeDef *TIMx) CLEAR_BIT(TIMx->CR2, TIM_CR2_CCPC); } +/** + * @brief Indicates whether the capture/compare control bits (CCxE, CCxNE and OCxM) preload is enabled. + * @rmtoll CR2 CCPC LL_TIM_CC_IsEnabledPreload + * @param TIMx Timer instance + * @retval State of bit (1 or 0). + */ +__STATIC_INLINE uint32_t LL_TIM_CC_IsEnabledPreload(const TIM_TypeDef *TIMx) +{ + return ((READ_BIT(TIMx->CR2, TIM_CR2_CCPC) == (TIM_CR2_CCPC)) ? 1UL : 0UL); +} + /** * @brief Set the updated source of the capture/compare control bits (CCxE, CCxNE and OCxM). * @note Macro IS_TIM_COMMUTATION_EVENT_INSTANCE(TIMx) can be used to check @@ -1562,7 +1565,7 @@ __STATIC_INLINE void LL_TIM_CC_SetDMAReqTrigger(TIM_TypeDef *TIMx, uint32_t DMAR * @arg @ref LL_TIM_CCDMAREQUEST_CC * @arg @ref LL_TIM_CCDMAREQUEST_UPDATE */ -__STATIC_INLINE uint32_t LL_TIM_CC_GetDMAReqTrigger(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_CC_GetDMAReqTrigger(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_BIT(TIMx->CR2, TIM_CR2_CCDS)); } @@ -1656,7 +1659,7 @@ __STATIC_INLINE void LL_TIM_CC_DisableChannel(TIM_TypeDef *TIMx, uint32_t Channe * @arg @ref LL_TIM_CHANNEL_CH4 * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_CC_IsEnabledChannel(TIM_TypeDef *TIMx, uint32_t Channels) +__STATIC_INLINE uint32_t LL_TIM_CC_IsEnabledChannel(const TIM_TypeDef *TIMx, uint32_t Channels) { return ((READ_BIT(TIMx->CCER, Channels) == (Channels)) ? 1UL : 0UL); } @@ -1757,7 +1760,7 @@ __STATIC_INLINE void LL_TIM_OC_SetMode(TIM_TypeDef *TIMx, uint32_t Channel, uint * @arg @ref LL_TIM_OCMODE_PWM1 * @arg @ref LL_TIM_OCMODE_PWM2 */ -__STATIC_INLINE uint32_t LL_TIM_OC_GetMode(TIM_TypeDef *TIMx, uint32_t Channel) +__STATIC_INLINE uint32_t LL_TIM_OC_GetMode(const TIM_TypeDef *TIMx, uint32_t Channel) { uint8_t iChannel = TIM_GET_CHANNEL_INDEX(Channel); const __IO uint32_t *pReg = (__IO uint32_t *)((uint32_t)((uint32_t)(&TIMx->CCMR1) + OFFSET_TAB_CCMRx[iChannel])); @@ -1815,7 +1818,7 @@ __STATIC_INLINE void LL_TIM_OC_SetPolarity(TIM_TypeDef *TIMx, uint32_t Channel, * @arg @ref LL_TIM_OCPOLARITY_HIGH * @arg @ref LL_TIM_OCPOLARITY_LOW */ -__STATIC_INLINE uint32_t LL_TIM_OC_GetPolarity(TIM_TypeDef *TIMx, uint32_t Channel) +__STATIC_INLINE uint32_t LL_TIM_OC_GetPolarity(const TIM_TypeDef *TIMx, uint32_t Channel) { uint8_t iChannel = TIM_GET_CHANNEL_INDEX(Channel); return (READ_BIT(TIMx->CCER, (TIM_CCER_CC1P << SHIFT_TAB_CCxP[iChannel])) >> SHIFT_TAB_CCxP[iChannel]); @@ -1876,7 +1879,7 @@ __STATIC_INLINE void LL_TIM_OC_SetIdleState(TIM_TypeDef *TIMx, uint32_t Channel, * @arg @ref LL_TIM_OCIDLESTATE_LOW * @arg @ref LL_TIM_OCIDLESTATE_HIGH */ -__STATIC_INLINE uint32_t LL_TIM_OC_GetIdleState(TIM_TypeDef *TIMx, uint32_t Channel) +__STATIC_INLINE uint32_t LL_TIM_OC_GetIdleState(const TIM_TypeDef *TIMx, uint32_t Channel) { uint8_t iChannel = TIM_GET_CHANNEL_INDEX(Channel); return (READ_BIT(TIMx->CR2, (TIM_CR2_OIS1 << SHIFT_TAB_OISx[iChannel])) >> SHIFT_TAB_OISx[iChannel]); @@ -1941,7 +1944,7 @@ __STATIC_INLINE void LL_TIM_OC_DisableFast(TIM_TypeDef *TIMx, uint32_t Channel) * @arg @ref LL_TIM_CHANNEL_CH4 * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_OC_IsEnabledFast(TIM_TypeDef *TIMx, uint32_t Channel) +__STATIC_INLINE uint32_t LL_TIM_OC_IsEnabledFast(const TIM_TypeDef *TIMx, uint32_t Channel) { uint8_t iChannel = TIM_GET_CHANNEL_INDEX(Channel); const __IO uint32_t *pReg = (__IO uint32_t *)((uint32_t)((uint32_t)(&TIMx->CCMR1) + OFFSET_TAB_CCMRx[iChannel])); @@ -2005,7 +2008,7 @@ __STATIC_INLINE void LL_TIM_OC_DisablePreload(TIM_TypeDef *TIMx, uint32_t Channe * @arg @ref LL_TIM_CHANNEL_CH4 * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_OC_IsEnabledPreload(TIM_TypeDef *TIMx, uint32_t Channel) +__STATIC_INLINE uint32_t LL_TIM_OC_IsEnabledPreload(const TIM_TypeDef *TIMx, uint32_t Channel) { uint8_t iChannel = TIM_GET_CHANNEL_INDEX(Channel); const __IO uint32_t *pReg = (__IO uint32_t *)((uint32_t)((uint32_t)(&TIMx->CCMR1) + OFFSET_TAB_CCMRx[iChannel])); @@ -2078,7 +2081,7 @@ __STATIC_INLINE void LL_TIM_OC_DisableClear(TIM_TypeDef *TIMx, uint32_t Channel) * @arg @ref LL_TIM_CHANNEL_CH4 * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_OC_IsEnabledClear(TIM_TypeDef *TIMx, uint32_t Channel) +__STATIC_INLINE uint32_t LL_TIM_OC_IsEnabledClear(const TIM_TypeDef *TIMx, uint32_t Channel) { uint8_t iChannel = TIM_GET_CHANNEL_INDEX(Channel); const __IO uint32_t *pReg = (__IO uint32_t *)((uint32_t)((uint32_t)(&TIMx->CCMR1) + OFFSET_TAB_CCMRx[iChannel])); @@ -2181,7 +2184,7 @@ __STATIC_INLINE void LL_TIM_OC_SetCompareCH4(TIM_TypeDef *TIMx, uint32_t Compare * @param TIMx Timer instance * @retval CompareValue (between Min_Data=0 and Max_Data=65535) */ -__STATIC_INLINE uint32_t LL_TIM_OC_GetCompareCH1(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_OC_GetCompareCH1(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_REG(TIMx->CCR1)); } @@ -2197,7 +2200,7 @@ __STATIC_INLINE uint32_t LL_TIM_OC_GetCompareCH1(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval CompareValue (between Min_Data=0 and Max_Data=65535) */ -__STATIC_INLINE uint32_t LL_TIM_OC_GetCompareCH2(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_OC_GetCompareCH2(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_REG(TIMx->CCR2)); } @@ -2213,7 +2216,7 @@ __STATIC_INLINE uint32_t LL_TIM_OC_GetCompareCH2(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval CompareValue (between Min_Data=0 and Max_Data=65535) */ -__STATIC_INLINE uint32_t LL_TIM_OC_GetCompareCH3(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_OC_GetCompareCH3(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_REG(TIMx->CCR3)); } @@ -2229,7 +2232,7 @@ __STATIC_INLINE uint32_t LL_TIM_OC_GetCompareCH3(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval CompareValue (between Min_Data=0 and Max_Data=65535) */ -__STATIC_INLINE uint32_t LL_TIM_OC_GetCompareCH4(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_OC_GetCompareCH4(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_REG(TIMx->CCR4)); } @@ -2329,7 +2332,7 @@ __STATIC_INLINE void LL_TIM_IC_SetActiveInput(TIM_TypeDef *TIMx, uint32_t Channe * @arg @ref LL_TIM_ACTIVEINPUT_INDIRECTTI * @arg @ref LL_TIM_ACTIVEINPUT_TRC */ -__STATIC_INLINE uint32_t LL_TIM_IC_GetActiveInput(TIM_TypeDef *TIMx, uint32_t Channel) +__STATIC_INLINE uint32_t LL_TIM_IC_GetActiveInput(const TIM_TypeDef *TIMx, uint32_t Channel) { uint8_t iChannel = TIM_GET_CHANNEL_INDEX(Channel); const __IO uint32_t *pReg = (__IO uint32_t *)((uint32_t)((uint32_t)(&TIMx->CCMR1) + OFFSET_TAB_CCMRx[iChannel])); @@ -2380,7 +2383,7 @@ __STATIC_INLINE void LL_TIM_IC_SetPrescaler(TIM_TypeDef *TIMx, uint32_t Channel, * @arg @ref LL_TIM_ICPSC_DIV4 * @arg @ref LL_TIM_ICPSC_DIV8 */ -__STATIC_INLINE uint32_t LL_TIM_IC_GetPrescaler(TIM_TypeDef *TIMx, uint32_t Channel) +__STATIC_INLINE uint32_t LL_TIM_IC_GetPrescaler(const TIM_TypeDef *TIMx, uint32_t Channel) { uint8_t iChannel = TIM_GET_CHANNEL_INDEX(Channel); const __IO uint32_t *pReg = (__IO uint32_t *)((uint32_t)((uint32_t)(&TIMx->CCMR1) + OFFSET_TAB_CCMRx[iChannel])); @@ -2455,7 +2458,7 @@ __STATIC_INLINE void LL_TIM_IC_SetFilter(TIM_TypeDef *TIMx, uint32_t Channel, ui * @arg @ref LL_TIM_IC_FILTER_FDIV32_N6 * @arg @ref LL_TIM_IC_FILTER_FDIV32_N8 */ -__STATIC_INLINE uint32_t LL_TIM_IC_GetFilter(TIM_TypeDef *TIMx, uint32_t Channel) +__STATIC_INLINE uint32_t LL_TIM_IC_GetFilter(const TIM_TypeDef *TIMx, uint32_t Channel) { uint8_t iChannel = TIM_GET_CHANNEL_INDEX(Channel); const __IO uint32_t *pReg = (__IO uint32_t *)((uint32_t)((uint32_t)(&TIMx->CCMR1) + OFFSET_TAB_CCMRx[iChannel])); @@ -2512,7 +2515,7 @@ __STATIC_INLINE void LL_TIM_IC_SetPolarity(TIM_TypeDef *TIMx, uint32_t Channel, * @arg @ref LL_TIM_IC_POLARITY_FALLING * @arg @ref LL_TIM_IC_POLARITY_BOTHEDGE */ -__STATIC_INLINE uint32_t LL_TIM_IC_GetPolarity(TIM_TypeDef *TIMx, uint32_t Channel) +__STATIC_INLINE uint32_t LL_TIM_IC_GetPolarity(const TIM_TypeDef *TIMx, uint32_t Channel) { uint8_t iChannel = TIM_GET_CHANNEL_INDEX(Channel); return (READ_BIT(TIMx->CCER, ((TIM_CCER_CC1NP | TIM_CCER_CC1P) << SHIFT_TAB_CCxP[iChannel])) >> @@ -2553,7 +2556,7 @@ __STATIC_INLINE void LL_TIM_IC_DisableXORCombination(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IC_IsEnabledXORCombination(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IC_IsEnabledXORCombination(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->CR2, TIM_CR2_TI1S) == (TIM_CR2_TI1S)) ? 1UL : 0UL); } @@ -2569,7 +2572,7 @@ __STATIC_INLINE uint32_t LL_TIM_IC_IsEnabledXORCombination(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval CapturedValue (between Min_Data=0 and Max_Data=65535) */ -__STATIC_INLINE uint32_t LL_TIM_IC_GetCaptureCH1(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IC_GetCaptureCH1(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_REG(TIMx->CCR1)); } @@ -2585,7 +2588,7 @@ __STATIC_INLINE uint32_t LL_TIM_IC_GetCaptureCH1(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval CapturedValue (between Min_Data=0 and Max_Data=65535) */ -__STATIC_INLINE uint32_t LL_TIM_IC_GetCaptureCH2(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IC_GetCaptureCH2(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_REG(TIMx->CCR2)); } @@ -2601,7 +2604,7 @@ __STATIC_INLINE uint32_t LL_TIM_IC_GetCaptureCH2(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval CapturedValue (between Min_Data=0 and Max_Data=65535) */ -__STATIC_INLINE uint32_t LL_TIM_IC_GetCaptureCH3(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IC_GetCaptureCH3(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_REG(TIMx->CCR3)); } @@ -2617,7 +2620,7 @@ __STATIC_INLINE uint32_t LL_TIM_IC_GetCaptureCH3(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval CapturedValue (between Min_Data=0 and Max_Data=65535) */ -__STATIC_INLINE uint32_t LL_TIM_IC_GetCaptureCH4(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IC_GetCaptureCH4(const TIM_TypeDef *TIMx) { return (uint32_t)(READ_REG(TIMx->CCR4)); } @@ -2664,7 +2667,7 @@ __STATIC_INLINE void LL_TIM_DisableExternalClock(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledExternalClock(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledExternalClock(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->SMCR, TIM_SMCR_ECE) == (TIM_SMCR_ECE)) ? 1UL : 0UL); } @@ -2813,7 +2816,7 @@ __STATIC_INLINE void LL_TIM_DisableMasterSlaveMode(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledMasterSlaveMode(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledMasterSlaveMode(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->SMCR, TIM_SMCR_MSM) == (TIM_SMCR_MSM)) ? 1UL : 0UL); } @@ -2974,7 +2977,7 @@ __STATIC_INLINE void LL_TIM_DisableAutomaticOutput(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledAutomaticOutput(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledAutomaticOutput(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->BDTR, TIM_BDTR_AOE) == (TIM_BDTR_AOE)) ? 1UL : 0UL); } @@ -3017,7 +3020,7 @@ __STATIC_INLINE void LL_TIM_DisableAllOutputs(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledAllOutputs(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledAllOutputs(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->BDTR, TIM_BDTR_MOE) == (TIM_BDTR_MOE)) ? 1UL : 0UL); } @@ -3190,7 +3193,7 @@ __STATIC_INLINE void LL_TIM_ClearFlag_UPDATE(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_UPDATE(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_UPDATE(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->SR, TIM_SR_UIF) == (TIM_SR_UIF)) ? 1UL : 0UL); } @@ -3212,7 +3215,7 @@ __STATIC_INLINE void LL_TIM_ClearFlag_CC1(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC1(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC1(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->SR, TIM_SR_CC1IF) == (TIM_SR_CC1IF)) ? 1UL : 0UL); } @@ -3234,7 +3237,7 @@ __STATIC_INLINE void LL_TIM_ClearFlag_CC2(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC2(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC2(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->SR, TIM_SR_CC2IF) == (TIM_SR_CC2IF)) ? 1UL : 0UL); } @@ -3256,7 +3259,7 @@ __STATIC_INLINE void LL_TIM_ClearFlag_CC3(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC3(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC3(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->SR, TIM_SR_CC3IF) == (TIM_SR_CC3IF)) ? 1UL : 0UL); } @@ -3278,7 +3281,7 @@ __STATIC_INLINE void LL_TIM_ClearFlag_CC4(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC4(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC4(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->SR, TIM_SR_CC4IF) == (TIM_SR_CC4IF)) ? 1UL : 0UL); } @@ -3300,7 +3303,7 @@ __STATIC_INLINE void LL_TIM_ClearFlag_COM(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_COM(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_COM(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->SR, TIM_SR_COMIF) == (TIM_SR_COMIF)) ? 1UL : 0UL); } @@ -3322,7 +3325,7 @@ __STATIC_INLINE void LL_TIM_ClearFlag_TRIG(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_TRIG(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_TRIG(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->SR, TIM_SR_TIF) == (TIM_SR_TIF)) ? 1UL : 0UL); } @@ -3344,7 +3347,7 @@ __STATIC_INLINE void LL_TIM_ClearFlag_BRK(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_BRK(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_BRK(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->SR, TIM_SR_BIF) == (TIM_SR_BIF)) ? 1UL : 0UL); } @@ -3367,7 +3370,7 @@ __STATIC_INLINE void LL_TIM_ClearFlag_CC1OVR(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC1OVR(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC1OVR(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->SR, TIM_SR_CC1OF) == (TIM_SR_CC1OF)) ? 1UL : 0UL); } @@ -3390,7 +3393,7 @@ __STATIC_INLINE void LL_TIM_ClearFlag_CC2OVR(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC2OVR(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC2OVR(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->SR, TIM_SR_CC2OF) == (TIM_SR_CC2OF)) ? 1UL : 0UL); } @@ -3413,7 +3416,7 @@ __STATIC_INLINE void LL_TIM_ClearFlag_CC3OVR(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC3OVR(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC3OVR(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->SR, TIM_SR_CC3OF) == (TIM_SR_CC3OF)) ? 1UL : 0UL); } @@ -3436,7 +3439,7 @@ __STATIC_INLINE void LL_TIM_ClearFlag_CC4OVR(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC4OVR(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsActiveFlag_CC4OVR(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->SR, TIM_SR_CC4OF) == (TIM_SR_CC4OF)) ? 1UL : 0UL); } @@ -3476,7 +3479,7 @@ __STATIC_INLINE void LL_TIM_DisableIT_UPDATE(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_UPDATE(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_UPDATE(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->DIER, TIM_DIER_UIE) == (TIM_DIER_UIE)) ? 1UL : 0UL); } @@ -3509,7 +3512,7 @@ __STATIC_INLINE void LL_TIM_DisableIT_CC1(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_CC1(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_CC1(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->DIER, TIM_DIER_CC1IE) == (TIM_DIER_CC1IE)) ? 1UL : 0UL); } @@ -3542,7 +3545,7 @@ __STATIC_INLINE void LL_TIM_DisableIT_CC2(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_CC2(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_CC2(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->DIER, TIM_DIER_CC2IE) == (TIM_DIER_CC2IE)) ? 1UL : 0UL); } @@ -3575,7 +3578,7 @@ __STATIC_INLINE void LL_TIM_DisableIT_CC3(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_CC3(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_CC3(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->DIER, TIM_DIER_CC3IE) == (TIM_DIER_CC3IE)) ? 1UL : 0UL); } @@ -3608,7 +3611,7 @@ __STATIC_INLINE void LL_TIM_DisableIT_CC4(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_CC4(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_CC4(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->DIER, TIM_DIER_CC4IE) == (TIM_DIER_CC4IE)) ? 1UL : 0UL); } @@ -3641,7 +3644,7 @@ __STATIC_INLINE void LL_TIM_DisableIT_COM(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_COM(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_COM(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->DIER, TIM_DIER_COMIE) == (TIM_DIER_COMIE)) ? 1UL : 0UL); } @@ -3674,7 +3677,7 @@ __STATIC_INLINE void LL_TIM_DisableIT_TRIG(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_TRIG(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_TRIG(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->DIER, TIM_DIER_TIE) == (TIM_DIER_TIE)) ? 1UL : 0UL); } @@ -3707,7 +3710,7 @@ __STATIC_INLINE void LL_TIM_DisableIT_BRK(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_BRK(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledIT_BRK(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->DIER, TIM_DIER_BIE) == (TIM_DIER_BIE)) ? 1UL : 0UL); } @@ -3747,7 +3750,7 @@ __STATIC_INLINE void LL_TIM_DisableDMAReq_UPDATE(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledDMAReq_UPDATE(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledDMAReq_UPDATE(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->DIER, TIM_DIER_UDE) == (TIM_DIER_UDE)) ? 1UL : 0UL); } @@ -3780,7 +3783,7 @@ __STATIC_INLINE void LL_TIM_DisableDMAReq_CC1(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledDMAReq_CC1(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledDMAReq_CC1(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->DIER, TIM_DIER_CC1DE) == (TIM_DIER_CC1DE)) ? 1UL : 0UL); } @@ -3813,7 +3816,7 @@ __STATIC_INLINE void LL_TIM_DisableDMAReq_CC2(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledDMAReq_CC2(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledDMAReq_CC2(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->DIER, TIM_DIER_CC2DE) == (TIM_DIER_CC2DE)) ? 1UL : 0UL); } @@ -3846,7 +3849,7 @@ __STATIC_INLINE void LL_TIM_DisableDMAReq_CC3(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledDMAReq_CC3(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledDMAReq_CC3(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->DIER, TIM_DIER_CC3DE) == (TIM_DIER_CC3DE)) ? 1UL : 0UL); } @@ -3879,7 +3882,7 @@ __STATIC_INLINE void LL_TIM_DisableDMAReq_CC4(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledDMAReq_CC4(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledDMAReq_CC4(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->DIER, TIM_DIER_CC4DE) == (TIM_DIER_CC4DE)) ? 1UL : 0UL); } @@ -3912,7 +3915,7 @@ __STATIC_INLINE void LL_TIM_DisableDMAReq_COM(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledDMAReq_COM(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledDMAReq_COM(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->DIER, TIM_DIER_COMDE) == (TIM_DIER_COMDE)) ? 1UL : 0UL); } @@ -3945,7 +3948,7 @@ __STATIC_INLINE void LL_TIM_DisableDMAReq_TRIG(TIM_TypeDef *TIMx) * @param TIMx Timer instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_TIM_IsEnabledDMAReq_TRIG(TIM_TypeDef *TIMx) +__STATIC_INLINE uint32_t LL_TIM_IsEnabledDMAReq_TRIG(const TIM_TypeDef *TIMx) { return ((READ_BIT(TIMx->DIER, TIM_DIER_TDE) == (TIM_DIER_TDE)) ? 1UL : 0UL); } @@ -4054,19 +4057,19 @@ __STATIC_INLINE void LL_TIM_GenerateEvent_BRK(TIM_TypeDef *TIMx) * @{ */ -ErrorStatus LL_TIM_DeInit(TIM_TypeDef *TIMx); +ErrorStatus LL_TIM_DeInit(const TIM_TypeDef *TIMx); void LL_TIM_StructInit(LL_TIM_InitTypeDef *TIM_InitStruct); -ErrorStatus LL_TIM_Init(TIM_TypeDef *TIMx, LL_TIM_InitTypeDef *TIM_InitStruct); +ErrorStatus LL_TIM_Init(TIM_TypeDef *TIMx, const LL_TIM_InitTypeDef *TIM_InitStruct); void LL_TIM_OC_StructInit(LL_TIM_OC_InitTypeDef *TIM_OC_InitStruct); -ErrorStatus LL_TIM_OC_Init(TIM_TypeDef *TIMx, uint32_t Channel, LL_TIM_OC_InitTypeDef *TIM_OC_InitStruct); +ErrorStatus LL_TIM_OC_Init(TIM_TypeDef *TIMx, uint32_t Channel, const LL_TIM_OC_InitTypeDef *TIM_OC_InitStruct); void LL_TIM_IC_StructInit(LL_TIM_IC_InitTypeDef *TIM_ICInitStruct); -ErrorStatus LL_TIM_IC_Init(TIM_TypeDef *TIMx, uint32_t Channel, LL_TIM_IC_InitTypeDef *TIM_IC_InitStruct); +ErrorStatus LL_TIM_IC_Init(TIM_TypeDef *TIMx, uint32_t Channel, const LL_TIM_IC_InitTypeDef *TIM_IC_InitStruct); void LL_TIM_ENCODER_StructInit(LL_TIM_ENCODER_InitTypeDef *TIM_EncoderInitStruct); -ErrorStatus LL_TIM_ENCODER_Init(TIM_TypeDef *TIMx, LL_TIM_ENCODER_InitTypeDef *TIM_EncoderInitStruct); +ErrorStatus LL_TIM_ENCODER_Init(TIM_TypeDef *TIMx, const LL_TIM_ENCODER_InitTypeDef *TIM_EncoderInitStruct); void LL_TIM_HALLSENSOR_StructInit(LL_TIM_HALLSENSOR_InitTypeDef *TIM_HallSensorInitStruct); -ErrorStatus LL_TIM_HALLSENSOR_Init(TIM_TypeDef *TIMx, LL_TIM_HALLSENSOR_InitTypeDef *TIM_HallSensorInitStruct); +ErrorStatus LL_TIM_HALLSENSOR_Init(TIM_TypeDef *TIMx, const LL_TIM_HALLSENSOR_InitTypeDef *TIM_HallSensorInitStruct); void LL_TIM_BDTR_StructInit(LL_TIM_BDTR_InitTypeDef *TIM_BDTRInitStruct); -ErrorStatus LL_TIM_BDTR_Init(TIM_TypeDef *TIMx, LL_TIM_BDTR_InitTypeDef *TIM_BDTRInitStruct); +ErrorStatus LL_TIM_BDTR_Init(TIM_TypeDef *TIMx, const LL_TIM_BDTR_InitTypeDef *TIM_BDTRInitStruct); /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_usart.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_usart.h index e07c2326..ed83b6c6 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_usart.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_usart.h @@ -345,7 +345,7 @@ typedef struct * @} */ -/** @defgroup USART_LL_EM_Exported_Macros_Helper Exported_Macros_Helper +/** @defgroup USART_LL_EM_Exported_Macros_Helper Exported Macros Helper * @{ */ @@ -432,7 +432,7 @@ __STATIC_INLINE void LL_USART_Disable(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabled(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabled(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR1, USART_CR1_UE) == (USART_CR1_UE)); } @@ -510,7 +510,7 @@ __STATIC_INLINE void LL_USART_SetTransferDirection(USART_TypeDef *USARTx, uint32 * @arg @ref LL_USART_DIRECTION_TX * @arg @ref LL_USART_DIRECTION_TX_RX */ -__STATIC_INLINE uint32_t LL_USART_GetTransferDirection(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetTransferDirection(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->CR1, USART_CR1_RE | USART_CR1_TE)); } @@ -544,7 +544,7 @@ __STATIC_INLINE void LL_USART_SetParity(USART_TypeDef *USARTx, uint32_t Parity) * @arg @ref LL_USART_PARITY_EVEN * @arg @ref LL_USART_PARITY_ODD */ -__STATIC_INLINE uint32_t LL_USART_GetParity(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetParity(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->CR1, USART_CR1_PS | USART_CR1_PCE)); } @@ -571,7 +571,7 @@ __STATIC_INLINE void LL_USART_SetWakeUpMethod(USART_TypeDef *USARTx, uint32_t Me * @arg @ref LL_USART_WAKEUP_IDLELINE * @arg @ref LL_USART_WAKEUP_ADDRESSMARK */ -__STATIC_INLINE uint32_t LL_USART_GetWakeUpMethod(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetWakeUpMethod(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->CR1, USART_CR1_WAKE)); } @@ -598,7 +598,7 @@ __STATIC_INLINE void LL_USART_SetDataWidth(USART_TypeDef *USARTx, uint32_t DataW * @arg @ref LL_USART_DATAWIDTH_8B * @arg @ref LL_USART_DATAWIDTH_9B */ -__STATIC_INLINE uint32_t LL_USART_GetDataWidth(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetDataWidth(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->CR1, USART_CR1_M)); } @@ -625,14 +625,14 @@ __STATIC_INLINE void LL_USART_SetOverSampling(USART_TypeDef *USARTx, uint32_t Ov * @arg @ref LL_USART_OVERSAMPLING_16 * @arg @ref LL_USART_OVERSAMPLING_8 */ -__STATIC_INLINE uint32_t LL_USART_GetOverSampling(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetOverSampling(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->CR1, USART_CR1_OVER8)); } /** * @brief Configure if Clock pulse of the last data bit is output to the SCLK pin or not - * @note Macro @ref IS_USART_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_USART_INSTANCE(USARTx) can be used to check whether or not * Synchronous mode is supported by the USARTx instance. * @rmtoll CR2 LBCL LL_USART_SetLastClkPulseOutput * @param USARTx USART Instance @@ -649,7 +649,7 @@ __STATIC_INLINE void LL_USART_SetLastClkPulseOutput(USART_TypeDef *USARTx, uint3 /** * @brief Retrieve Clock pulse of the last data bit output configuration * (Last bit Clock pulse output to the SCLK pin or not) - * @note Macro @ref IS_USART_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_USART_INSTANCE(USARTx) can be used to check whether or not * Synchronous mode is supported by the USARTx instance. * @rmtoll CR2 LBCL LL_USART_GetLastClkPulseOutput * @param USARTx USART Instance @@ -657,14 +657,14 @@ __STATIC_INLINE void LL_USART_SetLastClkPulseOutput(USART_TypeDef *USARTx, uint3 * @arg @ref LL_USART_LASTCLKPULSE_NO_OUTPUT * @arg @ref LL_USART_LASTCLKPULSE_OUTPUT */ -__STATIC_INLINE uint32_t LL_USART_GetLastClkPulseOutput(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetLastClkPulseOutput(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->CR2, USART_CR2_LBCL)); } /** * @brief Select the phase of the clock output on the SCLK pin in synchronous mode - * @note Macro @ref IS_USART_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_USART_INSTANCE(USARTx) can be used to check whether or not * Synchronous mode is supported by the USARTx instance. * @rmtoll CR2 CPHA LL_USART_SetClockPhase * @param USARTx USART Instance @@ -680,7 +680,7 @@ __STATIC_INLINE void LL_USART_SetClockPhase(USART_TypeDef *USARTx, uint32_t Cloc /** * @brief Return phase of the clock output on the SCLK pin in synchronous mode - * @note Macro @ref IS_USART_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_USART_INSTANCE(USARTx) can be used to check whether or not * Synchronous mode is supported by the USARTx instance. * @rmtoll CR2 CPHA LL_USART_GetClockPhase * @param USARTx USART Instance @@ -688,14 +688,14 @@ __STATIC_INLINE void LL_USART_SetClockPhase(USART_TypeDef *USARTx, uint32_t Cloc * @arg @ref LL_USART_PHASE_1EDGE * @arg @ref LL_USART_PHASE_2EDGE */ -__STATIC_INLINE uint32_t LL_USART_GetClockPhase(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetClockPhase(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->CR2, USART_CR2_CPHA)); } /** * @brief Select the polarity of the clock output on the SCLK pin in synchronous mode - * @note Macro @ref IS_USART_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_USART_INSTANCE(USARTx) can be used to check whether or not * Synchronous mode is supported by the USARTx instance. * @rmtoll CR2 CPOL LL_USART_SetClockPolarity * @param USARTx USART Instance @@ -711,7 +711,7 @@ __STATIC_INLINE void LL_USART_SetClockPolarity(USART_TypeDef *USARTx, uint32_t C /** * @brief Return polarity of the clock output on the SCLK pin in synchronous mode - * @note Macro @ref IS_USART_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_USART_INSTANCE(USARTx) can be used to check whether or not * Synchronous mode is supported by the USARTx instance. * @rmtoll CR2 CPOL LL_USART_GetClockPolarity * @param USARTx USART Instance @@ -719,14 +719,14 @@ __STATIC_INLINE void LL_USART_SetClockPolarity(USART_TypeDef *USARTx, uint32_t C * @arg @ref LL_USART_POLARITY_LOW * @arg @ref LL_USART_POLARITY_HIGH */ -__STATIC_INLINE uint32_t LL_USART_GetClockPolarity(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetClockPolarity(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->CR2, USART_CR2_CPOL)); } /** * @brief Configure Clock signal format (Phase Polarity and choice about output of last bit clock pulse) - * @note Macro @ref IS_USART_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_USART_INSTANCE(USARTx) can be used to check whether or not * Synchronous mode is supported by the USARTx instance. * @note Call of this function is equivalent to following function call sequence : * - Clock Phase configuration using @ref LL_USART_SetClockPhase() function @@ -754,7 +754,7 @@ __STATIC_INLINE void LL_USART_ConfigClock(USART_TypeDef *USARTx, uint32_t Phase, /** * @brief Enable Clock output on SCLK pin - * @note Macro @ref IS_USART_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_USART_INSTANCE(USARTx) can be used to check whether or not * Synchronous mode is supported by the USARTx instance. * @rmtoll CR2 CLKEN LL_USART_EnableSCLKOutput * @param USARTx USART Instance @@ -767,7 +767,7 @@ __STATIC_INLINE void LL_USART_EnableSCLKOutput(USART_TypeDef *USARTx) /** * @brief Disable Clock output on SCLK pin - * @note Macro @ref IS_USART_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_USART_INSTANCE(USARTx) can be used to check whether or not * Synchronous mode is supported by the USARTx instance. * @rmtoll CR2 CLKEN LL_USART_DisableSCLKOutput * @param USARTx USART Instance @@ -780,13 +780,13 @@ __STATIC_INLINE void LL_USART_DisableSCLKOutput(USART_TypeDef *USARTx) /** * @brief Indicate if Clock output on SCLK pin is enabled - * @note Macro @ref IS_USART_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_USART_INSTANCE(USARTx) can be used to check whether or not * Synchronous mode is supported by the USARTx instance. * @rmtoll CR2 CLKEN LL_USART_IsEnabledSCLKOutput * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledSCLKOutput(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledSCLKOutput(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR2, USART_CR2_CLKEN) == (USART_CR2_CLKEN)); } @@ -817,7 +817,7 @@ __STATIC_INLINE void LL_USART_SetStopBitsLength(USART_TypeDef *USARTx, uint32_t * @arg @ref LL_USART_STOPBITS_1_5 * @arg @ref LL_USART_STOPBITS_2 */ -__STATIC_INLINE uint32_t LL_USART_GetStopBitsLength(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetStopBitsLength(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->CR2, USART_CR2_STOP)); } @@ -875,14 +875,14 @@ __STATIC_INLINE void LL_USART_SetNodeAddress(USART_TypeDef *USARTx, uint32_t Nod * @param USARTx USART Instance * @retval Address of the USART node (Value between Min_Data=0 and Max_Data=255) */ -__STATIC_INLINE uint32_t LL_USART_GetNodeAddress(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetNodeAddress(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->CR2, USART_CR2_ADD)); } /** * @brief Enable RTS HW Flow Control - * @note Macro @ref IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not * Hardware Flow control feature is supported by the USARTx instance. * @rmtoll CR3 RTSE LL_USART_EnableRTSHWFlowCtrl * @param USARTx USART Instance @@ -895,7 +895,7 @@ __STATIC_INLINE void LL_USART_EnableRTSHWFlowCtrl(USART_TypeDef *USARTx) /** * @brief Disable RTS HW Flow Control - * @note Macro @ref IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not * Hardware Flow control feature is supported by the USARTx instance. * @rmtoll CR3 RTSE LL_USART_DisableRTSHWFlowCtrl * @param USARTx USART Instance @@ -908,7 +908,7 @@ __STATIC_INLINE void LL_USART_DisableRTSHWFlowCtrl(USART_TypeDef *USARTx) /** * @brief Enable CTS HW Flow Control - * @note Macro @ref IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not * Hardware Flow control feature is supported by the USARTx instance. * @rmtoll CR3 CTSE LL_USART_EnableCTSHWFlowCtrl * @param USARTx USART Instance @@ -921,7 +921,7 @@ __STATIC_INLINE void LL_USART_EnableCTSHWFlowCtrl(USART_TypeDef *USARTx) /** * @brief Disable CTS HW Flow Control - * @note Macro @ref IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not * Hardware Flow control feature is supported by the USARTx instance. * @rmtoll CR3 CTSE LL_USART_DisableCTSHWFlowCtrl * @param USARTx USART Instance @@ -934,7 +934,7 @@ __STATIC_INLINE void LL_USART_DisableCTSHWFlowCtrl(USART_TypeDef *USARTx) /** * @brief Configure HW Flow Control mode (both CTS and RTS) - * @note Macro @ref IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not * Hardware Flow control feature is supported by the USARTx instance. * @rmtoll CR3 RTSE LL_USART_SetHWFlowCtrl\n * CR3 CTSE LL_USART_SetHWFlowCtrl @@ -953,7 +953,7 @@ __STATIC_INLINE void LL_USART_SetHWFlowCtrl(USART_TypeDef *USARTx, uint32_t Hard /** * @brief Return HW Flow Control configuration (both CTS and RTS) - * @note Macro @ref IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not * Hardware Flow control feature is supported by the USARTx instance. * @rmtoll CR3 RTSE LL_USART_GetHWFlowCtrl\n * CR3 CTSE LL_USART_GetHWFlowCtrl @@ -964,7 +964,7 @@ __STATIC_INLINE void LL_USART_SetHWFlowCtrl(USART_TypeDef *USARTx, uint32_t Hard * @arg @ref LL_USART_HWCONTROL_CTS * @arg @ref LL_USART_HWCONTROL_RTS_CTS */ -__STATIC_INLINE uint32_t LL_USART_GetHWFlowCtrl(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetHWFlowCtrl(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->CR3, USART_CR3_RTSE | USART_CR3_CTSE)); } @@ -997,7 +997,7 @@ __STATIC_INLINE void LL_USART_DisableOneBitSamp(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledOneBitSamp(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledOneBitSamp(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR3, USART_CR3_ONEBIT) == (USART_CR3_ONEBIT)); } @@ -1042,7 +1042,7 @@ __STATIC_INLINE void LL_USART_SetBaudRate(USART_TypeDef *USARTx, uint32_t Periph * @arg @ref LL_USART_OVERSAMPLING_8 * @retval Baud Rate */ -__STATIC_INLINE uint32_t LL_USART_GetBaudRate(USART_TypeDef *USARTx, uint32_t PeriphClk, uint32_t OverSampling) +__STATIC_INLINE uint32_t LL_USART_GetBaudRate(const USART_TypeDef *USARTx, uint32_t PeriphClk, uint32_t OverSampling) { uint32_t usartdiv = 0x0U; uint32_t brrresult = 0x0U; @@ -1077,7 +1077,7 @@ __STATIC_INLINE uint32_t LL_USART_GetBaudRate(USART_TypeDef *USARTx, uint32_t Pe /** * @brief Enable IrDA mode - * @note Macro @ref IS_IRDA_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_IRDA_INSTANCE(USARTx) can be used to check whether or not * IrDA feature is supported by the USARTx instance. * @rmtoll CR3 IREN LL_USART_EnableIrda * @param USARTx USART Instance @@ -1090,7 +1090,7 @@ __STATIC_INLINE void LL_USART_EnableIrda(USART_TypeDef *USARTx) /** * @brief Disable IrDA mode - * @note Macro @ref IS_IRDA_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_IRDA_INSTANCE(USARTx) can be used to check whether or not * IrDA feature is supported by the USARTx instance. * @rmtoll CR3 IREN LL_USART_DisableIrda * @param USARTx USART Instance @@ -1103,20 +1103,20 @@ __STATIC_INLINE void LL_USART_DisableIrda(USART_TypeDef *USARTx) /** * @brief Indicate if IrDA mode is enabled - * @note Macro @ref IS_IRDA_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_IRDA_INSTANCE(USARTx) can be used to check whether or not * IrDA feature is supported by the USARTx instance. * @rmtoll CR3 IREN LL_USART_IsEnabledIrda * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledIrda(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledIrda(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR3, USART_CR3_IREN) == (USART_CR3_IREN)); } /** * @brief Configure IrDA Power Mode (Normal or Low Power) - * @note Macro @ref IS_IRDA_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_IRDA_INSTANCE(USARTx) can be used to check whether or not * IrDA feature is supported by the USARTx instance. * @rmtoll CR3 IRLP LL_USART_SetIrdaPowerMode * @param USARTx USART Instance @@ -1132,7 +1132,7 @@ __STATIC_INLINE void LL_USART_SetIrdaPowerMode(USART_TypeDef *USARTx, uint32_t P /** * @brief Retrieve IrDA Power Mode configuration (Normal or Low Power) - * @note Macro @ref IS_IRDA_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_IRDA_INSTANCE(USARTx) can be used to check whether or not * IrDA feature is supported by the USARTx instance. * @rmtoll CR3 IRLP LL_USART_GetIrdaPowerMode * @param USARTx USART Instance @@ -1140,7 +1140,7 @@ __STATIC_INLINE void LL_USART_SetIrdaPowerMode(USART_TypeDef *USARTx, uint32_t P * @arg @ref LL_USART_IRDA_POWER_NORMAL * @arg @ref LL_USART_PHASE_2EDGE */ -__STATIC_INLINE uint32_t LL_USART_GetIrdaPowerMode(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetIrdaPowerMode(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->CR3, USART_CR3_IRLP)); } @@ -1148,7 +1148,7 @@ __STATIC_INLINE uint32_t LL_USART_GetIrdaPowerMode(USART_TypeDef *USARTx) /** * @brief Set Irda prescaler value, used for dividing the USART clock source * to achieve the Irda Low Power frequency (8 bits value) - * @note Macro @ref IS_IRDA_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_IRDA_INSTANCE(USARTx) can be used to check whether or not * IrDA feature is supported by the USARTx instance. * @rmtoll GTPR PSC LL_USART_SetIrdaPrescaler * @param USARTx USART Instance @@ -1163,13 +1163,13 @@ __STATIC_INLINE void LL_USART_SetIrdaPrescaler(USART_TypeDef *USARTx, uint32_t P /** * @brief Return Irda prescaler value, used for dividing the USART clock source * to achieve the Irda Low Power frequency (8 bits value) - * @note Macro @ref IS_IRDA_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_IRDA_INSTANCE(USARTx) can be used to check whether or not * IrDA feature is supported by the USARTx instance. * @rmtoll GTPR PSC LL_USART_GetIrdaPrescaler * @param USARTx USART Instance * @retval Irda prescaler value (Value between Min_Data=0x00 and Max_Data=0xFF) */ -__STATIC_INLINE uint32_t LL_USART_GetIrdaPrescaler(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetIrdaPrescaler(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->GTPR, USART_GTPR_PSC)); } @@ -1184,7 +1184,7 @@ __STATIC_INLINE uint32_t LL_USART_GetIrdaPrescaler(USART_TypeDef *USARTx) /** * @brief Enable Smartcard NACK transmission - * @note Macro @ref IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not * Smartcard feature is supported by the USARTx instance. * @rmtoll CR3 NACK LL_USART_EnableSmartcardNACK * @param USARTx USART Instance @@ -1197,7 +1197,7 @@ __STATIC_INLINE void LL_USART_EnableSmartcardNACK(USART_TypeDef *USARTx) /** * @brief Disable Smartcard NACK transmission - * @note Macro @ref IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not * Smartcard feature is supported by the USARTx instance. * @rmtoll CR3 NACK LL_USART_DisableSmartcardNACK * @param USARTx USART Instance @@ -1210,20 +1210,20 @@ __STATIC_INLINE void LL_USART_DisableSmartcardNACK(USART_TypeDef *USARTx) /** * @brief Indicate if Smartcard NACK transmission is enabled - * @note Macro @ref IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not * Smartcard feature is supported by the USARTx instance. * @rmtoll CR3 NACK LL_USART_IsEnabledSmartcardNACK * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledSmartcardNACK(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledSmartcardNACK(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR3, USART_CR3_NACK) == (USART_CR3_NACK)); } /** * @brief Enable Smartcard mode - * @note Macro @ref IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not * Smartcard feature is supported by the USARTx instance. * @rmtoll CR3 SCEN LL_USART_EnableSmartcard * @param USARTx USART Instance @@ -1236,7 +1236,7 @@ __STATIC_INLINE void LL_USART_EnableSmartcard(USART_TypeDef *USARTx) /** * @brief Disable Smartcard mode - * @note Macro @ref IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not * Smartcard feature is supported by the USARTx instance. * @rmtoll CR3 SCEN LL_USART_DisableSmartcard * @param USARTx USART Instance @@ -1249,13 +1249,13 @@ __STATIC_INLINE void LL_USART_DisableSmartcard(USART_TypeDef *USARTx) /** * @brief Indicate if Smartcard mode is enabled - * @note Macro @ref IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not * Smartcard feature is supported by the USARTx instance. * @rmtoll CR3 SCEN LL_USART_IsEnabledSmartcard * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledSmartcard(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledSmartcard(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR3, USART_CR3_SCEN) == (USART_CR3_SCEN)); } @@ -1263,7 +1263,7 @@ __STATIC_INLINE uint32_t LL_USART_IsEnabledSmartcard(USART_TypeDef *USARTx) /** * @brief Set Smartcard prescaler value, used for dividing the USART clock * source to provide the SMARTCARD Clock (5 bits value) - * @note Macro @ref IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not * Smartcard feature is supported by the USARTx instance. * @rmtoll GTPR PSC LL_USART_SetSmartcardPrescaler * @param USARTx USART Instance @@ -1278,13 +1278,13 @@ __STATIC_INLINE void LL_USART_SetSmartcardPrescaler(USART_TypeDef *USARTx, uint3 /** * @brief Return Smartcard prescaler value, used for dividing the USART clock * source to provide the SMARTCARD Clock (5 bits value) - * @note Macro @ref IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not * Smartcard feature is supported by the USARTx instance. * @rmtoll GTPR PSC LL_USART_GetSmartcardPrescaler * @param USARTx USART Instance * @retval Smartcard prescaler value (Value between Min_Data=0 and Max_Data=31) */ -__STATIC_INLINE uint32_t LL_USART_GetSmartcardPrescaler(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetSmartcardPrescaler(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->GTPR, USART_GTPR_PSC)); } @@ -1292,7 +1292,7 @@ __STATIC_INLINE uint32_t LL_USART_GetSmartcardPrescaler(USART_TypeDef *USARTx) /** * @brief Set Smartcard Guard time value, expressed in nb of baud clocks periods * (GT[7:0] bits : Guard time value) - * @note Macro @ref IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not * Smartcard feature is supported by the USARTx instance. * @rmtoll GTPR GT LL_USART_SetSmartcardGuardTime * @param USARTx USART Instance @@ -1307,13 +1307,13 @@ __STATIC_INLINE void LL_USART_SetSmartcardGuardTime(USART_TypeDef *USARTx, uint3 /** * @brief Return Smartcard Guard time value, expressed in nb of baud clocks periods * (GT[7:0] bits : Guard time value) - * @note Macro @ref IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not * Smartcard feature is supported by the USARTx instance. * @rmtoll GTPR GT LL_USART_GetSmartcardGuardTime * @param USARTx USART Instance * @retval Smartcard Guard time value (Value between Min_Data=0x00 and Max_Data=0xFF) */ -__STATIC_INLINE uint32_t LL_USART_GetSmartcardGuardTime(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetSmartcardGuardTime(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->GTPR, USART_GTPR_GT) >> USART_POSITION_GTPR_GT); } @@ -1328,7 +1328,7 @@ __STATIC_INLINE uint32_t LL_USART_GetSmartcardGuardTime(USART_TypeDef *USARTx) /** * @brief Enable Single Wire Half-Duplex mode - * @note Macro @ref IS_UART_HALFDUPLEX_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_HALFDUPLEX_INSTANCE(USARTx) can be used to check whether or not * Half-Duplex mode is supported by the USARTx instance. * @rmtoll CR3 HDSEL LL_USART_EnableHalfDuplex * @param USARTx USART Instance @@ -1341,7 +1341,7 @@ __STATIC_INLINE void LL_USART_EnableHalfDuplex(USART_TypeDef *USARTx) /** * @brief Disable Single Wire Half-Duplex mode - * @note Macro @ref IS_UART_HALFDUPLEX_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_HALFDUPLEX_INSTANCE(USARTx) can be used to check whether or not * Half-Duplex mode is supported by the USARTx instance. * @rmtoll CR3 HDSEL LL_USART_DisableHalfDuplex * @param USARTx USART Instance @@ -1354,13 +1354,13 @@ __STATIC_INLINE void LL_USART_DisableHalfDuplex(USART_TypeDef *USARTx) /** * @brief Indicate if Single Wire Half-Duplex mode is enabled - * @note Macro @ref IS_UART_HALFDUPLEX_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_HALFDUPLEX_INSTANCE(USARTx) can be used to check whether or not * Half-Duplex mode is supported by the USARTx instance. * @rmtoll CR3 HDSEL LL_USART_IsEnabledHalfDuplex * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledHalfDuplex(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledHalfDuplex(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR3, USART_CR3_HDSEL) == (USART_CR3_HDSEL)); } @@ -1375,7 +1375,7 @@ __STATIC_INLINE uint32_t LL_USART_IsEnabledHalfDuplex(USART_TypeDef *USARTx) /** * @brief Set LIN Break Detection Length - * @note Macro @ref IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not * LIN feature is supported by the USARTx instance. * @rmtoll CR2 LBDL LL_USART_SetLINBrkDetectionLen * @param USARTx USART Instance @@ -1391,7 +1391,7 @@ __STATIC_INLINE void LL_USART_SetLINBrkDetectionLen(USART_TypeDef *USARTx, uint3 /** * @brief Return LIN Break Detection Length - * @note Macro @ref IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not * LIN feature is supported by the USARTx instance. * @rmtoll CR2 LBDL LL_USART_GetLINBrkDetectionLen * @param USARTx USART Instance @@ -1399,14 +1399,14 @@ __STATIC_INLINE void LL_USART_SetLINBrkDetectionLen(USART_TypeDef *USARTx, uint3 * @arg @ref LL_USART_LINBREAK_DETECT_10B * @arg @ref LL_USART_LINBREAK_DETECT_11B */ -__STATIC_INLINE uint32_t LL_USART_GetLINBrkDetectionLen(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_GetLINBrkDetectionLen(const USART_TypeDef *USARTx) { return (uint32_t)(READ_BIT(USARTx->CR2, USART_CR2_LBDL)); } /** * @brief Enable LIN mode - * @note Macro @ref IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not * LIN feature is supported by the USARTx instance. * @rmtoll CR2 LINEN LL_USART_EnableLIN * @param USARTx USART Instance @@ -1419,7 +1419,7 @@ __STATIC_INLINE void LL_USART_EnableLIN(USART_TypeDef *USARTx) /** * @brief Disable LIN mode - * @note Macro @ref IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not * LIN feature is supported by the USARTx instance. * @rmtoll CR2 LINEN LL_USART_DisableLIN * @param USARTx USART Instance @@ -1432,13 +1432,13 @@ __STATIC_INLINE void LL_USART_DisableLIN(USART_TypeDef *USARTx) /** * @brief Indicate if LIN mode is enabled - * @note Macro @ref IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not * LIN feature is supported by the USARTx instance. * @rmtoll CR2 LINEN LL_USART_IsEnabledLIN * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledLIN(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledLIN(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR2, USART_CR2_LINEN) == (USART_CR2_LINEN)); } @@ -1493,7 +1493,7 @@ __STATIC_INLINE void LL_USART_ConfigAsyncMode(USART_TypeDef *USARTx) * - IREN bit in the USART_CR3 register, * - HDSEL bit in the USART_CR3 register. * This function also sets the USART in Synchronous mode. - * @note Macro @ref IS_USART_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_USART_INSTANCE(USARTx) can be used to check whether or not * Synchronous mode is supported by the USARTx instance. * @note Call of this function is equivalent to following function call sequence : * - Clear LINEN in CR2 using @ref LL_USART_DisableLIN() function @@ -1531,7 +1531,7 @@ __STATIC_INLINE void LL_USART_ConfigSyncMode(USART_TypeDef *USARTx) * - IREN bit in the USART_CR3 register, * - HDSEL bit in the USART_CR3 register. * This function also set the UART/USART in LIN mode. - * @note Macro @ref IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not * LIN feature is supported by the USARTx instance. * @note Call of this function is equivalent to following function call sequence : * - Clear CLKEN in CR2 using @ref LL_USART_DisableSCLKOutput() function @@ -1571,7 +1571,7 @@ __STATIC_INLINE void LL_USART_ConfigLINMode(USART_TypeDef *USARTx) * - SCEN bit in the USART_CR3 register, * - IREN bit in the USART_CR3 register, * This function also sets the UART/USART in Half Duplex mode. - * @note Macro @ref IS_UART_HALFDUPLEX_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_HALFDUPLEX_INSTANCE(USARTx) can be used to check whether or not * Half-Duplex mode is supported by the USARTx instance. * @note Call of this function is equivalent to following function call sequence : * - Clear LINEN in CR2 using @ref LL_USART_DisableLIN() function @@ -1610,7 +1610,7 @@ __STATIC_INLINE void LL_USART_ConfigHalfDuplexMode(USART_TypeDef *USARTx) * This function also configures Stop bits to 1.5 bits and * sets the USART in Smartcard mode (SCEN bit). * Clock Output is also enabled (CLKEN). - * @note Macro @ref IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_SMARTCARD_INSTANCE(USARTx) can be used to check whether or not * Smartcard feature is supported by the USARTx instance. * @note Call of this function is equivalent to following function call sequence : * - Clear LINEN in CR2 using @ref LL_USART_DisableLIN() function @@ -1652,7 +1652,7 @@ __STATIC_INLINE void LL_USART_ConfigSmartcardMode(USART_TypeDef *USARTx) * - SCEN bit in the USART_CR3 register, * - HDSEL bit in the USART_CR3 register. * This function also sets the UART/USART in IRDA mode (IREN bit). - * @note Macro @ref IS_IRDA_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_IRDA_INSTANCE(USARTx) can be used to check whether or not * IrDA feature is supported by the USARTx instance. * @note Call of this function is equivalent to following function call sequence : * - Clear LINEN in CR2 using @ref LL_USART_DisableLIN() function @@ -1734,7 +1734,7 @@ __STATIC_INLINE void LL_USART_ConfigMultiProcessMode(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_PE(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_PE(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->SR, USART_SR_PE) == (USART_SR_PE)); } @@ -1745,7 +1745,7 @@ __STATIC_INLINE uint32_t LL_USART_IsActiveFlag_PE(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_FE(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_FE(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->SR, USART_SR_FE) == (USART_SR_FE)); } @@ -1756,7 +1756,7 @@ __STATIC_INLINE uint32_t LL_USART_IsActiveFlag_FE(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_NE(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_NE(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->SR, USART_SR_NE) == (USART_SR_NE)); } @@ -1767,7 +1767,7 @@ __STATIC_INLINE uint32_t LL_USART_IsActiveFlag_NE(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_ORE(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_ORE(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->SR, USART_SR_ORE) == (USART_SR_ORE)); } @@ -1778,7 +1778,7 @@ __STATIC_INLINE uint32_t LL_USART_IsActiveFlag_ORE(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_IDLE(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_IDLE(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->SR, USART_SR_IDLE) == (USART_SR_IDLE)); } @@ -1789,7 +1789,7 @@ __STATIC_INLINE uint32_t LL_USART_IsActiveFlag_IDLE(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_RXNE(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_RXNE(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->SR, USART_SR_RXNE) == (USART_SR_RXNE)); } @@ -1800,7 +1800,7 @@ __STATIC_INLINE uint32_t LL_USART_IsActiveFlag_RXNE(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_TC(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_TC(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->SR, USART_SR_TC) == (USART_SR_TC)); } @@ -1811,33 +1811,33 @@ __STATIC_INLINE uint32_t LL_USART_IsActiveFlag_TC(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_TXE(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_TXE(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->SR, USART_SR_TXE) == (USART_SR_TXE)); } /** * @brief Check if the USART LIN Break Detection Flag is set or not - * @note Macro @ref IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not * LIN feature is supported by the USARTx instance. * @rmtoll SR LBD LL_USART_IsActiveFlag_LBD * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_LBD(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_LBD(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->SR, USART_SR_LBD) == (USART_SR_LBD)); } /** * @brief Check if the USART CTS Flag is set or not - * @note Macro @ref IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not * Hardware Flow control feature is supported by the USARTx instance. * @rmtoll SR CTS LL_USART_IsActiveFlag_nCTS * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_nCTS(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_nCTS(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->SR, USART_SR_CTS) == (USART_SR_CTS)); } @@ -1848,7 +1848,7 @@ __STATIC_INLINE uint32_t LL_USART_IsActiveFlag_nCTS(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_SBK(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_SBK(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR1, USART_CR1_SBK) == (USART_CR1_SBK)); } @@ -1859,7 +1859,7 @@ __STATIC_INLINE uint32_t LL_USART_IsActiveFlag_SBK(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_RWU(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsActiveFlag_RWU(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR1, USART_CR1_RWU) == (USART_CR1_RWU)); } @@ -1983,7 +1983,7 @@ __STATIC_INLINE void LL_USART_ClearFlag_RXNE(USART_TypeDef *USARTx) /** * @brief Clear LIN Break Detection Flag - * @note Macro @ref IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not * LIN feature is supported by the USARTx instance. * @rmtoll SR LBD LL_USART_ClearFlag_LBD * @param USARTx USART Instance @@ -1996,7 +1996,7 @@ __STATIC_INLINE void LL_USART_ClearFlag_LBD(USART_TypeDef *USARTx) /** * @brief Clear CTS Interrupt Flag - * @note Macro @ref IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not * Hardware Flow control feature is supported by the USARTx instance. * @rmtoll SR CTS LL_USART_ClearFlag_nCTS * @param USARTx USART Instance @@ -2072,7 +2072,7 @@ __STATIC_INLINE void LL_USART_EnableIT_PE(USART_TypeDef *USARTx) /** * @brief Enable LIN Break Detection Interrupt - * @note Macro @ref IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not * LIN feature is supported by the USARTx instance. * @rmtoll CR2 LBDIE LL_USART_EnableIT_LBD * @param USARTx USART Instance @@ -2100,7 +2100,7 @@ __STATIC_INLINE void LL_USART_EnableIT_ERROR(USART_TypeDef *USARTx) /** * @brief Enable CTS Interrupt - * @note Macro @ref IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not * Hardware Flow control feature is supported by the USARTx instance. * @rmtoll CR3 CTSIE LL_USART_EnableIT_CTS * @param USARTx USART Instance @@ -2168,7 +2168,7 @@ __STATIC_INLINE void LL_USART_DisableIT_PE(USART_TypeDef *USARTx) /** * @brief Disable LIN Break Detection Interrupt - * @note Macro @ref IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not * LIN feature is supported by the USARTx instance. * @rmtoll CR2 LBDIE LL_USART_DisableIT_LBD * @param USARTx USART Instance @@ -2196,7 +2196,7 @@ __STATIC_INLINE void LL_USART_DisableIT_ERROR(USART_TypeDef *USARTx) /** * @brief Disable CTS Interrupt - * @note Macro @ref IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not * Hardware Flow control feature is supported by the USARTx instance. * @rmtoll CR3 CTSIE LL_USART_DisableIT_CTS * @param USARTx USART Instance @@ -2213,7 +2213,7 @@ __STATIC_INLINE void LL_USART_DisableIT_CTS(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_IDLE(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_IDLE(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR1, USART_CR1_IDLEIE) == (USART_CR1_IDLEIE)); } @@ -2224,7 +2224,7 @@ __STATIC_INLINE uint32_t LL_USART_IsEnabledIT_IDLE(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_RXNE(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_RXNE(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR1, USART_CR1_RXNEIE) == (USART_CR1_RXNEIE)); } @@ -2235,7 +2235,7 @@ __STATIC_INLINE uint32_t LL_USART_IsEnabledIT_RXNE(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_TC(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_TC(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR1, USART_CR1_TCIE) == (USART_CR1_TCIE)); } @@ -2246,7 +2246,7 @@ __STATIC_INLINE uint32_t LL_USART_IsEnabledIT_TC(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_TXE(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_TXE(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR1, USART_CR1_TXEIE) == (USART_CR1_TXEIE)); } @@ -2257,20 +2257,20 @@ __STATIC_INLINE uint32_t LL_USART_IsEnabledIT_TXE(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_PE(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_PE(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR1, USART_CR1_PEIE) == (USART_CR1_PEIE)); } /** * @brief Check if the USART LIN Break Detection Interrupt is enabled or disabled. - * @note Macro @ref IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_LIN_INSTANCE(USARTx) can be used to check whether or not * LIN feature is supported by the USARTx instance. * @rmtoll CR2 LBDIE LL_USART_IsEnabledIT_LBD * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_LBD(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_LBD(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR2, USART_CR2_LBDIE) == (USART_CR2_LBDIE)); } @@ -2281,20 +2281,20 @@ __STATIC_INLINE uint32_t LL_USART_IsEnabledIT_LBD(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_ERROR(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_ERROR(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR3, USART_CR3_EIE) == (USART_CR3_EIE)); } /** * @brief Check if the USART CTS Interrupt is enabled or disabled. - * @note Macro @ref IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not + * @note Macro IS_UART_HWFLOW_INSTANCE(USARTx) can be used to check whether or not * Hardware Flow control feature is supported by the USARTx instance. * @rmtoll CR3 CTSIE LL_USART_IsEnabledIT_CTS * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_CTS(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledIT_CTS(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR3, USART_CR3_CTSIE) == (USART_CR3_CTSIE)); } @@ -2335,7 +2335,7 @@ __STATIC_INLINE void LL_USART_DisableDMAReq_RX(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledDMAReq_RX(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledDMAReq_RX(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR3, USART_CR3_DMAR) == (USART_CR3_DMAR)); } @@ -2368,7 +2368,7 @@ __STATIC_INLINE void LL_USART_DisableDMAReq_TX(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval State of bit (1 or 0). */ -__STATIC_INLINE uint32_t LL_USART_IsEnabledDMAReq_TX(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_IsEnabledDMAReq_TX(const USART_TypeDef *USARTx) { return (READ_BIT(USARTx->CR3, USART_CR3_DMAT) == (USART_CR3_DMAT)); } @@ -2380,7 +2380,7 @@ __STATIC_INLINE uint32_t LL_USART_IsEnabledDMAReq_TX(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval Address of data register */ -__STATIC_INLINE uint32_t LL_USART_DMA_GetRegAddr(USART_TypeDef *USARTx) +__STATIC_INLINE uint32_t LL_USART_DMA_GetRegAddr(const USART_TypeDef *USARTx) { /* return address of DR register */ return ((uint32_t) &(USARTx->DR)); @@ -2400,7 +2400,7 @@ __STATIC_INLINE uint32_t LL_USART_DMA_GetRegAddr(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval Value between Min_Data=0x00 and Max_Data=0xFF */ -__STATIC_INLINE uint8_t LL_USART_ReceiveData8(USART_TypeDef *USARTx) +__STATIC_INLINE uint8_t LL_USART_ReceiveData8(const USART_TypeDef *USARTx) { return (uint8_t)(READ_BIT(USARTx->DR, USART_DR_DR)); } @@ -2411,7 +2411,7 @@ __STATIC_INLINE uint8_t LL_USART_ReceiveData8(USART_TypeDef *USARTx) * @param USARTx USART Instance * @retval Value between Min_Data=0x00 and Max_Data=0x1FF */ -__STATIC_INLINE uint16_t LL_USART_ReceiveData9(USART_TypeDef *USARTx) +__STATIC_INLINE uint16_t LL_USART_ReceiveData9(const USART_TypeDef *USARTx) { return (uint16_t)(READ_BIT(USARTx->DR, USART_DR_DR)); } @@ -2489,10 +2489,10 @@ __STATIC_INLINE void LL_USART_RequestExitMuteMode(USART_TypeDef *USARTx) /** @defgroup USART_LL_EF_Init Initialization and de-initialization functions * @{ */ -ErrorStatus LL_USART_DeInit(USART_TypeDef *USARTx); -ErrorStatus LL_USART_Init(USART_TypeDef *USARTx, LL_USART_InitTypeDef *USART_InitStruct); +ErrorStatus LL_USART_DeInit(const USART_TypeDef *USARTx); +ErrorStatus LL_USART_Init(USART_TypeDef *USARTx, const LL_USART_InitTypeDef *USART_InitStruct); void LL_USART_StructInit(LL_USART_InitTypeDef *USART_InitStruct); -ErrorStatus LL_USART_ClockInit(USART_TypeDef *USARTx, LL_USART_ClockInitTypeDef *USART_ClockInitStruct); +ErrorStatus LL_USART_ClockInit(USART_TypeDef *USARTx, const LL_USART_ClockInitTypeDef *USART_ClockInitStruct); void LL_USART_ClockStructInit(LL_USART_ClockInitTypeDef *USART_ClockInitStruct); /** * @} diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_usb.h b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_usb.h index a7114cd0..caabcf33 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_usb.h +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Inc/stm32f4xx_ll_usb.h @@ -37,18 +37,24 @@ extern "C" { */ /* Exported types ------------------------------------------------------------*/ +#ifndef HAL_USB_TIMEOUT +#define HAL_USB_TIMEOUT 0xF000000U +#endif /* define HAL_USB_TIMEOUT */ + +#ifndef HAL_USB_CURRENT_MODE_MAX_DELAY_MS +#define HAL_USB_CURRENT_MODE_MAX_DELAY_MS 200U +#endif /* define HAL_USB_CURRENT_MODE_MAX_DELAY_MS */ /** * @brief USB Mode definition */ -#if defined (USB_OTG_FS) || defined (USB_OTG_HS) typedef enum { - USB_DEVICE_MODE = 0, - USB_HOST_MODE = 1, - USB_DRD_MODE = 2 -} USB_OTG_ModeTypeDef; + USB_DEVICE_MODE = 0, + USB_HOST_MODE = 1, + USB_DRD_MODE = 2 +} USB_ModeTypeDef; /** * @brief URB States definition @@ -61,7 +67,7 @@ typedef enum URB_NYET, URB_ERROR, URB_STALL -} USB_OTG_URBStateTypeDef; +} USB_URBStateTypeDef; /** * @brief Host channel States definition @@ -71,13 +77,14 @@ typedef enum HC_IDLE = 0, HC_XFRC, HC_HALTED, + HC_ACK, HC_NAK, HC_NYET, HC_STALL, HC_XACTERR, HC_BBLERR, HC_DATATGLERR -} USB_OTG_HCStateTypeDef; +} USB_HCStateTypeDef; /** @@ -85,40 +92,41 @@ typedef enum */ typedef struct { - uint32_t dev_endpoints; /*!< Device Endpoints number. + uint8_t dev_endpoints; /*!< Device Endpoints number. This parameter depends on the used USB core. This parameter must be a number between Min_Data = 1 and Max_Data = 15 */ - uint32_t Host_channels; /*!< Host Channels number. + uint8_t Host_channels; /*!< Host Channels number. This parameter Depends on the used USB core. This parameter must be a number between Min_Data = 1 and Max_Data = 15 */ - uint32_t speed; /*!< USB Core speed. - This parameter can be any value of @ref PCD_Speed/HCD_Speed - (HCD_SPEED_xxx, HCD_SPEED_xxx) */ + uint8_t dma_enable; /*!< USB DMA state. + If DMA is not supported this parameter shall be set by default to zero */ - uint32_t dma_enable; /*!< Enable or disable of the USB embedded DMA used only for OTG HS. */ + uint8_t speed; /*!< USB Core speed. + This parameter can be any value of @ref PCD_Speed/HCD_Speed + (HCD_SPEED_xxx, HCD_SPEED_xxx) */ - uint32_t ep0_mps; /*!< Set the Endpoint 0 Max Packet size. */ + uint8_t ep0_mps; /*!< Set the Endpoint 0 Max Packet size. */ - uint32_t phy_itface; /*!< Select the used PHY interface. - This parameter can be any value of @ref PCD_PHY_Module/HCD_PHY_Module */ + uint8_t phy_itface; /*!< Select the used PHY interface. + This parameter can be any value of @ref PCD_PHY_Module/HCD_PHY_Module */ - uint32_t Sof_enable; /*!< Enable or disable the output of the SOF signal. */ + uint8_t Sof_enable; /*!< Enable or disable the output of the SOF signal. */ - uint32_t low_power_enable; /*!< Enable or disable the low power mode. */ + uint8_t low_power_enable; /*!< Enable or disable the low Power Mode. */ - uint32_t lpm_enable; /*!< Enable or disable Link Power Management. */ + uint8_t lpm_enable; /*!< Enable or disable Link Power Management. */ - uint32_t battery_charging_enable; /*!< Enable or disable Battery charging. */ + uint8_t battery_charging_enable; /*!< Enable or disable Battery charging. */ - uint32_t vbus_sensing_enable; /*!< Enable or disable the VBUS Sensing feature. */ + uint8_t vbus_sensing_enable; /*!< Enable or disable the VBUS Sensing feature. */ - uint32_t use_dedicated_ep1; /*!< Enable or disable the use of the dedicated EP1 interrupt. */ + uint8_t use_dedicated_ep1; /*!< Enable or disable the use of the dedicated EP1 interrupt. */ - uint32_t use_external_vbus; /*!< Enable or disable the use of the external VBUS. */ + uint8_t use_external_vbus; /*!< Enable or disable the use of the external VBUS. */ -} USB_OTG_CfgTypeDef; +} USB_CfgTypeDef; typedef struct { @@ -140,25 +148,25 @@ typedef struct uint8_t data_pid_start; /*!< Initial data PID This parameter must be a number between Min_Data = 0 and Max_Data = 1 */ + uint32_t maxpacket; /*!< Endpoint Max packet size + This parameter must be a number between Min_Data = 0 and Max_Data = 64KB */ + + uint8_t *xfer_buff; /*!< Pointer to transfer buffer */ + + uint32_t xfer_len; /*!< Current transfer length */ + + uint32_t xfer_count; /*!< Partial transfer length in case of multi packet transfer */ + uint8_t even_odd_frame; /*!< IFrame parity This parameter must be a number between Min_Data = 0 and Max_Data = 1 */ uint16_t tx_fifo_num; /*!< Transmission FIFO number This parameter must be a number between Min_Data = 1 and Max_Data = 15 */ - uint32_t maxpacket; /*!< Endpoint Max packet size - This parameter must be a number between Min_Data = 0 and Max_Data = 64KB */ - - uint8_t *xfer_buff; /*!< Pointer to transfer buffer */ - uint32_t dma_addr; /*!< 32 bits aligned transfer buffer address */ - uint32_t xfer_len; /*!< Current transfer length */ - uint32_t xfer_size; /*!< requested transfer size */ - - uint32_t xfer_count; /*!< Partial transfer length in case of multi packet transfer */ -} USB_OTG_EPTypeDef; +} USB_EPTypeDef; typedef struct { @@ -179,8 +187,13 @@ typedef struct (HCD_DEVICE_SPEED_xxx) */ uint8_t do_ping; /*!< Enable or disable the use of the PING protocol for HS mode. */ + uint8_t do_ssplit; /*!< Enable start split transaction in HS mode. */ + uint8_t do_csplit; /*!< Enable complete split transaction in HS mode. */ + uint8_t ep_ss_schedule; /*!< Enable periodic endpoint start split schedule . */ + uint32_t iso_splt_xactPos; /*!< iso split transfer transaction position. */ - uint8_t process_ping; /*!< Execute the PING protocol for HS mode. */ + uint8_t hub_port_nbr; /*!< USB HUB port number */ + uint8_t hub_addr; /*!< USB HUB address */ uint8_t ep_type; /*!< Endpoint Type. This parameter can be any value of @ref USB_LL_EP_Type */ @@ -193,7 +206,7 @@ typedef struct uint8_t *xfer_buff; /*!< Pointer to transfer buffer. */ - uint32_t XferSize; /*!< OTG Channel transfer size. */ + uint32_t XferSize; /*!< OTG Channel transfer size. */ uint32_t xfer_len; /*!< Current transfer length. */ @@ -208,15 +221,21 @@ typedef struct uint32_t dma_addr; /*!< 32 bits aligned transfer buffer address. */ uint32_t ErrCnt; /*!< Host channel error count. */ + uint32_t NyetErrCnt; /*!< Complete Split NYET Host channel error count. */ - USB_OTG_URBStateTypeDef urb_state; /*!< URB state. - This parameter can be any value of @ref USB_OTG_URBStateTypeDef */ + USB_URBStateTypeDef urb_state; /*!< URB state. + This parameter can be any value of @ref USB_URBStateTypeDef */ - USB_OTG_HCStateTypeDef state; /*!< Host Channel state. - This parameter can be any value of @ref USB_OTG_HCStateTypeDef */ -} USB_OTG_HCTypeDef; -#endif /* defined (USB_OTG_FS) || defined (USB_OTG_HS) */ + USB_HCStateTypeDef state; /*!< Host Channel state. + This parameter can be any value of @ref USB_HCStateTypeDef */ +} USB_HCTypeDef; +typedef USB_ModeTypeDef USB_OTG_ModeTypeDef; +typedef USB_CfgTypeDef USB_OTG_CfgTypeDef; +typedef USB_EPTypeDef USB_OTG_EPTypeDef; +typedef USB_URBStateTypeDef USB_OTG_URBStateTypeDef; +typedef USB_HCStateTypeDef USB_OTG_HCStateTypeDef; +typedef USB_HCTypeDef USB_OTG_HCTypeDef; /* Exported constants --------------------------------------------------------*/ @@ -244,18 +263,6 @@ typedef struct * @} */ -/** @defgroup USB_LL Device Speed - * @{ - */ -#define USBD_HS_SPEED 0U -#define USBD_HSINFS_SPEED 1U -#define USBH_HS_SPEED 0U -#define USBD_FS_SPEED 2U -#define USBH_FSLS_SPEED 1U -/** - * @} - */ - /** @defgroup USB_LL_Core_Speed USB Low Layer Core Speed * @{ */ @@ -319,7 +326,7 @@ typedef struct /** * @} */ - +#endif /* defined (USB_OTG_FS) || defined (USB_OTG_HS) */ /** @defgroup USB_LL_EP0_MPS USB Low Layer EP0 MPS * @{ */ @@ -331,6 +338,18 @@ typedef struct * @} */ +/** @defgroup USB_LL_EP_Type USB Low Layer EP Type + * @{ + */ +#define EP_TYPE_CTRL 0U +#define EP_TYPE_ISOC 1U +#define EP_TYPE_BULK 2U +#define EP_TYPE_INTR 3U +#define EP_TYPE_MSK 3U +/** + * @} + */ + /** @defgroup USB_LL_EP_Speed USB Low Layer EP Speed * @{ */ @@ -341,18 +360,30 @@ typedef struct * @} */ -/** @defgroup USB_LL_EP_Type USB Low Layer EP Type +/** @defgroup USB_LL_CH_PID_Type USB Low Layer Channel PID Type * @{ */ -#define EP_TYPE_CTRL 0U -#define EP_TYPE_ISOC 1U -#define EP_TYPE_BULK 2U -#define EP_TYPE_INTR 3U -#define EP_TYPE_MSK 3U +#define HC_PID_DATA0 0U +#define HC_PID_DATA2 1U +#define HC_PID_DATA1 2U +#define HC_PID_SETUP 3U /** * @} */ +/** @defgroup USB_LL Device Speed + * @{ + */ +#define USBD_HS_SPEED 0U +#define USBD_HSINFS_SPEED 1U +#define USBH_HS_SPEED 0U +#define USBD_FS_SPEED 2U +#define USBH_FSLS_SPEED 1U +/** + * @} + */ + +#if defined (USB_OTG_FS) || defined (USB_OTG_HS) /** @defgroup USB_LL_STS_Defines USB Low Layer STS Defines * @{ */ @@ -375,6 +406,16 @@ typedef struct * @} */ +/** @defgroup USB_LL_HFIR_Defines USB Low Layer frame interval Defines + * @{ + */ +#define HFIR_6_MHZ 6000U +#define HFIR_60_MHZ 60000U +#define HFIR_48_MHZ 48000U +/** + * @} + */ + /** @defgroup USB_LL_HPRT0_PRTSPD_SPEED_Defines USB Low Layer HPRT0 PRTSPD Speed Defines * @{ */ @@ -390,16 +431,21 @@ typedef struct #define HCCHAR_BULK 2U #define HCCHAR_INTR 3U -#define HC_PID_DATA0 0U -#define HC_PID_DATA2 1U -#define HC_PID_DATA1 2U -#define HC_PID_SETUP 3U - #define GRXSTS_PKTSTS_IN 2U #define GRXSTS_PKTSTS_IN_XFER_COMP 3U #define GRXSTS_PKTSTS_DATA_TOGGLE_ERR 5U #define GRXSTS_PKTSTS_CH_HALTED 7U +#define CLEAR_INTERRUPT_MASK 0xFFFFFFFFU + +#define HC_MAX_PKT_CNT 256U +#define ISO_SPLT_MPS 188U + +#define HCSPLT_BEGIN 1U +#define HCSPLT_MIDDLE 2U +#define HCSPLT_END 3U +#define HCSPLT_FULL 4U + #define TEST_J 1U #define TEST_K 2U #define TEST_SE0_NAK 3U @@ -423,13 +469,9 @@ typedef struct + USB_OTG_HOST_CHANNEL_BASE\ + ((i) * USB_OTG_HOST_CHANNEL_SIZE))) -#endif /* defined (USB_OTG_FS) || defined (USB_OTG_HS) */ #define EP_ADDR_MSK 0xFU - -#ifndef USE_USB_DOUBLE_BUFFER -#define USE_USB_DOUBLE_BUFFER 1U -#endif /* USE_USB_DOUBLE_BUFFER */ +#endif /* defined (USB_OTG_FS) || defined (USB_OTG_HS) */ /** * @} */ @@ -460,55 +502,55 @@ HAL_StatusTypeDef USB_EnableGlobalInt(USB_OTG_GlobalTypeDef *USBx); HAL_StatusTypeDef USB_DisableGlobalInt(USB_OTG_GlobalTypeDef *USBx); HAL_StatusTypeDef USB_SetTurnaroundTime(USB_OTG_GlobalTypeDef *USBx, uint32_t hclk, uint8_t speed); HAL_StatusTypeDef USB_SetCurrentMode(USB_OTG_GlobalTypeDef *USBx, USB_OTG_ModeTypeDef mode); -HAL_StatusTypeDef USB_SetDevSpeed(USB_OTG_GlobalTypeDef *USBx, uint8_t speed); +HAL_StatusTypeDef USB_SetDevSpeed(const USB_OTG_GlobalTypeDef *USBx, uint8_t speed); HAL_StatusTypeDef USB_FlushRxFifo(USB_OTG_GlobalTypeDef *USBx); HAL_StatusTypeDef USB_FlushTxFifo(USB_OTG_GlobalTypeDef *USBx, uint32_t num); -HAL_StatusTypeDef USB_ActivateEndpoint(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep); -HAL_StatusTypeDef USB_DeactivateEndpoint(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep); -HAL_StatusTypeDef USB_ActivateDedicatedEndpoint(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep); -HAL_StatusTypeDef USB_DeactivateDedicatedEndpoint(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep); +HAL_StatusTypeDef USB_ActivateEndpoint(const USB_OTG_GlobalTypeDef *USBx, const USB_OTG_EPTypeDef *ep); +HAL_StatusTypeDef USB_DeactivateEndpoint(const USB_OTG_GlobalTypeDef *USBx, const USB_OTG_EPTypeDef *ep); +HAL_StatusTypeDef USB_ActivateDedicatedEndpoint(const USB_OTG_GlobalTypeDef *USBx, const USB_OTG_EPTypeDef *ep); +HAL_StatusTypeDef USB_DeactivateDedicatedEndpoint(const USB_OTG_GlobalTypeDef *USBx, const USB_OTG_EPTypeDef *ep); HAL_StatusTypeDef USB_EPStartXfer(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep, uint8_t dma); -HAL_StatusTypeDef USB_EP0StartXfer(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep, uint8_t dma); -HAL_StatusTypeDef USB_WritePacket(USB_OTG_GlobalTypeDef *USBx, uint8_t *src, +HAL_StatusTypeDef USB_WritePacket(const USB_OTG_GlobalTypeDef *USBx, uint8_t *src, uint8_t ch_ep_num, uint16_t len, uint8_t dma); -void *USB_ReadPacket(USB_OTG_GlobalTypeDef *USBx, uint8_t *dest, uint16_t len); -HAL_StatusTypeDef USB_EPSetStall(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep); -HAL_StatusTypeDef USB_EPClearStall(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep); -HAL_StatusTypeDef USB_EPStopXfer(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep); -HAL_StatusTypeDef USB_SetDevAddress(USB_OTG_GlobalTypeDef *USBx, uint8_t address); -HAL_StatusTypeDef USB_DevConnect(USB_OTG_GlobalTypeDef *USBx); -HAL_StatusTypeDef USB_DevDisconnect(USB_OTG_GlobalTypeDef *USBx); +void *USB_ReadPacket(const USB_OTG_GlobalTypeDef *USBx, uint8_t *dest, uint16_t len); +HAL_StatusTypeDef USB_EPSetStall(const USB_OTG_GlobalTypeDef *USBx, const USB_OTG_EPTypeDef *ep); +HAL_StatusTypeDef USB_EPClearStall(const USB_OTG_GlobalTypeDef *USBx, const USB_OTG_EPTypeDef *ep); +HAL_StatusTypeDef USB_EPStopXfer(const USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep); +HAL_StatusTypeDef USB_SetDevAddress(const USB_OTG_GlobalTypeDef *USBx, uint8_t address); +HAL_StatusTypeDef USB_DevConnect(const USB_OTG_GlobalTypeDef *USBx); +HAL_StatusTypeDef USB_DevDisconnect(const USB_OTG_GlobalTypeDef *USBx); HAL_StatusTypeDef USB_StopDevice(USB_OTG_GlobalTypeDef *USBx); -HAL_StatusTypeDef USB_ActivateSetup(USB_OTG_GlobalTypeDef *USBx); -HAL_StatusTypeDef USB_EP0_OutStart(USB_OTG_GlobalTypeDef *USBx, uint8_t dma, uint8_t *psetup); -uint8_t USB_GetDevSpeed(USB_OTG_GlobalTypeDef *USBx); -uint32_t USB_GetMode(USB_OTG_GlobalTypeDef *USBx); -uint32_t USB_ReadInterrupts(USB_OTG_GlobalTypeDef *USBx); -uint32_t USB_ReadDevAllOutEpInterrupt(USB_OTG_GlobalTypeDef *USBx); -uint32_t USB_ReadDevOutEPInterrupt(USB_OTG_GlobalTypeDef *USBx, uint8_t epnum); -uint32_t USB_ReadDevAllInEpInterrupt(USB_OTG_GlobalTypeDef *USBx); -uint32_t USB_ReadDevInEPInterrupt(USB_OTG_GlobalTypeDef *USBx, uint8_t epnum); +HAL_StatusTypeDef USB_ActivateSetup(const USB_OTG_GlobalTypeDef *USBx); +HAL_StatusTypeDef USB_EP0_OutStart(const USB_OTG_GlobalTypeDef *USBx, uint8_t dma, const uint8_t *psetup); +uint8_t USB_GetDevSpeed(const USB_OTG_GlobalTypeDef *USBx); +uint32_t USB_GetMode(const USB_OTG_GlobalTypeDef *USBx); +uint32_t USB_ReadInterrupts(USB_OTG_GlobalTypeDef const *USBx); +uint32_t USB_ReadChInterrupts(const USB_OTG_GlobalTypeDef *USBx, uint8_t chnum); +uint32_t USB_ReadDevAllOutEpInterrupt(const USB_OTG_GlobalTypeDef *USBx); +uint32_t USB_ReadDevOutEPInterrupt(const USB_OTG_GlobalTypeDef *USBx, uint8_t epnum); +uint32_t USB_ReadDevAllInEpInterrupt(const USB_OTG_GlobalTypeDef *USBx); +uint32_t USB_ReadDevInEPInterrupt(const USB_OTG_GlobalTypeDef *USBx, uint8_t epnum); void USB_ClearInterrupts(USB_OTG_GlobalTypeDef *USBx, uint32_t interrupt); HAL_StatusTypeDef USB_HostInit(USB_OTG_GlobalTypeDef *USBx, USB_OTG_CfgTypeDef cfg); -HAL_StatusTypeDef USB_InitFSLSPClkSel(USB_OTG_GlobalTypeDef *USBx, uint8_t freq); -HAL_StatusTypeDef USB_ResetPort(USB_OTG_GlobalTypeDef *USBx); -HAL_StatusTypeDef USB_DriveVbus(USB_OTG_GlobalTypeDef *USBx, uint8_t state); -uint32_t USB_GetHostSpeed(USB_OTG_GlobalTypeDef *USBx); -uint32_t USB_GetCurrentFrame(USB_OTG_GlobalTypeDef *USBx); +HAL_StatusTypeDef USB_InitFSLSPClkSel(const USB_OTG_GlobalTypeDef *USBx, uint8_t freq); +HAL_StatusTypeDef USB_ResetPort(const USB_OTG_GlobalTypeDef *USBx); +HAL_StatusTypeDef USB_DriveVbus(const USB_OTG_GlobalTypeDef *USBx, uint8_t state); +uint32_t USB_GetHostSpeed(USB_OTG_GlobalTypeDef const *USBx); +uint32_t USB_GetCurrentFrame(USB_OTG_GlobalTypeDef const *USBx); HAL_StatusTypeDef USB_HC_Init(USB_OTG_GlobalTypeDef *USBx, uint8_t ch_num, uint8_t epnum, uint8_t dev_address, uint8_t speed, uint8_t ep_type, uint16_t mps); HAL_StatusTypeDef USB_HC_StartXfer(USB_OTG_GlobalTypeDef *USBx, USB_OTG_HCTypeDef *hc, uint8_t dma); -uint32_t USB_HC_ReadInterrupt(USB_OTG_GlobalTypeDef *USBx); -HAL_StatusTypeDef USB_HC_Halt(USB_OTG_GlobalTypeDef *USBx, uint8_t hc_num); -HAL_StatusTypeDef USB_DoPing(USB_OTG_GlobalTypeDef *USBx, uint8_t ch_num); +uint32_t USB_HC_ReadInterrupt(const USB_OTG_GlobalTypeDef *USBx); +HAL_StatusTypeDef USB_HC_Halt(const USB_OTG_GlobalTypeDef *USBx, uint8_t hc_num); +HAL_StatusTypeDef USB_DoPing(const USB_OTG_GlobalTypeDef *USBx, uint8_t ch_num); HAL_StatusTypeDef USB_StopHost(USB_OTG_GlobalTypeDef *USBx); -HAL_StatusTypeDef USB_ActivateRemoteWakeup(USB_OTG_GlobalTypeDef *USBx); -HAL_StatusTypeDef USB_DeActivateRemoteWakeup(USB_OTG_GlobalTypeDef *USBx); +HAL_StatusTypeDef USB_ActivateRemoteWakeup(const USB_OTG_GlobalTypeDef *USBx); +HAL_StatusTypeDef USB_DeActivateRemoteWakeup(const USB_OTG_GlobalTypeDef *USBx); #endif /* defined (USB_OTG_FS) || defined (USB_OTG_HS) */ /** diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Release_Notes.html b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Release_Notes.html index 1fa10375..997f3386 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Release_Notes.html +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Release_Notes.html @@ -1,11459 +1,4189 @@ - - - - - - - - Release Notes for STM32F4xx HAL Drivers - -
-
-

 

-
- - - - - - -
- - - - - - - - - -
-

Back to - Release page

-
-

Release - - - - Notes for STM32F4xx HAL Drivers

-

Copyright 2017 - STMicroelectronics

-

-
-

 

- - - - - - -
- - - - - - - - - -
-

Update - - - - History

V1.8.1 - / 24-June-2022

-
-

Main - - - - - Changes
-

-
-
  • General updates to fix HAL ETH defects and implementation enhancements.
  • HAL - updates

    - - -
      -
    • HAL ETH update
      • Remove useless assert_param(IS_ETH_MAC_ADDRESS0123(MacAddr)) from static function ETH_MACAddressConfig().
      • Replace hard coded Rx buffer size (1000U) by macro ETH_RX_BUF_SIZE.
      • Correct -bit positions when getting MAC and DMA configurations and replace -‘UnicastSlowProtocolPacketDetect’ by ‘UnicastPausePacketDetect’ in the -MAC default configuration structure.
      • Ensure a delay of 4 TX_CLK/RX_CLK cycles between two successive write operations to the same register.
      • Disable DMA transmission in both HAL_ETH_Stop_IT() and HAL_ETH_Stop() APIs.

V1.8.0 - / 11-February-2022

-
-

Main - - - - - Changes
-

-
-
    -
  • General updates to fix known defects and implementation enhancements.
  • -
  • All source files: update disclaimer to add reference to the new license agreement.
    -
  • -
  • The following changes done on the HAL drivers require an update of the application code based on older HAL versions
    • Rework of HAL Ethernet driver to resolve problems and improve performance (compatibility break). 
    • A new HAL Ethernet driver has been redesigned with new APIs, to bypass limitations with previous HAL Ethernet driver version.
    • The new HAL Ethernet driver is the -recommended version. It is located as usual in -Drivers/STM32F4xx_HAL_Driver/Src and Drivers/STM32F4xx_HAL_Driver/Inc -folders. 
      • It can be enabled through switch HAL_ETH_MODULE_ENABLED in stm32f4xx_hal_conf.h
    • The legacy HAL Ethernet driver is also -present in the release in Drivers/STM32F4xx_HAL_Driver/Src/Legacy and -Drivers/STM32F4xx_HAL_Driver/Inc/Legacy folders for software -compatibility reasons.
      • Its usage is not recommended as -deprecated. It can however be enabled through switch -HAL_ETH_LEGACY_MODULE_ENABLED in stm32f4xx_hal_conf.h
    -
  • HAL - updates

    - - -
      - - - - -
    • HAL ETH update -
        -
      • Entire receive process reworked.
      • Resolve the problem of received data corruption.
      • Implement transmission in interrupt mode.
      • Handle one interrupt for multiple transmitted packets.
      • Implement APIs to handle PTP feature.
      • Implement APIs to handle Timestamp feature.
      • Add support of receive buffer unavailable.
      • -
      • Update HAL_ETH_IRQHandler() to handle receive buffer unavailable.
      • - -
    • -
    • HAL SMBUS update
    • -
        -
      • Update to fix issue of mismatched data -received by master in case of data size to be transmitted by the slave -is greater than the data size to be received by the master.
      • -
          -
        • Add flush on TX register.
        • -
        -
      -
    • HAL TIM update
    • -
        -
      • __LL_TIM_CALC_PSC() macro update to round up the evaluate value when the fractional part of the division is greater than 0.5.
      • -
      - -
    • HAL LPTIM update
    • -
        -
      • Add check on PRIMASK register to prevent from enabling unwanted global interrupts within LPTIM_Disable() and LL_LPTIM_Disable()
      • -
      -
    • HAL UART update
    • -
        -
      • Add const qualifier for read only pointers.
      • -
      • Improve header description of UART_WaitOnFlagUntilTimeout() function.
      • -
      • Add a check on the UART parity before enabling the parity error interruption.
      • -
      • Fix typo in UART_IT_TXE bit description.
        -
      • - -
      -
    • HAL IRDA update
    • -
        -
      • Improve header description of IRDA_WaitOnFlagUntilTimeout() function.
      • -
      • Add a check on the IRDA parity before enabling the parity error interrupt.
      • -
      • Add const qualifier for read only pointers.
      • -
      -
    • HAL SMARTCARD update
    • -
        -
      • Improve header description of SMARTCARD_WaitOnFlagUntilTimeout() function
      • -
      • Add const qualifier for read only pointers.
      • -
      -
    • HAL NOR update
    • -
        -
      • Apply adequate commands according to the command set field value
      • -
      • command set 1 for Micron JS28F512P33
      • -
      • command set 2 for Micron M29W128G and Cypress S29GL128P
      • -
      • Add new command operations:
      • -
          -
        • NOR_CMD_READ_ARRAY
        • -
        • NOR_CMD_WORD_PROGRAM
        • -
        • NOR_CMD_BUFFERED_PROGRAM
        • -
        • NOR_CMD_CONFIRM
        • -
        • NOR_CMD_BLOCK_ERASE
        • -
        • NOR_CMD_BLOCK_UNLOCK
        • -
        • NOR_CMD_READ_STATUS_REG
        • -
        • NOR_CMD_CLEAR_STATUS_REG
        • -
        -
      • Update some APIs in order to be compliant for memories with different command set, the updated APIs are:
      • -
          -
        • HAL_NOR_Init()
        • -
        • HAL_NOR_Read_ID()
        • -
        • HAL_NOR_ReturnToReadMode()
        • -
        • HAL_NOR_Read()
        • -
        • HAL_NOR_Program()
        • -
        • HAL_NOR_ReadBuffer()
        • -
        • HAL_NOR_ProgramBuffer()
        • -
        • HAL_NOR_Erase_Block()
        • -
        • HAL_NOR_Erase_Chip()
        • -
        • HAL_NOR_GetStatus()
        • -
        -
      • Align HAL_NOR_Init() API with core of the function when write operation is disabled to avoid HardFault.
      • -
      -
    • HAL SDMMC update
    • -
        -
      • Take into account the voltage range in the CMD1 command.
      • -
      • Add new LL function to have correct response for MMC driver.
      • -
      • Update the driver to have all fields correctly initialized.
      • -
      • Add an internal variable to manage the power class and call it before to update speed of bus width.
      • -
      • Add new API to get the value of the Extended CSD register and populate the ExtCSD field of the MMC handle.
      • -
      • In HAL_MMC_InitCard(), call to SDIO_PowerState_ON() moved after -__HAL_MMC_ENABLE() to ensure MMC clock is enabled before the call to -HAL_Delay() from within SDIO_PowerState_ON().
      • -
      -
    • HAL DMA update
    • -
        -
      • Manage the case of an invalid value of CallbackID passed to the HAL_DMA_RegisterCallback() API.
      • -
      -
    • HAL LTDC update
    • -
        -
      • Update HAL_LTDC_DeInit() to fix MCU Hang up during LCD turn OFF.
      • -
      - -
    • HAL I2C update
    • -
        -
      • Update to fix issue detected due to low system frequency execution (HSI).
      • -
      • Declare an internal macro link to DMA macro to check remaining data: I2C_GET_DMA_REMAIN_DATA
      • -
      • Update HAL I2C Master Receive IT process to safe manage data N= 2 and N= 3.
      • -
          -
        • Disable RxNE interrupt if nothing to do.
        • -
        -
      -
    • HAL USART update
    • -
        -
      • Improve header description of USART_WaitOnFlagUntilTimeout() function.
      • -
      • Add a check on the USART parity before enabling the parity error interrupt.
      • -
      • Add const qualifier for read only pointers.
      • -
      -
    • HAL/LL ADC update
    • -
        -
      • Update LL_ADC_IsActiveFlag_MST_EOCS() API to get the appropriate flag.
      • -
      • Better performance by removing multiple volatile reads or writes in interrupt handler.
      • -
      -
    • HAL FMPI2C update
      -
    • -
        -
      • Update to handle errors in polling mode.
      • -
          -
        • Rename I2C_IsAcknowledgeFailed() to I2C_IsErrorOccurred() and correctly manage when error occurs.
        • -
        -
      -
    • HAL EXTI update
    • -
        -
      • Update HAL_EXTI_GetConfigLine() API to fix wrong calculation of GPIOSel value.
      • -
      -
    • HAL QSPI update
    • -
        -
      • Update HAL_QSPI_Abort() and  -HAL_QSPI_Abort_IT() APIs to check on QSPI BUSY flag status before -executing the abort procedure.
      • -
      - -
    • HAL/LL RTC cleanup
    • -
        -
      • Use bits definitions from CMSIS Device header file instead of hard-coded values.
      • -
      • Wrap comments to be 80-character long and correct typos.
      • -
      • Move constants RTC_IT_TAMP. from hal_rtc.h to hal_rtc_ex.h.
      • -
      • Gather all instructions related to exiting the "init" mode into new function RTC_ExitInitMode().
      • -
      • Add -new macro -assert_param(IS_RTC_TAMPER_FILTER_CONFIG_CORRECT(sTamper->Filter, -sTamper->Trigger)) to check tamper filtering is disabled in case -tamper events are triggered on signal edges.
      • -
      • Rework functions HAL_RTCEx_SetTamper() and HAL_RTCEx_SetTamper_IT() to:
      • -
          -
        • Write in TAFCR register in one single access instead of two.
        • -
        • Avoid modifying user structure sTamper.
        • -
        -
      • Remove functions LL_RTC_EnablePushPullMode() and LL_RTC_DisablePushPullMode() as related to non-supported features.
      • -
      • Remove any reference to non-supported features (e.g., LL_RTC_ISR_TAMP3F).
      • -
      • Remove -useless conditional defines as corresponding features are supported by -all part-numbers (e.g., #if defined(RTC_TAFCR_TAMPPRCH)).
      • -
      -
    • HAL USB OTG update
    • -
        - - -
      • Fix USB_FlushRxFifo() and USB_FlushTxFifo() APIs by adding check on AHB master IDLE state before flushing the USB FIFO
      • -
      • Fix to avoid resetting host channel direction during channel halt
      • -
      • Fix to report correct received amount of data with USB DMA enabled
      • -
      • Fix to avoid compiler optimization on count variable used for USB HAL timeout loop check
      • -
      • Add missing registered callbacks check for HAL_HCD_HC_NotifyURBChange_Callback()
      • -
      • Add new API HAL_PCD_SetTestMode() APIs to handle USB device high speed Test modes
      • -
      • Setting SNAK for EPs not required during device reset
      • -
      • Update USB IRQ handler to enable EP OUT disable
      • -
      • Add support of USB IN/OUT Iso incomplete
      • -
      • Fix USB BCD data contact timeout
        -
        -
      • - -
      -
    -
  • + + + + + + + Release Notes for STM32F4xx HAL Drivers + + + + + + +
    +
    +
    +

    Release Notes for STM32F4xx HAL Drivers

    +

    Copyright © 2017 STMicroelectronics
    +

    + +
    +

    Purpose

    +

    The STM32Cube HAL and LL, an STM32 abstraction layer embedded software, ensure maximized portability across STM32 portfolio.

    +

    The Portable APIs layer provides a generic, multi instanced and simple set of APIs to interact with the upper layer (application, libraries and stacks). It is composed of native and extended APIs set. It is directly built around a generic architecture and allows the build-upon layers, like the middleware layer, to implement its functions without knowing in-depth the used STM32 device. This improves the library code reusability and guarantees an easy portability on other devices and STM32 families.

    +

    The Low Layer (LL) drivers are part of the STM32Cube firmware HAL that provide basic set of optimized and one shot services. The Low layer drivers, contrary to the HAL ones are not Fully Portable across the STM32 families; the availability of some functions depend on the physical availability of the relative features on the product. The Low Layer (LL) drivers are designed to offer the following features:

    +
      +
    • New set of inline function for direct and atomic register access
    • +
    • One-shot operations that can be used by the HAL drivers or from application level.
    • +
    • Fully Independent from HAL and can be used in standalone usage (without HAL drivers)
    • +
    • Full features coverage of the all the supported peripherals.
    -

    V1.7.13 - / 16-July-2021

    -
    -

    Main - - - - - Changes
    -

    -
    -
    • -

      HAL - updates

      - -
        -
      • HAL EXTI - update -
        • Update - HAL_EXTI_GetConfigLine() - API to set default - configuration value of - Trigger and GPIOSel - before checking each - corresponding registers.
        -
      • -
      • HAL GPIO - update -
        • Update - HAL_GPIO_Init() API to - avoid the configuration - of PUPDR register when - Analog mode is selected.
        -
      • -
      • HAL DMA - update -
        • Update - HAL_DMA_IRQHandler() API - to set the DMA state - before unlocking access - to the DMA handle.
        -
      • -
      • HAL/LL ADC - update -
        • Update - LL_ADC_DeInit() API to - clear missing SQR3 - register.
        • Update - LL_ADC_DMA_GetRegAddr() - API to prevent unused - argument compilation - warning.
        • Update HAL - timeout mechanism to - avoid false timeout - detection in case of - preemption.
        -
      • -
      • HAL CAN - update -
        • Update - HAL_CAN_Init() API to be - aligned with referance - manual and to avoid - timeout error:
        -
      • -
      • HAL/LL - RTC_BKP update -
        • Update - __HAL_RTC_…(__HANDLE__, - …) macros to access - registers through - (__HANDLE__)->Instance - pointer and avoid - “unused variable” - warnings.
        • Correct month - management in - IS_LL_RTC_MONTH() macro.
        -
      • -
      • HAL RNG - update -
        • Update timeout - mechanism to avoid false - timeout detection in - case of preemption.
        -
      • -
      • HAL QSPI - update -
        • ES0305 - workaround disabled for - STM32412xx devices.
        -
      • -
      • HAL I2C - update -
        • Update - HAL_I2C_Mem_Write_DMA() - and - HAL_I2C_Mem_Read_DMA() - APIs to initialize - Devaddress, Memaddress - and EventCount - parameters.
        • Update to - prevent several calls of - Start bit: -
          • Update - I2C_MemoryTransmit_TXE_BTF() - API to increment - EventCount.
          -
        • Update to - avoid I2C interrupt in - endless loop: -
          • Update - HAL_I2C_Master_Transmit_IT(), - HAL_I2C_Master_Receive_IT(), - - HAL_I2C_Master_Transmit_DMA() - - and - HAL_I2C_Master_Receive_DMA() - APIs to unlock the - I2C peripheral - before generating - the start.
          -
        • Update to use - the right macro to clear - I2C ADDR flag inside - I2C_Slave_ADDR() API as - it’s indicated in the - reference manual.
        • Update - I2C_IsAcknowledgeFailed() - API to avoid I2C in busy - state if NACK received - after transmitting - register address.
        • Update - HAL_I2C_EV_IRQHandler() - and - I2C_MasterTransmit_BTF() - APIs to correctly manage - memory transfers: -
          • Add check - on memory mode - before calling - callbacks - procedures.
          -
        -
      • -
      • LL USART - update -
        • Handling of - UART concurrent register - access in case of race - condition between Tx and - Rx transfers (HAL UART - and LL LPUART)
        -
      • -
      • HAL SMBUS - update -
        • Updated - HAL_SMBUS_ER_IRQHandler() - API to return the - correct error code - “SMBUS_FLAG_PECERR” in - case of packet error - occurs.
        -
      • -
      • HAL/LL SPI - update -
        • Updated to fix - MISRA-C 2012 Rule-13.2.
        • Update - LL_SPI_TransmitData8() - API to avoid casting the - result to 8 bits.
        -
      • -
      • HAL UART - update -
        • Fix wrong - comment related to RX - pin configuration within - the description section
        • Correction on - UART ReceptionType - management in case of - ReceptionToIdle API are - called from RxEvent - callback
        • Handling of - UART concurrent register - access in case of race - condition between Tx and - Rx transfers (HAL UART - and LL LPUART) -
          • Update CAN - Initialization - sequence to set - "request - initialization" bit - before exit from - sleep mode.
          -
        -
      • -
      • HAL USB - update -
        • HAL PCD: add - fix transfer complete - for IN Interrupt - transaction in single - buffer mode
        • Race condition - in USB PCD control - endpoint receive ISR.
        -
      -

    V1.7.12 - / 26-March-2021

    -

    Main - - - - - Changes

    -
      -
    • HAL
    • -
        -
      • HAL/LL - USART update
      • -
          -
        • Fix typo in - USART_Receive_IT() and - USART_TransmitReceive_IT() - APIs to avoid possible - compilation issues if the - UART driver files are not - included.
        • -
        -
      -
    -

    V1.7.11 - / 12-February-2021

    -

    Main - - - - - Changes

    -
      -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • Added - - - - new HAL - - - - FMPSMBUS extended driver - - - - - to support FMPSMBUS fast Mode - Plus.
    • -
    • Removed - - - - “register” keyword to be - compliant with new C++ rules:
    • -
        -
      • The register - storage class specifier was - deprecated in C++11 and - removed in C++17.
      • -
      -
    • HAL
    • -
        -
      • HAL update
      • -
      • General - updates to fix known defects - and enhancements - implementation.
      • -
      • Added new - defines for ARM compiler V6:
      • -
          -
        • __weak
        • -
        • __packed
        • -
        • __NOINLINE
        • -
        -
      • Updated HAL TimeBase - TIM, RTC alarm and RTC WakeUp - templates for more robustness
      • -
          -
        • Updated Hal_Init_Tick() - API to propoerty - store the priority when - using the non-default time - base.
        • -
        -
      • Updated - PPP_MODULE_ENABLED for - FMPSMBUS.
      • -
      • HAL/LL ADC update
      • -
          -
        • Updated to - add include of the LL ADC - driver.
        • -
        • Updated the - following APIs to set status - HAL_ADC_STATE_ERROR_INTERNAL - and error code - HAL_ADC_ERROR_INTERNAL when - error occurs:
        • -
            -
          • HAL_ADC_Start()
          • -
          • HAL_ADC_Start_IT()
          • -
          • HAL_ADC_Start_DMA()
          • -
          • HAL_ADCEx_InjectedStart()
          • -
          • HAL_ADCEx_InjectedStart_IT()
          • -
          • HAL_ADCEx_MultiModeStart_DMA()
          • -
          -
        • Updated HAL_ADC_Stop_DMA() - API to check if DMA state is - Busy before calling HAL_DMA_Abort() - - - - - API to avoid DMA internal - error.
        • -
        • Updated - IS_ADC_CHANNEL to support - temperature sensor for:
        • -
            -
          • STM32F411xE
          • -
          • STM32F413xx
          • -
          • STM32F423xx
          • -
          -
        • Fixed wrong - defined values for:
        • -
            -
          • LL_ADC_MULTI_REG_DMA_LIMIT_3
          • -
          • LL_ADC_MULTI_REG_DMA_UNLMT_3
          • -
          -
        • Added - __LL_ADC_CALC_VREFANALOG_VOLTAGE() - macro to evaluate analog - reference voltage.
        • -
        • Removed - __LL_ADC_CALC_TEMPERATURE() - macro for STM32F4x9 devices - as the TS_CAL2 is not - available.
        • -
        -
      • HAL/LL DAC update
      • -
          -
        • Added restruction - on DAC Channel 2 defines and - parametres.
        • -
        • HAL_DAC_MSPINIT_CB_ID - - - - - and HAL_DAC_MSPDEINIT_CB_ID - used instead of - HAL_DAC_MSP_INIT_CB_ID and - HAL_DAC_MSP_DEINIT_CB_ID.
        • -
        • Updated to - support dual mode:
        • -
            -
          • Added two - new APIs:
          • -
              -
            • HAL_DACEx_DualStart()
            • -
            • HAL_DACEx_DualStop()
            • -
            -
          -
        • Added - position bit definition to - be used instead of - __DAC_MASK_SHIFT macro
        • -
            -
          • __DAC_MASK_SHIFT - - - - macro has been removed.
          • -
          -
        • Updated HAL_DAC_Start_DMA() - API to return HAL_ERROR when - error occurs.
        • -
        • Updated HAL_DAC_Stop_DMA() - API to not return HAL_ERROR - when DAC is already - disabled.
        • -
        -
      • HAL CEC update
      • -
          -
        • Updated HAL_CEC_IRQHandler() - API to avoid appending an - extra byte to the end of a - message.
        • -
        -
      • HAL/LL GPIO update
      • -
          -
        • Updated - IS_GPIO_AF() to - add missing values for - STM32F401xC and STM32F401xE - devices:
        • -
            -
          • GPIO_AF3_TIM9
          • -
          • GPIO_AF3_TIM10
          • -
          • GPIO_AF3_TIM11
          • -
          -
        • Updated - LL/HAL GPIO_TogglePin() - APIs to allow multi Pin’s - toggling.
        • -
        • Updated HAL_GPIO_Init() - API to avoid the - configuration of PUPDR - register when Analog mode is - selected.
        • -
        -
      • HAL/LL RCC update
      • -
          -
        • Updated HAL_RCC_OscConfig() - API to add missing checks - and to don’t return - HAL_ERROR if request repeats - the current PLL - configuration.
        • -
        • Updated - IS_RCC_PLLN_VALUE(VALUE) - macro in case of STM32F411xE - device in order to - be aligned with reference - manual.
        • -
        -
      • HAL SD update
      • -
          -
        • Update - function SD_FindSCR() - to resolve issue of FIFO - blocking when reading.
        • -
        • Update - read/write functions in DMA - mode in - - - - order to - force the DMA direction, - updated functions:
        • -
            -
          • HAL_SD_ReadBlocks_DMA()
          • -
          • HAL_SD_WriteBlocks_DMA()
          • -
          -
        • Add the - block size settings in the - initialization functions and - remove it from read/write - transactions to avoid - repeated and inefficient - reconfiguration, updated - functions:
        • -
            -
          • HAL_SD_InitCard()
          • -
          • HAL_SD_GetCardStatus()
          • -
          • HAL_SD_ConfigWideBusOperation(
          • -
          • HAL_SD_ReadBlocks()
          • -
          • HAL_SD_WriteBlocks()
          • -
          • HAL_SD_ReadBlocks_IT()
          • -
          • HAL_SD_WriteBlocks_IT()
          • -
          • HAL_SD_ReadBlocks_DMA()
          • -
          • HAL_SD_WriteBlocks_DMA()
          • -
          -
        -
      • HAL MMC update
      • -
          -
        • Add the - block size settings in the - initialization function and - remove it from read/write - transactions to avoid - repeated and inefficient - reconfiguration, updated - functions:
        • -
            -
          • HAL_MMC_InitCard()
          • -
          • HAL_MMC_ReadBlocks()
          • -
          • HAL_MMC_WriteBlocks()
          • -
          • HAL_MMC_ReadBlocks_IT()
          • -
          • HAL_MMC_WriteBlocks_IT()
          • -
          • HAL_MMC_ReadBlocks_DMA()
          • -
          • HAL_MMC_WriteBlocks_DMA()
          • -
          -
        • Update - read/write functions in DMA - mode in - - - - order to - force the DMA direction, - updated functions:
        • -
            -
          • HAL_MMC_ReadBlocks_DMA()
          • -
          • HAL_MMC_WriteBlocks_DMA()
          • -
          -
        • Deploy new - functions MMC_ReadExtCSD() - and SDMMC_CmdSendEXTCSD - () that read and check the - sectors number of the - device in order to resolve - the issue of wrongly reading - big memory size.
        • -
        -
      • HAL NAND - update
      • -
          -
        • Update - functions - HAL_NAND_Read_SpareArea_16b() - and - HAL_NAND_Write_SpareArea_16b() - to fix column address - calculation issue.
        • -
        -
      • LL SDMMC - update
      • -
          -
        • Update the - definition of - SDMMC_DATATIMEOUT constant in - - - - order to - allow the user to redefine - it in his proper - application.
        • -
        • Remove - 'register' storage class - specifier from LL SDMMC - driver.
        • -
        • Deploy new - functions MMC_ReadExtCSD() - and SDMMC_CmdSendEXTCSD - () that read and check the - sectors number of the device - in order to resolve the - issue of wrongly reading big - memory size.
        • -
        -
      • HAL SMBUS update
      • -
          -
        • Support for - Fast Mode Plus to be SMBUS - rev 3 compliant.
        • -
        • Added HAL_FMPSMBUSEx_EnableFastModePlus() - and HAL_FMPSMBUSEx_DisableFastModePlus() - - - - - APIs to manage Fm+.
        • -
        • Updated SMBUS_MasterTransmit_BTF() - , SMBUS_MasterTransmit_TXE() - - - - - and SMBUS_MasterReceive_BTF() - - - - - APIs to allow stop - generation when CurrentXferOptions - is different from - SMBUS_FIRST_FRAME and - SMBUS_NEXT_FRAME.
        • -
        • Updated SMBUS_ITError() - API to correct the twice - call of HAL_SMBUS_ErrorCallback.
        • -
        -
      • HAL SPI update
      • -
          -
        • Updated HAL_SPI_Init() - API
        • -
            -
          • To avoid - setting the BaudRatePrescaler - in case of Slave Motorola - Mode.
          • -
          • Use the bit-mask - for SPI configuration.
          • -
          -
        • Updated - Transmit/Receive processes - in half-duplex mode
        • -
            -
          • Disable - the SPI instance before - setting BDIOE bit.
          • -
          -
        • Fixed wrong - timeout management
        • -
        • Calculate - Timeout based on a software - loop to avoid blocking issue - if Systick - is disabled.
        • -
        -
      • HAL - SPDIFRX update
      • -
          -
        • Remove - 'register' storage class - specifier from HAL SPDIFRX - driver.
        • -
        -
      • HAL I2S update
      • -
          -
        • Updated - I2SEx APIs to correctly - support circular transfers
        • -
            -
          • Updated - I2SEx_TxRxDMACplt() - API to manage DMA circular - mode.
          • -
          -
        • Updated - HAL_I2SEx_TransmitReceive_DMA() - API to set hdmatx - (transfert - callback and half) to NULL.
        • -
        -
      • HAL SAI update
      • -
          -
        • Updated to - avoid the incorrect - left/right synchronization.
        • -
            -
          • Updated HAL_SAI_Transmit_DMA() - API to follow the sequence - described in the reference - manual for slave - transmitter mode.
          • -
          -
        • Updated HAL_SAI_Init() - API to correct the formula - in case of SPDIF is wrong.
        • -
        -
      • HAL CRYP update
      • -
          -
        • Updated HAL_CRYP_SetConfig() - and HAL_CRYP_GetConfig() - - - - - APIs to set/get the - continent of KeyIVConfigSkip - correctly.
        • -
        -
      • HAL EXTI update
      • -
          -
        • __EXTI_LINE__ - - - - is now used instead of - __LINE__ which is a standard - C macro.
        • -
        -
      • HAL DCMI
      • -
          -
        • Support of - HAL callback registration - feature for DCMI extended - driver.
        • -
        -
      • HAL/LL TIM update
      • -
          -
        • Updated HAL_TIMEx_OnePulseN_Start() - and HAL_TIMEx_OnePulseN_Stop() - - - - - APIs (pooling and IT mode) - to take into consideration - all OutputChannel - parameters.
        • -
        • Corrected - reversed description of - TIM_LL_EC_ONEPULSEMODE One - Pulse Mode.
        • -
        • Updated LL_TIM_GetCounterMode() - API to return the correct - counter mode.
        • -
        -
      • HAL/LL - SMARTCARD update
      • -
          -
        • Fixed - invalid initialization of - SMARTCARD configuration by - removing FIFO mode - configuration as it is not - member of SMARTCARD_InitTypeDef - Structure.
        • -
        • Fixed typos - in SMARTCARD State - definition description
        • -
        -
      • HAL/LL IRDA update
      • -
          -
        • Fixed typos - in IRDA State definition - description
        • -
        -
      • LL USART update
      • -
          -
        • Remove - useless check on maximum BRR - value by removing - IS_LL_USART_BRR_MAX() - macro.
        • -
        • Update - USART polling and - interruption processes to - fix issues related to - accesses out of user - specified buffer.
        • -
        -
      • HAL USB update
      • -
          -
        • Enhanced - USB OTG host HAL with USB - DMA is enabled:
        • -
            -
          • fixed - ping and data toggle - issue,
          • -
          • reworked - Channel error report - management
          • -
          -
        -
      -
    -

    V1.7.10 - - - - / 22-October-2020

    -

    Main Changes

    -
      -
    • General - - - - updates to fix known defects.
    • -
    • HAL/LL - - - - I2C update
    • -
    -
      -
        -
      •   Update - to fix hardfault - issue with HAL_I2C_Mem_Write_DMA() - API:
      • -
          -
        •   - Abort the right ongoing DMA - transfer when memory write - access request operation - failed: fix typo “hdmarx” - replaced by “hdmatx”
        • -
        -
      -
    -

    V1.7.9 - - - - / 14-August-2020

    -

    Main Changes

    -
      -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • HAL/LL - - - - I2C update
    • -
    -
      -
        -
      •   Update - HAL_I2C_ER_IRQHandler() - API to fix acknowledge failure - issue with I2C memory IT - processes
      • -
          -
        •   Add - stop condition generation - when NACK occurs.
        • -
        -
      •   Update - I2C_DMAXferCplt(), - - - - - I2C_DMAError() and - I2C_DMAAbort() APIs to fix hardfault - issue when hdmatx - and hdmarx - parameters in i2c handle - aren't initialized (NULL - pointer).
      • -
          -
        •   Add - additional check on - hi2c->hdmtx - and hi2c->hdmarx - before resetting DMA Tx/Rx - complete callbacks
        • -
        -
      •   Update - Sequential transfer APIs to - adjust xfermode - condition.
      • -
          -
        •   - - - - - Replace hi2c->XferCount - < MAX_NBYTE_SIZE by - hi2c->XferCount - <= MAX_NBYTE_SIZE which - corresponds to a case - without reload
        • -
        -
      -
    -
      -
    •  HAL/LL - - - - USB update
    • -
        -
      •   Bug - - - - fix: USB_ReadPMA() - and USB_WritePMA() - - - - - by ensuring 16-bits access to - USB PMA memory
      • -
      •   Bug - - - - fix: correct USB RX count - calculation
      • -
      •   Fix - USB Bulk transfer double - buffer mode
      • -
      •   Remove - register keyword from USB - defined macros as no more - supported by C++ compiler
      • -
      •   Minor - rework on USBD_Start() - and USBD_Stop() - - - - - APIs: stopping device will be - handled by HAL_PCD_DeInit() - - - - - API.
      • -
      •   Remove - non used API for USB device - mode.
      • -
      -
    -

    V1.7.8 - - - - / 12-February-2020

    -

    Main Changes

    -
      -
    • Add - new HAL FMPSMBUS and LL - - - - FMPI2C drivers
    • -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    -
      -
    • Update - - - - HAL CRYP driver to support block - by block decryption without - reinitializes the IV and KEY for - each call.
    • -
    • Improve - - - - code quality by fixing - MisraC-2012 violations
    • -
    • HAL/LL - - - - USB update
    • -
        -
      •  Add - handling USB host babble error - interrupt
      • -
      •  Fix - Enabling ULPI interface for - platforms that integrates USB - HS PHY
      • -
      •  Fix - Host data toggling for IN Iso - transfers
      • -
      •  Ensure - to disable USB EP during - endpoint deactivation
      • -
      -
    • HAL - - - - CRYP update
    • -
        -
      •  Update - HAL CRYP driver to support - block by block decryption - without initializing the IV - and KEY at each call.
      • -
          -
        • Add new - CRYP Handler parameters: "KeyIVConfig" - and "SizesSum"
        • -
        • Add new - CRYP init - parameter: "KeyIVConfigSkip"
        • -
        -
      -
    • HAL - - - - I2S update
    • -
        -
      • Update - HAL_I2S_DMAStop() - API to be more safe
      • -
          -
        • Add a check - on BSY, TXE and RXNE flags - before disabling the I2S
        • -
        -
      • Update - HAL_I2S_DMAStop() - API to fix multi-call transfer - issue(to avoid re-initializing - the I2S for the next - transfer).
      • -
          -
        • Add - __HAL_I2SEXT_FLUSH_RX_DR() - and __HAL_I2S_FLUSH_RX_DR() - macros to flush the - remaining data inside DR - registers.
        • -
        • Add new ErrorCode - define: - HAL_I2S_ERROR_BUSY_LINE_RX
        • -
        -
      -
    -

    V1.7.7 - - - - / 06-December-2019

    -

    Main Changes

    -
      -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • HAL - - - - Generic update
    • -
        -
      • HAL_SetTickFreq(): update to - restore the previous tick - frequency when HAL_InitTick() - - - - - configuration failed.
      • -
      -
    • HAL/LL - - - - GPIO update
    • -
        -
      • Update GPIO - initialization sequence to - - - - avoid unwanted pulse on GPIO Pin's
      • -
      -
    • HAL - - - - EXTI update
    • -
    -
      -
        -
      • General - update to enhance HAL EXTI - driver robustness 
      • -
          -
        • Add - additional assert check on - EXTI config lines
        • -
        • Update to - compute EXTI line mask - before read/write access - to EXTI registers
          -
          -
        • -
        -
      • Update EXTI - callbacks management to be - compliant with reference - manual: only one PR - register for rising and - falling interrupts.
      • -
          -
        • Update - parameters in EXTI_HandleTypeDef - structure: merge HAL - EXTI RisingCallback - and FallingCallback - in only one PendingCallback
        • -
        • Remove - HAL_EXTI_RISING_CB_ID and - HAL_EXTI_FALLING_CB_ID - values from EXTI_CallbackIDTypeDef - enumeration.
          -
          -
        • -
        -
      • Update - HAL_EXTI_IRQHandler() - API to serve interrupts - correctly.
      • -
          -
        • Update to - compute EXTI line mask - before handle - EXTI interrupt.
        • -
        -
      • Update to - support GPIO port - interrupts:
      • -
          -
        • Add new "GPIOSel" - parameter in EXTI_ConfigTypeDef - structure
        • -
        -
      -
    • HAL/LL - - - - RCC update
    • -
        -
      • Update HAL_RCCEx_PeriphCLKConfig() - API to support PLLI2S - configuration for STM32F42xxx - and STM32F43xxx devices
      • -
      • Update the HAL_RCC_ClockConfig() - and HAL_RCC_DeInit() - - - - - API to don't overwrite the - custom tick priority
      • -
      • Fix LL_RCC_DeInit() - failure detected with gcc - compiler and high optimization - level is selected(-03)
      • -
      • Update HAL_RCC_OscConfig() - API to don't return - HAL_ERROR if request repeats - the current PLL configuration
      • -
      -
    • HAL - - - - ADC update
    • -
        -
      • Update LL_ADC_REG_Init() - to fix wrong ADC CR1 register - configuration
      • -
          -
        • The ADC - sequencer length is part - of ADC SQR1 - register not of ADC CR1 - register
        • -
        -
      -
    • HAL - - - - CRYP update
    • -
        -
      • Update HAL_CRYP_Encrypt() - and HAL_CRYP_Decrypt() - - - - - APIs to take into - consideration the datatype fed - to the DIN register (1-, 8-, - 16-, or 32-bit data) when - padding the last block of the - payload, in case the size of - this last block is less than - 128 bits.
      • -
      -
    • HAL - - - - RNG update
    • -
        -
      • Update HAL_RNG_IRQHandler() - API to fix error code - management issue: error code - is assigned - "HAL_RNG_ERROR_CLOCK" in case - of clock error and - "HAL_RNG_ERROR_SEED" in case - of seed error, not the - opposite.
      • -
      -
    • HAL - - - - DFSDM update
    • -
        -
      • Update DFSDM_GetChannelFromInstance() - API to remove unreachable - check condition
      • -
      -
    • HAL - - - - DMA update
    • -
        -
      • Update HAL_DMA_Start_IT() - API to omit the FIFO error
      • -
      -
    • HAL - - - - FLASH update
    • -
        -
      • Update FLASH_Program_DoubleWord() - API to fix with EWARM high - level optimization issue
      • -
      -
    • HAL - - - - QSPI update
    • -
        -
      • Remove Lock - mechanism from HAL_QSPI_Init() - and HAL_QSPI_DeInit() - - - - - APIs
      • -
      -
    • HAL - - - - HASH update
    • -
        -
      • Null pointer - on handler "hhash" - is now checked before - accessing structure member "hhash->Init.DataType" - in the following API:
      • -
          -
        • HAL_HASH_Init()
        • -
        -
      • Following interrupt-based - APIs have been added. - Interrupt mode could allow the - MCU to enter "Sleep" mode - while a data block is being - processed. Please refer to the - "##### How to use this driver - #####" section for details - about their use.
      • -
          -
        • HAL_HASH_SHA1_Accmlt_IT()
        • -
        • HAL_HASH_MD5_Accmlt_IT()
        • -
        • HAL_HASHEx_SHA224_Accmlt_IT()
        • -
        • HAL_HASHEx_SHA256_Accmlt_IT()
        • -
        -
      • Following aliases - have been added (just for - clarity sake) as they - shall be used at the end - of the computation of a - multi-buffers message and not - at the start:
      • -
          -
        • HAL_HASH_SHA1_Accmlt_End() - to be used instead of - HAL_HASH_SHA1_Start()
        • -
        • HAL_HASH_MD5_Accmlt_End() - to be used instead of - HAL_HASH_MD5_Start()
        • -
        • HAL_HASH_SHA1_Accmlt_End_IT() - to be used instead of - HAL_HASH_SHA1_Start_IT()
        • -
        • HAL_HASH_MD5_Accmlt_End_IT() - to be used instead of - HAL_HASH_MD5_Start_IT()
        • -
        • HAL_HASHEx_SHA224_Accmlt_End() - to be used instead of - HAL_HASHEx_SHA224_Start()
        • -
        • HAL_HASHEx_SHA256_Accmlt_End() - to be used instead of - HAL_HASHEx_SHA256_Start()
        • -
        -
      -
    -
      -
        -
          -
        • HAL_HASHEx_SHA224_Accmlt_End_IT() - to be used instead of - HAL_HASHEx_SHA224_Start_IT()
        • -
        • HAL_HASHEx_SHA256_Accmlt_End_IT() - to be used instead of - HAL_HASHEx_SHA256_Start_IT()
        • -
        -
      -
    -
      -
        -
      • MISRAC-2012 - rule R.5.1 (identifiers - shall be distinct in the first - 31 characters) constrained the - naming of the above listed - aliases (e.g. - HAL_HASHEx_SHA256_Accmlt_End() - could not be named - HAL_HASHEx_SHA256_Accumulate_End(). - - - - - Otherwise the name would have - conflicted with - HAL_HASHEx_SHA256_Accumulate_End_IT()). - In - - - - - order to - have aligned names following - APIs have been renamed:
      • -
      -
    -
      -
        -
          -
            -
          • HAL_HASH_MD5_Accumulate() - renamed - HAL_HASH_MD5_Accmlt()
          • -
          • HAL_HASH_SHA1_Accumulate() - renamed - HAL_HASH_SHA1_Accmlt()
          • -
          • HAL_HASHEx_SHA224_Accumulate() - renamed - HAL_HASHEx_SHA224_Accmlt()
          • -
          -
        -
      -
    -
      -
        -
          -
            -
          • HAL_HASHEx_SHA256_Accumulate() - renamed - HAL_HASHEx_SHA256_Accmlt()
          • -
          -
        -
      -
    -
      -
        -
      • HASH handler - state is no more - reset to HAL_HASH_STATE_READY - once DMA has been started - in the following APIs:
      • -
          -
        • HAL_HASH_MD5_Start_DMA()
        • -
        • HAL_HMAC_MD5_Start_DMA()
        • -
        • HAL_HASH_SHA1_Start_DMA()
        • -
        • HAL_HMAC_SHA1_Start_DMA()
        • -
        -
      • HASH phase - state is now set to - HAL_HASH_PHASE_READY once - the digest has been read - in the following APIs:
      • -
          -
        • HASH_IT()
        • -
        • HMAC_Processing()
        • -
        • HASH_Start()
        • -
        • HASH_Finish()
        • -
        -
      • Case of a - large buffer scattered around - in memory each piece of which - is not necessarily a multiple - - - - of 4 bytes in length.
      • -
          -
        • In section - "##### How to use this - driver #####", sub-section - "*** Remarks on message - length ***" added to provide - recommendations to follow in - such case.
        • -
        • No - modification of the driver - as the root-cause is at - design-level.
        • -
        -
      -
    -
      -
    • HAL CAN - - - - update
    • -
        -
      • HAL_CAN_GetRxMessage() update to - get the correct value for the - RTR (type of frame for - the message that will be - transmitted) field in the CAN_RxHeaderTypeDef - structure.
      • -
      -
    • HAL - - - - DCMI update
    • -
        -
      • Add new HAL_DCMI_ConfigSyncUnmask() - API to set embedded - synchronization delimiters - unmasks.
      • -
      -
    • HAL - - - - RTC update
    • -
        -
      • Following IRQ - handlers' implementation has - been aligned with the - STM32Cube firmware - specification (in case of - interrupt lines shared by - multiple events, first check - the IT enable bit is set then - check the IT flag is set too):
      • -
          -
        • HAL_RTC_AlarmIRQHandler()
        • -
        • HAL_RTCEx_WakeUpTimerIRQHandler()
        • -
        • HAL_RTCEx_TamperTimeStampIRQHandler()
        • -
        -
      -
    -
      -
    • HAL - - - - WWDG update
    • -
        -
      • In "##### - WWDG Specific features #####" - descriptive comment section:
      • -
          -
        • Maximal prescaler - value has been corrected (8 - instead of 128).
        • -
        • Maximal APB - frequency has been corrected - (42MHz instead of 56MHz) and - possible timeout values - updated.
        • -
        -
      -
    • HAL - - - - DMA2D update
    • -
    -
      -
        -
      • Add the - following API's to Start DMA2D - CLUT Loading.
      • -
          -
        • HAL_DMA2D_CLUTStartLoad() - Start DMA2D CLUT Loading.
        • -
        • HAL_DMA2D_CLUTStartLoad_IT() - Start DMA2D CLUT Loading - with interrupt enabled.
        • -
        -
      • The following - old wrong services will be - kept in the HAL DCMI driver - for legacy purpose and a - specific Note is added:
      • -
          -
        • HAL_DMA2D_CLUTLoad() - can be replaced with - HAL_DMA2D_CLUTStartLoad()
        • -
        • HAL_DMA2D_CLUTLoad_IT() can - - - - - be replaced with - HAL_DMA2D_CLUTStartLoad_IT()
        • -
        • HAL_DMA2D_ConfigCLUT() - can be omitted as the config - can be performed using - the HAL_DMA2D_CLUTStartLoad() - API.
        • -
        -
      -
    -
      -
    • HAL - - - - SDMMC update
    • -
        -
      • Fix  - typo in "FileFormatGroup" - parameter in the HAL_MMC_CardCSDTypeDef - and HAL_SD_CardCSDTypeDef - structures 
      • -
      • Fix an - improve handle state and - error management
      • -
      • Rename the - defined MMC card capacity type - to be more meaningful:
      • -
          -
        • Update MMC_HIGH_VOLTAGE_CARD to - - - - - MMC LOW_CAPACITY_CARD
        • -
        • Update MMC_DUAL_VOLTAGE_CRAD - to MMC_HIGH_CAPACITY_CARD
        • -
        -
      • Fix - management of peripheral - flags depending on commands - or data transfers
      • -
          -
        • Add new - defines - "SDIO_STATIC_CMD_FLAGS" - and "SDIO_STATIC_DATA_FLAGS" 
        • -
        • Updates HAL - - - - SD and HAL MMC drivers to - manage the new SDIO static - flags.
          -
          -
        • -
        -
      • Due to - limitation SDIO hardware flow - control indicated in Errata - Sheet:
      • -
          -
        • In 4-bits - bus wide mode, do not use - the HAL_SD_WriteBlocks_IT() - or HAL_SD_WriteBlocks() - - - - - APIs otherwise underrun will - occur and it isn't possible - to activate the flow - control.
        • -
        • Use DMA - mode when using 4-bits bus - wide mode or decrease the - SDIO_CK frequency.
        • -
        -
      -
    • HAL - - - - UART update
    • -
        -
      • Update UART - polling processes to handle - efficiently the Lock mechanism
      • -
          -
        •  Move - the process unlock at the - top of the HAL_UART_Receive() - and HAL_UART_Transmit() - - - - - API.
        • -
        -
      • Fix baudrate - calculation error for clock - higher than 172Mhz
      • -
          -
        • Add a - forced cast on - UART_DIV_SAMPLING8() and - UART_DIV_SAMPLING16() - macros.
        • -
        • Remove - useless parenthesis from - UART_DIVFRAQ_SAMPLING8(), - UART_DIVFRAQ_SAMPLING16(), - UART_BRR_SAMPLING8() and - UART_BRR_SAMPLING16() macros - to solve some MISRA - warnings.
        • -
        -
      • Update UART - interruption handler to manage - correctly the overrun interrupt
      • -
          -
        • Add in - the HAL_UART_IRQHandler() - API a check on - USART_CR1_RXNEIE bit when an - overrun interrupt occurs.
        • -
        -
      • Fix baudrate - calculation error UART9 - and UART10
      • -
          -
        • In UART_SetConfig() - API fix UART9 and UART10 - clock source when computing - baudrate - values by adding a check on - these instances and setting - clock sourcePCLK2 instead of - PCLK1.
        • -
        -
      • Update UART_SetConfig() - API
      • -
          -
        • Split - HAL_RCC_GetPCLK1Freq() - and HAL_RCC_GetPCLK2Freq() - macros from the - UART_BRR_SAMPLING8() and - UART_BRR_SAMPLING8() - macros 
        • -
        -
      -
    • HAL - - - - USART update
    • -
        -
      • Fix baudrate - calculation error for clock - higher than 172Mhz
      • -
          -
        • Add a - forced cast on USART_DIV() - macro.
        • -
        • Remove - useless parenthesis - from USART_DIVFRAQ() - macro to solve some MISRA - warnings.
        • -
        -
      • Update USART - interruption handler to manage - correctly the overrun interrupt
      • -
          -
        • Add in - the HAL_USART_IRQHandler() - API a check on - USART_CR1_RXNEIE bit when an - overrun interrupt occurs.
        • -
        -
      • Fix baudrate - calculation error UART9 - and UART10
      • -
          -
        • In USART_SetConfig() - API fix UART9 and UART10 - clock source when computing - baudrate - values by adding a check on - these instances and setting - clock sourcePCLK2 instead of - PCLK1.
        • -
        -
      • Update USART_SetConfig() - API
      • -
          -
        • Split - HAL_RCC_GetPCLK1Freq() - and HAL_RCC_GetPCLK2Freq() - macros from the USART_BRR() - macro
        • -
        -
      -
    • HAL - - - - IRDA update
    • -
        -
      • Fix baudrate - calculation error for clock - higher than 172Mhz
      • -
          -
        • Add a - forced cast on IRDA_DIV() - macro.
        • -
        • Remove - useless parenthesis - from IRDA_DIVFRAQ() - macro to solve some - MISRA warnings.
        • -
        -
      • Update IRDA - interruption handler to manage - correctly the overrun interrupt
      • -
          -
        • Add in - the HAL_IRDA_IRQHandler() - API a check on - USART_CR1_RXNEIE bit when an - overrun interrupt occurs.
        • -
        -
      • Fix baudrate - calculation error UART9 - and UART10
      • -
          -
        • In IRDA_SetConfig() - API fix UART9 and UART10 - clock source when computing - baudrate - values by adding a check on - these instances and setting - clock sourcePCLK2 instead of - PCLK1.
        • -
        -
      • Update IRDA_SetConfig() - API
      • -
          -
        • Split - HAL_RCC_GetPCLK1Freq() - and HAL_RCC_GetPCLK2Freq() - macros from the IRDA_BRR() - macro
        • -
        -
      -
    • HAL - - - - SMARTCARD update
    • -
        -
      • Fix baudrate - calculation error for clock - higher than 172Mhz
      • -
          -
        • Add a - forced cast on SMARTCARD_DIV() - macro.
        • -
        • Remove useless parenthesis - - - - - from SMARTCARD_DIVFRAQ() - macro to solve some - MISRA warnings.
        • -
        -
      • Update - SMARTCARD interruption handler - to manage correctly the - overrun interrupti
      • -
          -
        • Add in - the HAL_SMARTCARD_IRQHandler() - API a check on - USART_CR1_RXNEIE bit when an - overrun interrupt occurs.
        • -
        -
      • Update SMARTCARD_SetConfig() - API
      • -
          -
        • Split - HAL_RCC_GetPCLK1Freq() - and HAL_RCC_GetPCLK2Freq() - macros from the - SMARTCARD_BRR() macro
        • -
        -
      -
    • HAL - - - - TIM update
    • -
        -
      • Add new - macros to enable and disable - the fast mode when using the - one pulse mode to output a - waveform with a minimum delay
      • -
          -
        • __HAL_TIM_ENABLE_OCxFAST() - and __HAL_TIM_DISABLE_OCxFAST().
        • -
        -
      • Update - Encoder interface mode to - keep TIM_CCER_CCxNP - bits low
      • -
          -
        • Add TIM_ENCODERINPUTPOLARITY_RISING - - - - - and - TIM_ENCODERINPUTPOLARITY_FALLING - definitions to determine - encoder input polarity.
        • -
        • Add - IS_TIM_ENCODERINPUT_POLARITY() - macro to check the - encoder input polarity.
        • -
        • Update HAL_TIM_Encoder_Init() - API 
        • -
            -
          • Replace - IS_TIM_IC_POLARITY() - macro by - IS_TIM_ENCODERINPUT_POLARITY() - macro.
          • -
          -
        -
      • Update TIM - remapping input configuration - in HAL_TIMEx_RemapConfig() - API
      • -
          -
        • Remove - redundant check on - LPTIM_OR_TIM5_ITR1_RMP bit - and replace it by check on - LPTIM_OR_TIM9_ITR1_RMP bit.
        • -
        -
      • Update HAL_TIMEx_MasterConfigSynchronization() - API to avoid functional errors - and assert fails when using - some TIM instances as input - trigger.
      • -
          -
        • Replace IS_TIM_SYNCHRO_INSTANCE() - macro by - IS_TIM_MASTER_INSTANCE() - macro. 
        • -
        • Add IS_TIM_SLAVE_INSTANCE() - macro to check on - TIM_SMCR_MSM bit.
        • -
        -
      • Add lacking - TIM input remapping definition 
      • -
          -
        • Add - LL_TIM_TIM11_TI1_RMP_SPDIFRX - and - LL_TIM_TIM2_ITR1_RMP_ETH_PTP.
        • -
        • Add lacking - definition for linked - LPTIM_TIM input trigger remapping  
        • -
            -
          • Add - following definitions - - - - - : - LL_TIM_TIM9_ITR1_RMP_TIM3_TRGO, - LL_TIM_TIM9_ITR1_RMP_LPTIM, - - - - LL_TIM_TIM5_ITR1_RMP_TIM3_TRGO, - - - - - LL_TIM_TIM5_ITR1_RMP_LPTIM, - - - - LL_TIM_TIM1_ITR2_RMP_TIM3_TRGO - - - - and - LL_TIM_TIM1_ITR2_RMP_LPTIM.
          • -
          • Add a new - mechanism in LL_TIM_SetRemap() - API to remap TIM1, TIM9, - and TIM5 input - triggers mapped on LPTIM - register. 
          • -
          -
        -
      -
    • HAL - - - - LPTIM update
    • -
        -
      • Add a polling - mechanism to check - on LPTIM_FLAG_XXOK flags - in different API 
      • -
          -
        • Add  - LPTIM_WaitForFlag() API to - wait for flag set.
        • -
        • Perform new - checks on - HAL_LPTIM_STATE_TIMEOUT.
        • -
        -
      • Add lacking - definitions of LPTIM input - trigger remapping and its - related API
      • -
          -
            -
          • LL_LPTIM_INPUT1_SRC_PAD_AF, - - - - - LL_LPTIM_INPUT1_SRC_PAD_PA4, - - - - LL_LPTIM_INPUT1_SRC_PAD_PB9 - - - - and - LL_LPTIM_INPUT1_SRC_TIM_DAC.
          • -
          • Add a new - API LL_LPTIM_SetInput1Src() - to access to the LPTIM_OR - register and remap the - LPTIM input trigger.
          • -
          -
        -
      • Perform a new - check on indirect EXTI23 line - associated to the LPTIM wake - up timer
      • -
          -
        • Condition - the use of the LPTIM Wake-up - Timer associated EXTI - line configuration's - macros by EXTI_IMR_MR23 - bit in different API - - - - :
        • -
            -
          • __HAL_LPTIM_WAKEUPTIMER_EXTI_ENABLE/DDISABLE_FALLING_EDGE()
          • -
          • __HAL_LPTIM_WAKEUPTIMER_EXTI_DISABLE_FALLING_EDGE()
          • -
          • __HAL_LPTIM_WAKEUPTIMER_EXTI_ENABLE_RISING_EDGE(
          • -
          • __HAL_LPTIM_WAKEUPTIMER_EXTI_DISABLE_RISING_EDGE()
          • -
          • __HAL_LPTIM_WAKEUPTIMER_EXTI_ENABLE_RISING_FALLING_EDGE()
          • -
          • __HAL_LPTIM_WAKEUPTIMER_EXTI_DISABLE_RISING_FALLING_EDGE()
          • -
          • __HAL_LPTIM_WAKEUPTIMER_EXTI_GET_FLAG()
          • -
          • __HAL_LPTIM_WAKEUPTIMER_EXTI_CLEAR_FLAG()
          • -
          • __HAL_LPTIM_WAKEUPTIMER_EXTI_GENERATE_SWIT(
          • -
          -
        • Update HAL_LPTIM_TimeOut_Start_IT(), HAL_LPTIM_TimeOut_Stop_IT(), - - - - - HAL_LPTIM_Counter_Start_IT() - - - - - and HAL_LPTIM_Counter_Stop_IT() - - - - - API by adding Enable/Disable - rising edge trigger on - the LPTIM Wake-up Timer - Exti - line.
        • -
        • Add - __HAL_LPTIM_WAKEUPTIMER_EXTI_CLEAR_FLAG() - in the end of the HAL_LPTIM_IRQHandler() - - - - - API conditioned by - EXTI_IMR_MR23 bit.
        • -
        -
      -
    • HAL - - - - I2C update
    • -
        -
      • Update - HAL_I2C_EV_IRQHandler() - API to fix I2C send break - issue 
      • -
          -
        • Add - additional check on - hi2c->hdmatx, - hdmatx->XferCpltCallback, - hi2c->hdmarx, - hdmarx->XferCpltCallback - in I2C_Master_SB() - API to avoid enabling - DMA request when IT - mode is used.
        • -
        -
      • Update - HAL_I2C_ER_IRQHandler() - API to fix acknowledge failure - issue with I2C memory IT - processes
      • -
          -
        •  Add stop - - - - - condition generation when - NACK occurs.
        • -
        -
      • Update - HAL_I2C_Init() - API to force software reset - before setting new I2C - configuration
      • -
      • Update HAL - I2C processes to report ErrorCode - when wrong I2C start condition - occurs
      • -
          -
        •  Add - new ErrorCode - define: HAL_I2C_WRONG_START
        • -
        •  Set ErrorCode - parameter in I2C handle - to HAL_I2C_WRONG_START
        • -
        -
      • Update I2C_DMAXferCplt(), - - - - - I2C_DMAError() and - I2C_DMAAbort() APIs to fix hardfault - issue when hdmatx - and hdmarx parameters - - - - - in i2c handle aren't - initialized (NULL pointer).
      • -
          -
        • Add - additional check on - hi2c->hdmtx - and hi2c->hdmarx - before resetting DMA - Tx/Rx complete callbacks
        • -
        -
      -
    • HAL - - - - FMPI2C update
    • -
        -
      • Fix HAL - FMPI2C slave interrupt - handling issue with I2C - sequential transfers.
      • -
          -
        • Update - FMPI2C_Slave_ISR_IT() - and FMPI2C_Slave_ISR_DMA() - APIs to check on STOP - condition and handle it - before clearing the ADDR - flag
        • -
        -
      -
    • HAL - - - - NAND update
    • -
        -
      • Update - HAL_NAND_Write_Page_8b(), - - - - HAL_NAND_Write_Page_16b() - and  - HAL_NAND_Write_SpareArea_16b() - to manage correctly the time - out condition.
      • -
      -
    • HAL - - - - SAI update
    • -
        -
      • Optimize SAI_DMATxCplt() - and SAI_DMARxCplt() - - - - - APIs to check on "Mode" - parameter instead of CIRC - bit in the CR register.
      • -
      • Remove unused - SAI_FIFO_SIZE define
      • -
      • Update HAL_SAI_Receive_DMA() - programming sequence to be inline - with reference manual
      • -
      -
    -

    V1.7.6 - - - - / 12-April-2019

    -

    Main Changes

    -
      -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • HAL - - - - I2C update
    • -
        -
      • Fix I2C send - break issue in IT processes
      • -
          -
        • Add - additional check on - hi2c->hdmatx - and hi2c->hdmarx to - - - - - avoid the DMA request - enable when IT mode is used.
        • -
        -
      -
    • HAL - - - - SPI update
    • -
        -
      • Update to - implement Erratasheet: - BSY bit may stay high at the - end of a data transfer in - Slave mode
      • -
      -
    • LL - - - - LPTIM update
    • -
        -
      • Fix - compilation errors with LL_LPTIM_WriteReg() - and LL_LPTIM_ReadReg() - - - - - macros
      • -
      -
    • HAL - - - - SDMMC update
    • -
        -
      • Fix - preprocessing compilation - issue with SDIO - STA STBITERR interrupt
      • -
      -
    • HAL/LL - - - - USB update
    • -
        -
      • Updated USB_WritePacket(), - - - - - USB_ReadPacket() - - - - - APIs to prevent compilation - warning with GCC GNU v8.2.0
      • -
      • Rework USB_EPStartXfer() - API to enable the USB endpoint - before unmasking the TX FiFo - empty interrupt in case DMA is - not used
      • -
      • USB HAL_HCD_Init() - and HAL_PCD_Init() - - - - - APIs updated to avoid enabling - USB DMA feature for OTG FS - instance, USB DMA feature is - available only on OTG HS - Instance
      • -
      • Remove - duplicated line in hal_hcd.c - header file comment section -
      • -
      • Rework USB - HAL driver to use instance PCD_SPEED_xxx, - HCD_SPEED_xx - speeds instead of OTG register - Core speed definition during - the instance initialization
      • -
      • Software - Quality improvement with a fix - of CodeSonar - warning on PCD_Port_IRQHandler() - and  HCD_Port_IRQHandler() - - - - - interrupt handlers
      • -
      -
    -

    V1.7.5 - - - - / 08-February-2019

    -

    Main Changes

    -
      -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • General - - - - updates to fix CodeSonar - compilation warnings
    • -
    • General - - - - updates to fix SW4STM32 - compilation errors under Linux
    • -
    • General - - - - updates to fix the user manual - .chm files
    • -
    • Add - support of HAL callback - registration feature
    • -
    -
      -
    • Add - new HAL - - - - EXTI driver
    • -
    • Add - new HAL - - - - SMBUS driver
    • -
    • The - - - - following changes done on the - HAL drivers require an update - on the application code based - on older HAL versions
    • -
        -
      • Rework of HAL - CRYP driver (compatibility - break)
      • -
          -
        • HAL CRYP - driver has been redesigned - with new API's, to bypass - limitations on data - Encryption/Decryption - management present with - previous HAL CRYP driver - version.
        • -
        • The new HAL - CRYP driver is the - recommended version. It is - located as usual in - Drivers/STM32F4xx_HAL_Driver/Src - and - Drivers/STM32f4xx_HAL_Driver/Inc - folders. It can be enabled - through switch - HAL_CRYP_MODULE_ENABLED in - stm32f4xx_hal_conf.h
        • -
        • The legacy - HAL CRYP driver is no longer - supported.
        • -
        -
      • Add new AutoReloadPreload - field in TIM_Base_InitTypeDef - structure to allow the - possibilities to enable or - disable the TIM Auto Reload - Preload.
      • -
      -
    -
      -
    • HAL/LL - - - - Generic update
    • -
        -
      • Add support - of HAL callback - registration feature
      • -
          -
        • The feature - disabled by default is - available for the following - HAL drivers:
        • -
            -
          • ADC, - CAN, CEC, CRYP, DAC, - DCMI, DFSDM, DMA2D, DSI, - ETH, HASH, HCD, I2C, - FMPI2C, SMBUS,
            - UART, USART, IRDA, - SMARTCARD, LPTIM, LTDC, - MMC, NAND, NOR, - PCCARD, PCD, QSPI, RNG,

            -
            RTC, - SAI, SD, SDRAM, SRAM, - SPDIFRX, SPI, I2S, TIM, - and WWDG
          • -
          -
        • The feature - may be enabled individually - per HAL PPP driver - by setting the corresponding - definition USE_HAL_PPP_REGISTER_CALLBACKS - - - - - to 1U in - stm32f4xx_hal_conf.h project - configuration file (template - file - stm32f4xx_hal_conf_template.h - available from  - - - - Drivers/STM32F4xx_HAL_Driver/Inc)
        • -
        • Once enabled - - - - , the user - application may resort to HAL_PPP_RegisterCallback() - - - - - to register specific - callback function(s) and - unregister it(them) with HAL_PPP_UnRegisterCallback().
        • -
        -
      • General - updates to fix MISRA 2012 - compilation errors
      • -
          -
        • Replace HAL_GetUID() - API by HAL_GetUIDw0(), - HAL_GetUIDw1() and - HAL_GetUIDw2()
        • -
        • HAL_IS_BIT_SET()/HAL_IS_BIT_CLR() - macros implementation update
        • -
        • "stdio.h" - include updated with "stddef.h"
        • -
        -
      -
    • HAL - - - - GPIO  - - - - update
    • -
        -
      • Add missing - define for SPI3 alternate - function "GPIO_AF5_SPI3" for - STM32F401VE devices
      • -
      • Remove - "GPIO_AF9_TIM14" from defined - alternate function list for - STM32F401xx devices
      • -
      • HAL_GPIO_TogglePin() reentrancy - robustness improvement
      • -
      • HAL_GPIO_DeInit() API update - to avoid potential pending - interrupt after call
      • -
      • Update - GPIO_GET_INDEX() - API for more compliance with - STM32F412Vx/STM32F412Rx/STM32F412Cx - devices
      • -
      • Update - GPIO_BRR registers with - Reference Manual regarding - registers and bit definition - values
      • -
      -
    • HAL - - - - CRYP update
    • -
        -
      • The CRYP_InitTypeDef - is no more - supported, changed by CRYP_ConfigTypedef - to allow changing parameters - using HAL_CRYP_setConfig() - API without reinitialize the - CRYP IP using the HAL_CRYP_Init() - - - - - API
      • -
      • New - parameters added in the CRYP_ConfigTypeDef - structure: B0 and DataWidthUnit
      • -
      • Input data - size parameter is added in the - CRYP_HandleTypeDef - structure
      • -
      • Add new APIs - to manage the CRYP - configuration:
      • -
          -
        •  HAL_CRYP_SetConfig()
        • -
        • HAL_CRYP_GetConfig()
        • -
        -
      • Add new APIs - to manage the Key derivation:
      • -
          -
        • HAL_CRYPEx_EnableAutoKeyDerivation()
        • -
        • HAL_CRYPEx_DisableAutoKeyDerivation()
        • -
        -
      • Add new APIs - to encrypt and decrypt data:
      • -
          -
        • HAL_CRYP_Encypt()
        • -
        • HAL_CRYP_Decypt()
        • -
        • HAL_CRYP_Encypt_IT()
        • -
        • HAL_CRYP_Decypt_IT()
        • -
        • HAL_CRYP_Encypt_DMA()
        • -
        • HAL_CRYP_Decypt_DMA()
        • -
        -
      • Add new APIs - to generate TAG:
      • -
          -
        • HAL_CRYPEx_AESGCM_GenerateAuthTAG()
        • -
        • HAL_CRYPEx_AESCCM_Generago teAuthTAG()
        • -
        -
      -
    • HAL - - - - LPTIM update
    • -
        -
      • Remove - useless LPTIM Wakeup EXTI - related macros from HAL_LPTIM_TimeOut_Start_IT() - API
      • -
      -
    • HAL - - - - I2C update
    • -
        -
      • I2C API - changes for MISRA-C 2012 - compliancy:
      • -
          -
        • Rename - HAL_I2C_Master_Sequential_Transmit_IT() - to - HAL_I2C_Master_Seq_Transmit_IT()
        • -
        • Rename - HAL_I2C_Master_Sequentiel_Receive_IT() - to - HAL_I2C_Master_Seq_Receive_IT()
        • -
        • Rename - HAL_I2C_Slave_Sequentiel_Transmit_IT() - to - HAL_I2C_Slave_Seq_Transmit_IT() -
        • -
        • Rename - HAL_I2C_Slave_Sequentiel_Receive_DMA() - to - HAL_I2C_Slave_Seq_Receive_DMA()
        • -
        -
      • SMBUS defined - flags are removed as not used - by the HAL I2C driver
      • -
          -
        • I2C_FLAG_SMBALERT
        • -
        • I2C_FLAG_TIMEOUT
        • -
        • I2C_FLAG_PECERR
        • -
        • I2C_FLAG_SMBHOST
        • -
        • I2C_FLAG_SMBDEFAULT
        • -
        -
      • Add support - of I2C repeated start feature - in DMA Mode:
      • -
          -
        • With the - following new API's
        • -
            -
          • HAL_I2C_Master_Seq_Transmit_DMA()
          • -
          • HAL_I2C_Master_Seq_Receive_DMA()
          • -
          • HAL_I2C_Slave_Seq_Transmit_DMA()
          • -
          • HAL_I2C_Slave_Seq_Receive_DMA()
          • -
          -
        -
      • Add new I2C - transfer options to easy - manage the sequential transfers
      • -
          -
        • I2C_FIRST_AND_NEXT_FRAME
        • -
        • I2C_LAST_FRAME_NO_STOP
        • -
        • I2C_OTHER_FRAME
        • -
        • I2C_OTHER_AND_LAST_FRAME
        • -
        -
      -
    • HAL - - - - FMPI2C update
    • -
        -
      • I2C API - changes for MISRA-C 2012 - compliancy:
      • -
          -
        • Rename - HAL_FMPI2C_Master_Sequential_Transmit_IT() - to - HAL_FMPI2C_Master_Seq_Transmit_IT()
        • -
        • Rename - HAL_FMPI2C_Master_Sequentiel_Receive_IT() - to - HAL_FMPI2C_Master_Seq_Receive_IT()
        • -
        • Rename - HAL_FMPI2C_Master_Sequentiel_Transmit_DMA() - to - HAL_FMPI2C_Master_Seq_Transmit_DMA() -
        • -
        • Rename - HAL_FMPI2C_Master_Sequentiel_Receive_DMA() - to - HAL_FMPI2C_Master_Seq_Receive_DMA()
        • -
        -
      • Rename - FMPI2C_CR1_DFN to - FMPI2C_CR1_DNF for more - compliance with Reference - Manual regarding registers and - bit definition naming
      • -
      • Add support - of I2C repeated start feature - in DMA Mode:
      • -
          -
        • With the - following new API's
        • -
            -
          • HAL_FMPI2C_Master_Seq_Transmit_DMA()
          • -
          • HAL_FMPI2C_Master_Seq_Receive_DMA()
          • -
          • HAL_FMPI2C_Slave_Seq_Transmit_DMA()
          • -
          • HAL_FMPI2C_Slave_Seq_Receive_DMA()
          • -
          -
        -
      -
    • HAL - - - - FLASH update
    • -
        -
      • Update the FLASH_OB_GetRDP() - API to return the correct RDP - level
      • -
      -
    • HAL  - RCC - update
    • -
        -
      • Remove GPIOD - CLK macros for STM32F412Cx - devices (X = D)
      • -
      • Remove GPIOE - CLK macros for - STM32F412Rx\412Cx devices: (X - = E)
      • -
      • Remove - GPIOF/G CLK macros for - STM32F412Vx\412Rx\412Cx - devices (X= F or G)
      • -
          -
        • __HAL_RCC_GPIOX_CLK_ENABLE()
        • -
        • __HAL_RCC_GPIOX_CLK_DISABLE()
        • -
        • __HAL_RCC_GPIOX_IS_CLK_ENABLED()
        • -
        • __HAL_RCC_GPIOX_IS_CLK_DISABLED()
        • -
        • __HAL_RCC_GPIOX_FORCE_RESET()
        • -
        -
      -
    • HAL - - - - RNG update
    • -
        -
      • Update to - manage RNG error code:
      • -
          -
        • Add ErrorCode - parameter in HAL RNG Handler - structure
        • -
        -
      -
    • LL - - - - ADC update
    • -
        -
      • Add - __LL_ADC_CALC_TEMPERATURE() - helper macro to calculate the - temperature (unit: degree - Celsius) from ADC conversion - data of internal temperature - sensor.
      • -
      • Fix ADC - channels configuration issues - on STM32F413xx/423xx devices
      • -
          -
        • To allow - possibility to switch - between VBAT and TEMPERATURE - channels configurations
        • -
        -
      • HAL_ADC_Start(), HAL_ADC_Start_IT() - - - - - and HAL_ADC_Start_DMA() - - - - - update to prevention from - starting ADC2 or ADC3 once - multimode is enabled
      • -
      -
    • HAL - - - - DFSDM  - - - - update
    • -
        -
      • General - updates to be compliant with - DFSDM bits naming used in - CMSIS files.
      • -
      -
    • HAL - - - - CAN  - - - - update
    • -
        -
      • Update - possible values list for FilterActivation - parameter in CAN_FilterTypeDef - structure
      • -
          -
        • CAN_FILTER_ENABLE - - - - - instead of ENABLE
        • -
        • CAN_FILTER_DISABLE - - - - - instead of DISABLE
        • -
        -
      -
    • HAL - - - - CEC  - - - - update
    • -
        -
      • Update HAL - CEC State management method:
      • -
          -
        • Remove HAL_CEC_StateTypeDef - structure parameters
        • -
        • Add new - defines for CEC states
        • -
        -
      -
    • HAL - - - - DMA  - - - - update
    • -
        -
      • Add clean of - callbacks in HAL_DMA_DeInit() - API
      • -
      -
    • HAL - - - - DMA2D  - - - - update
    • -
        -
      • Remove unused - DMA2D_ColorTypeDef structure - to be compliant with MISRAC - 2012 Rule 2.3
      • -
      • General - update to use dedicated - defines for - DMA2D_BACKGROUND_LAYER and - DMA2D_FOREGROUND_LAYER instead - of numerical values: 0/1.
      • -
      -
    • HAL - - - - DSI  - - - - update
    • -
        -
      • Fix read - multibyte issue: remove extra - call to __HAL_UNLOCK__ from DSI_ShortWrite() - API.
      • -
      -
    -
      -
    • HAL/LL - - - - RTC update
    • -
    -
      -
        -
      • HAL/ LL drivers - optimization
      • -
          -
        • HAL driver: - remove unused variables
        • -
        • LL driver: - getter APIs optimization
        • -
        -
      -
    • HAL - - - - PWR update
    • -
        -
      • Remove the - followings API's as feature - not supported by - STM32F469xx/479xx devices
      • -
          -
        • HAL_PWREx_EnableWakeUpPinPolarityRisingEdge()
        • -
        • HAL_PWREx_EnableWakeUpPinPolarityRisingEdge()
        • -
        -
      -
    • HAL - - - - SPI update
    • -
        -
      • Update HAL_SPI_StateTypeDef - structure to add new state: - HAL_SPI_STATE_ABORT
      • -
      -
    • HAL/LL - - - - TIM update
    • -
        -
      • Add new AutoReloadPreload - field in TIM_Base_InitTypeDef - structure
      • -
          -
        • Refer to - the TIM examples to identify - the changes -
        • -
        -
      • Move the - following TIM structures from - stm32f4xx_hal_tim_ex.h into - stm32f4xx_hal_tim.h
      • -
          -
        • TIM_MasterConfigTypeDef
        • -
        • TIM_BreakDeadTimeConfigTypeDef
        • -
        -
      • Add new TIM - Callbacks API's:
      • -
          -
        • HAL_TIM_PeriodElapsedHalfCpltCallback()
        • -
        • HAL_TIM_IC_CaptureHalfCpltCallback()
        • -
        • HAL_TIM_PWM_PulseFinishedHalfCpltCallback()
        • -
        • HAL_TIM_TriggerHalfCpltCallback()
        • -
        -
      • TIM API - changes for MISRA-C 2012 - compliancy:
      • -
          -
        • Rename HAL_TIM_SlaveConfigSynchronization - to HAL_TIM_SlaveConfigSynchro
        • -
        • Rename HAL_TIM_SlaveConfigSynchronization_IT - to HAL_TIM_SlaveConfigSynchro_IT
        • -
        • Rename HAL_TIMEx_ConfigCommutationEvent - to HAL_TIMEx_ConfigCommutEvent
        • -
        • Rename HAL_TIMEx_ConfigCommutationEvent_IT - to HAL_TIMEx_ConfigCommutEvent_IT
        • -
        • Rename HAL_TIMEx_ConfigCommutationEvent_DMA - to HAL_TIMEx_ConfigCommutEvent_DMA
        • -
        • Rename HAL_TIMEx_CommutationCallback - to HAL_TIMEx_CommutCallback
        • -
        • Rename HAL_TIMEx_DMACommutationCplt - to TIMEx_DMACommutationCplt
        • -
        -
      -
    -
      -
    • HAL/LL - - - - USB update
    • -
        -
      • Rework USB - interrupt handler and improve - HS DMA support in Device mode
      • -
      • Fix BCD - handling fr OTG - instance in device mode
      • -
      • cleanup - reference to low speed in - device mode
      • -
      • allow writing - TX FIFO in case of transfer - length is equal to available - space in the TX FIFO
      • -
      • Fix Toggle - OUT interrupt channel in host - mode
      • -
      • Update USB - OTG max number of endpoints (6 - FS and 9 HS instead of 5 and - 8)
      • -
      • Update USB - OTG IP to enable internal - transceiver when starting USB - device after committee BCD negotiation
      • -
      -
    • LL - - - - IWDG update
    • -
        -
      • Update LL - inline macros to use IWDGx - parameter instead of IWDG - instance defined in CMSIS device
      • -
      -
    -

    V1.7.4 - - - - / 02-February-2018

    -

    Main Changes

    -
      -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • HAL update
    • -
        -
      • Update UNUSED() - macro implementation to avoid - GCC warning
      • -
          -
        • The warning - is detected when the UNUSED() - macro is called from C++ - file
        • -
        -
      • Update to - make RAMFUNC define as generic - type instead of HAL_StatusTypdef - type.
      • -
      -
    • HAL - - - - FLASH update
    • -
        -
      • Update - the prototypes of the - following APIs after change on - RAMFUNC defines 
      • -
          -
        • HAL_FLASHEx_StopFlashInterfaceClk()
        • -
        • HAL_FLASHEx_StartFlashInterfaceClk()
        • -
        • HAL_FLASHEx_EnableFlashSleepMode()
        • -
        • HAL_FLASHEx_DisableFlashSleepMode()
        • -
        -
      -
    • HAL - - - - SAI update
    • -
        -
      • Update HAL_SAI_DMAStop() - and HAL_SAI_Abort() - - - - - process to fix the lock/unlock - audio issue
      • -
      -
    -

    V1.7.3 - - - - / 22-December-2017

    -

    Main Changes

    -
      -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • The - - - - following changes done on the - HAL drivers require an update - on the application code based - on older HAL versions
    • -
        -
      • Rework of - HAL CAN driver - (compatibility break) 
      • -
          -
        • A new HAL - CAN driver has been - redesigned with new APIs, to - bypass limitations on CAN - Tx/Rx FIFO management - present with previous HAL - CAN driver version.
        • -
        • The new HAL - CAN driver is the - recommended version. It is - located as usual in - Drivers/STM32F4xx_HAL_Driver/Src - and - Drivers/STM32f4xx_HAL_Driver/Inc - folders. It can be enabled - through switch - HAL_CAN_MODULE_ENABLED in - stm32f4xx_hal_conf.h
        • -
        • The legacy - HAL CAN driver is also - present in the release in - Drivers/STM32F4xx_HAL_Driver/Src/Legacy - - - - and - Drivers/STM32F4xx_HAL_Driver/Inc/Legacy - folders for software - compatibility reasons. Its - usage is not recommended as - deprecated. It can - however be enabled through - switch - HAL_CAN_LEGACY_MODULE_ENABLED - in stm32f4xx_hal_conf.h
        • -
        -
      -
    • HAL update
    • -
        -
      • Update HAL - driver to allow user to change - systick - period to 1ms, 10 ms - or 100 ms :
      • -
          -
        • Add the - following API's - - - - :  
        • -
            -
          • HAL_GetTickPrio(): - Returns a tick priority.
          • -
          • HAL_SetTickFreq(): Sets - new tick frequency.
          • -
          • HAL_GetTickFreq(): - Returns tick frequency.
          • -
          -
        • Add HAL_TickFreqTypeDef - enumeration for the - different Tick Frequencies: - 10 Hz, 100 Hz and 1KHz - (default).
        • -
        -
      -
    • HAL - - - - CAN update
    • -
        -
      • Fields of CAN_InitTypeDef - structure are reworked:
      • -
          -
        • SJW to SyncJumpWidth, - BS1 to TimeSeg1, BS2 to - TimeSeg2, TTCM to TimeTriggeredMode, - ABOM to AutoBusOff, - AWUM to AutoWakeUp, - NART to AutoRetransmission - (inversed), RFLM to ReceiveFifoLocked - and TXFP to TransmitFifoPriority
        • -
        -
      • HAL_CAN_Init() is split - into both HAL_CAN_Init() - - - - - and HAL_CAN_Start() - - - - - API's
      • -
      • HAL_CAN_Transmit() is replaced - by HAL_CAN_AddTxMessage() - - - - - to place Tx Request, then HAL_CAN_GetTxMailboxesFreeLevel() - - - - - for polling until completion.
      • -
      • HAL_CAN_Transmit_IT() is replaced - by HAL_CAN_ActivateNotification() - - - - - to enable transmit IT, then HAL_CAN_AddTxMessage() - - - - - for place Tx request.
      • -
      • HAL_CAN_Receive() is replaced - by HAL_CAN_GetRxFifoFillLevel() - - - - - for polling until reception, - then HAL_CAN_GetRxMessage() - - - - -
        - to get Rx message.
      • -
      • HAL_CAN_Receive_IT() is replaced - by HAL_CAN_ActivateNotification() to - - - - - enable receive IT, then HAL_CAN_GetRxMessage()
        - in the receivecallback - to get Rx message
      • -
      • HAL_CAN_Slepp() is renamed - as HAL_CAN_RequestSleep()
      • -
      • HAL_CAN_TxCpltCallback() is split - into - HAL_CAN_TxMailbox0CompleteCallback(), -HAL_CAN_TxMailbox1CompleteCallback() -and HAL_CAN_TxMailbox2CompleteCallback().
      • -
      • HAL_CAN_RxCpltCallback is split - into HAL_CAN_RxFifo0MsgPendingCallback() - and - HAL_CAN_RxFifo1MsgPendingCallback().
      • -
      • More complete - "How to use the new driver" is - detailed in the driver header - section itself.
      • -
      -
    • HAL - - - - FMPI2C update
    • -
        -
      • Add new - option - FMPI2C_LAST_FRAME_NO_STOP for - the sequential transfer management
      • -
          -
        • This option - allows to manage a restart - condition after several call - of the same master - sequential interface. 
        • -
        -
      -
    • HAL - - - - RCC update
    • -
        -
      • Add new HAL macros
      • -
          -
        • __HAL_RCC_GET_RTC_SOURCE() - allowing to get the RTC - clock source
        • -
        • __HAL_RCC_GET_RTC_HSE_PRESCALER() - allowing to get the HSE - clock divider for RTC - peripheral
        • -
        -
      • Ensure reset - of CIR and CSR registers when - issuing HAL_RCC_DeInit()/LL_RCC_DeInit - functions
      • -
      • Update HAL_RCC_OscConfig() to - - - - - keep backup domain enabled - when configuring - respectively LSE and RTC - clock source
      • -
      • Add new HAL - interfaces allowing to control - the activation or deactivation - of PLLI2S and PLLSAI:
      • -
          -
        • HAL_RCCEx_EnablePLLI2S()
        • -
        • HAL_RCCEx_DisablePLLI2S()
        • -
        • HAL_RCCEx_EnablePLLSAI()
        • -
        • HAL_RCCEx_DisablePLLSAI()
        • -
        -
      -
    -
      -
    • LL - - - - RCC update 
    • -
        -
      • Add new LL - RCC macro
      • -
          -
        • LL_RCC_PLL_SetMainSource() allowing - to configure PLL main clock - source
        • -
        -
      -
    • LL - - - - FMC / LL FSMC update
    • -
        -
      • Add clear of - the PTYP bit to select the - PCARD mode in FMC_PCCARD_Init() - / FSMC_PCCARD_Init()
      • -
      -
    -

    V1.7.2 - - - - / 06-October-2017

    -

    Main Changes

    -
      -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • Fix - compilation warning with - GCC compiler
    • -
    • Remove - - - - Date and version - from header files
    • -
    • Update - - - - HAL drivers to refer to the - new CMSIS bit position - defines instead of usage the - POSITION_VAL() - macro
    • -
    • HAL - - - - Generic update
    • -
        -
      • stm32f4xx_hal_def.h - - - - file changes: 
      • -
          -
        • Update - __weak and __packed defined - values for ARM compiler
        • -
        • Update - __ALIGN_BEGIN and - __ALIGN_END defined values - for ARM compiler
        • -
        -
      • stm32f4xx_ll_system.h - - - - - file: - add LL_SYSCFG_REMAP_SDRAM - define
      • -
      -
    • HAL - - - - ADC update
    • -
        -
      • Fix wrong - definition of ADC channel - temperature sensor for - STM32F413xx and STM32F423xx - devices.
      • -
      -
    • HAL - - - - DMA update
    • -
        -
      • Update values - - - - for the following defines: - DMA_FLAG_FEIF0_4 and - DMA_FLAG_DMEIF0_4 
      • -
      -
    • HAL - - - - DSI update
    • -
        -
      • Fix Extra - warning with SW4STM32 compiler
      • -
      • Fix DSI - display issue when using EWARM - w/ high level optimization 
      • -
      • Fix - MISRAC errors
      • -
      -
    • HAL - - - - FLASH update
    • -
        -
      • HAL_FLASH_Unlock() update to - return state error when the - FLASH is already unlocked
      • -
      -
    • HAL - - - - FMPI2C update
    • -
        -
      • Update - Interface APIs headers to - remove confusing message about - device address
      • -
      • Update - FMPI2C_WaitOnRXNEFlagUntilTimeout() - to resolve a race condition - between STOPF and RXNE Flags
      • -
      • Update - FMPI2C_TransferConfig() - to fix wrong bit management.
      • -
      • Update code - comments to use DMA stream - instead of DMA channel
      • -
      -
    -
      -
    • HAL - - - - PWR update
    • -
        -
      • HAL_PWR_EnableWakeUpPin() update - description to add support of - PWR_WAKEUP_PIN2 and - PWR_WAKEUP_PIN3
      • -
      -
    • HAL - - - - NOR update
    • -
        -
      • Add the - support of STM32F412Rx devices
      • -
      -
    • HAL - - - - I2C update
    • -
        -
      • Update - Interface APIs headers to - remove confusing mesage - about device address
      • -
      • Update - I2C_MasterReceive_RXNE() - and I2C_MasterReceive_BTF() - static APIs to fix bad - Handling of NACK in I2C master - receive process.
      • -
      -
    -
      -
    • HAL - - - - RCC update
    • -
        -
      • Update HAL_RCC_GetOscConfig() - API to:
      • -
          -
        • set PLLR in - the RCC_OscInitStruct
        • -
        • check on - null pointer
        • -
        -
      • Update HAL_RCC_ClockConfig() - API to:
      • -
          -
        • check on - null pointer
        • -
        • optimize code - - - - size by updating the - handling method of the SWS bits
        • -
        • update to use  - - - - - __HAL_FLASH_GET_LATENCY() - - - - flash macro instead of using - direct register access - to LATENCY bits in - FLASH ACR register.
        • -
        -
      • Update HAL_RCC_DeInit() -  and LL_RCC_DeInit() - - - - - APIs to
      • -
          -
        • Be able to - return HAL/LL status
        • -
        • Add checks - for HSI, PLL and PLLI2S -  ready - before modifying RCC CFGR - registers
        • -
        • Clear all - interrupt falgs
        • -
        • Initialize - systick - interrupt period
        • -
        -
      • Update HAL_RCC_GetSysClockFreq() - to avoid risk of rounding - error which may leads to a - wrong returned value. 
      • -
      -
    -

     

    -

     

    -
      -
    • HAL - - - - RNG update
    • -
        -
      • HAL_RNG_Init() remove - Lock()/Unlock()
      • -
      -
    • HAL - - - - MMC update
    • -
        -
      • HAL_MMC_Erase() - API: add missing () to - fix compilation warning - detected with SW4STM32 when - extra feature is enabled.
      • -
      -
    • HAL - - - - RTC update
    • -
        -
      • HAL_RTC_Init() API: update - to force the wait for synchro - before setting TAFCR register - when BYPSHAD bit in CR - register is 0.
      • -
      -
    • HAL - - - - SAI update
    • -
        -
      • Update HAL_SAI_DMAStop() - API to flush fifo - after disabling SAI
      • -
      -
    • HAL - - - - I2S update
    • -
        -
      • Update I2S - DMA fullduplex process to - handle I2S Rx and Tx DMA Half - transfer complete callback
      • -
      -
    • HAL - - - - TIM update
    • -
        -
      • Update HAL_TIMEx_OCN_xxxx() - and HAL_TIMEx_PWMN_xxx() - - - - - API description to remove - support of TIM_CHANNEL_4
      • -
      -
    • LL - - - - DMA update
    • -
        -
      • Update to - clear DMA flags using WRITE_REG() - instead SET_REG() API to avoid - read access to the IFCR - register that is write only.
      • -
      -
    • LL - - - - RTC update
    • -
        -
      • Fix warning - with static analyzer
      • -
      -
    • LL - - - - USART update
    • -
        -
      • Add assert - macros to check USART BaudRate - register
      • -
      -
    • LL - - - - I2C update
    • -
        -
      • Rename - IS_I2C_CLOCK_SPEED() - and IS_I2C_DUTY_CYCLE() - respectively to - IS_LL_I2C_CLOCK_SPEED() and - IS_LL_I2C_DUTY_CYCLE() to - avoid incompatible macros - redefinition.
      • -
      -
    • LL - - - - TIM update
    • -
        -
      • Update LL_TIM_EnableUpdateEvent() - API to clear UDIS bit in TIM - CR1 register instead of - setting it.
      • -
      • Update LL_TIM_DisableUpdateEvent() - API to set UDIS bit in TIM CR1 - register instead of clearing - it.
      • -
      -
    • LL - - - - USART update
    • -
        -
      • Fix MISRA - error w/ IS_LL_USART_BRR() - macro
      • -
      • Fix wrong - check when UART10 instance is - used
      • -
      -
    -

    V1.7.1 - - - - / 14-April-2017

    -

    Main Changes

    -
      -
    • Update - - - - CHM UserManuals - to support LL drivers
    • -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • HAL - - - - CAN update
    • -
        -
      • Add - management of overrun - error. 
      • -
      • Allow - possibility to receive - messages from the 2 RX FIFOs - in parallel via interrupt.
      • -
      • Fix message - - - - lost issue with specific - sequence of transmit requests.
      • -
      • Handle - transmission failure with - error callback, when NART is - enabled.
      • -
      • Add - __HAL_CAN_CANCEL_TRANSMIT() - call to abort transmission - when timeout is reached
      • -
      -
    -
      -
    • HAL - - - - PWR update
    • -
        -
      • HAL_PWREx_EnterUnderDriveSTOPMode() API: remove - check on UDRDY flag
      • -
      -
    -
      -
    • LL - - - - ADC update
    • -
        -
      • Fix wrong ADC - group injected sequence configuration
      • -
          -
        • LL_ADC_INJ_SetSequencerRanks() and LL_ADC_INJ_GetSequencerRanks() - - - - - API's update to take in - consideration the ADC number - of conversions
        • -
        • Update - the defined values for - ADC group injected seqencer - ranks 
        • -
        -
      -
    -

    V1.7.0 - - - - / 17-February-2017

    -

    Main Changes

    -
      -
    • Add - - - - Low Layer drivers allowing - performance and footprint optimization
    • -
        -
      • Low Layer drivers - APIs provide register level - programming: require deep - knowledge of peripherals - described in STM32F4xx - Reference Manuals
      • -
      • Low Layer - drivers are available for: - ADC, Cortex, CRC, DAC, - DMA, DMA2D, EXTI, GPIO, I2C, - IWDG, LPTIM, PWR, RCC, RNG, - RTC, SPI, TIM, USART, WWDG - peripherals and additionnal - Low Level Bus, System and - Utilities APIs.
      • -
      • Low Layer drivers - APIs are implemented as static - inline function in new Inc/stm32f4xx_ll_ppp.h files - - - - - for PPP peripherals, there is - no configuration file and each stm32f4xx_ll_ppp.h file - - - - - must be included in user code.
      • -
      -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • Fix extra - - - - warnings with GCC compiler
    • -
    • HAL - drivers clean up: remove - double casting 'uint32_t' and 'U'
    • -
    • Add - new HAL - - - - MMC driver
    • -
    • The - - - - following changes done on the - HAL drivers require an update - on the application code based - on older HAL versions
    • -
        -
      • HAL SD update
      • -
          -
        • Overall - rework of the driver for a - more - efficient implementation
        • -
            -
          • Modify - initialization API and structures
          • -
          • Modify - Read / Write sequences: - separate transfer process - and SD Cards state management 
          • -
          • Adding - interrupt mode for Read / - Write operations
          • -
          • Update - the HAL_SD_IRQHandler - function by optimizing the - management of interrupt errors
          • -
          -
        • Refer to - the following example to - identify the changes: BSP - example and USB_Device/MSC_Standalone - application
        • -
        -
      • HAL NAND update
      • -
          -
        • Modify NAND_AddressTypeDef, - NAND_DeviceConfigTypeDef - and NAND_HandleTypeDef - structures fields
        • -
        • Add new HAL_NAND_ConfigDevice - API
        • -
        -
      • HAL DFSDM update
      • -
          -
        • Add - support of Multichannel - Delay feature
        • -
            -
          • Add HAL_DFSDM_ConfigMultiChannelDelay - API
          • -
          • The - following APIs are moved - to internal static - functions: HAL_DFSDM_ClockIn_SourceSelection, - HAL_DFSDM_ClockOut_SourceSelection, - HAL_DFSDM_DataInX_SourceSelection - (X=0,2,4,6), HAL_DFSDM_BitStreamClkDistribution_Config
          • -
          -
        -
      • HAL I2S update
      • -
          -
        • Add specific - - - - - callback API to manage I2S - full duplex end of transfer - process:
        • -
            -
          • HAL_I2S_TxCpltCallback() - and - HAL_I2S_RxCpltCallback() - API's will be replaced - with only - HAL_I2SEx_TxRxCpltCallback() - API. 
          • -
          -
        -
      -
    • HAL - - - - update
    • -
        -
      • Modifiy default HAL_Delay - implementation to guarantee - minimum delay 
      • -
      -
    • HAL - - - - Cortex update
    • -
        -
      • Move HAL_MPU_Disable() - and HAL_MPU_Enable() - - - - - from stm32f4xx_hal_cortex.h to - stm32f4xx_hal_cortex.c
      • -
      • Clear the - whole MPU control register - in HAL_MPU_Disable() - API
      • -
      -
    • HAL - - - - FLASH update
    • -
        -
      • IS_FLASH_ADDRESS() - macro update to support OTP - range
      • -
      • FLASH_Program_DoubleWord(): Replace - 64-bit accesses with 2 - double-words operations
      • -
      -
    • LL - - - - GPIO update
    • -
        -
      • Update - IS_GPIO_PIN() - macro implementation to be - more safe
      • -
      -
    • LL - - - - RCC update
    • -
        -
      • Update - IS_RCC_PLLQ_VALUE() - macro implementation: the - minimum accepted value is - 2 instead of 4
      • -
      • Rename - RCC_LPTIM1CLKSOURCE_PCLK - define to - RCC_LPTIM1CLKSOURCE_PCLK1
      • -
      • Fix - compilation issue w/ - __HAL_RCC_USB_OTG_FS_IS_CLK_ENABLED() - and - __HAL_RCC_USB_OTG_FS_IS_CLK_DISABLED() - macros for STM32F401xx devices
      • -
      • Add the - following is clock - enabled macros for STM32F401xx - devices
      • -
          -
        •  __HAL_RCC_SDIO_IS_CLK_ENABLED()
        • -
        • __HAL_RCC_SPI4_IS_CLK_ENABLED()
        • -
        • __HAL_RCC_TIM10_IS_CLK_ENABLED()
        • -
        -
      • Add the - following is clock - enabled macros for STM32F410xx - devices
      • -
          -
        •  __HAL_RCC_CRC_IS_CLK_ENABLED()
        • -
        • __HAL_RCC_RNG_IS_CLK_ENABLED()
        • -
        -
      • Update HAL_RCC_DeInit() - to reset the RCC clock - configuration to the default - reset state.
      • -
      • Remove macros - to configure BKPSRAM from - STM32F401xx devices 
      • -
      • Update to - refer to AHBPrescTable[] - and APBPrescTable[] - - - - - tables defined in - system_stm32f4xx.c file - instead of APBAHBPrescTable[] - - - - - table.
      • -
      -
    • HAL - - - - FMPI2C update
    • -
        -
      • Add - FMPI2C_FIRST_AND_NEXT_FRAME - define in Sequential - Transfer Options
      • -
      -
    • HAL - - - - ADC update
    • -
        -
      • HAL_ADCEx_InjectedConfigChannel(): update the - external trigger injected - condition
      • -
      -
    • HAL - - - - DMA update
    • -
        -
      • HAL_DMA_Init(): update to - check compatibility between - FIFO threshold level and size - of the memory burst 
      • -
      -
    • HAL - - - - QSPI update
    • -
        -
      • QSPI_HandleTypeDef structure: - Update transfer parameters on - uint32_t instead of uint16_t
      • -
      -
    • HAL - - - - UART/USART/IrDA/SMARTCARD update
    • -
        -
      • DMA Receive - process; the code has been - updated to clear the USART - OVR flag before - enabling DMA receive - request.
      • -
      • UART_SetConfig() update to - manage correctly USART6 - instance that is not available - on STM32F410Tx devices
      • -
      -
    • HAL - - - - CAN update
    • -
        -
      • Remove Lock - mechanism from HAL_CAN_Transmit_IT() - and HAL_CAN_Receive_IT() - - - - - processes
      • -
      -
    • HAL - - - - TIM update
    • -
        -
      • Add - __HAL_TIM_MOE_DISABLE_UNCONDITIONALLY() - macro to disable Master output - without check on TIM channel - state. 
      • -
      • Update HAL_TIMEx_ConfigBreakDeadTime() - to fix TIM BDTR register - corruption.
      • -
      -
    • HAL - - - - I2C update
    • -
        -
      • Update - HAL_I2C_Master_Transmit() - and HAL_I2C_Slave_Transmit() - to avoid sending extra - bytes at the end of the - transmit processes
      • -
      • Update - HAL_I2C_Mem_Read() - API to fix wrong check on - misused parameter “Size”
      • -
      • Update - I2C_MasterReceive_RXNE() - and I2C_MasterReceive_BTF() - static APIs to enhance Master - sequential reception process.
      • -
      -
    • HAL - - - - SPI update
    • -
        -
      • Add transfer - abort APIs and associated - callbacks in interrupt mode
      • -
          -
        • HAL_SPI_Abort()
        • -
        • HAL_SPI_Abort_IT()
        • -
        • HAL_SPI_AbortCpltCallback()
        • -
        -
      -
    • HAL - - - - I2S update
    • -
        -
      • Add specific - - - - - callback API to manage I2S - full duplex end of transfer - process:
      • -
          -
        • HAL_I2S_TxCpltCallback() - and HAL_I2S_RxCpltCallback() - API's will be replaced with - only - HAL_I2SEx_TxRxCpltCallback() - API. 
        • -
        -
      • Update I2S - Transmit/Receive polling - process to manage Overrun - and Underrun errors
      • -
      • Move - the I2S clock input - frequency calculation to - HAL RCC driver.
      • -
      • Update the - HAL I2SEx driver to keep only - full duplex feature.
      • -
      • HAL_I2S_Init() - API updated to
      • -
          -
        • Fix wrong - I2S clock calculation when - PCM mode is used.
        • -
        • Return - state HAL_I2S_ERROR_PRESCALER when - the I2S clock is wrongly configured
        • -
        -
      -
    -
      -
    • HAL - - - - LTDC update
    • -
        -
      • Optimize HAL_LTDC_IRQHandler() - function by using direct - register read
      • -
      • Rename the - following API's
      • -
          -
        • HAL_LTDC_Relaod() by HAL_LTDC_Reload() 
        • -
        • HAL_LTDC_StructInitFromVideoConfig() by HAL_LTDCEx_StructInitFromVideoConfig()
        • -
        • HAL_LTDC_StructInitFromAdaptedCommandConfig() by HAL_LTDCEx_StructInitFromAdaptedCommandConfig()
        • -
        -
      • Add new - defines for LTDC layers - (LTDC_LAYER_1 / LTDC_LAYER_2)
      • -
      • Remove unused - asserts
      • -
      -
    • HAL - - - - USB PCD - update
    • -
        -
      • Flush all TX - FIFOs on USB Reset
      • -
      • Remove Lock - mechanism from HAL_PCD_EP_Transmit() - and HAL_PCD_EP_Receive() - - - - - API's
      • -
      -
    -
      -
    • LL - - - - USB update
    • -
        -
      • Enable DMA - Burst mode for USB OTG HS
      • -
      • Fix SD card - detection issue
      • -
      -
    • LL - - - - SDMMC update
    • -
        -
      • Add new SDMMC_CmdSDEraseStartAdd, - SDMMC_CmdSDEraseEndAdd, - SDMMC_CmdOpCondition - and SDMMC_CmdSwitch - functions
      • -
      -
    -

    V1.6.0 - - - - / 04-November-2016

    -

    Main Changes

    -
      -
    • Add support of STM32F413xx - and STM32F423xx - devices
    • -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • HAL - - - - CAN update
    • -
        -
      • Update to add - the support of 3 CAN management
      • -
      -
    • HAL - - - - CRYP update
    • -
        -
      • Update to add - the support of AES features
      • -
      -
    • HAL - - - - DFSDM update
    • -
        -
      • Add - definitions for new external - trigger filters
      • -
      • Add - definition for new Channels - 4, 5, 6 and 7
      • -
      • Add - functions and API for Filter - state configuration and management
      • -
      • Add new - functions: 
      • -
          -
        • HAL_DFSDM_BitstreamClock_Start()
        • -
        • HAL_DFSDM_BitstreamClock_Stop()
        • -
        • HAL_DFSDM_BitStreamClkDistribution_Config(
        • -
        -
      -
    • HAL - - - - DMA
    • -
        -
      • Add the - support of DMA Channels from - 8 to 15
      • -
      • Update HAL_DMA_DeInit() - function with the check on - DMA stream instance
      • -
      -
    • HAL - - - - DSI update
    • -
    -
      -
        -
      • Update HAL_DSI_ConfigHostTimeouts() - and HAL_DSI_Init() - - - - - functions to avoid scratch in - DSI_CCR register
      • -
      -
    • HAL - - - - FLASH update
    • -
        -
      • Enhance FLASH_WaitForLastOperation() - function implementation
      • -
      • Update - __HAL_FLASH_GET_FLAG() - macro implementation
      • -
      -
    • HAL - - - - GPIO update
    • -
        -
      • Add - specific alternate functions - definitions
      • -
      -
    • HAL - - - - I2C update
    • -
        -
      • Update I2C_DMAError() - function implementation to - ignore DMA FIFO error
      • -
      -
    • HAL - - - - I2S update
    • -
        -
      • Enhance - HAL_I2S_Init() - implementation to test on - PCM_SHORT and PCM_LONG - standards
      • -
      -
    • HAL - - - - IRDA update
    • -
        -
      • Add new - functions and call backs for - Transfer Abort
      • -
          -
        • HAL_IRDA_Abort()
        • -
        • HAL_IRDA_AbortTransmit()
        • -
        • HAL_IRDA_AbortReceive()
        • -
        • HAL_IRDA_Abort_IT()
        • -
        • HAL_IRDA_AbortTransmit_IT()
        • -
        • HAL_IRDA_AbortReceive_IT()
        • -
        • HAL_IRDA_AbortCpltCallback()
        • -
        • HAL_IRDA_AbortTransmitCpltCallback()
        • -
        -
      -
    -
      -
        -
          -
        • HAL_IRDA_AbortReceiveCpltCallback()
        • -
        -
      -
    • HAL - - - - PCD update
    • -
    -
      -
        -
      • Update HAL_PCD_GetRxCount() -  function implementation
      • -
      -
    • HAL - - - - RCC update
    • -
        -
      • Update - __HAL_RCC_HSE_CONFIG() - macro implementation
      • -
      • Update __HAL_RCC_LSE_CONFIG() - macro implementation
      • -
      -
    • HAL - - - - SMARTCARD update
    • -
    -
      -
        -
      • Add new - functions and call backs for - Transfer Abort
      • -
          -
        • HAL_ SMARTCARD_Abort()
        • -
        • HAL_ SMARTCARD_AbortTransmit()
        • -
        • HAL_ SMARTCARD_AbortReceive()
        • -
        • HAL_ SMARTCARD_Abort_IT()
        • -
        • HAL_ SMARTCARD_AbortTransmit_IT()
        • -
        • HAL_ SMARTCARD_AbortReceive_IT()
        • -
        • HAL_ SMARTCARD_AbortCpltCallback()
        • -
        • HAL_ SMARTCARD_AbortTransmitCpltCallback()
        • -
        • HAL_ SMARTCARD_AbortReceiveCpltCallback()
        • -
        -
      -
    • HAL - - - - TIM update
    • -
        -
      • Update HAL_TIMEx_RemapConfig() - function to manage TIM - internal trigger remap: - LPTIM or TIM3_TRGO
      • -
      -
    • HAL - - - - UART update
    • -
        -
      • Add - Transfer abort functions and - callbacks
      • -
      -
    • HAL - - - - USART update
    • -
        -
      • Add - Transfer abort functions and - callbacks
      • -
      -
    -

    V1.5.2 - - - - / 22-September-2016

    -

    Main Changes

    -
      -
    • HAL - - - - I2C update
    • -
        -
      • Fix wrong - behavior in consecutive - transfers in case of single - byte transmission - (Master/Memory Receive
        - interfaces)
      • -
      • Update - HAL_I2C_Master_Transmit_DMA() - / - HAL_I2C_Master_Receive_DMA()/ - HAL_I2C_Slave_Transmit_DMA()
        - and - HAL_I2C_Slave_Receive_DMA() to - manage addressing phase - through interruption instead - of polling
      • -
      • Add - a check on I2C handle - state at start of all I2C - API's to ensure that I2C is ready
      • -
      • Update I2C - API's (Polling, IT and DMA - interfaces) to manage I2C XferSize - and XferCount - handle parameters instead of - API size parameter to help - user to get information of - counter in case of - error. 
      • -
      • Update Abort - functionality to manage DMA - use case
      • -
      -
    • HAL - - - - FMPI2C update
    • -
        -
      • Update to - disable Own Address - before setting the new Own - Address - configuration:
      • -
          -
        • Update - HAL_FMPI2C_Init() - to disable FMPI2C_OARx_EN - bit before any - configuration in OARx - registers
        • -
        -
      -
    • HAL - - - - CAN update
    • -
        -
      • Update CAN - receive processes to set CAN - RxMsg - FIFONumber - parameter
      • -
      -
    • HAL - - - - UART update
    • -
        -
      • Update UART - - - - handle TxXferCount - and RxXferCount parameters - - - - - as volatile to avoid - eventual issue with High Speed - optimization  
      • -
      -
    -

    V1.5.1 - - - - / 01-July-2016

    -

    Main Changes

    -
      -
    • HAL - - - - GPIO update
    • -
        -
      • HAL_GPIO_Init()/HAL_GPIO_DeInit() - API's: - update GPIO_GET_INDEX() - macro implementation to - support all GPIO's
      • -
      -
    • HAL - - - - SPI update
    • -
        -
      • Fix - regression issue: - retore HAL_SPI_DMAPause() - and HAL_SPI_DMAResume() API's -
      • -
      -
    • HAL - - - - RCC update
    • -
        -
      • Fix FSMC - macros compilation warnings - with STM32F412Rx devices
      • -
      -
    • HAL - - - - DMA update
    • -
        -
      • HAL_DMA_PollFortransfer() API clean - up
        -
        -
      • -
      -
    • HAL - - - - PPP update(PPP refers to - IRDA, UART, USART and SMARTCARD)
    • -
        -
      • Update - - - - HAL_PPP_IRQHandler() - to add a check on interrupt - source before managing the - error 
      • -
      -
    -
      -
    • HAL - - - - QSPI update
    • -
        -
      • Implement - workaround to fix the - limitation pronounced - - - - in - the Errata - sheet 2.1.8 section: - In some specific cases, - DMA2 data corruption - occurs when managing AHB - and APB2 peripherals in a - concurrent way
      • -
      -
    -

    V1.5.0 - - - - / 06-May-2016

    -

    Main Changes

    -
      -
    • Add support of STM32F412cx, - - - - - STM32F412rx, STM32F412vx and - STM32F412zx devices
    • -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • Add - new HAL driver for DFSDM peripheral
    • -
    • Enhance - - - - HAL delay and time base - implementation:
    • -
        -
      • Add new - drivers - stm32f4xx_hal_timebase_rtc_alarm_template.c - and - stm32f4xx_hal_timebase_rtc_wakeup_template.c - which override the native HAL - time base functions (defined - as weak) to either use the RTC - as time base tick source. For - more details about the usage - of these drivers, please refer - to HAL\HAL_TimeBase_RTC - examples and - - - - - FreeRTOS-based - - - - - applications
      • -
      -
    • The - - - - following changes done on the - HAL drivers require an update - on the application code based - on HAL V1.4.4
    • -
        -
      • HAL UART, - USART, IRDA, SMARTCARD, SPI, - I2C,FMPI2C, - - - - - QSPI (referenced - as PPP here - - - - - below) drivers
      • -
          -
        • Add PPP - error management during DMA - process. This requires the - following updates - on user application:
        • -
            -
          • Configure - and enable the PPP IRQ in - HAL_PPP_MspInit() - function
          • -
          • In stm32f4xx_it.c - - - - - file, PPP_IRQHandler() - function: add - - - - - a call to HAL_PPP_IRQHandler() - - - - - function
          • -
          • Add and - customize the Error - Callback API: HAL_PPP_ErrorCallback()
          • -
          -
        -
      • HAL I2C, FMPI2C (referenced - as PPP here - - - - - below) drivers:
      • -
          -
        • Update to - avoid waiting on STOPF/BTF/AF - - - - - flag under DMA ISR by using - the PPP - - - - end of transfer interrupt in - the DMA transfer process. This - - - - - requires the following - updates on user - application:
        • -
            -
          • Configure - and enable the PPP IRQ in - HAL_PPP_MspInit() - function
          • -
          • In stm32f4xx_it.c - - - - - file, PPP_IRQHandler() - function: add - - - - - a call to HAL_PPP_IRQHandler() - - - - - function
          • -
          -
        -
      • HAL I2C driver:
      • -
          -
        • I2C - transfer processes IT - update: NACK during - addressing phase is managed - through I2C Error - interrupt instead of - HAL state
        • -
        -
      -
    -
      -
        -
      • HAL IWDG driver: - rework overall driver for - better implementation
      • -
          -
        • Remove HAL_IWDG_Start(), HAL_IWDG_MspInit() - - - - - and HAL_IWDG_GetState() APIs
        • -
        -
      • HAL WWDG driver: - rework overall driver for - better implementation
      • -
          -
        • Remove HAL_WWDG_Start(), HAL_WWDG_Start_IT(), HAL_WWDG_MspDeInit() - - - - - and HAL_WWDG_GetState() - - - - - APIs 
        • -
        • Update - the HAL_WWDG_Refresh(WWDG_HandleTypeDef *hwwdg, - uint32_t counter) -  function and API -  by removing the -  "counter" parameter
        • -
        -
      • HAL QSPI - driver:  Enhance - the DMA transmit process - by using PPP TC - interrupt instead of waiting - on TC flag under DMA - ISR. This requires the - following updates on user - application:
      • -
          -
        • Configure - and enable the QSPI IRQ - in HAL_QSPI_MspInit() - function
        • -
        • In stm32f4xx_it.c - - - - - file, QSPI_IRQHandler() - function: add - - - - a call to HAL_QSPI_IRQHandler() - - - - - function
        • -
        -
      • HAL CEC - driver:  Overall - driver rework with - compatibility break versus - previous HAL version
      • -
          -
        • Remove HAL - CEC polling Process - functions: HAL_CEC_Transmit() - and HAL_CEC_Receive()
        • -
        • Remove HAL - CEC receive interrupt - process function HAL_CEC_Receive_IT() - and enable the "receive" -  mode during the Init - phase
        • -
        • Rename HAL_CEC_GetReceivedFrameSize() - funtion - to HAL_CEC_GetLastReceivedFrameSize()
        • -
        • Add new HAL - APIs: HAL_CEC_SetDeviceAddress() - and HAL_CEC_ChangeRxBuffer()
        • -
        • Remove - the 'InitiatorAddress' - field from the CEC_InitTypeDef - structure and manage - it as a parameter in - the HAL_CEC_Transmit_IT() - function
        • -
        • Add new - parameter 'RxFrameSize' - in HAL_CEC_RxCpltCallback() - function
        • -
        • Move CEC Rx - buffer pointer from CEC_HandleTypeDef - structure to CEC_InitTypeDef - structure
        • -
        -
      -
    -
      -
    • HAL - - - - RCC update
    • -
        -
      • Update HAL_RCC_ClockConfig() - function to adjust the SystemCoreClock
      • -
      • Rename macros - and Literals:
      • -
          -
        • RCC_PERIPHCLK_CK48 by RCC_PERIPHCLK_CLK48
        • -
        • IS_RCC_CK48CLKSOURCE by - - - - - IS_RCC_CLK48CLKSOURCE
        • -
        • RCC_CK48CLKSOURCE_PLLSAIP - - - - - by RCC_CLK48CLKSOURCE_PLLSAIP
        • -
        • RCC_SDIOCLKSOURCE_CK48 - - - - by RCC_SDIOCLKSOURCE_CLK48
        • -
        • RCC_CK48CLKSOURCE_PLLQ - - - - - by RCC_CLK48CLKSOURCE_PLLQ
        • -
        -
      • Update HAL_RCCEx_GetPeriphCLKConfig() - and HAL_RCCEx_PeriphCLKConfig() - - - - - functions to support TIM Prescaler - for STM32F411xx devices
      • -
      • HAL_RCCEx_PeriphCLKConfig() API: update - to fix the RTC clock - configuration issue
      • -
      -
    • HAL - - - - CEC update
    • -
        -
      • Overall - driver rework with break - of compatibility with HAL - V1.4.4
      • -
          -
        • Remove the - HAL CEC polling Process: HAL_CEC_Transmit() - and HAL_CEC_Receive()
        • -
        -
      -
    -
      -
        -
          -
        • Remove the - HAL CEC receive interrupt - process (HAL_CEC_Receive_IT()) - - - - - and manage the "Receive" - mode enable within the Init - phase
        • -
        • Rename HAL_CEC_GetReceivedFrameSize() - function to HAL_CEC_GetLastReceivedFrameSize() - - - - - function
        • -
        • Add new HAL - APIs: HAL_CEC_SetDeviceAddress() - and HAL_CEC_ChangeRxBuffer()
        • -
        • Remove - the 'InitiatorAddress' - field from the CEC_InitTypeDef - structure and manage - it as a parameter in - the HAL_CEC_Transmit_IT() - function
        • -
        • Add new - parameter 'RxFrameSize' - in HAL_CEC_RxCpltCallback() - function
        • -
        • Move CEC Rx - buffer pointer from CEC_HandleTypeDef - structure to CEC_InitTypeDef - structure
        • -
        -
      • Update driver - to implement the new CEC state - machine:
      • -
          -
        • Add - new "rxState" field - - - - - in CEC_HandleTypeDef - structure to provide the CEC state - - - - - information related to Rx Operations
        • -
        • Rename - "state" field in CEC_HandleTypeDef - structure to "gstate": - - - - CEC state - - - - - information related to - global Handle management and - Tx Operations
        • -
        • Update CEC - process to manage the new - CEC states.
        • -
        • Update - __HAL_CEC_RESET_HANDLE_STATE() - macro to handle the new CEC - state parameters (gState, - rxState)
        • -
        -
      -
    -
      -
    • HAL - - - - UART, USART, SMARTCARD and - IRDA (referenced - - - - as PPP here below) update
    • -
        -
      • Update - Polling management:
      • -
          -
        • The user - Timeout value must be - estimated for the overall - process duration: the - Timeout measurement is - cumulative
        • -
        -
      • Update DMA - process:
      • -
          -
        • Update the - management of PPP peripheral - errors during DMA process. - This requires the following - updates in user application:
        • -
            -
          • Configure - and enable the PPP IRQ in - HAL_PPP_MspInit() - function
          • -
          • In - stm32f4xx_it.c file, PPP_IRQHandler() - function: add a call to HAL_PPP_IRQHandler() - - - - - function
          • -
          • Add and - customize the Error - Callback API: HAL_PPP_ErrorCallback()
          • -
          -
        -
      -
    • HAL - - - - FMC update
    • -
        -
      • Update FMC_NORSRAM_Init() - to remove the Burst access - mode configuration
      • -
      • Update FMC_SDRAM_Timing_Init() - to fix initialization issue - when configuring 2 SDRAM banks
      • -
      -
    • HAL - - - - HCD update
    • -
        -
      • Update HCD_Port_IRQHandler() - to unmask disconnect IT only - when the port is disabled
      • -
      -
    • HAL - - - - I2C/FMPI2C - update
    • -
        -
      • Update Polling - - - - - management:
      • -
          -
        • The Timeout - value must be estimated for - the overall process - duration: the - Timeout measurement is - cumulative
        • -
        -
      • Add the - management of Abort - service: Abort DMA - transfer through interrupt
      • -
          -
        • In the case - of Master Abort IT transfer - usage:
        • -
            -
          • Add new - - - - user HAL_I2C_AbortCpltCallback() - to inform user of the end - of abort process
          • -
          • A new - abort state is defined in - the HAL_I2C_StateTypeDef structure
          • -
          -
        -
      • Add the - management of I2C peripheral - errors, ACK failure and STOP - condition detection during DMA - process. This requires the - following updates on user - application:
      • -
          -
        • Configure - and enable the I2C IRQ in - HAL_I2C_MspInit() - function
        • -
        • In - stm32f4xx_it.c file, I2C_IRQHandler() - function: add a call to - HAL_I2C_IRQHandler() - function
        • -
        • Add and - customize the Error Callback - API: HAL_I2C_ErrorCallback()
        • -
        • Refer to - the I2C_EEPROM or - I2C_TwoBoards_ComDMA project - examples usage of the API
        • -
        -
      • NACK error - during addressing phase is - returned through interrupt - instead of previously through - I2C transfer API's
      • -
      • I2C - addressing phase is updated to - be managed using interrupt - instead of polling (Only - for HAL I2C driver)
      • -
          -
        • Add new - static functions to manage - I2C SB, ADDR and ADD10 flags
        • -
        -
      -
    • HAL - - - - SPI update
    • -
    -
      -
        -
      • Overall - driver optimization to improve - performance in - polling/interrupt mode to - reach maximum peripheral frequency
      • -
          -
        • Polling - mode:
        • -
            -
          • Replace - the use of SPI_WaitOnFlagUnitTimeout() - function by "if" statement - to check on RXNE/TXE flage - while transferring data
          • -
          -
        -
      -
    -
      -
        -
          -
        •  Interrupt - - - - mode:
        • -
            -
          • Minimize - access on SPI registers
          • -
          -
        • All modes:
        • -
            -
          • Add the - USE_SPI_CRC switch to - minimize the number of - statements when CRC - calculation is disabled
          • -
          • Update timeout - - - - - management to check on - global processes
          • -
          • Update - error code management in - all processes
          • -
          -
        -
      • Update DMA - process:
      • -
          -
        • Add the - management of SPI peripheral - errors during DMA process. - This requires the following - updates in the user - application:
        • -
            -
          • Configure - and enable the SPI IRQ in - HAL_SPI_MspInit() - function
          • -
          • In - stm32f4xx_it.c file, SPI_IRQHandler() - function: add a call to HAL_SPI_IRQHandler() - - - - - function
          • -
          • Add and - customize the Error - Callback API: HAL_SPI_ErrorCallback()
          • -
          • Refer to - the following example - which describe the - changes: SPI_FullDuplex_ComDMA
          • -
          -
        -
      • Fix - regression in polling mode:
      • -
          -
        • Add - preparing data to transmit - in case of slave mode in HAL_SPI_TransmitReceive() - and HAL_SPI_Transmit()
        • -
        • Add to - manage properly the overrun - flag at the end of a HAL_SPI_TransmitReceive()
        • -
        -
      • Fix - regression in interrupt mode:
      • -
          -
        • Add a wait - on TXE flag in SPI_CloseTx_ISR() - and in SPI_CloseTxRx_ISR()
        • -
        • Add to - manage properly - the overrun flag in SPI_CloseRxTx_ISR() - and SPI_CloseRx_ISR()
        • -
        -
      -
    -
      -
    • HAL - - - - DMA2D update
    • -
        -
      • Update the - HAL_DMA2D_DeInit() - function to:
      • -
          -
        • Abort - transfer in case of ongoing - DMA2D transfer
        • -
        • Reset DMA2D - control registers
        • -
        -
      • Update - HAL_DMA2D_Abort() - to disable DMA2D interrupts - after stopping transfer
      • -
      • Optimize - HAL_DMA2D_IRQHandler() - by reading status registers - only once
      • -
      • Update - HAL_DMA2D_ProgramLineEvent() - function to:
      • -
          -
        • Return HAL - error state in case of wrong - line value
        • -
        • Enable line - interrupt after setting the - line watermark configuration
        • -
        -
      • Add new - HAL_DMA2D_CLUTLoad() - and HAL_DMA2D_CLUTLoad_IT() functions - - - - - to start DMA2D CLUT loading
      • -
          -
        • HAL_DMA2D_CLUTLoading_Abort() - function to abort the DMA2D - CLUT loading
        • -
        • HAL_DMA2D_CLUTLoading_Suspend() - function to suspend the - DMA2D CLUT loading
        • -
        • HAL_DMA2D_CLUTLoading_Resume() - function to resume the DMA2D - CLUT loading
        • -
        -
      • Add new DMA2D - dead time management:
      • -
          -
        • HAL_DMA2D_EnableDeadTime() - function to enable DMA2D - dead time feature
        • -
        • HAL_DMA2D_DisableDeadTime() - function to disable DMA2D - dead time feature
        • -
        • HAL_DMA2D_ConfigDeadTime() - function to configure dead - time
        • -
        -
      • Update the - name of DMA2D Input/Output - color mode defines to be more - - - - clear for - user (DMA2D_INPUT_XXX for - input layers Colors, - DMA2D_OUTPUT_XXX for output - framebuffer Colors)
      • -
      -
    -
      -
    • HAL - - - - LTDC update
    • -
    -
      -
        -
      • Update HAL_LTDC_IRQHandler() - to manage the case of reload - interrupt
      • -
      • Add new - callback API HAL_LTDC_ReloadEventCallback()
      • -
      • Add HAL_LTDC_Reload() - to configure LTDC reload - feature
      • -
      • Add new No - Reload LTDC variant APIs
      • -
          -
        • HAL_LTDC_ConfigLayer_NoReload() to - configure the LTDC Layer - according to the specified - without reloading
        • -
        • HAL_LTDC_SetWindowSize_NoReload() to set - the LTDC window size without - reloading
        • -
        • HAL_LTDC_SetWindowPosition_NoReload() to set - the LTDC window position - without reloading
        • -
        • HAL_LTDC_SetPixelFormat_NoReload() to - reconfigure the pixel format - without reloading
        • -
        • HAL_LTDC_SetAlpha_NoReload() to - reconfigure the layer alpha - value without reloading
        • -
        • HAL_LTDC_SetAddress_NoReload() to - reconfigure the frame buffer - Address without reloading
        • -
        • HAL_LTDC_SetPitch_NoReload() to - reconfigure the pitch for - specific cases
        • -
        • HAL_LTDC_ConfigColorKeying_NoReload() to - configure the color keying - without reloading
        • -
        • HAL_LTDC_EnableColorKeying_NoReload() to enable - the color keying without - reloading
        • -
        • HAL_LTDC_DisableColorKeying_NoReload() to - disable the color keying - without reloading
        • -
        • HAL_LTDC_EnableCLUT_NoReload() to enable - the color lookup table - without reloading
        • -
        • HAL_LTDC_DisableCLUT_NoReload() to - disable the color lookup - table without reloading
        • -
        • Note: Variant - functions with “_NoReload” - post fix allows to set the - LTDC configuration/settings - without immediate reload. - This is useful in case when - the program requires to - modify several LTDC settings - (on one or both layers) then - applying (reload) these - settings in one shot by - calling the function “HAL_LTDC_Reload
        • -
        -
      -
    • HAL - - - - RTC update 
    • -
        -
      • Add new - timeout implementation based - on cpu - cycles - for ALRAWF, ALRBWF - and WUTWF flags
      • -
      -
    -
      -
    • HAL - - - - SAI update
    • -
        -
      • Update SAI - state in case of TIMEOUT error - within the HAL_SAI_Transmit() - / HAL_SAI_Receive()
      • -
      • Update HAL_SAI_IRQHandler:
      • -
          -
        • Add error - management in case DMA - errors through XferAbortCallback() - and HAL_DMA_Abort_IT()
        • -
        • Add error - management in case of IT
        • -
        -
      • Move SAI_BlockSynchroConfig() - and SAI_GetInputClock() - - - - - functions to - stm32f4xx_hal_sai.c/.h files - (extension files are kept - empty for projects - compatibility reason)
      • -
      -
    -
      -
    • HAL - - - - DCMI update
    • -
        -
      • Rename DCMI_DMAConvCplt - to DCMI_DMAXferCplt
      • -
      • Update HAL_DCMI_Start_DMA() - function to Enable the - DCMI peripheral
      • -
      • Add new - timeout implementation based - on cpu - cycles for DCMI stop
      • -
      • Add HAL_DCMI_Suspend() - function to suspend DCMI - capture
      • -
      • Add HAL_DCMI_Resume() - function to resume capture - after DCMI suspend
      • -
      • Update lock - mechanism for DCMI process
      • -
      • Update HAL_DCMI_IRQHandler() - function to:
      • -
          -
        • Add error - management in case DMA - errors through XferAbortCallback() - and HAL_DMA_Abort_IT()
        • -
        • Optimize - code by using direct - register read
        • -
        -
      -
    -
      -
    • HAL - - - - DMA - update
    • -
        -
      • Add new APIs - HAL_DMA_RegisterCallback() - and HAL_DMA_UnRegisterCallback - to register/unregister the - different callbacks identified - by the enum - typedef HAL_DMA_CallbackIDTypeDef
      • -
      • Add new API HAL_DMA_Abort_IT() - to abort DMA transfer under - interrupt context
      • -
          -
        • The new - registered Abort callback is - called when DMA transfer - abortion is completed
        • -
        -
      • Add the check - of compatibility between FIFO - threshold level and size of - the memory burst in the HAL_DMA_Init() - API
      • -
      • Add new Error - Codes: HAL_DMA_ERROR_PARAM, - HAL_DMA_ERROR_NO_XFER and - HAL_DMA_ERROR_NOT_SUPPORTED
      • -
      • Remove all - DMA states related to - MEM0/MEM1 in HAL_DMA_StateTypeDef
      • -
      -
    • HAL - - - - IWDG - update
    • -
        -
      • Overall - rework of the driver for a - more - efficient implementation
      • -
          -
        • Remove the - following APIs:
        • -
            -
          • HAL_IWDG_Start()
          • -
          • HAL_IWDG_MspInit()
          • -
          • HAL_IWDG_GetState()
          • -
          -
        • Update - implementation:
        • -
            -
          • HAL_IWDG_Init(): this - function insures the - configuration and the - start of the IWDG counter
          • -
          • HAL_IWDG_Refresh(): this - function insures the - reload of the IWDG counter
          • -
          -
        • Refer to - the following example to - identify the changes: IWDG_Example
        • -
        -
      -
    • HAL - - - - LPTIM - update
    • -
        -
      • Update HAL_LPTIM_TimeOut_Start_IT() - and HAL_LPTIM_Counter_Start_IT( - ) APIs to configure WakeUp - Timer EXTI interrupt to be - able to wakeup - MCU from low power mode by - pressing the EXTI line.
      • -
      • Update HAL_LPTIM_TimeOut_Stop_IT() - and HAL_LPTIM_Counter_Stop_IT( - ) APIs to disable WakeUp - Timer EXTI interrupt. 
      • -
      -
    • HAL - - - - NOR update
    • -
        -
      • Update - NOR_ADDR_SHIFT macro implementation
      • -
      -
    • HAL - - - - PCD update
    • -
        -
      • Update HAL_PCD_IRQHandler() - to get HCLK frequency before - setting TRDT value
      • -
      -
    • HAL - - - - QSPI - update
    • -
    -
      -
        -
      • Update to - manage QSPI error management - during DMA process
      • -
      • Improve the - DMA transmit process by using - QSPI TC interrupt instead of - waiting loop on TC flag under - DMA ISR
      • -
      • These two - improvements require the - following updates on user - application:
      • -
          -
        • Configure - and enable the QSPI IRQ in HAL_QSPI_MspInit() - function
        • -
        • In - stm32f4xx_it.c file, QSPI_IRQHandler() - function: add a call to HAL_QSPI_IRQHandler() - - - - function
        • -
        • Add and - customize the Error Callback - API: HAL_QSPI_ErrorCallback()
        • -
        -
      • Add the - management of non-blocking - transfer abort service: HAL_QSPI_Abort_IT(). - - - - - In this case the user must:
      • -
          -
        • Add new - callback HAL_QSPI_AbortCpltCallback() - to inform user at the end of - abort process
        • -
        • A new value - of State in the HAL_QSPI_StateTypeDef - provides the current state - during the abort phase
        • -
        -
      • Polling - management update:
      • -
          -
        • The Timeout - value user must be estimated - for the overall process - duration: the - Timeout measurement is - cumulative. 
        • -
        -
      • Refer to the - following examples, which - describe the changes:
      • -
          -
        • QSPI_ReadWrite_DMA
        • -
        • QSPI_MemoryMapped
        • -
        • QSPI_ExecuteInPlace
        • -
        -
      -
    -
      -
        -
      • Add two new - APIs for the QSPI fifo - threshold:
      • -
          -
        • HAL_QSPI_SetFifoThreshold(): - configure the FIFO threshold - of the QSPI
        • -
        • HAL_QSPI_GetFifoThreshold(): give the - current FIFO threshold
        • -
        -
      • Fix wrong - data size management in HAL_QSPI_Receive_DMA()
      • -
      -
    -
      -
    • HAL - - - - ADC update
    • -
        -
      • Add new - __HAL_ADC_PATH_INTERNAL_VBAT_DISABLE() - macro for STM32F42x and - STM32F43x devices to - provide the possibility - to convert VrefInt - channel when both VrefInt - and Vbat - channels are selected.
      • -
      -
    • HAL - - - - SPDIFRX update
    • -
        -
      • Overall driver - update for wait on flag - management optimization 
      • -
      -
    • HAL - - - - WWDG update 
    • -
        -
      • Overall - rework of the driver for more - efficient implementation
      • -
          -
        • Remove the - following APIs:
        • -
            -
          • HAL_WWDG_Start()
          • -
          • HAL_WWDG_Start_IT()
          • -
          • HAL_WWDG_MspDeInit()
          • -
          • HAL_WWDG_GetState()
          • -
          -
        • Update - implementation:
        • -
            -
          • HAL_WWDG_Init()
          • -
              -
            • A new - - - - parameter in the Init - Structure: EWIMode
            • -
            -
          • HAL_WWDG_MspInit()
          • -
          • HAL_WWDG_Refresh(
          • -
              -
            • This - function insures the - reload of the counter
            • -
            • The - "counter" parameter has - been removed
            • -
            -
          • HAL_WWDG_IRQHandler()
          • -
          • HAL_WWDG_EarlyWakeupCallback() is the - new prototype of HAL_WWDG_WakeUpCallback()
          • -
          -
        -
      • Refer to the - following example to identify - the changes: WWDG_Example
      • -
      -
    -

    V1.4.4 - - - - / 22-January-2016

    -

    Main Changes

    -
      -
    • HAL - - - - Generic update
    • -
        -
      • stm32f4xx_hal_conf_template.h
      • -
          -
        • Optimize - HSE Startup Timeout value - from 5000ms to 100 ms
        • -
        • Add new - define LSE_STARTUP_TIMEOUT
        • -
        • Add new - define USE_SPI_CRC for code - cleanup when the CRC - calculation is disabled.
        • -
        -
      • Update HAL - drivers to support MISRA C - 2004 rule 10.6
      • -
      • Add new - template driver to - configure timebase - using TIMER :
      • -
          -
        • stm32f4xx_hal_timebase_tim_template.c
        • -
        -
      -
    -
      -
    • HAL - - - - CAN update
    • -
        -
      • Update HAL_CAN_Transmit() - and HAL_CAN_Transmit_IT() - - - - - functions to unlock - process when all Mailboxes are - busy
      • -
      -
    -
      -
    • HAL - - - - DSI update
    • -
        -
      • Update HAL_DSI_SetPHYTimings() - functions to use the correct - mask
      • -
      -
    • HAL - - - - UART update
    • -
        -
      • Several - update on HAL UART driver to - implement the new UART state - machine: 
      • -
          -
        • Add new - field in UART_HandleTypeDef - structure: "rxState", - - - - - UART state information - related to Rx Operations
        • -
        • Rename - "state" field in UART_HandleTypeDef - structure by "gstate": - - - - UART state information - related to global Handle - management and Tx Operations
        • -
        • Update UART - process to manage the new - UART states.
        • -
        • Update - __HAL_UART_RESET_HANDLE_STATE() - macro to handle the new UART - state parameters (gState, - rxState)
        • -
        -
      • Update - UART_BRR_SAMPLING16() and - UART_BRR_SAMPLING8() Macros to - fix wrong baudrate - calculation.
      • -
      -
    -
      -
    • HAL - - - - IRDA update
    • -
        -
      • Several - update on HAL IRDA driver to - implement the new UART state - machine: 
      • -
          -
        • Add new - field in IRDA_HandleTypeDef - structure: "rxState", - - - - - IRDA state information - related to Rx Operations
        • -
        • Rename - "state" field in UART_HandleTypeDef - structure by "gstate": - - - - IRDA state information - related to global Handle - management and Tx Operations
        • -
        • Update IRDA - process to manage the new - UART states.
        • -
        • Update - __HAL_IRDA_RESET_HANDLE_STATE() - macro to handle the new IRDA - state parameters (gState, - rxState)
        • -
        -
      • Removal of - IRDA_TIMEOUT_VALUE define
      • -
      • Update IRDA_BRR() - Macro to fix wrong baudrate - calculation
      • -
      -
    • HAL - - - - SMARTCARD update
    • -
        -
      • Several - update on HAL SMARTCARD driver - to implement the new UART - state machine: 
      • -
          -
        • Add new - field in SMARTCARD_HandleTypeDef - structure: "rxState", - - - - - SMARTCARDstate - information related to Rx Operations
        • -
        • Rename - "state" field in UART_HandleTypeDef - structure by "gstate": - - - - SMARTCARDstate - information related to - global Handle management and - Tx Operations
        • -
        • Update SMARTCARD - - - - - process to manage the new - UART states.
        • -
        • Update - __HAL_SMARTCARD_RESET_HANDLE_STATE() - macro to handle the - new SMARTCARD state - parameters (gState, - rxState)
        • -
        -
      • Update - SMARTCARD_BRR() - macro to fix wrong baudrate - calculation
      • -
      -
    -
      -
    • HAL  - RCC - update
    • -
        -
      • Add new - default define value for HSI - calibration - "RCC_HSICALIBRATION_DEFAULT"
      • -
      • Optimize - Internal oscillators and PLL - startup timeout 
      • -
      • Update to - avoid the disable for HSE/LSE - oscillators before setting the - new RCC HSE/LSE configuration - and add the following notes in - HAL_RCC_OscConfig() - API description:
      • -
      -
    -

         - - - -       -     -     -     -       * - @note   Transitions LSE - Bypass to LSE On and LSE On to LSE - Bypass are not
    -
        - - - - -     -     -     -     -     -      - *         -     supported by this - API. User should request a - transition to LSE Off
    -
        - - - - -     -     -     -     -     -      - *         -     first and then LSE - On or LSE Bypass.
    -
        - - - - -     -     -     -     -     -      * - @note   Transition HSE - Bypass to HSE On and HSE On to HSE - Bypass are not
    -
        - - - - -     -     -     -     -     -      - *         -     supported by this - API. User should request a - transition to HSE Off
    -
        - - - - -     -     -     -         -      - *         -     first and then HSE - On or HSE Bypass.

    -
      -
        -
      • Optimize - the HAL_RCC_ClockConfig() - API implementation.
      • -
      -
    -
      -
    • HAL - - - - DMA2D update
    • -
        -
      • Update - HAL_DMA2D_Abort() - Function to end current DMA2D - transfer properly
      • -
      • Update - HAL_DMA2D_PollForTransfer() - function to add poll for - background CLUT loading (layer - 0 and layer 1).
      • -
      • Update - HAL_DMA2D_PollForTransfer() - to set the corresponding ErrorCode - in case of error occurrence
      • -
      • Update - HAL_DMA2D_ConfigCLUT() - function to fix wrong CLUT - size and color mode settings
      • -
      • Removal of - useless macro __HAL_DMA2D_DISABLE()
      • -
      • Update - HAL_DMA2D_Suspend() - to manage correctly the case - where no transfer is on going
      • -
      • Update - HAL_DMA2D_Resume() to - - - - - manage correctly the case - where no transfer is on going
      • -
      • Update - HAL_DMA2D_Start_IT() - to enable all required - interrupts before enabling the - transfer.
      • -
      • Add - HAL_DMA2D_CLUTLoad_IT() - Function to allow loading a - CLUT with interruption model.
      • -
      •  Update - HAL_DMA2D_IRQHandler() - to manage the following - cases :
        -
        -
      • -
          -
        • CLUT - transfer complete
        • -
        • CLUT access - error
        • -
        • Transfer - watermark reached
        • -
        -
      • Add new - Callback APIs:
      • -
          -
        •  HAL_DMA2D_LineEventCallback() - to signal a transfer - watermark reached event
        • -
        •  HAL_DMA2D_CLUTLoadingCpltCallback() - to signal a CLUT loading - complete event
        • -
        -
      -
    -
      -
        -
      • Miscellaneous - Improvement:
      • -
          -
        • Add - "HAL_DMA2D_ERROR_CAE" new - define for CLUT Access error - management.
        • -
        • Add     assert_param” - used for parameters check is - now done on the top of the - exported functions : before - locking the process using - __HAL_LOCK
        • -
        -
      -
    -

     

    -
      -
    • HAL - - - - I2C update
    • -
        -
      • Add support - of I2C repeated start feature:
      • -
          -
        • With the - following new API's
        • -
            -
          • HAL_I2C_Master_Sequential_Transmit_IT()
          • -
          • HAL_I2C_Master_Sequential_Receive_IT()
          • -
          • HAL_I2C_Master_Abort_IT()
          • -
          • HAL_I2C_Slave_Sequential_Transmit_IT()
          • -
          • HAL_I2C_Slave_Sequential_Receive_IT()
          • -
          • HAL_I2C_EnableListen_IT()
          • -
          • HAL_I2C_DisableListen_IT()
          • -
          -
        • Add new - user callbacks:
        • -
            -
          • HAL_I2C_ListenCpltCallback()
          • -
          • HAL_I2C_AddrCallback()
          • -
          -
        -
      • Update to - generate STOP condition when a - acknowledge failure error is detected
      • -
      • Several - update on HAL I2C driver to - implement the new I2C state - machine: 
      • -
          -
        • Add new API - to get the I2C mode: - HAL_I2C_GetMode()
        • -
        • Update I2C - process to manage the new - I2C states.
        • -
        -
      • Fix wrong behaviour - in single byte transmission 
      • -
      • Update I2C_WaitOnFlagUntilTimeout() to - - - - - manage the NACK feature.
      • -
      • Update  I2C - transmission process to - support the case data size - equal 0
      • -
      -
    -
      -
    • HAL - - - - FMPI2C update
    • -
        -
      • Add support - of FMPI2C repeated start - feature:
      • -
          -
        • With the - following new API's
        • -
            -
          • HAL_FMPI2C_Master_Sequential_Transmit_IT()
          • -
          • HAL_FMPI2C_Master_Sequential_Receive_IT()
          • -
          • HAL_FMPI2C_Master_Abort_IT()
          • -
          • HAL_FMPI2C_Slave_Sequential_Transmit_IT()
          • -
          • HAL_FMPI2C_Slave_Sequential_Receive_IT()
          • -
          • HAL_FMPI2C_EnableListen_IT()
          • -
          • HAL_FMPI2C_DisableListen_IT()
          • -
          -
        • Add new - user callbacks:
        • -
            -
          • HAL_FMPI2C_ListenCpltCallback()
          • -
          • HAL_FMPI2C_AddrCallback()
          • -
          -
        -
      • Several - update on HAL I2C driver to - implement the new I2C state - machine: 
      • -
          -
        • Add new API - to get the FMPI2C mode: - HAL_FMPI2C_GetMode()
        • -
        • Update - FMPI2C process to manage the - new FMPI2C states.
        • -
        -
      -
    -
      -
    • HAL - - - - SPI update
    • -
        -
      • Major Update - to improve performance in - polling/interrupt mode to - reach max frequency:
      • -
          -
        • Polling mode - - - - :
        • -
            -
          • Replace - use of SPI_WaitOnFlagUnitTimeout() - funnction - by "if" statement to check - on RXNE/TXE flage - while transferring data.
          • -
          • Use API - data pointer instead of - SPI handle data pointer.
          • -
          • Use a Goto - implementation instead of - "if..else" - statements.
          • -
          -
        -
      -
    -
      -
        -
          -
        • Interrupt - mode
        • -
            -
          • Minimize - access on SPI registers.
          • -
          • Split the - SPI modes into dedicated - static functions to - minimize checking - statements under HAL_IRQHandler():
          • -
              -
            • 1lines/2lines - - - - modes
            • -
            • 8 bit/ - 16 bits data formats
            • -
            • CRC - calculation - enabled/disabled.
            • -
            -
          • Remove - waiting loop under ISR - when closing - - - -  the - communication.
          • -
          -
        • All - modes:  
        • -
            -
          • Adding - switch USE_SPI_CRC to - minimize number of - statements when CRC - calculation is disabled.
          • -
          • Update - Timeout management to - check on global process.
          • -
          • Update - Error code management in - all processes.
          • -
          -
        -
      • Add note to - the max frequencies reached in - all modes.
      • -
      • Add note - about Master Receive mode restrictions :
      • -
          -
        • Master - - - - Receive mode restriction:
          -       - (#) In Master - unidirectional receive-only - mode (MSTR =1, BIDIMODE=0, - RXONLY=0) or
          -           - - - - - bidirectional receive mode - (MSTR=1, BIDIMODE=1, - BIDIOE=0), to ensure that - the SPI
          -           - - - - does not initiate a new - transfer the following - procedure has to be - respected:
          -           - - - - (##) HAL_SPI_DeInit()
          -           - - - - (##) HAL_SPI_Init() - - - -
        • -
        -
      -
    -
      -
    • HAL - - - - SAI update
    • -
        -
      • Update for - proper management of the - external synchronization input - selection
      • -
          -
        • update - of HAL_SAI_Init - () funciton
        • -
        • update - definition of SAI_Block_SyncExt - and SAI_Block_Synchronization - groups
        • -
        -
      • Update - SAI_SLOTACTIVE_X -  defines - values
      • -
      • Update HAL_SAI_Init() - function for proper companding - mode management
      • -
      • Update SAI_Transmit_ITxxBit() - functions to add the check on - transfer counter before - writing new data to SAIx_DR - registers
      • -
      • Update SAI_FillFifo() - function to avoid issue when - the number of data to transmit - is smaller than the FIFO size
      • -
      • Update HAL_SAI_EnableRxMuteMode() - function for proper mute - management
      • -
      • Update SAI_InitPCM() - function to support 24bits - configuration
      • -
      -
    • HAL - - - - ETH update
    • -
        -
      • Removal of - ETH MAC debug register defines
      • -
      -
    • HAL - - - - FLASH update
    • -
        -
      • Update FLASH_MassErase() - function to apply correctly - voltage range parameter
      • -
      -
    • HAL - - - - I2S update
    • -
        -
      • Update I2S_DMATxCplt() - and I2S_DMARxCplt() to manage - properly FullDuplex - mode without any risk of - missing data.
      • -
      -
    • LL - - - - FMC update
    • -
        -
      • Update the FMC_NORSRAM_Init() - function to use BurstAccessMode - field properly
      • -
      -
    • LL - - - - FSMC  - - - - update
    • -
        -
      • Update the FSMC_NORSRAM_Init() - function to use BurstAccessMode - field properly
      • -
      -
    -


    -
    -

    -

    V1.4.4 - / 11-December-2015

    -

    Main Changes

    -
      -
    • HAL - - - - Generic update
    • -
        -
      • Update HAL - weak empty callbacks to - prevent unused argument - compilation warnings with some - compilers by calling the - following line:
      • -
          -
        • UNUSED(hppp);
        • -
        -
      • STM32Fxxx_User_Manual.chm - - - - - files regenerated for HAL - V1.4.3
      • -
      -
    • HAL - - - - ETH update 
    • -
        -
      • Update HAL_ETH_Init() - function to add timeout on the - Software reset management
      • -
      -
    -

    V1.4.2 - - - - / 10-November-2015

    -

    Main Changes

    -
      -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • One - - - - change done on the HAL CRYP - requires an update on the - application code based on HAL - V1.4.1
    • -
        -
      • Update HAL_CRYP_DESECB_Decrypt() - API to invert pPlainData - and pCypherData - parameters
      • -
      -
    • HAL - - - - generic - update
    • -
        -
      • Update HAL - weak empty callbacks to - prevent unused argument - compilation warnings with some - compilers by calling the - following line:
      • -
          -
        • UNUSED(hppp);
        • -
        -
      -
    -
      -
    • HAL - - - - CORTEX update
    • -
        -
      • Remove - duplication for - __HAL_CORTEX_SYSTICKCLK_CONFIG() - macro
      • -
      -
    -
      -
    • HAL - - - - HASH update
    • -
        -
      • Rename HAL_HASH_STATETypeDef - to HAL_HASH_StateTypeDef
      • -
      • Rename HAL_HASH_PhaseTypeDef - to HAL_HASH_PhaseTypeDef
      • -
      -
    • HAL - - - - RCC update
    • -
        -
      • Add new - macros __HAL_RCC_PPP_IS_CLK_ENABLED() - to check on Clock - enable/disable status
      • -
      • Update - __HAL_RCC_USB_OTG_FS_CLK_DISABLE() - macro to remove the disable - for the SYSCFG
      • -
      • Update HAL_RCC_MCOConfig() - API to use new defines for the - GPIO Speed
      • -
      • Generic - update to improve the - PLL VCO min - value(100MHz): PLLN, PLLI2S - and PLLSAI min value is 50 - instead of 192
      • -
      -
    • HAL - - - - FLASH update
    • -
        -
      • __HAL_FLASH_INSTRUCTION_CACHE_RESET() - macro: update to reset -  ICRST bit in - the ACR register after - setting it.
      • -
      • Update to - support until 15 FLASH wait - state (FLASH_LATENCY_15) for - STM32F446xx devices -
      • -
      -
    -

    §  - - - - - HAL CRYP update

    -
      -
        -
      • Update HAL_CRYP_DESECB_Decrypt() - API to fix the inverted pPlainData - and pCypherData - parameters issue
      • -
      -
    • HAL - - - - I2S update
    • -
        -
      • Update - HAL_I2S_Init() - API to call - __HAL_RCC_I2S_CONFIG() macro - when external I2S clock is - selected
      • -
      -
    • HAL - - - - LTDC update
    • -
        -
      • Update HAL_LTDC_SetWindowPosition() - API to configure - Immediate reload register - instead of vertical blanking - reload register.
      • -
      -
    • HAL - - - - TIM update
    • -
        -
      • Update HAL_TIM_ConfigClockSource() - API to check only the - required parameters
      • -
      -
    • HAL - - - - NAND update
    • -
        -
      • Update - HAL_NAND_Read_Page()/HAL_NAND_Write_Page()/HAL_NAND_Read_SpareArea() - APIs to manage correctly the - NAND Page access
      • -
      -
    • HAL - - - - CAN update
    • -
        -
      • Update to use - "=" instead of "|=" to clear - flags in the MSR, TSR, RF0R - and RF1R registers
      • -
      -
    • HAL - - - - HCD update
    • -
        -
      • Fix typo in - __HAL_USB_OTG_HS_WAKEUP_EXTI_ENABLE_RISING_FALLING_EDGE() - macro implementation
      • -
      -
    • HAL - - - - PCD update
    • -
        -
      • Update HAL_PCD_IRQHandler() - API to avoid issue - when DMA mode enabled for - Status Phase IN stage
      • -
      -
    • LL - - - - FMC update
    • -
        -
      • Update the FMC_NORSRAM_Extended_Timing_Init() - API to remove the check - on CLKDIvison - and DataLatency - parameters
      • -
      • Update the FMC_NORSRAM_Init() - API to add a check on the PageSize - parameter for STM32F42/43xx - devices
      • -
      -
    • LL - - - - FSMC update
    • -
        -
      • Update the FSMC_NORSRAM_Extended_Timing_Init() - API to remove the check - on CLKDIvison - and DataLatency - parameters
      • -
      -
    -

    V1.4.1 - - - - / 09-October-2015

    -

    Main Changes

    -
      -
    • HAL - - - - DSI update
    • -
        -
      • Update TCCR - register assigned value - in HAL_DSI_ConfigHostTimeouts() - function
      • -
      • Update WPCR - register assigned value - in HAL_DSI_Init(), - - - - - HAL_DSI_SetSlewRateAndDelayTuning(), - - - - - HAL_DSI_SetSlewRateAndDelayTuning(), - - - - - HAL_DSI_SetLowPowerRXFilter() - - - - - / HAL_DSI_SetSDD(), - - - - - HAL_DSI_SetLanePinsConfiguration(), - - - - - HAL_DSI_SetPHYTimings(), - - - - - HAL_DSI_ForceTXStopMode(), - - - - - HAL_DSI_ForceRXLowPower(), - - - - - HAL_DSI_ForceDataLanesInRX(), - - - - - HAL_DSI_SetPullDown() - - - - - and HAL_DSI_SetContentionDetectionOff() - - - - - functions
      • -
      • Update - DSI_HS_PM_ENABLE define value
      • -
      • Implement - workaround for the hardware - limitation: “The time to - activate the clock between HS - transmissions is not - calculated correctly”
      • -
      -
    -

    V1.4.0 - - - - / 14-August-2015

    -

    Main Changes

    -
      -
    • Add - support of STM32F469xx, STM32F479xx, - STM32F410Cx, STM32F410Rx - and STM32F410Tx  - devices
    • -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • Add - new HAL drivers for DSI and LPTIM - - - - - peripherals
    • -
    -
      -
    • HAL - - - - ADC update
    • -
        -
      • Rename - ADC_CLOCKPRESCALER_PCLK_DIV2 - define to - ADC_CLOCK_SYNC_PCLK_DIV2
      • -
      • Rename - ADC_CLOCKPRESCALER_PCLK_DIV4 - define to - ADC_CLOCK_SYNC_PCLK_DIV4
      • -
      • Rename - ADC_CLOCKPRESCALER_PCLK_DIV6 - define to - ADC_CLOCK_SYNC_PCLK_DIV6
      • -
      • Rename - ADC_CLOCKPRESCALER_PCLK_DIV8 - define to - ADC_CLOCK_SYNC_PCLK_DIV8
      • -
      -
    • HAL - - - - CORTEX update
    • -
        -
      • Add specific - API for MPU management
      • -
          -
        • add MPU_Region_InitTypeDef - structure
        • -
        • add new - function HAL_MPU_ConfigRegion()
        • -
        -
      -
    • HAL - - - - DMA update
    • -
        -
      • Overall driver - update for code optimization
      • -
          -
        • add StreamBaseAddress - and StreamIndex - new fields in the DMA_HandleTypeDef - structure
        • -
        • add DMA_Base_Registers - private structure
        • -
        • add static - function DMA_CalcBaseAndBitshift()
        • -
        • update HAL_DMA_Init() - function to use the new - added static function
        • -
        • update HAL_DMA_DeInit() - function to optimize clear - flag operations
        • -
        • update HAL_DMA_Start_IT() - function to optimize - interrupts enable
        • -
        • update HAL_DMA_PollForTransfer() - function to optimize check - on flags
        • -
        • update HAL_DMA_IRQHandler() - function to optimize - interrupt flag management
        • -
        -
      -
    • HAL - - - - FLASH update
    • -
        -
      • update HAL_FLASH_Program_IT() - function by removing the - pending flag clear
      • -
      • update HAL_FLASH_IRQHandler() - function to improve erase - operation procedure
      • -
      • update FLASH_WaitForLastOperation() - function by checking on end of - operation flag
      • -
      -
    • HAL - - - - GPIO update
    • -
        -
      • Rename - GPIO_SPEED_LOW define to - GPIO_SPEED_FREQ_LOW
      • -
      • Rename - GPIO_SPEED_MEDIUM define to - GPIO_SPEED_FREQ_MEDIUM
      • -
      • Rename - GPIO_SPEED_FAST define to - GPIO_SPEED_FREQ_HIGH
      • -
      • Rename - GPIO_SPEED_HIGH define to - GPIO_SPEED_FREQ_VERY_HIGH
      • -
      -
    • HAL - - - - I2S update
    • -
        -
      • Move - I2S_Clock_Source defines to - extension file to properly add - the support of STM32F410xx devices
      • -
      -
    • HAL - - - - LTDC update
    • -
        -
      • rename HAL_LTDC_LineEvenCallback() - function to HAL_LTDC_LineEventCallback()
      • -
      • add new - function HAL_LTDC_SetPitch()
      • -
      • add new - functions HAL_LTDC_StructInitFromVideoConfig() - and HAL_LTDC_StructInitFromAdaptedCommandConfig() - - - - - applicable only to STM32F469xx - and STM32F479xx devices
      • -
      -
    • HAL - - - - PWR update
    • -
        -
      • move - __HAL_PWR_VOLTAGESCALING_CONFIG() - macro to extension file
      • -
      • move - PWR_WAKEUP_PIN2 define to - extension file
      • -
      • add - PWR_WAKEUP_PIN3 define, - applicable only to STM32F10xx - devices
      • -
      • add new - functions HAL_PWREx_EnableWakeUpPinPolarityRisingEdge() - and HAL_PWREx_EnableWakeUpPinPolarityFallingEdge(), - - - - - applicable only to STM32F469xx - and STM32F479xx devices
      • -
      -
    -
      -
    • HAL - - - - RTC update
    • -
        -
      • Update HAL_RTCEx_SetWakeUpTimer() - and HAL_RTCEx_SetWakeUpTimer_IT() - - - - - functions to properly check on - the WUTWF flag
      • -
      -
    • HAL - - - - TIM update
    • -
        -
      • add new - defines TIM_SYSTEMBREAKINPUT_HARDFAULT,  - - - - TIM_SYSTEMBREAKINPUT_PVD - - - - - and - TIM_SYSTEMBREAKINPUT_HARDFAULT_PVD, - applicable only to STM32F410xx - devices
      • -
      -
    -

    V1.3.2 - - - - / 26-June-2015

    -

    Main Changes

    -
      -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • One - - - - changes - done on the HAL may require an - update on the application code - based on HAL V1.3.1
    • -
        -
      • HASH IT - process: update to call the HAL_HASH_InCpltCallback() - at the end of the complete - buffer instead of every each - 512 bits
      • -
      -
    -
      -
    • HAL - - - - RCC update
    • -
        -
      • HAL_RCCEx_PeriphCLKConfig() updates:
      • -
          -
        • Update the - LSE check condition after - backup domain reset: - update to check LSE - ready flag when LSE - oscillator is already - enabled instead of check on - LSE oscillator only when LSE - is used as RTC clock source
        • -
        • Use the - right macro to check the - PLLI2SQ parameters -
        • -
        -
      -
    -
      -
    • HAL - - - - RTC update
    • -
        -
      • __HAL_RTC_TAMPER_TIMESTAMP_EXTI_GET_FLAG() - macro: fix implementation - issue
      • -
      • __HAL_RTC_ALARM_GET_IT(), - - - - - __HAL_RTC_ALARM_CLEAR_FLAG(), - __HAL_RTC_WAKEUPTIMER_CLEAR_FLAG(), - - - - - __HAL_RTC_TIMESTAMP_CLEAR_FLAG() - - - - and - __HAL_RTC_TAMPER_CLEAR_FLAG() - macros implementation changed: - remove unused cast
      • -
      • IS_RTC_TAMPER() - macro: update to use literal - instead of hardcoded - value 
      • -
      • Add new - parameter SecondFraction - in RTC_TimeTypeDef - structure
      • -
      • HAL_RTC_GetTime() API update - to support the new - parameter SecondFraction -
      • -
      -
    • HAL - - - - ADC update
    • -
        -
      • Add new - literal: - ADC_INJECTED_SOFTWARE_START to - be used as possible value for - the ExternalTrigInjecConvEdge - parameter in the ADC_InitTypeDef - structure to select the ADC - software trigger mode.
      • -
      -
    • HAL - - - - FLASH update
    • -
        -
      • FLASH_OB_GetRDP() API update - to return uint8_t instead of FlagStatus
      • -
      •  __HAL_FLASH_GET_LATENCY() - new macro add to get the flash - latency
      • -
      -
    • HAL - - - - SPI update
    • -
        -
      • Fix the wrong - definition of - HAL_SPI_ERROR_FLAG literal
      • -
      -
    • HAL - - - - I2S update
    • -
        -
      • HAL_I2S_Transmit() - API update to check on busy - flag only for I2S slave mode
      • -
      -
    • HAL - - - - CRC update
    • -
        -
      • __HAL_CRC_SET_IDR() - macro implementation change to - use WRITE_REG() instead of - MODIFY_REG()
      • -
      -
    • HAL - - - - DMA2D update
    • -
        -
      • HAL_DMA2D_ConfigLayer() - API update to use "=" instead - of "|=" to erase BGCOLR and - FGCOLR registers before - setting the new configuration
      • -
      -
    • HAL - - - - HASH update
    • -
        -
      • HAL_HASH_MODE_Start_IT() (MODE - - - - stands for MD5, SHA1, - SHA224 and SHA36) updates:
      • -
          -
        • Fix processing - - - - - fail for small input buffers
        • -
        • Update to - unlock the process and - call return - HAL_OK at the end of - HASH processing to avoid - incorrectly repeating software
        • -
        • Update to - properly manage the HashITCounter
        • -
        • Update to - call the HAL_HASH_InCpltCallback() - at the end of the complete - buffer instead of every each - 512 bits
        • -
        -
      • __HAL_HASH_GET_FLAG() - update to  check the - right register when the DINNE - flag  is selected
      • -
      • HAL_HASH_SHA1_Accumulate() - updates:
      • -
          -
        • Add - a call to the new - IS_HASH_SHA1_BUFFER_SIZE() - macro to check the size - parameter. 
        • -
        • Add the - following note in API description
        • -
        -
      -
    -
    -

     * - - - - - @note  - - - - - Input buffer - size in bytes must be a multiple - of 4 otherwise the digest - computation is corrupted.

    -
    -
      -
    • HAL - - - - RTC update
    • -
        -
      • Update to - define hardware - independent literals names:
      • -
          -
        • Rename - RTC_TAMPERPIN_PC13 by - - - -  RTC_TAMPERPIN_DEFAULT
        • -
        • Rename - RTC_TAMPERPIN_PA0 by - RTC_TAMPERPIN_POS1
        • -
        • Rename - RTC_TAMPERPIN_PI8 by - RTC_TAMPERPIN_POS1
        • -
        • Rename - RTC_TIMESTAMPPIN_PC13 by - RTC_TIMESTAMPPIN_DEFAULT
        • -
        • Rename - RTC_TIMESTAMPPIN_PA0 by - RTC_TIMESTAMPPIN_POS1
        • -
        • Rename - RTC_TIMESTAMPPIN_PI8 by - RTC_TIMESTAMPPIN_POS1
        • -
        -
      -
    • HAL - - - - ETH update
    • -
        -
      • Remove - duplicated IS_ETH_DUPLEX_MODE() - and IS_ETH_RX_MODE() macros
      • -
      • Remove - illegal space - ETH_MAC_READCONTROLLER_FLUSHING - macro
      • -
      • Update - ETH_MAC_READCONTROLLER_XXX - defined values (XXX can be - IDLE, READING_DATA and - READING_STATUS)
      • -
      -
    • HAL - - - - PCD update
    • -
        -
      • HAL_PCD_IRQHandler API: fix the - bad Configuration of - Turnaround Time
      • -
      -
    • HAL - - - - HCD update
    • -
        -
      • Update to use - local variable in USB - Host channel re-activation
      • -
      -
    • LL - - - - FMC update
    • -
        -
      • FMC_SDRAM_SendCommand() API: remove - the following line: return - HAL_ERROR;
      • -
      -
    • LL - - - - USB update
    • -
        -
      • USB_FlushTxFifo API: - update to flush all Tx FIFO
      • -
      • Update to use - local variable in USB - Host channel re-activation
      • -
      -
    -

    V1.3.1 - - - - / 25-Mars-2015

    -

    Main Changes

    -
      -
    • HAL - - - - PWR update
    • -
        -
      • Fix - compilation issue with - STM32F417xx product: - update STM32F17xx - by STM32F417xx
      • -
      -
    • HAL - - - - SPI update
    • -
        -
      • Remove unused - variable to avoid warning with - TrueSTUDIO 
      • -
      -
    • HAL - - - - I2C update
    • -
        -
      • I2C - Polling/IT/DMA processes: move - the wait loop on busy - flag at the top of the - processes, to ensure that - software not perform any write - access to I2C_CR1 register - before hardware - clearing STOP bit and to - avoid - - - - - also the - waiting loop on BUSY flag - under I2C/DMA ISR.
      • -
      • Update busy - flag Timeout value
      • -
      • I2C Master - Receive Processes update to - disable ACK before generate - the STOP 
      • -
      -
    • HAL - - - - DAC update
    • -
        -
      • Fix V1.3.0 - regression issue with DAC - software trigger configuration
      • -
      -
    -

    V1.3.0 - - - - / 09-Mars-2015

    -

    Main Changes

    -
      -
    • Add - support of STM32F446xx devices
    • -
    • General - - - - updates to fix known defects and - enhancements implementation
    • -
    • Add - new HAL drivers for CEC, - QSPI, FMPI2C and SPDIFRX - - - - peripherals
    • -
    • Two - - - - changes done on the HAL - requires an update on the - application code based on HAL - V1.2.0
    • -
        -
      • Overall SAI - driver rework to have - exhaustive support of the - peripheral features: details - are provided in HAL SAI update - - - - section below --> Compatibility - - - - - with previous version is impacted
      • -
      • CRYP driver - updated to support multi instance,so - user must ensure that the - new parameter Instance is - initalized - in his application(CRYPHandle.Instance - = CRYP) 
      • -
      -
    -
      -
    • HAL - - - - Generic update
    • -
        -
      • stm32f4xx_hal_def.h
      • -
          -
        • Remove NULL - definition and add - include for stdio.h
        • -
        -
      • stm32_hal_legacy.h
      • -
          -
        • Update method - - - - to manage deference in - alias implementation between - all STM32 families
        • -
        -
      • stm32f4xx_hal_ppp.c
      • -
          -
        • HAL_PPP_Init(): update - to force the - HAL_PPP_STATE_RESET before - calling the HAL_PPP_MspInit()
        • -
        -
      -
    -
      -
    • HAL - - - - RCC update
    • -
        -
      • Add new - function HAL_RCCEx_GetPeriphCLKFreq()
      • -
      • Move RCC_PLLInitTypeDef - structure to extension file - and add the new PLLR field - specific to STM32F446xx devices
      • -
      • Move the - following functions to - extension file and add a - __weak attribute in generic driver - - - - : this - update is related to new - system clock source (PLL/PLLR) - added and only available for - STM32F44xx devices
      • -
          -
        • HAL_RCC_OscConfig()
        • -
        • HAL_RCC_GetSysClockFreq()
        • -
        • HAL_RCC_GetOscConfig()
        • -
        -
      • Move the - following macro to extension - file as they have device - dependent implementation
      • -
          -
        • __HAL_RCC_PLL_CONFIG()
        • -
        • __HAL_RCC_PLLI2S_CONFIG()
        • -
        • __HAL_RCC_I2S_CONFIG()
        • -
        -
      • Add new - structure RCC_PLLI2SInitTypeDef - containing new PLLI2S - division factors used only w/ - STM32F446xx devices
      • -
      • Add new - structure RCC_PLLSAIInitTypeDef - containing new PLLSAI - division factors used only w/ - STM32F446xx devices
      • -
      • Add new RCC_PeriphCLKInitTypeDef - to support the peripheral - source clock selection for (I2S, - - - - SAI, SDIO, FMPI2C, CEC, - SPDIFRX and CLK48)
      • -
      • Update the HAL_RCCEx_PeriphCLKConfig() - and HAL_RCCEx_GetPeriphCLKConfig() - - - - - functions to support the - new peripherals Clock source - selection
      • -
      • Add __HAL_RCC_PLL_CONFIG() - macro (the number of parameter - and the implementation depend - on the device part number)
      • -
      • Add __HAL_RCC_PLLI2S_CONFIG() - macro(the number of parameter - and the implementation depend - on device part number)
      • -
      • Update __HAL_RCC_PLLSAI_CONFIG() - macro to support new PLLSAI - factors (PLLSAIM and - PLLSAIP)
      • -
      • Add new - macros for clock - enable/Disable for the - following peripherals (CEC, - - - - SPDIFRX, SAI2, QUADSPI)
      • -
      • Add the - following new macros for clock - source selection - - - - :
      • -
          -
        • __HAL_RCC_SAI1_CONFIG() - / - __HAL_RCC_GET_SAI1_SOURCE()
        • -
        • __HAL_RCC_SAI2_CONFIG() - / - __HAL_RCC_GET_SAI2_SOURCE()
        • -
        • __HAL_RCC_I2S1_CONFIG() - / - __HAL_RCC_GET_I2S1_SOURCE()
        • -
        • __HAL_RCC_I2S2_CONFIG() - / - __HAL_RCC_GET_I2S2_SOURCE()
        • -
        • __HAL_RCC_CEC_CONFIG() - / - __HAL_RCC__GET_CEC_SOURCE() -
        • -
        • __HAL_RCC_FMPI2C1_CONFIG() - / - __HAL_RCC_GET_FMPI2C1_SOURCE() -
        • -
        • __HAL_RCC_SDIO_CONFIG() - / - __HAL_RCC_GET_SDIO_SOURCE() -
        • -
        • __HAL_RCC_CLK48_CONFIG() - / - __HAL_RCC_GET_CLK48_SOURCE() -
        • -
        • __HAL_RCC_SPDIFRXCLK_CONFIG() - / - __HAL_RCC_GET_SPDIFRX_SOURCE()
        • -
        -
      • __HAL_RCC_PPP_CLK_ENABLE(): - - - - - Implement workaround to cover - RCC limitation regarding - peripheral enable delay
      • -
      • HAL_RCC_OscConfig() fix - issues: 
      • -
          -
        • Add a check - on LSERDY flag when - LSE_BYPASS is selected as - new state for LSE - oscillator.
        • -
        -
      • Add - new possible value RCC_PERIPHCLK_PLLI2S - - - - to be selected as PeriphClockSelection - parameter in the - - - - -  RCC_PeriphCLKInitTypeDef - structure to allow the - possibility to output the - PLLI2S on MCO without - activating the I2S or the SAI.
      • -
      • __HAL_RCC_HSE_CONFIG() -  macro: add - the comment below:
      • -
      -
    -
    -

     * - - - - - @note   Transition - HSE Bypass to HSE On and HSE - On to HSE Bypass are not - supported by this macro.
    -  *         - - - - User should request a - transition to HSE Off first - and then HSE On or HSE Bypass.

    -
    -
      -
        -
      • __HAL_RCC_LSE_CONFIG()  macro: add - the comment below:
      • -
      -
    -
    -

      * - - - - - @note   Transition - LSE Bypass to LSE On and LSE - On to LSE Bypass are not - supported by this macro.
    -   - *         - User should request a - transition to LSE Off first - and then LSE On or LSE Bypass.

    -
    -
      -
        -
      • Add the - following new macros for - PLL source and PLLM selection - - - - :
      • -
          -
        • __HAL_RCC_PLL_PLLSOURCE_CONFIG()
        • -
        • __HAL_RCC_PLL_PLLM_CONFIG()
        • -
        -
      • Macros - rename:
      • -
          -
        • HAL_RCC_OTGHS_FORCE_RESET() -by HAL_RCC_USB_OTG_HS_FORCE_RESET()
        • -
        • HAL_RCC_OTGHS_RELEASE_RESET() -by HAL_RCC_USB_OTG_HS_RELEASE_RESET()
        • -
        • HAL_RCC_OTGHS_CLK_SLEEP_ENABLE() -by HAL_RCC_USB_OTG_HS_CLK_SLEEP_ENABLE()
        • -
        • HAL_RCC_OTGHS_CLK_SLEEP_DISABLE() -by HAL_RCC_USB_OTG_HS_CLK_SLEEP_DISABLE()
        • -
        • HAL_RCC_OTGHSULPI_CLK_SLEEP_ENABLE() -by HAL_RCC_USB_OTG_HS_ULPI_CLK_SLEEP_ENABLE()
        • -
        • HAL_RCC_OTGHSULPI_CLK_SLEEP_DISABLE() -by HAL_RCC_USB_OTG_HS_ULPI_CLK_SLEEP_DISABLE()
        • -
        -
      • Add __HAL_RCC_SYSCLK_CONFIG() - new macro to configure the - system clock source (SYSCLK)
      • -
      • __HAL_RCC_GET_SYSCLK_SOURCE() - updates:
      • -
          -
        • Add new RCC - Literals:
        • -
            -
          • RCC_SYSCLKSOURCE_STATUS_HSI
          • -
          • RCC_SYSCLKSOURCE_STATUS_HSE
          • -
          • RCC_SYSCLKSOURCE_STATUS_PLLCLK
          • -
          • RCC_SYSCLKSOURCE_STATUS_PLLRCLK
          • -
          -
        •  Update - - - - macro description to refer - to the literals above -
        • -
        -
      -
    • HAL - - - - PWR update
    • -
        -
      • Add new - define PWR_WAKEUP_PIN2
      • -
      • Add new API - to Control/Get VOS bits - of CR register
      • -
          -
        • HAL_PWR_HAL_PWREx_ControlVoltageScaling()
        • -
        • HAL_PWREx_GetVoltageRange()
        • -
        -
      • __HAL_PWR_ - VOLTAGESCALING_CONFIG(): Implement - workaround to cover VOS - limitation delay when PLL is - enabled after setting the VOS - configuration
      • -
      -
    • HAL - - - - GPIO update
    • -
        -
      • Add the new - Alternate functions literals - related to remap for SPI, - - - - USART, I2C, SPDIFRX, CEC - and QSPI
      • -
      • HAL_GPIO_DeInit(): - - - - Update to check if GPIO - Pin x is already used in EXTI - mode on another GPIO Port - before De-Initialize the EXTI - registers
      • -
      -
    • HAL - - - - FLASH update
    • -
        -
      • __HAL_FLASH_INSTRUCTION_CACHE_RESET() - macro: update to reset -  ICRST bit in - the ACR register after - setting it.
      • -
      • __HAL_FLASH_DATA_CACHE_RESET() macro: - - - - - update to reset -  DCRST bit in the ACR - register after setting it.
      • -
      -
    • HAL - - - - ADC update
    • -
        -
      • Add new - literal: ADC_SOFTWARE_START to - be used as possible value for - the ExternalTrigConv - parameter in the ADC_InitTypeDef - structure to select the ADC - software trigger mode.
      • -
      • IS_ADC_CHANNEL() - macro update to don't assert - stop the ADC_CHANNEL_TEMPSENSOR - value
      • -
      • HAL_ADC_PollForConversion(): update to - manage particular case when - ADC configured in DMA mode and - ADC sequencer with several - ranks and polling for end of - each conversion
      • -
      • HAL_ADC_Start()/HAL_ADC_Start_IT() - /HAL_ADC_Start_DMA() - - - - - update:
      • -
          -
        • unlock the - process before starting the - ADC software conversion.
        • -
        • Optimize - the ADC stabilization delays
        • -
        -
      • __HAL_ADC_GET_IT_SOURCE() - update macro implementation
      • -
      • Add more - details in 'How to use this - driver' section
      • -
      -
    • HAL - - - - DAC update
    • -
        -
      • Add new macro - to check if the specified DAC - interrupt source is enabled or - disabled
      • -
          -
        • __HAL_DAC_GET_IT_SOURCE()
        • -
        -
      • HAL_DACEx_TriangleWaveGeneration() update to - use DAC CR bit mask definition
      • -
      • HAL_DACEx_NoiseWaveGeneration() update to - use DAC CR bit mask definition
      • -
      -
    • HAL - - - - CAN update
    • -
        -
      • CanTxMsgTypeDef structure: - update to use uint8_t Data[8] - - - - - instead of - uint32_t Data[8]
      • -
      • CanRxMsgTypeDef structure: - update to use uint8_t Data[8] - instead of - uint32_t Data[8]
      • -
      -
    -
      -
    • HAL - - - - RTC update
    • -
        -
      • Update to - use CMSIS mask definition - instead of hardcoded values (EXTI_IMR_IM17, - EXTI_IMR_IM19..)
      • -
      -
    • HAL - - - - LTDC update
    • -
        -
      • LTDC_SetConfig() update to - allow the drawing - of partial bitmap in - active layer.
      • -
      -
    • HAL - - - - USART update
    • -
        -
      • HAL_USART_Init() fix USART - baud rate configuration - issue: USART baud rate is - twice Higher than expected
      • -
      -
    • HAL - - - - SMARTCARD update
    • -
        -
      • HAL_SMARTCARD_Transmit_IT() update to - force the disable for the ERR - interrupt to avoid the OVR - interrupt
      • -
      • HAL_SMARTCARD_IRQHandler() - update check condition - for transmission end
      • -
      • Clean up: - remove the following - literals that aren't used in - smartcard mode
      • -
          -
        • SMARTCARD_PARITY_NONE
        • -
        • SMARTCARD_WORDLENGTH_8B
        • -
        • SMARTCARD_STOPBITS_1
        • -
        • SMARTCADR_STOPBITS_2
        • -
        -
      -
    • HAL - - - - SPI update
    • -
        -
      • HAL_SPI_Transmit_DMA()/HAL_SPI_Receive_DMA()/HAL_SPI_TarnsmitReceive_DMA() - update to unlock - the process before - enabling the SPI peripheral
      • -
      • HAL_SPI_Transmit_DMA() update to - manage correctly the DMA RX - stream in SPI Full duplex mode
      • -
      • Section - SPI_Exported_Functions_Group2 update - to remove duplication in *.chm - UM
      • -
      -
    • HAL - - - - CRYP update
    • -
        -
      • Update to - manage multi instance:
      • -
          -
        • Add new - parameter Instance in the CRYP_HandleTypeDef - Handle structure.
        • -
        • Add new - parameter in all HAL CRYP - macros
        • -
            -
          • example: __HAL_CRYP_ENABLE() -  updated by - __HAL_CRYP_ENABLE(__HANDLE__)
          • -
          -
        -
      -
    • HAL - - - - DCMI update
    • -
        -
      • Add an - extension - driver stm32f4xx_hal_dcmi_ex.c/h - to manage the support of new - Black and White feature -
      • -
      • Add  __weak attribute - for HAL_DCMI_Init() - function and add a new - implementation in the - extension driver to manage the - black and white configuration - only available in the  - STM32F446xx devices. -
      • -
      • Move DCMI_InitTypeDef - structure to extension driver - and add the - following new fields - related to black and white - feature: ByteSelectModeByteSelectStartLineSelectMode - and LineSelectStart
      • -
      -
    • HAL - - - - PCD update
    • -
        -
      • Add the - support of LPM feature
      • -
          -
        • add PCD_LPM_StateTypeDef - enum
        • -
        • update PCD_HandleTypeDef - structure to support the LPM - feature
        • -
        • add new - functions HAL_PCDEx_ActivateLPM(), - - - - - HAL_PCDEx_DeActivateLPM() - - - - - and HAL_PCDEx_LPM_Callback() - - - - - in the - stm32f4xx_hal_pcd_ex.h/.c - files
        • -
        -
      -
    • HAL - - - - TIM update
    • -
        -
      • Add  - TIM_TIM11_SPDIFRX - - - - define
      • -
      -
    • HAL - - - - SAI update
    • -
        -
      • Add - stm32f4xx_hal_sai_ex.h/.c - files for the SAI_BlockSynchroConfig() - and the SAI_GetInputClock() - - - - - management
      • -
      • Add new - defines HAL_SAI_ERROR_AFSDET, - HAL_SAI_ERROR_LFSDET, - HAL_SAI_ERROR_CNREADY, - HAL_SAI_ERROR_WCKCFG, - HAL_SAI_ERROR_TIMEOUT in the SAI_Error_Code - group
      • -
      • Add new - defines SAI_SYNCEXT_DISABLE, - SAI_SYNCEXT_IN_ENABLE, - SAI_SYNCEXT_OUTBLOCKA_ENABLE, - SAI_SYNCEXT_OUTBLOCKB_ENABLE - for the SAI External - synchronization
      • -
      • Add new - defines SAI_I2S_STANDARD, - SAI_I2S_MSBJUSTIFIED, - SAI_I2S_LSBJUSTIFIED, - SAI_PCM_LONG and SAI_PCM_SHORT - for the SAI Supported protocol
      • -
      • Add new - defines - SAI_PROTOCOL_DATASIZE_16BIT, - SAI_PROTOCOL_DATASIZE_16BITEXTENDED, - SAI_PROTOCOL_DATASIZE_24BIT - and - SAI_PROTOCOL_DATASIZE_32BIT - for SAI protocol data size
      • -
      • Add SAI - Callback prototype definition
      • -
      • Update SAI_InitTypeDef - structure by adding new - fields: SynchroExt, - Mckdiv, - MonoStereoMode, - CompandingMode, - TriState
      • -
      • Update SAI_HandleTypeDef - structure:
      • -
          -
        • remove - uint16_t *pTxBuffPtr, - *pRxBuffPtr, - TxXferSize, - RxXferSize, - TxXferCount - and RxXferCount - and replace them - respectively by uint8_t *pBuffPtr, - uint16_t XferSize and - - - - - uint16_t XferCount
        • -
        • add mutecallback - field
        • -
        • add struct - __SAI_HandleTypeDef - *hsai field
        • -
        -
      • Remove - SAI_CLKSOURCE_PLLR and - SAI_CLOCK_PLLSRC defines
      • -
      • Add - SAI_CLKSOURCE_NA define
      • -
      • Add - SAI_AUDIO_FREQUENCY_MCKDIV - define
      • -
      • Add - SAI_SPDIF_PROTOCOL define
      • -
      • Add - SAI_SYNCHRONOUS_EXT define
      • -
      • Add new - functions HAL_SAI_InitProtocol(), - - - - - HAL_SAI_Abort(), - - - - - HAL_SAI_EnableTxMuteMode(), - - - - - HAL_SAI_DisableTxMuteMode(), - - - - - HAL_SAI_EnableRxMuteMode(), - - - - - HAL_SAI_DisableRxMuteMode()
      • -
      • Update HAL_SAI_Transmit(), - - - - - HAL_SAI_Receive(), - - - - - HAL_SAI_Transmit_IT(), - - - - - HAL_SAI_Receive_IT(), - - - - - HAL_SAI_Transmit_DMA(), - - - - - HAL_SAI_Receive_DMA() - - - - - functions to use uint8_t *pData - instead of uint16_t *pData - --> This update is mainly - impacting the compatibility - with previous driver - version.
      • -
      -
    • HAL - - - - I2S update
    • -
        -
      • Split the - following - functions between Generic - and Extended API based on full - duplex management and add the - attribute __weak in the - Generic API
      • -
          -
        • HAL_I2S_Init(), -HAL_I2S_DMAPause(), HAL_I2S_DMAStop(), HAL_I2S_DMAResume(), HAL_I2S_IRQHandle() -
        • -
        -
      • Move the - following static functions - from generic to extension driver
      • -
          -
        •  I2S_DMARxCplt() - and I2S_DMATxCplt()
        • -
        -
      • Remove static - attribute from I2S_Transmit_IT() - and I2S_Receive_IT() functions
      • -
      • Move I2SxEXT() - macro to extension file
      • -
      • Add - I2S_CLOCK_PLLR and - I2S_CLOCK_PLLSRC defines for - I2S clock source
      • -
      • Add new - function I2S_GetInputClock()
      • -
      -
    • HAL - - - - LL FMC update
    • -
        -
      • Add WriteFifo - and PageSize - fields in the FMC_NORSRAM_InitTypeDef - structure
      • -
      • Add - FMC_PAGE_SIZE_NONE, - FMC_PAGE_SIZE_128, - FMC_PAGE_SIZE_256, - FMC_PAGE_SIZE_1024, - FMC_WRITE_FIFO_DISABLE, - FMC_WRITE_FIFO_ENABLE defines
      • -
      • Update FMC_NORSRAM_Init(), - - - - - FMC_NORSRAM_DeInit() - - - - - and FMC_NORSRAM_Extended_Timing_Init() functions
      • -
      -
    • HAL - - - - LL USB update
    • -
        -
      • Update USB_OTG_CfgTypeDef - structure to support LPM, lpm_enable - field added
      • -
      • Update USB_HostInit() - and USB_DevInit() - - - - - functions to support the VBUS - Sensing B activation
      • -
      -
    -

    V1.2.0 - - - - / 26-December-2014

    -

    Main Changes

    -
      -
    • Maintenance - - - - release to fix known defects - and enhancements implementation
    • -
    -
      -
    • Macros - - - - and literals renaming to - ensure compatibles across - STM32 series, - backward compatibility - maintained thanks to new added - file stm32_hal_legacy.h under - - - - /Inc/Legacy
    • -
    • Add - *.chm UM for all drivers, a UM - is provided for each superset RPN
    • -
    • Update - - - - drivers to be C++ compliant
    • -
    • Several - - - - update on source code - formatting, for better UM - generation (i.e. - Doxygen - tags updated)
    • -
    • Two - - - - changes done on the HAL - requires an update on the - application code based on HAL - V1.1.0
    • -
        -
      • LSI_VALUE constant has - been corrected in - stm32f4xx_hal_conf.h file, its - value changed from 40 KHz - to 32 KHz
      • -
      • UART, USART, - IRDA and SMARTCARD - (referenced as PPP - here below) drivers: - in DMA transmit process, the - code has been updated to avoid - waiting on TC flag under DMA - ISR, PPP TC interrupt - is used instead. Below the - update to be done on user - application:
      • -
          -
        • Configure - and enable the USART IRQ in - HAL_PPP_MspInit() - function
        • -
        • In - stm32f4xx_it.c file, PPP_IRQHandler() - function: add a call to HAL_PPP_IRQHandler() - - - - function
        • -
        -
      -
    -
      -
    • HAL - - - - generic - update
    • -
    -
      -
        -
      • stm32f4xx_hal_def.h
      • -
          -
        • Update NULL - definition to fix C++ - compilation issue
        • -
        • Add UNUSED() - macro
        • -
        • Add a new - define __NOINLINE to be used - for the no inline code - independent from tool chain
        • -
        -
      • stm32f4xx_hal_conf_template.h
      • -
          -
        • LSI_VALUE constant - has been corrected, - its value changed from 40 KHz - to 32 KHz
        • -
        -
      -
    -
      -
        -
      • Update all - macros and literals naming to - be uper - case
      • -
      • ErrorCode parameter in - PPP_HandleTypeDef - structure updated - to uint32_t instead - of enum HAL_PPP_ErrorTypeDef
      • -
      • Remove the - - - - unused FLAG and IT assert macros
      • -
      -
    • HAL - - - - ADC update
    • -
        -
      • Fix temperature - - - - - sensor channel configuration - issue for STM32F427/437xx - - - - -  and STM32F429/439xx - - - - devices
      • -
      -
    • HAL - - - - DAC update
    • -
        -
      • HAL_DAC_ConfigChannel(): update the - access to the DAC peripheral - registers via the hdac - handle instance
      • -
      • HAL_DAC_IRQHandler(): update to - check on both DAC_FLAG_DMAUDR1 - and DAC_FLAG_DMAUDR2
      • -
      • HAL_DACEx_NoiseWaveGenerate(): update to - reset DAC CR register before - setting the new DAC - configuration
      • -
      • HAL_DACEx_TriangleWaveGenerate(): update to - reset DAC CR register before - setting the new DAC - configuration
      • -
      -
    • HAL - - - - CAN update
    • -
        -
      • Unlock the - CAN process when communication - error occurred
      • -
      -
    • HAL - - - - CORTEX update
    • -
        -
      • Add new macro - IS_NVIC_DEVICE_IRQ() - to check on negative values of - IRQn - parameter
      • -
      -
    -

    §  - - - - - HAL CRYP update

    -
      -
        -
      • HAL_CRYP_DESECB_Decrypt_DMA(): fix the - inverted pPlainData - and pCypherData - parameters issue
      • -
      • CRYPEx_GCMCCM_SetInitVector(): remove - the IVSize - parameter as the key length - 192bits and 256bits are not - supported by this version
      • -
      • Add restriction for - - - - - the CCM Encrypt/Decrypt API's - that only DataType - equal to 8bits is supported
      • -
      • HAL_CRYPEx_AESGCM_Finish():
      • -
          -
        • Add restriction - - - - - that the implementation is - limited to 32bits inputs - data length  (Plain/Cyphertext, - - - - Header) compared with GCM stadards - specifications (800-38D)
        • -
        • Update Size - parameter on 32bits instead - of 16bits
        • -
        • Fix issue - with 16-bit Data Type: - update to use intrinsic __ROR() - instead of __REV16()
        • -
        -
      -
    -

    §  - - - - - HAL DCMI update

    -
      -
        -
      • HAL_DCMI_ConfigCROP(): Invert - assert macros to check Y0 and - Ysize - parameters
      • -
      -
    -

    §  - - - - - HAL DMA update

    -
      -
        -
      • HAL_DMA_Init(): Update to - - - - - clear the DBM bit in the - SxCR - register before setting the - new configuration
      • -
      • DMA_SetConfig(): - add to clear the DBM - bit in the SxCR - register
      • -
      -
    -

    §  - - - - - HAL FLASH update

    -
      -
        -
      • Add "HAL_" - prefix in the defined values - for the FLASH error code
      • -
          -
        • Example: FLASH_ERROR_PGP - renamed by HAL_FLASH_ERROR_PGP
        • -
        -
      • Clear the - - - - Flash ErrorCode - in the FLASH_WaitForLastOperation() - function
      • -
      • Update FLASH_SetErrorCode() - function to use "|=" - operant to update the Flash ErrorCode - parameter in the FLASH handle
      • -
      • IS_FLASH_ADDRESS(): Update the - macro check using '<=' - condition instead of '<'
      • -
      • IS_OPTIONBYTE(): Update the - macro check using '<=' - condition instead of '<'
      • -
      • Add "FLASH_" - - - - - prefix in the defined values - of FLASH Type Program - parameter
      • -
          -
        • Example: TYPEPROGRAM_BYTE - renamed by FLASH_TYPEPROGRAM_BYTE
        • -
        -
      • Add "FLASH_" - - - - - prefix in the defined values - of FLASH Type Erase parameter
      • -
          -
        • Example: TYPEERASE_SECTORS - renamed by FLASH_TYPEERASE_SECTORS
        • -
        -
      • Add "FLASH_" - - - - - prefix in the defined values - of FLASH Voltage Range - parameter
      • -
          -
        • Example: VOLTAGE_RANGE_1 - renamed by FLASH_VOLTAGE_RANGE_1
        • -
        -
      • Add "OB_" - - - - - prefix in the defined values - of FLASH WRP State parameter
      • -
          -
        • Example: WRPSTATE_ENABLE - renamed by OB_WRPSTATE_ENABLE
        • -
        -
      • Add "OB_" - - - - - prefix in the defined values - of the FLASH PCROP State - parameter
      • -
          -
        • PCROPSTATE_DISABLE  - updated by OB_PCROP_STATE_DISABLE
        • -
        • PCROPSTATE_ENABLE -  updated by OB_PCROP_STATE_ENABLE
        • -
        -
      • Change - "OBEX" prefix by - "OPTIONBYTE" prefix in these - defines:
      • -
          -
        • OBEX_PCROP - - - - by OPTIONBYTE_PCROP 
        • -
        • OBEX_BOOTCONFIG - - - - by OPTIONBYTE_BOOTCONFIG
        • -
        -
      -
    -

    §  - - - - - HAL ETH update

    -
      -
        -
      • Fix macros - naming typo
      • -
      -
    -
      -
        -
          -
        • Update - __HAL_ETH_EXTI_SET_RISING_EGDE_TRIGGER() - by - __HAL_ETH_EXTI_SET_RISING_EDGE_TRIGGER()
        • -
        • Update - __HAL_ETH_EXTI_SET_FALLING_EGDE_TRIGGER() -by __HAL_ETH_EXTI_SET_FALLING_EDGE_TRIGGER()
        • -
        -
      -
    -

    §  - - - - - HAL PWR update

    -
      -
        -
      • Add new API - to manage SLEEPONEXIT and - SEVONPEND bits of SCR register
      • -
          -
        • HAL_PWR_DisableSleepOnExit()
        • -
        • HAL_PWR_EnableSleepOnExit()
        • -
        • HAL_PWR_EnableSEVOnPend()
        • -
        • HAL_PWR_DisableSEVOnPend()
        • -
        -
      • HAL_PWR_EnterSTOPMode()
      • -
          -
        • Update to - - - - clear the CORTEX SLEEPDEEP - bit of SCR register - before entering in sleep mode
        • -
        • Update - usage of __WFE() - in low power entry function: - if there is a pending event, - calling __WFE() will not - enter the CortexM4 core to - sleep mode. The solution is - to made the call below; the - first __WFE() - is always ignored and clears - the event if one was already - pending, the second is - always applied
        • -
        -
      -
    -
    -

    __SEV()
    -
    __WFE()
    -
    __WFE()

    -
    -
      -
        -
      • Add - new PVD configuration modes
      • -
          -
        • PWR_PVD_MODE_NORMAL
        • -
        • PWR_PVD_MODE_EVENT_RISING 
        • -
        • PWR_PVD_MODE_EVENT_FALLING
        • -
        • PWR_PVD_MODE_EVENT_RISING_FALLING
        • -
        -
      • Add new - macros to manage PVD Trigger
      • -
          -
        • __HAL_PWR_PVD_EXTI_ENABLE_RISING_EDGE()
        • -
        • __HAL_PWR_PVD_EXTI_DISABLE_RISING_EDGE(
        • -
        • __HAL_PWR_PVD_EXTI_ENABLE_FALLING_EDGE()
        • -
        • __HAL_PWR_PVD_EXTI_DISABLE_FALLING_EDGE()
        • -
        • __HAL_PWR_PVD_EXTI_ENABLE_RISING_FALLING_EDGE()
        • -
        • __HAL_PWR_PVD_EXTI_DISABLE_RISING_FALLING_EDGE()
        • -
        -
      • PVD macros:
      • -
          -
        • Remove the - __EXTILINE__ parameter
        • -
        • Update to - use prefix "__HAL_PWR_PVD_" - instead of  prefix - "__HAL_PVD"
        • -
        -
      -
    -
      -
        -
      • Rename HAL_PWR_PVDConfig() - by HAL_PWR_ConfigPVD()
      • -
      • Rename HAL_PWREx_ActivateOverDrive() - by HAL_PWREx_EnableOverDrive() - - - - -
      • -
      • Rename HAL_PWREx_DeactivateOverDrive() - by HAL_PWREx_DisableOverDrive() - - - - -
      • -
      -
    • HAL - - - - GPIO update
    • -
        -
      • HAL_GPIO_Init()/HAL_GPIO_DeInit(): add a call - to the CMSIS assert macro - to check GPIO instance: - IS_GPIO_ALL_INSTANCE() 
      • -
      • HAL_GPIO_WritePin(): update to - write in BSRR register
      • -
      • Rename GPIO_GET_SOURCE() - by GET_GPIO_INDEX() and - - - - move this later to file  - stm32f4xx_hal_gpio_ex.h
      • -
      • Add new - define for alternate function - GPIO_AF5_SPI3 for - STM32F429xx/439xx and - STM32F427xx/437xx devices
      • -
      -
    • HAL - - - - HASH update
    • -
        -
      • HAL_HASH_MD5_Start_IT(): - - - - - fix input - address management issue
      • -
      -
    • HAL - - - - RCC update
    • -
        -
      • Rename the - following Macros
      • -
          -
        • __PPP_CLK_ENABLE()  - - - - - by - __HAL_RCC_PPP_CLK_ENABLE()
        • -
        • __PPP_CLK_DISABLE()  - - - - - by - __HAL_RCC_PPP_CLK_DISABLE()
        • -
        • __PPP_FORCE_RESET()  - - - - - by - __HAL_RCC_PPP_FORCE_RESET()
        • -
        • __PPP_RELEASE_RESET()  - - - - - by - __HAL_RCC_PPP_RELEASE_RESET()
        • -
        • __PPP_CLK_SLEEP_ENABLE() - by - __HAL_RCC_PPP_CLK_SLEEP_ENABLE()
        • -
        • __PPP_CLK_SLEEP_DISABLE() - by - __HAL_RCC_PPP_CLK_SLEEP_DISABLE()
        • -
        -
      • IS_RCC_PLLSAIN_VALUE() - macro: update the check - condition
      • -
      • Add - description of RCC known Limitations
      • -
      • Rename HAL_RCC_CCSCallback() - by HAL_RCC_CSSCallback()
      • -
      • HAL_RCC_OscConfig() fix - issues: 
      • -
          -
        • Remove the - disable of HSE - oscillator when - HSE_BYPASS is used as - system clock source or as - PPL clock source
        • -
        • Add a check - on HSERDY flag - when HSE_BYPASS is - selected as new state - for HSE oscillator.
        • -
        -
      • Rename - __HAL_RCC_I2SCLK() - by __HAL_RCC_I2S_Config()
      • -
      -
    -

    §  - - - - - HAL I2S update

    -
      -
        -
      • HAL_I2S_Init(): add check - on I2S instance - using CMSIS macro IS_I2S_ALL_INSTANCE() 
      • -
      • HAL_I2S_IRQHandler() - update for compliancy w/ C++
      • -
      • Add use - of tmpreg - variable in __HAL_I2S_CLEAR_OVRFLAG() - and __HAL_I2S_CLEAR_UDRFLAG() - macro for compliancy with C++
      • -
      • HAL_I2S_GetError(): update to - return uint32_t instead of - HAL_I2S_ErrorTypeDef - enumeration
      • -
      -
    -

    §  - - - - - HAL I2C update

    -
      -
        -
      • Update to - - - - - clear the POS bit in the - CR1 register at the end - of HAL_I2C_Master_Read_IT() - and HAL_I2C_Mem_Read_IT() - process
      • -
      • Rename - HAL_I2CEx_DigitalFilter_Config()  - - - - by - HAL_I2CEx_ConfigDigitalFilter() -
      • -
      • Rename - HAL_I2CEx_AnalogFilter_Config()  - - - - by - HAL_I2CEx_ConfigAnalogFilter() -
      • -
      • Add use - of tmpreg - variable in __HAL_I2C_CLEAR_ADDRFLAG() - and __HAL_I2C_CLEAR_STOPFLAG() - macro for compliancy with - C++
      • -
      -
    • HAL - - - - IrDA update
    • -
        -
      • DMA transmit - process; the code has been - updated to avoid waiting on TC - flag under DMA ISR, IrDA TC - interrupt is used instead. - Below the update to be done on - user application:
      • -
          -
        • Configure - and enable the USART IRQ in - HAL_IRDA_MspInit() - function
        • -
        • In - stm32f4xx_it.c file, UASRTx_IRQHandler() - function: add a call to HAL_IRDA_IRQHandler() - - - - function
        • -
        -
      • IT transmit - process; the code has been - updated to avoid waiting on TC - flag under IRDA ISR, IrDA TC - interrupt is used instead. No - impact on user application
      • -
      • Rename - Macros: add prefix "__HAL"
      • -
          -
        • __IRDA_ENABLE() - by __HAL_IRDA_ENABLE()
        • -
        • __IRDA_DISABLE() - by __HAL_IRDA_DISABLE()
        • -
        -
      • Add new user - macros to manage the sample - method feature
      • -
          -
        • __HAL_IRDA_ONE_BIT_SAMPLE_ENABLE()
        • -
        • __HAL_IRDA_ONE_BIT_SAMPLE_DISABLE()
        • -
        -
      • HAL_IRDA_Transmit_IT(): update to - remove the enable of the - parity error interrupt
      • -
      • Add use - of tmpreg - variable in __HAL_IRDA_CLEAR_PEFLAG() - macro for compliancy with - C++
      • -
      • HAL_IRDA_Transmit_DMA() update to - follow the - right procedure - "Transmission using DMA"  - in the reference manual
      • -
          -
        • Add clear - the TC flag in the SR - register before enabling the - DMA transmit - request
        • -
        -
      -
    • HAL - - - - IWDG update
    • -
        -
      • Rename the - defined IWDG keys: 
      • -
          -
        • KR_KEY_RELOAD - - - - by IWDG_KEY_RELOAD
        • -
        • KR_KEY_ENABLE - - - - by IWDG_KEY_ENABLE
        • -
        • KR_KEY_EWA - by - IWDG_KEY_WRITE_ACCESS_ENABLE
        • -
        • KR_KEY_DWA - by - IWDG_KEY_WRITE_ACCESS_DISABLE
        • -
        -
      •  Add new - macros - __HAL_IWDG_RESET_HANDLE_STATE() - and - __HAL_IWDG_CLEAR_FLAG() 
      • -
      • Update - __HAL_IWDG_ENABLE_WRITE_ACCESS() - and - __HAL_IWDG_DISABLE_WRITE_ACCESS() - as private macro
      • -
      -
    -

    §  - - - - - HAL SPI update

    -
      -
        -
      • HAL_SPI_TransmitReceive_DMA() update to - remove the  DMA Tx Error - Callback initialization when - SPI RxOnly - mode is selected
      • -
      • Add use of UNUSED(tmpreg) - in __HAL_SPI_CLEAR_MODFFLAG(), - __HAL_SPI_CLEAR_OVRFLAG(), - __HAL_SPI_CLEAR_FREFLAG() to - fix "Unused variable" warning - with TrueSTUDIO.
      • -
      • Rename - Literals: remove "D" from - "DISABLED" and "ENABLED"
      • -
          -
        • SPI_TIMODE_DISABLED by - - - - - SPI_TIMODE_DISABLE
        • -
        • SPI_TIMODE_ENABLED by SPI_TIMODE_ENABLE
        • -
        • SPI_CRCCALCULATION_DISABLED - by - - - - -  SPI_CRCCALCULATION_DISABLE
        • -
        • SPI_CRCCALCULATION_ENABLED - by - - - - -  SPI_CRCCALCULATION_ENABLE
        • -
        -
      • Add use - of tmpreg - variable in __HAL_SPI_CLEAR_MODFFLAG(), - - - - - __HAL_SPI_CLEAR_FREFLAG() and - __HAL_SPI_CLEAR_OVRFLAG() - macros for compliancy - with C++
      • -
      -
    -

    §  - - - - - HAL SDMMC update

    -
      -
        -
      • IS_SDIO_ALL_INSTANCE() -  macro moved to CMSIS - files
      • -
      -
    • HAL - - - - LTDC update
    • -
        -
      • HAL_LTDC_ConfigCLUT: optimize - the function when pixel format -is LTDC_PIXEL_FORMAT_AL44 
      • -
          -
        • Update the - size of color look up table - to 16 instead of 256 when - the pixel format - is LTDC_PIXEL_FORMAT_AL44 -
        • -
        -
      -
    • HAL - - - - NAND update
    • -
        -
      • Rename NAND - Address structure to NAND_AddressTypeDef - instead of NAND_AddressTypedef
      • -
      • Update the - used algorithm of these functions
      • -
          -
        • HAL_NAND_Read_Page()
        • -
        • HAL_NAND_Write_Page()
        • -
        • HAL_NAND_Read_SpareArea()
        • -
        • HAL_NAND_Write_SpareArea()
        • -
        -
      • HAL_NAND_Write_Page(): move - initialization of tickstart - before while loop
      • -
      • HAL_NAND_Erase_Block(): add whait - until NAND status is ready - before exiting this function
      • -
      -
    • HAL - - - - NOR update
    • -
        -
      • Rename NOR - Address structure to NOR_AddressTypeDef - instead of NOR_AddressTypedef
      • -
      • NOR Status - literals renamed
      • -
          -
        • NOR_SUCCESS - by HAL_NOR_STATUS_SUCCESS
        • -
        • NOR_ONGOING - by HAL_NOR_STATUS_ONGOING
        • -
        • NOR_ERROR - by HAL_NOR_STATUS_ERROR
        • -
        • NOR_TIMEOUT - by HAL_NOR_STATUS_TIMEOUT
        • -
        -
      • HAL_NOR_GetStatus() update to - fix Timeout issue - and exit from waiting - loop when timeout occurred
      • -
      -
    • HAL - - - - PCCARD update
    • -
        -
      • Rename PCCARD - Address structure to HAL_PCCARD_StatusTypeDef - instead of CF_StatusTypedef
      • -
      • PCCARD Status - literals renamed
      • -
          -
        • CF_SUCCESS - by HAL_PCCARD_STATUS_SUCCESS
        • -
        • CF_ONGOING - by HAL_PCCARD_STATUS_ONGOING
        • -
        • CF_ERROR - by HAL_PCCARD_STATUS_ERROR
        • -
        • CF_TIMEOUT - by HAL_PCCARD_STATUS_TIMEOUT
        • -
        -
      • Update "CF" - by "PCCARD" in functions, - literals - and macros
      • -
      -
    • HAL - - - - PCD update
    • -
        -
      • Rename functions
      • -
          -
        • HAL_PCD_ActiveRemoteWakeup() by HAL_PCD_ActivateRemoteWakeup()
        • -
        • HAL_PCD_DeActiveRemoteWakeup() by HAL_PCD_DeActivateRemoteWakeup()
        • -
        -
      • Rename literals
      • -
          -
        • USB_FS_EXTI_TRIGGER_RISING_EDGE - - - - - by - USB_OTG_FS_WAKEUP_EXTI_RISING_EDGE
        • -
        • USB_FS_EXTI_TRIGGER_FALLING_EDGE - - - - - by - USB_OTG_FS_WAKEUP_EXTI_FALLING_EDGE
        • -
        • USB_FS_EXTI_TRIGGER_BOTH_EDGE() - by - USB_OTG_FS_WAKEUP_EXTI_RISING_FALLING_EDGE
        • -
        • USB_HS_EXTI_TRIGGER_RISING_EDGE - - - - - by - USB_OTG_HS_WAKEUP_EXTI_RISING_EDGE 
        • -
        • USB_HS_EXTI_TRIGGER_FALLING_EDGE - - - - - by - USB_OTG_HS_WAKEUP_EXTI_FALLING_EDGE
        • -
        • USB_HS_EXTI_TRIGGER_BOTH_EDGE - - - - - by - USB_OTG_HS_WAKEUP_EXTI_RISING_FALLING_EDGE
        • -
        • USB_HS_EXTI_LINE_WAKEUP - - - - - by - USB_OTG_HS_EXTI_LINE_WAKEUP
        • -
        • USB_FS_EXTI_LINE_WAKEUP - - - - - by - USB_OTG_FS_EXTI_LINE_WAKEUP
        • -
        -
      • Rename USB - EXTI macros (FS, HS - - - - referenced as SUBBLOCK - here below)
      • -
          -
        • __HAL_USB_SUBBLOCK_EXTI_ENABLE_IT() -  by  - __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_ENABLE_IT()  
        • -
        • __HAL_USB_SUBBLOCK_EXTI_DISABLE_IT() - by __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_DISABLE_IT()
        • -
        • __HAL_USB_SUBBLOCK_EXTI_GET_FLAG() - by __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_GET_FLAG() 
        • -
        • __HAL_USB_SUBBLOCK_EXTI_CLEAR_FLAG() - by __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_CLEAR_FLAG()
        • -
        • __HAL_USB_SUBBLOCK_EXTI_SET_RISING_EGDE_TRIGGER() - by __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_ENABLE_RISING_EDGE()
        • -
        • __HAL_USB_SUBBLOCK_EXTI_SET_FALLING_EGDE_TRIGGER() - by __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_ENABLE_FALLING_EDGE()
        • -
        • __HAL_USB_SUBBLOCK_EXTI_SET_FALLINGRISING_TRIGGER() - by __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_ENABLE_RISING_FALLING_EDGE()
        • -
        • __HAL_USB_SUBBLOCK_EXTI_GENERATE_SWIT()  - - - - - by __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_GENERATE_SWIT()                                       -
        • -
        -
      -
    -
      -
    • HAL - - - - RNG update
    • -
        -
      • Add new functions
      • -
          -
        • HAL_RNG_GenerateRandomNumber(): to - generate a 32-bits random - number, - return - random value in argument and - return HAL status.
        • -
        • HAL_RNG_GenerateRandomNumber_IT(): to -  start generation of - the 32-bits random - number, user should call - the HAL_RNG_ReadLastRandomNumber() - - - - - function under the HAL_RNG_ReadyCallback() - - - - to get the generated random - value.
        • -
        • HAL_RNG_ReadLastRandomNumber(): to - return the last random value - stored in the RNG handle
        • -
        -
      • HAL_RNG_GetRandomNumber(): return - value update (obsolete), - replaced by HAL_RNG_GenerateRandomNumber()
      • -
      • HAL_RNG_GetRandomNumber_IT(): wrong - implementation (obsolete), - replaced by HAL_RNG_GenerateRandomNumber_IT()
      • -
      • __HAL_RNG_CLEAR_FLAG() - macro (obsolete), replaced by - new __HAL_RNG_CLEAR_IT() macro
      • -
      • Add new - define for RNG ready - interrupt:  RNG_IT_DRDY
      • -
      -
    • HAL - - - - RTC update
    • -
        -
      • HAL_RTC_GetTime() and HAL_RTC_GetDate(): - - - - - add the comment below
      • -
      -
    -
    -
    -

      - - - - - * @note You must call HAL_RTC_GetDate() - after HAL_RTC_GetTime() - - - - - to unlock the values
    -
      - - - - - * in the higher-order - calendar shadow registers to - ensure consistency between - the time and date values.
    -
      - - - - - * Reading RTC current time - locks the values in calendar - shadow registers until - Current date is read. 

    -
    -
    -
      -
        -
      • Rename - literals: add prefix "__HAL"
      • -
          -
        • FORMAT_BIN by HAL_FORMAT_BIN
        • -
        • FORMAT_BCD - by HAL_FORMAT_BCD
        • -
        -
      • Rename macros - (ALARM, WAKEUPTIMER and - TIMESTAMP referenced - as SUBBLOCK here - below)
      • -
          -
        • __HAL_RTC_EXTI_ENABLE_IT() - by  __HAL_RTC_SUBBLOCK_EXTI_ENABLE_IT()
        • -
        • __HAL_RTC_EXTI_DISABLE_IT() - by  __HAL_RTC_SUBBLOCK_EXTI_DISABLE_IT()
        • -
        • __HAL_RTC_EXTI_CLEAR_FLAG() - by  __HAL_RTC_SUBBLOCK_EXTI_CLEAR_FLAG()
        • -
        • __HAL_RTC_EXTI_GENERATE_SWIT() - by __HAL_RTC_SUBBLOCK_EXTI_GENERATE_SWIT()
        • -
        -
      • Add new - macros (ALARM, - WAKEUPTIMER and TAMPER_TIMESTAMP - - - - referenced as SUBBLOCK - here below)
      • -
          -
        • __HAL_RTC_SUBBLOCK_GET_IT_SOURCE(
        • -
        • __HAL_RTC_SUBBLOCK_EXTI_ENABLE_EVENT()
        • -
        • __HAL_RTC_SUBBLOCK_EXTI_DISABLE_EVENT()
        • -
        • __HAL_RTC_SUBBLOCK_EXTI_ENABLE_FALLING_EDGE()
        • -
        • __HAL_RTC_SUBBLOCK_EXTI_DISABLE_FALLING_EDGE()
        • -
        • __HAL_RTC_SUBBLOCK_EXTI_ENABLE_RISING_EDGE()
        • -
        • __HAL_RTC_SUBBLOCK_EXTI_DISABLE_RISING_EDGE()
        • -
        •  __HAL_RTC_SUBBLOCK_EXTI_ENABLE_RISING_FALLING_EDGE()
        • -
        •  __HAL_RTC_SUBBLOCK_EXTI_DISABLE_RISING_FALLING_EDGE()
        • -
        •  __HAL_RTC_SUBBLOCK_EXTI_GET_FLAG()
        • -
        -
      -
    • HAL - - - - SAI update
    • -
        -
      • Update - SAI_STREOMODE by SAI_STEREOMODE
      • -
      • Update FIFO - status Level defines in upper - case
      • -
      • Rename - literals: remove "D" from - "DISABLED" and "ENABLED"
      • -
          -
        • SAI_OUTPUTDRIVE_DISABLED - - - - -  by - SAI_OUTPUTDRIVE_DISABLE
        • -
        • SAI_OUTPUTDRIVE_ENABLED - - - - -  by - SAI_OUTPUTDRIVE_ENABLE
        • -
        • SAI_MASTERDIVIDER_ENABLED  by - SAI_MASTERDIVIDER_ENABLE
        • -
        • SAI_MASTERDIVIDER_DISABLED  by - SAI_MASTERDIVIDER_DISABLE
        • -
        -
      -
    -
      -
    • HAL - - - - SD update
    • -
        -
      • Rename - SD_CMD_SD_APP_STAUS by SD_CMD_SD_APP_STATUS
      • -
      • SD_PowerON() updated to - add 1ms required power up - waiting time before starting - the SD initialization sequence
      • -
      • SD_DMA_RxCplt()/SD_DMA_TxCplt(): - - - - - add a call to - HAL_DMA_Abort()
      • -
      • HAL_SD_ReadBlocks() update to - set the defined - DATA_BLOCK_SIZE as SDIO DataBlockSize - parameter
      • -
      • HAL_SD_ReadBlocks_DMA()/HAL_SD_WriteBlocks_DMA() - update to call the HAL_DMA_Start_IT() - - - - function with DMA Datalength - set to BlockSize/4  - - - - - as the DMA is - configured in word 
      • -
      -
    • HAL - - - - SMARTCARD update 
    • -
        -
      • DMA transmit - process; the code has been - updated to avoid waiting on TC - flag under DMA ISR, SMARTCARD - TC interrupt is used instead. - Below the update to be done on - user application:
      • -
          -
        • Configure - and enable the USART IRQ in - HAL_SAMRTCARD_MspInit() - function
        • -
        • In - stm32f4xx_it.c file, UASRTx_IRQHandler() - function: add a call to HAL_SMARTCARD_IRQHandler() - - - - - function
        • -
        -
      • IT transmit - process; the code has been - updated to avoid waiting on TC - flag under SMARTCARD - ISR, SMARTCARD TC - interrupt is used instead. No - impact on user application
      • -
      • Rename - macros: add prefix "__HAL"
      • -
          -
        • __SMARTCARD_ENABLE() - by __HAL_SMARTCARD_ENABLE()
        • -
        • __SMARTCARD_DISABLE() - by __HAL_SMARTCARD_DISABLE()
        • -
        • __SMARTCARD_ENABLE_IT() - by - __HAL_SMARTCARD_ENABLE_IT()
        • -
        • __SMARTCARD_DISABLE_IT() - by - __HAL_SMARTCARD_DISABLE_IT()
        • -
        • __SMARTCARD_DMA_REQUEST_ENABLE() - by - __HAL_SMARTCARD_DMA_REQUEST_ENABLE()
        • -
        • __SMARTCARD_DMA_REQUEST_DISABLE() - by - __HAL_SMARTCARD_DMA_REQUEST_DISABLE()
        • -
        -
      • Rename - literals: remove "D" from - "DISABLED" and "ENABLED"
      • -
          -
        • SMARTCARD_NACK_ENABLED by - - - - - SMARTCARD_NACK_ENABLE
        • -
        • SMARTCARD_NACK_DISABLED by SMARTCARD_NACK_DISABLE
        • -
        -
      • Add new user - macros to manage the sample - method feature
      • -
          -
        • __HAL_SMARTCARD_ONE_BIT_SAMPLE_ENABLE()
        • -
        • __HAL_SMARTCARD_ONE_BIT_SAMPLE_DISABLE()
        • -
        -
      • Add use - of tmpreg - variable in - __HAL_SMARTCARD_CLEAR_PEFLAG() - macro for compliancy with - C++
      • -
      • HAL_SMARTCARD_Transmit_DMA() update to - follow the - right procedure - "Transmission using DMA"  - in the reference manual
      • -
          -
        • Add clear - the TC flag in the SR - register before enabling the - DMA transmit - request
        • -
        -
      -
    • HAL - - - - TIM update
    • -
        -
      • Add - TIM_CHANNEL_ALL as possible - value for all Encoder - Start/Stop APIs Description
      • -
      • HAL_TIM_OC_ConfigChannel() remove call - to IS_TIM_FAST_STATE() assert - macro
      • -
      • HAL_TIM_PWM_ConfigChannel() add a call - to IS_TIM_FAST_STATE() assert - macro to check the OCFastMode - parameter
      • -
      • HAL_TIM_DMADelayPulseCplt() Update to - set the TIM Channel before to - call  HAL_TIM_PWM_PulseFinishedCallback()
      • -
      • HAL_TIM_DMACaptureCplt() update to - set the TIM Channel before to - call  HAL_TIM_IC_CaptureCallback()
      • -
      • TIM_ICx_ConfigChannel() update - to fix Timer CCMR1 register - corruption when setting ICFilter - parameter
      • -
      • HAL_TIM_DMABurst_WriteStop()/HAL_TIM_DMABurst_ReadStop() - update to abort the DMA - transfer for the specifc - TIM channel
      • -
      • Add new - function for TIM Slave - configuration in IT mode: - HAL_TIM_SlaveConfigSynchronization_IT(
      • -
      • HAL_TIMEx_ConfigBreakDeadTime() add an - assert check on Break & DeadTime - parameters values
      • -
      • HAL_TIMEx_OCN_Start_IT() add the - enable of Break Interrupt for - all output modes
      • -
      • Add new - macros to ENABLE/DISABLE URS - bit in TIM CR1 register:
      • -
          -
        • __HAL_TIM_URS_ENABLE()
        • -
        • __HAL_TIM_URS_DISABLE()
        • -
        -
      • Add new macro - for TIM Edge modification: - __HAL_TIM_SET_CAPTUREPOLARITY()
      • -
      -
    • HAL - - - - UART update
    • -
        -
      • Add IS_LIN_WORD_LENGTH() - and - IS_LIN_OVERSAMPLING()  - macros: to check respectively - WordLength - and OverSampling - parameters in LIN mode
      • -
      • DMA transmit - process; the code has been - updated to avoid waiting on TC - flag under DMA ISR, UART TC - interrupt is used instead. - Below the update to be done on - user application:
      • -
          -
        • Configure - and enable the USART IRQ in - HAL_UART_MspInit() - function
        • -
        • In - stm32f4xx_it.c file, USARTx_IRQHandler() - function: add a call to HAL_UART_IRQHandler() - - - - function
        • -
        -
      • IT transmit - process; the code has been - updated to avoid waiting on TC - flag under UART ISR, UART - TC interrupt is used instead. - No impact on user application
      • -
      • Rename - macros:
      • -
          -
        • __HAL_UART_ONEBIT_ENABLE() - by - __HAL_UART_ONE_BIT_SAMPLE_ENABLE()
        • -
        • __HAL_UART_ONEBIT_DISABLE() - by - __HAL_UART_ONE_BIT_SAMPLE_DISABLE()
        • -
        -
      • Rename - literals:
      • -
          -
        • UART_WAKEUPMETHODE_IDLELINE by - - - - - UART_WAKEUPMETHOD_IDLELINE
        • -
        • UART_WAKEUPMETHODE_ADDRESSMARK by -UART_WAKEUPMETHOD_ADDRESSMARK
        • -
        -
      • Add use - of tmpreg - variable in __HAL_UART_CLEAR_PEFLAG() - macro for compliancy with - C++
      • -
      • HAL_UART_Transmit_DMA() update to - follow the right procedure - "Transmission using DMA" in - the reference manual
      • -
          -
        • Add clear - the TC flag in the SR - register before enabling the - DMA transmit - request
        • -
        -
      -
    • HAL - - - - USART update
    • -
        -
      • DMA transmit - process; the code has been - updated to avoid waiting on TC - flag under DMA ISR, USART TC - interrupt is used instead. - Below the update to be done on - user application:
      • -
          -
        • Configure - and enable the USART IRQ in - HAL_USART_MspInit() - function
        • -
        • In - stm32f4xx_it.c file, USARTx_IRQHandler() - function: add a call to HAL_USART_IRQHandler() - - - - - function
        • -
        -
      • IT transmit - process; the code has been - updated to avoid waiting on TC - flag under USART ISR, - USART TC interrupt is used - instead. No impact on user - application
      • -
      • HAL_USART_Init() update - to enable the USART - oversampling by 8 by default - in order to reach max USART - frequencies
      • -
      • USART_DMAReceiveCplt() update - to set the new USART state - after checking on the - old state
      • -
      • HAL_USART_Transmit_DMA()/HAL_USART_TransmitReceive_DMA() - update to - follow the - right procedure - "Transmission using DMA"  - in the reference manual
      • -
          -
        • Add clear - the TC flag in the SR - register before enabling the - DMA transmit - request
        • -
        -
      • Rename - macros:
      • -
          -
        • __USART_ENABLE() - by __HAL_USART_ENABLE()
        • -
        • __USART_DISABLE() - by __HAL_USART_DISABLE()
        • -
        • __USART_ENABLE_IT() - by __HAL_USART_ENABLE_IT()
        • -
        • __USART_DISABLE_IT() - by __HAL_USART_DISABLE_IT()
        • -
        -
      • Rename - literals: remove "D" from - "DISABLED" and "ENABLED"
      • -
          -
        • USART_CLOCK_DISABLED by - - - - - USART_CLOCK_DISABLE
        • -
        • USART_CLOCK_ENABLED by - - - - - USART_CLOCK_ENABLE
        • -
        • USARTNACK_ENABLED - - - - by USART_NACK_ENABLE
        • -
        • USARTNACK_DISABLED - - - - by USART_NACK_DISABLE
        • -
        -
      • Add new user - macros to manage the sample - method feature
      • -
          -
        • __HAL_USART_ONE_BIT_SAMPLE_ENABLE()
        • -
        • __HAL_USART_ONE_BIT_SAMPLE_DISABLE()
        • -
        -
      • Add use - of tmpreg - variable in __HAL_USART_CLEAR_PEFLAG() - macro for compliancy with - C++
      • -
      -
    • HAL - - - - WWDG update
    • -
        -
      • Add new - parameter in - __HAL_WWDG_ENABLE_IT() - macro
      • -
      • Add new - macros to manage WWDG IT & - correction:
      • -
          -
        • __HAL_WWDG_DISABLE()
        • -
        • __HAL_WWDG_DISABLE_IT()
        • -
        • __HAL_WWDG_GET_IT()
        • -
        • __HAL_WWDG_GET_IT_SOURCE()
        • -
        -
      -
    -

    V1.1.0 - - - - / 19-June-2014

    -

    Main Changes

    -
      -
    • Add - support of STM32F411xE devices
    • -
    -
      -
    • HAL - - - - generic - update
    • -
        -
      • Enhance HAL - delay and time base implementation
      • -
          -
        • Systick timer is - used by default as source of - time base, but user can - eventually implement his - proper time base source (a general - - - - purpose - timer for example or other - time source)
        • -
        • Functions - affecting time base - configurations are declared - as __Weak to make override - possible in case of other - implementations in user - file, for more details - please refer to HAL_TimeBase - example
        • -
        -
      • Fix flag - clear procedure: use atomic - write operation "=" instead of - ready-modify-write operation - "|=" or "&="
      • -
      • Fix on - Timeout management, Timeout - value set to 0 passed to API - automatically exits the - function after checking the - flag without any wait
      • -
      • Common update - for the following - communication peripherals: - SPI, UART, USART and IRDA
      • -
          -
        • Add DMA - circular mode support
        • -
        • Remove lock - from recursive process
        • -
        -
      • Add new macro - __HAL_RESET_HANDLE_STATE to - reset a given handle state
      • -
      • Add a new - attribute for functions - executed from internal SRAM - and depending from - Compiler implementation
      • -
      • When USE_RTOS - == 1 (in - stm32l0xx_hal_conf.h), the - __HAL_LOCK() - is not defined instead of - being defined empty
      • -
      • Miscellaneous - comments and formatting update
      • -
      • stm32f4xx_hal_conf_template.h
      • -
          -
        • Add a new - define for LSI default value - LSI_VALUE
        • -
        • Add a new - define for LSE default value - LSE_VALUE
        • -
        • Add a new - define for Tick interrupt - priority TICK_INT_PRIORITY - (needed for the enhanced - time base implementation)
        • -
        -
      • Important - - - - - Note: - aliases has been added for any - API naming change, to keep - compatibility with previous version
      • -
      -
    • HAL - - - - GPIO update
    • -
    -
      -
        -
      • Add a new - macro __HAL_GPIO_EXTI_GENERATE_SWIT() - to manage the generation of - software interrupt on selected - EXTI line
      • -
      • HAL_GPIO_Init(): use - temporary variable when - modifying the registers, to - avoid unexpected transition in - the GPIO pin configuration
      • -
      • Remove - IS_GET_GPIO_PIN macro
      • -
      • Add a new - function HAL_GPIO_LockPin()
      • -
      • Private Macro - __HAL_GET_GPIO_SOURCE renamed - into GET_GPIO_SOURCE
      • -
      • Add the - support of STM32F411xx devices - - - - - : add the - new Alternate functions values - related to new remap added for - SPI, USART, I2C
      • -
      • Update the - following HAL GPIO macros - description: rename EXTI_Linex - by GPIO_PIN_x
      • -
          -
        • __HAL_GPIO_EXTI_CLEAR_IT()
        • -
        • __HAL_GPIO_EXTI_GET_IT()
        • -
        • __HAL_GPIO_EXTI_CLEAR_FLAG()
        • -
        • __HAL_GPIO_EXTI_GET_FLAG()
        • -
        -
      -
    -

    §  - - - - - HAL DMA update

    -
      -
        -
      • Fix in HAL_DMA_PollForTransfer() - to:
      • -
          -
        • set DMA - error code in case of - HAL_ERROR status
        • -
        • set HAL - Unlock before DMA state update
        • -
        -
      -
    -

    §  - - - - - HAL DMA2D - update

    -
      -
        -
      • Add - configuration of source - address in case of A8 or A4 - M2M_PFC DMA2D mode
      • -
      -
    • HAL - - - - FLASH update
    • -
    -
      -
        -
      • Functions - reorganization update, - depending on the features - supported by each STM32F4 device
      • -
      • Add new - driver - (stm32f4xx_hal_flash_ramfunc.h/.c) - to manage function executed - from RAM, these functions are - available only for STM32F411xx - Devices
      • -
          -
        • FLASH_StopFlashInterfaceClk()  : - Stop the flash interface - while System Run
        • -
        • FLASH_StartFlashInterfaceClk() : Stop the - flash interface while System - Run
        • -
        • FLASH_EnableFlashSleepMode() : Enable - the flash sleep while System - Run
        • -
        • FLASH_DisableFlashSleepMode() :  - Disable the flash sleep - while System Run
        • -
        -
      -
    -
      -
    • HAL - - - - PWR update
    • -
    -
      -
        -
      • HAL_PWR_PVDConfig(): add clear - of the EXTI trigger before new - configuration
      • -
      • Fix in HAL_PWR_EnterSTANDBYMode() - to not clear Wakeup flag - (WUF), which need to be - cleared at application level - before to call this function
      • -
      • HAL_PWR_EnterSLEEPMode()
      • -
          -
        • Remove - disable and enable of SysTick - Timer
        • -
        • Update - usage of __WFE() - in low power entry function: - if there is a pending event, - calling __WFE() will not - enter the CortexM4 core to - sleep mode. The solution is - to made the call below; the - first __WFE() - is always ignored and clears - the event if one was already - pending, the second is - always applied
        • -
        -
      -
    -
    -

    __SEV()
    -
    __WFE()
    -
    __WFE()

    -
    -
      -
        -
      • Add new macro - for software event generation - __HAL_PVD_EXTI_GENERATE_SWIT()
      • -
      • Remove the - following defines form Generic - driver and add them under - extension driver because they - are only used within extension - functions.
      • -
          -
        • CR_FPDS_BB: - used within HAL_PWREx_EnableFlashPowerDown() - function
        • -
        • CSR_BRE_BB: - used within HAL_PWREx_EnableBkUpReg() - function
        • -
        -
      • Add the - support of STM32F411xx devices - add the define STM32F411xE
      • -
          -
        • For - STM32F401xC, STM32F401xE and - STM32F411xE devices add the - following functions used to - enable or disable the low - voltage mode for regulators
        • -
        -
      -
    -
      -
        -
          -
            -
          • HAL_PWREx_EnableMainRegulatorLowVoltage()
          • -
          • HAL_PWREx_DisableMainRegulatorLowVoltage()
          • -
          • HAL_PWREx_EnableLowRegulatorLowVoltage()
          • -
          • HAL_PWREx_DisableLowRegulatorLowVoltage()
          • -
          -
        -
      • For - STM32F42xxx/43xxx devices, add - a new function for Under - Driver management as the macro - already added for this mode is - not sufficient: HAL_PWREx_EnterUnderDriveSTOPMode()
      • -
      -
    -
      -
    • HAL - - - - RCC update
    • -
        -
      • In HAL_RCC_ClockConfig() - function: update the AHB clock - divider before clock switch to - new source
      • -
      • Allow to - calibrate the HSI when it is - used as system clock source
      • -
      • Rename the - following macros
      • -
          -
        • __OTGFS_FORCE_RESET - - - - ()  by - __USB_OTG_FS_FORCE_RESET()
        • -
        • __OTGFS_RELEASE_RESET - - - - ()  by  -__USB_OTG_FS_RELEASE_RESET()
        • -
        • __OTGFS_CLK_SLEEP_ENABLE - - - - - ()  by  -__USB_OTG_FS_CLK_SLEEP_ENABLE()
        • -
        • __OTGFS_CLK_SLEEP_DISABLE - - - - - () by  __USB_OTG_FS_CLK_SLEEP_DISABLE()
        • -
        -
      -
    -

     

    -
      -
        -
      • Add new field - PLLI2SM in - RCC_PLLI2SInitTypeDef - structure, this division - factor is added for PLLI2S VCO - input clock only STM32F411xE - devices => the FW - compatibility is broken vs. - STM32F401xx devices
      • -
      • Update HAL_RCCEx_PeriphCLKConfig() - and  HAL_RCCEx_GetPeriphCLKConfig()  - - - - - functions to support the new - PLLI2SM
      • -
      • Add new - function to manage the new LSE - mode - - - - - : HAL_RCCEx_SelectLSEMode()
      • -
      • Reorganize - the macros depending from - Part number used and make them - more clear
      • -
      -
    -

    §  HAL - - - - UART update

    -
      -
        -
      • Add new - macros to control CTS and RTS
      • -
      • Add specific - macros to manage the flags - cleared only by a software sequence
      • -
          -
        • __HAL_UART_CLEAR_PEFLAG()
        • -
        • __HAL_UART_CLEAR_FEFLAG()
        • -
        • __HAL_UART_CLEAR_NEFLAG()
        • -
        • __HAL_UART_CLEAR_OREFLAG()
        • -
        • __HAL_UART_CLEAR_IDLEFLAG()
        • -
        -
      • Add several - enhancements without affecting - the driver functionalities -
      • -
          -
        • Remove the - check on RXNE set after - reading the Data in the DR - register
        • -
        • Update the - transmit processes to use - TXE instead of TC
        • -
        • Update HAL_UART_Transmit_IT() - to enable UART_IT_TXE - instead of UART_IT_TC
        • -
        -
      -
    -

    §  - - - - - HAL USART - update

    -
      -
        -
      • Add specific - macros to manage the flags - cleared only by a software sequence
      • -
          -
        • __HAL_USART_CLEAR_PEFLAG()
        • -
        • __HAL_USART_CLEAR_FEFLAG()
        • -
        • __HAL_USART_CLEAR_NEFLAG()
        • -
        • __HAL_USART_CLEAR_OREFLAG()
        • -
        • __HAL_USART_CLEAR_IDLEFLAG()
        • -
        -
      • Update HAL_USART_Transmit_IT() - to enable USART_IT_TXE - instead of USART_IT_TC
      • -
      -
    -

    §  - - - - - HAL IRDA update

    -
      -
        -
      • Add specific - macros to manage the flags - cleared only by a software sequence
      • -
          -
        • __HAL_IRDA_CLEAR_PEFLAG()
        • -
        • __HAL_ - IRDA _CLEAR_FEFLAG()
        • -
        • __HAL_ - IRDA _CLEAR_NEFLAG()
        • -
        • __HAL_ - IRDA _CLEAR_OREFLAG()
        • -
        • __HAL_ - IRDA _CLEAR_IDLEFLAG()
        • -
        -
      • Add several - enhancements without affecting - the driver functionalities
      • -
      -
    -
      -
        -
          -
        • Remove the - check on RXNE set after - reading the Data in the DR - register
        • -
        • Update HAL_IRDA_Transmit_IT() - to enable IRDA_IT_TXE - instead of IRDA_IT_TC
        • -
        -
      • Add the - following APIs used within DMA - process -
      • -
          -
        • HAL_StatusTypeDef HAL_IRDA_DMAPause(IRDA_HandleTypeDef - *hirda);
        • -
        • HAL_StatusTypeDef HAL_IRDA_DMAResume(IRDA_HandleTypeDef - *hirda);
        • -
        • HAL_StatusTypeDef HAL_IRDA_DMAStop(IRDA_HandleTypeDef - *hirda); -
        • -
        • void HAL_IRDA_TxHalfCpltCallback(IRDA_HandleTypeDef - *hirda);
        • -
        • void HAL_IRDA_RxHalfCpltCallback(IRDA_HandleTypeDef - *hirda);
        • -
        -
      -
    -

    §  - - - - - HAL SMARTCARD - update

    -
      -
        -
      • Add specific - macros to manage the flags - cleared only by a software sequence
      • -
          -
        • __HAL_SMARTCARD_CLEAR_PEFLAG()
        • -
        • __HAL_SMARTCARD_CLEAR_FEFLAG()
        • -
        • __HAL_SMARTCARD_CLEAR_NEFLAG()
        • -
        • __HAL_SMARTCARD_CLEAR_OREFLAG()
        • -
        • __HAL_SMARTCARD_CLEAR_IDLEFLAG()
        • -
        -
      • Add several - enhancements without affecting - the driver functionalities
      • -
          -
        • Add a new - state HAL_SMARTCARD_STATE_BUSY_TX_RX - and all processes has been - updated accordingly
        • -
        • Update HAL_SMARTCARD_Transmit_IT() - to enable SMARTCARD_IT_TXE - instead of SMARTCARD_IT_TC
        • -
        -
      -
    -
      -
    • HAL - - - - SPI update
    • -
        -
      • Bugs fix
      • -
          -
        • SPI - interface is used in - synchronous polling mode: at - high clock rates like SPI prescaler - 2 and 4, calling
          - HAL_SPI_TransmitReceive() - returns with error - HAL_TIMEOUT
          -
        • -
        • HAL_SPI_TransmitReceive_DMA() does not - clean up the TX DMA, so any - subsequent SPI calls return - the DMA error
        • -
        • HAL_SPI_Transmit_DMA() is failing - when data size is equal to 1 - byte -
        • -
        -
      • Add the - following APIs used within the - DMA process
      • -
      -
    -
      -
        -
          -
        • HAL_StatusTypeDef HAL_SPI_DMAPause(SPI_HandleTypeDef - *hspi);
        • -
        • HAL_StatusTypeDef HAL_SPI_DMAResume(SPI_HandleTypeDef - *hspi);
        • -
        • HAL_StatusTypeDef HAL_SPI_DMAStop(SPI_HandleTypeDef - *hspi);
        • -
        • void HAL_SPI_TxHalfCpltCallback(SPI_HandleTypeDef - *hspi);
        • -
        • void HAL_SPI_RxHalfCpltCallback(SPI_HandleTypeDef - *hspi);
        • -
        • void HAL_SPI_TxRxHalfCpltCallback(SPI_HandleTypeDef - *hspi);
        • -
        -
      -
    -
      -
    • HAL - - - - RNG update
    • -
        -
          -
        • Add a - conditional define to make - this driver visible for all - STM32F4xx devices except - STM32F401xx and STM32F411xx - Devices.
        • -
        -
      -
    • HAL - - - - CRC update
    • -
        -
          -
        • These - macros are added to - read/write the CRC IDR - register: __HAL_CRC_SET_IDR() - and __HAL_CRC_GET_IDR()
        • -
        -
      -
    -
      -
    • HAL - - - - DAC update
    • -
        -
      • Enhance the - DMA channel configuration when - used with DAC
      • -
      -
    • HAL - - - - TIM update
    • -
        -
      • HAL_TIM_IRQHandler(): update to - check the input capture - channel 3 and 4 in CCMR2 - instead of CCMR1
      • -
      • __HAL_TIM_PRESCALER() - updated to use '=' instead of - '|='
      • -
      • Add the - following macro in TIM HAL - driver
      • -
          -
        • __HAL_TIM_GetCompare() -
        • -
        • __HAL_TIM_GetCounter() -
        • -
        • __HAL_TIM_GetAutoreload() -
        • -
        • __HAL_TIM_GetClockDivision() -
        • -
        • __HAL_TIM_GetICPrescaler()
        • -
        -
      -
    • HAL - - - - SDMMC update
    • -
    -
      -
        -
      • Use of CMSIS - constants instead of magic values
      • -
      • Miscellaneous - update in functions internal - coding
      • -
      -
    • HAL - - - - NAND update
    • -
        -
      • Fix - - - - issue of macros returning - wrong address for NAND blocks
      • -
      • Fix - - - - issue for read/write NAND - page/spare area
      • -
      -
    • HAL - - - - NOR update
    • -
        -
      • Add - - - - the NOR address bank macro - used within the API
      • -
      • Update - - - - NOR API implementation to - avoid the use of NOR address - bank hard coded
      • -
      -
    • HAL - - - - HCD update
    • -
        -
      • HCD_StateTypeDef structure - members renamed
      • -
      • These macro are renamed
      • -
          -
        • __HAL_GET_FLAG(__HANDLE__, - - - - - __INTERRUPT__)    - - - - by - __HAL_HCD_GET_FLAG(__HANDLE__, - __INTERRUPT__)
        • -
        • __HAL_CLEAR_FLAG(__HANDLE__, - - - - - __INTERRUPT__) by - __HAL_HCD_CLEAR_FLAG(__HANDLE__, - __INTERRUPT__) 
        • -
        • __HAL_IS_INVALID_INTERRUPT(__HANDLE__)  - - - - - by - __HAL_HCD_IS_INVALID_INTERRUPT(__HANDLE__)  -
        • -
        -
      -
    • HAL - - - - PCD update
    • -
        -
      • HAL_PCD_SetTxFiFo() and HAL_PCD_SetRxFiFo() - - - - - renamed into HAL_PCDEx_SetTxFiFo() - - - - - and HAL_PCDEx_SetRxFiFo() - - - - - and moved to the extension - files - stm32f4xx_hal_pcd_ex.h/.c
      • -
      • PCD_StateTypeDef structure - members renamed
      • -
      • Fix incorrect - masking of TxFIFOEmpty
      • -
      • stm32f4xx_ll_usb.c: - - - - fix issue in HS mode
      • -
      • New macros added
      • -
          -
        • __HAL_PCD_IS_PHY_SUSPENDED()
        • -
        • __HAL_USB_HS_EXTI_GENERATE_SWIT()
        • -
        • __HAL_USB_FS_EXTI_GENERATE_SWIT()
        • -
        -
      • These macro are renamed
      • -
          -
        • __HAL_GET_FLAG(__HANDLE__, - - - - - __INTERRUPT__)    - - - - by - __HAL_PCD_GET_FLAG(__HANDLE__, - __INTERRUPT__)
        • -
        • __HAL_CLEAR_FLAG(__HANDLE__, - - - - - __INTERRUPT__) by - __HAL_PCD_CLEAR_FLAG(__HANDLE__, - __INTERRUPT__) 
        • -
        • __HAL_IS_INVALID_INTERRUPT(__HANDLE__)  - - - - - by - __HAL_PCD_IS_INVALID_INTERRUPT(__HANDLE__)  -
        • -
        • __HAL_PCD_UNGATE_CLOCK(__HANDLE__) - - - - - by - __HAL_PCD_UNGATE_PHYCLOCK(__HANDLE__)
        • -
        • __HAL_PCD_GATE_CLOCK(__HANDLE__) - - - - - by - __HAL_PCD_GATE_PHYCLOCK(__HANDLE__)
        • -
        -
      -
    • HAL - - - - ETH update
    • -
        -
      • Update HAL_ETH_GetReceivedFrame_IT() - function to return HAL_ERROR - if the received packet is not - complete
      • -
      • Use HAL_Delay() - instead of counting loop
      • -
      •  __HAL_ETH_MAC_CLEAR_FLAG() - macro is removed: the MACSR - register is read only
      • -
      • Add the - following macros used to Wake - up the device from STOP mode - by Ethernet event - - - - :
      • -
          -
        • __HAL_ETH_EXTI_ENABLE_IT()
        • -
        • __HAL_ETH_EXTI_DISABLE_IT()
        • -
        • __HAL_ETH_EXTI_GET_FLAG()
        • -
        • __HAL_ETH_EXTI_CLEAR_FLAG()
        • -
        • __HAL_ETH_EXTI_SET_RISING_EGDE_TRIGGER()
        • -
        • __HAL_ETH_EXTI_SET_FALLING_EGDE_TRIGGER()
        • -
        • __HAL_ETH_EXTI_SET_FALLINGRISING_TRIGGER()
        • -
        -
      -
    • HAL - - - - WWDG update
    • -
        -
      • Update macro - parameters to use underscore: - __XXX__
      • -
      • Use of CMSIS - constants instead of magic values
      • -
      • Use - MODIFY_REG macro in HAL_WWDG_Init()
      • -
      • Add - IS_WWDG_ALL_INSTANCE in HAL_WWDG_Init() - and HAL_WWDG_DeInit()
      • -
      -
    • HAL - - - - IWDG update
    • -
        -
      • Use WRITE_REG - instead of SET_BIT for all - IWDG macros
      • -
      • __HAL_IWDG_CLEAR_FLAG - - - - - removed: no IWDG flag cleared - by access to SR register
      • -
      • Use - MODIFY_REG macro in HAL_IWDG_Init()
      • -
      • Add - IS_IWDG_ALL_INSTANCE in HAL_IWDG_Init()Add - - - - - the following macros used to - Wake
      • -
      -
    -

    V1.0.0 - - - - / 18-February-2014

    -

    Main Changes

    -
      -
    • First - - - - official release 
    • -
    -

    -

    For - - - - - complete documentation on STM32 - Microcontrollers visit www.st.com/STM32

    -
-

 

-
-
-

-
-
-

 

-
-
- \ No newline at end of file + +
+

Update History

+
+ +
+

Main Changes

+
    +
  • General updates to fix known defects and implementation enhancements.
  • +
  • HAL code quality enhancement for MISRA-C Rule-8.13 by adding const qualifiers.
  • +
  • HAL Generic update +
      +
    • Allow redefinition of macro UNUSED(x).
    • +
  • +
  • HAL RCC update +
      +
    • Correct the configuration macro of I2S clock source and the value of I2S external clock source.
    • +
    • Set the minimum value of PLLM.
    • +
    • Update the rest values for the macro __HAL_RCC_Axxx_FORCE_RESET() to avoid setting reserved bits.
    • +
  • +
  • HAL PWR update +
      +
    • Add a call to UNUSED() macro to avoid the generation of a warning related to the unused argument ‘Regulator’.
    • +
    • Add PWR_SLEEPENTRY_WFE_NO_EVT_CLEAR mode to avoid systematic clear of event in HAL_PWR_EnterSLEEPMode() and HAL_PWR_EnterSTOPMode().
    • +
  • +
  • HAL Cortex update +
      +
    • Add the following new HAL/LL CORTEX APIs to clear PWR pending event: +
        +
      • LL_LPM_ClearEvent().
      • +
      • HAL_CORTEX_ClearEvent().
      • +
    • +
  • +
  • HAL EXTI update +
      +
    • Fix computation of pExtiConfig->GPIOSel in HAL_EXTI_GetConfigLine.
    • +
  • +
  • HAL/LL ADC update +
      +
    • Cast both LL_ADC_REG_ReadConversionData6() and LL_ADC_REG_ReadConversionData8() returned values from uint16_t to uint8_t to be consistent with prototypes.
    • +
    • Add a call to UNUSED() macro in LL_ADC_DMA_GetRegAddr() API to prevent compilation warning due to unused ‘Register’ parameter.
    • +
  • +
  • HAL DAC update +
      +
    • Fix incorrect word ‘surcharged’ in functions headers.
    • +
    • Updated DAC buffer calibration according to RM.
    • +
  • +
  • HAL CEC update +
      +
    • Better performance by removing multiple volatile reads or writes in interrupt handler.
    • +
  • +
  • HAL RTC update +
      +
    • Check if the RTC calendar has been previously initialized before entering initialization mode.
    • +
    • Remove macro __HAL_RTC_TAMPER_GET_IT() as it is redundant with macro __HAL_RTC_TAMPER_GET_FLAG() and create an alias into the stm32_hal_legacy.h file.
    • +
    • Correct misleading note about shadow registers.
    • +
  • +
  • HAL CRYP update +
      +
    • Update Crypt/Decrypt IT processes to avoid Computation Completed IRQ fires before the DINR pointer increment.
    • +
  • +
  • HAL HASH update +
      +
    • HAL code quality enhancement for MISRA-C2012 Rule-2.2_c.
    • +
  • +
  • HAL TIM update +
      +
    • Align TIM_TIM2_ETH_PTP definition with the reference manual specification.
    • +
    • Improve driver robustness against wrong period values.
    • +
    • Improve driver robustness against wrong DMA related parameters.
    • +
    • Improve period configuration parameter check.
    • +
    • Remove Lock management from callback management functions.
    • +
    • Remove multiple volatile reads or writes in interrupt handler for better performance.
    • +
    • Improve HAL TIM driver’s operational behavior.
    • +
    • Remove unnecessary change of MOE bitfield in LL_TIM_BDTR_Init().
    • +
  • +
  • HAL LPTIM update +
      +
    • Apply same naming rules to clear FLAG related functions.
    • +
    • Remove Lock management from callback management functions.
    • +
  • +
  • HAL CAN update +
      +
    • Removal of never reached code.
    • +
    • Improve protection against bad inputs.
    • +
  • +
  • HAL DSI update +
      +
    • Update to align DSI ULPS entry and exit sequences with reference manual.
    • +
  • +
  • HAL ETH update +
      +
    • Update Rx descriptor tail pointer management to avoid race condition.
    • +
  • +
  • HAL UART update +
      +
    • New API HAL_UARTEx_GetRxEventType() to retrieve the type of event that has led the RxEventCallback execution.
    • +
    • Removal of HAL_LOCK/HAL_UNLOCK() calls in HAL UART Tx and Rx APIs.
    • +
    • Removal of __HAL_LOCK() from HAL_xxx_RegisterCallback()/HAL_xxx_UnRegisterCallback().
    • +
    • Avoid ORE flag to be cleared by a transmit process in polling mode.
    • +
    • Rework of UART_WaitOnFlagUntilTimeout() API to avoid being stuck forever when UART overrun error occurs and to enhance behavior.
    • +
  • +
  • HAL USART update +
      +
    • Removal of __HAL_LOCK() from HAL_xxx_RegisterCallback()/HAL_xxx_UnRegisterCallback().
    • +
  • +
  • HAL I2C update +
      +
    • Declare an internal macro link to DMA macro to check remaining data: I2C_GET_DMA_REMAIN_DATA.
    • +
    • Update I2C_MemoryTransmit_TXE_BTF process to disable TXE and BTF interrupts if nothing to do.
    • +
    • Clear TXE Flag at the end of transfer.
    • +
    • Move polling code of HAL memory interface through interrupt management to prevent timeout issue using HAL MEM interface through FreeRTOS.
    • +
    • Update I2C_IsErrorOccurred to return error if timeout is detected.
    • +
    • Clear the ADDRF flag only when direction is confirmed as changed, to prevent that the ADDRF flag is cleared too early when the restart is received.
    • +
    • Update HAL_I2C_Master_Transmit_IT to return HAL_BUSY instead of HAL_ERROR when timeout occur and I2C_FLAG_BUSY is SET.
    • +
    • Clear ACK bit once 3 bytes to read remain to be able to send the NACK once the transfer ends.
    • +
    • Duplicate the test condition after timeout detection to avoid false timeout detection.
    • +
    • Update HAL_I2C_IsDeviceReady() API to support 10_bit addressing mode: macro I2C_GENERATE_START is updated.
    • +
    • Update HAL I2C driver to prefetch data before starting the transmission: implementation of errata sheet workaround I2C2-190208 : Transmission stalled after first byte.
    • +
    • Update the HAL I2C driver to disactivate all interrupts after the end of transaction.
    • +
    • Update HAL_I2C_Init API to clear ADD10 bit in 7 bit addressing mode.
    • +
    • Solve Slave No stretch not functional by using HAL Slave interface.
    • +
    • Update HAL_FMPI2C_Mem_Write_IT API to initialize XferSize at 0.
    • +
    • Update I2C_Slave_ISR_IT, I2C_Slave_ISR_DMA and I2C_ITSlaveCplt to prevent the call of HAL_I2C_ListenCpltCallback twice.
    • +
    • Update I2C_WaitOnRXNEFlagUntilTimeout to check I2C_FLAG_AF independently from I2C_FLAG_RXNE.
    • +
    • Clear ACK bit once 3 bytes to read remain to be able to send the NACK once the transfer ends.
    • +
    • Remove the unusable code in function HAL_I2C_IsDeviceReady.
    • +
    • Update HAL_I2C_Master_Abort_IT to support memory abort transfer.
    • +
    • Update LL_I2C_HandleTranfer function to prevent undefined behavior of volatile usage before updating the CR2 register.
    • +
    • Update I2C_WaitOnFlagUntilTimeout to handle error case.
    • +
    • Update the HAL I2C driver to reset PreviousState to I2C_STATE_NONE at the end of transfer.
    • +
  • +
  • HAL SMBUS update +
      +
    • Update to fix issue of mismatched data received by master in case of data size to be transmitted by the slave is greater than the data size to be received by the master.
    • +
    • Add flush on TX register.
    • +
    • Change previous state from HAL_SMBUS_STATE_READY to HAL_SMBUS_STATE_NONE at the end of transfer.
    • +
    • Update HAL SMBUS driver to prefetch data before starting the transmission: implementation of errata sheet workaround I2C2-190208 : Transmission stalled after first byte.
    • +
  • +
  • HAL SAI update +
      +
    • Improve audio quality (avoid potential glitch).
    • +
    • Fix incorrect word ‘surcharged’.
    • +
  • +
  • HAL SPI update +
      +
    • Fix driver to don’t update state in case of error (HAL_SPI_STATE_READY will be set only in case of HAL_TIMEOUT).
    • +
    • Update HAL_SPI_TransmitReceive API to set the bit CRCNEXT in case of one byte transaction.
    • +
    • Update IT API to enable interrupts after process unlock.
    • +
    • Add wait on flag TXE to be set at the end of transaction to be aligned with reference manual.
    • +
  • +
  • HAL SPDIFRX update +
      +
    • Prevent hard fault by checking DMA usage.
    • +
    • Tuning of default SPDIFRX timeout.
    • +
  • +
  • HAL USB OTG update +
      +
    • ll_usb.c fix added to USB_ClearInterrupts(), should write “1†to clear the interrupt status bits of OTG_FS_GINTSTS register.
    • +
    • ll_usb.c: remove useless software setting to setup the frame interval at 80%.
    • +
    • ll_usb.c, hal_hcd.c: adding support of hub split transactions.
    • +
    • ll_usb.c: improve delay management to set core mode.
    • +
    • ll_usb.c, hal_pcd.c: fix device connection in case battery charging used with HS instance linked to internal FS PHY.
    • +
    • ll_usb.c: increase timeout value to allow core reset to complete.
    • +
  • +
  • HAL IRDA update +
      +
    • Removal of __HAL_LOCK() from HAL_xxx_RegisterCallback()/HAL_xxx_UnRegisterCallback().
    • +
  • +
  • HAL SMARTCARD update +
      +
    • Removal of __HAL_LOCK() from HAL_xxx_RegisterCallback()/HAL_xxx_UnRegisterCallback().
    • +
  • +
  • HAL SDMMC update +
      +
    • Update HAL SD processes to manage STBITERR flag.
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • General updates to fix HAL ETH defects and implementation enhancements.
  • +
  • HAL updates +
      +
    • HAL ETH update +
        +
      • Remove useless assert_param(IS_ETH_MAC_ADDRESS0123(MacAddr)) from static function ETH_MACAddressConfig().
      • +
      • Replace hard coded Rx buffer size (1000U) by macro ETH_RX_BUF_SIZE.
      • +
      • Correct bit positions when getting MAC and DMA configurations and replace ‘UnicastSlowProtocolPacketDetect’ by ‘UnicastPausePacketDetect’ in the MAC default configuration structure.
      • +
      • Ensure a delay of 4 TX_CLK/RX_CLK cycles between two successive write operations to the same register.
      • +
      • Disable DMA transmission in both HAL_ETH_Stop_IT() and HAL_ETH_Stop() APIs.
      • +
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • General updates to fix known defects and implementation enhancements.
  • +
  • All source files: update disclaimer to add reference to the new license agreement.
  • +
  • The following changes done on the HAL drivers require an update of the application code based on older HAL versions +
      +
    • Rework of HAL Ethernet driver to resolve problems and improve performance (compatibility break).
    • +
    • A new HAL Ethernet driver has been redesigned with new APIs, to bypass limitations with previous HAL Ethernet driver version.
    • +
    • The new HAL Ethernet driver is the recommended version. It is located as usual in Drivers/STM32F4xx_HAL_Driver/Src and Drivers/STM32F4xx_HAL_Driver/Inc folders. +
        +
      • It can be enabled through switch HAL_ETH_MODULE_ENABLED in stm32f4xx_hal_conf.h
      • +
    • +
    • The legacy HAL Ethernet driver is also present in the release in Drivers/STM32F4xx_HAL_Driver/Src/Legacy and Drivers/STM32F4xx_HAL_Driver/Inc/Legacy folders for software compatibility reasons. +
        +
      • Its usage is not recommended as deprecated. It can however be enabled through switch HAL_ETH_LEGACY_MODULE_ENABLED in stm32f4xx_hal_conf.h
      • +
    • +
  • +
  • HAL update +
      +
    • HAL ETH update +
        +
      • Entire receive process reworked.
      • +
      • Resolve the problem of received data corruption.
      • +
      • Implement transmission in interrupt mode.
      • +
      • Handle one interrupt for multiple transmitted packets.
      • +
      • Implement APIs to handle PTP feature.
      • +
      • Implement APIs to handle Timestamp feature.
      • +
      • Add support of receive buffer unavailable.
      • +
      • Update HAL_ETH_IRQHandler() to handle receive buffer unavailable.
      • +
    • +
    • HAL SMBUS update +
        +
      • Update to fix issue of mismatched data received by master in case of data size to be transmitted by the slave is greater than the data size to be received by the master. +
          +
        • Add flush on TX register.
        • +
      • +
    • +
    • HAL TIM update +
        +
      • __LL_TIM_CALC_PSC() macro update to round up the evaluate value when the fractional part of the division is greater than 0.5.
      • +
    • +
    • HAL LPTIM update +
        +
      • Add check on PRIMASK register to prevent from enabling unwanted global interrupts within LPTIM_Disable() and LL_LPTIM_Disable()
      • +
    • +
    • HAL UART update +
        +
      • Add const qualifier for read only pointers.
      • +
      • Improve header description of UART_WaitOnFlagUntilTimeout() function.
      • +
      • Add a check on the UART parity before enabling the parity error interruption.
      • +
      • Fix typo in UART_IT_TXE bit description.
      • +
    • +
    • HAL IRDA update +
        +
      • Improve header description of IRDA_WaitOnFlagUntilTimeout() function.
      • +
      • Add a check on the IRDA parity before enabling the parity error interrupt.
      • +
      • Add const qualifier for read only pointers.
      • +
    • +
    • HAL SMARTCARD update +
        +
      • Improve header description of SMARTCARD_WaitOnFlagUntilTimeout() function
      • +
      • Add const qualifier for read only pointers.
      • +
    • +
    • HAL NOR update +
        +
      • Apply adequate commands according to the command set field value
      • +
      • command set 1 for Micron JS28F512P33
      • +
      • command set 2 for Micron M29W128G and Cypress S29GL128P
      • +
      • Add new command operations: +
          +
        • NOR_CMD_READ_ARRAY
        • +
        • NOR_CMD_WORD_PROGRAM
        • +
        • NOR_CMD_BUFFERED_PROGRAM
        • +
        • NOR_CMD_CONFIRM
        • +
        • NOR_CMD_BLOCK_ERASE
        • +
        • NOR_CMD_BLOCK_UNLOCK
        • +
        • NOR_CMD_READ_STATUS_REG
        • +
        • NOR_CMD_CLEAR_STATUS_REG
        • +
      • +
      • Update some APIs in order to be compliant for memories with different command set, the updated APIs are: +
          +
        • HAL_NOR_Init()
        • +
        • HAL_NOR_Read_ID()
        • +
        • HAL_NOR_ReturnToReadMode()
        • +
        • HAL_NOR_Read()
        • +
        • HAL_NOR_Program()
        • +
        • HAL_NOR_ReadBuffer()
        • +
        • HAL_NOR_ProgramBuffer()
        • +
        • HAL_NOR_Erase_Block()
        • +
        • HAL_NOR_Erase_Chip()
        • +
        • HAL_NOR_GetStatus()
        • +
      • +
      • Align HAL_NOR_Init() API with core of the function when write operation is disabled to avoid HardFault.
      • +
    • +
    • HAL SDMMC update +
        +
      • Take into account the voltage range in the CMD1 command.
      • +
      • Add new LL function to have correct response for MMC driver.
      • +
      • Update the driver to have all fields correctly initialized.
      • +
      • Add an internal variable to manage the power class and call it before to update speed of bus width.
      • +
      • Add new API to get the value of the Extended CSD register and populate the ExtCSD field of the MMC handle.
      • +
      • In HAL_MMC_InitCard(), call to SDIO_PowerState_ON() moved after __HAL_MMC_ENABLE() to ensure MMC clock is enabled before the call to HAL_Delay() from within SDIO_PowerState_ON().
      • +
    • +
    • HAL DMA update +
        +
      • Manage the case of an invalid value of CallbackID passed to the HAL_DMA_RegisterCallback() API.
      • +
    • +
    • HAL LTDC update +
        +
      • Update HAL_LTDC_DeInit() to fix MCU Hang up during LCD turn OFF.
      • +
    • +
    • HAL I2C update +
        +
      • Update to fix issue detected due to low system frequency execution (HSI).
      • +
      • Declare an internal macro link to DMA macro to check remaining data: I2C_GET_DMA_REMAIN_DATA
      • +
      • Update HAL I2C Master Receive IT process to safe manage data N= 2 and N= 3. +
          +
        • Disable RxNE interrupt if nothing to do.
        • +
      • +
    • +
    • HAL USART update +
        +
      • Improve header description of USART_WaitOnFlagUntilTimeout() function.
      • +
      • Add a check on the USART parity before enabling the parity error interrupt.
      • +
      • Add const qualifier for read only pointers.
      • +
    • +
    • HAL/LL ADC update +
        +
      • Update LL_ADC_IsActiveFlag_MST_EOCS() API to get the appropriate flag.
      • +
      • Better performance by removing multiple volatile reads or writes in interrupt handler.
      • +
    • +
    • HAL FMPI2C update +
        +
      • Update to handle errors in polling mode. +
          +
        • Rename I2C_IsAcknowledgeFailed() to I2C_IsErrorOccurred() and correctly manage when error occurs.
        • +
      • +
    • +
    • HAL EXTI update +
        +
      • Update HAL_EXTI_GetConfigLine() API to fix wrong calculation of GPIOSel value.
      • +
    • +
    • HAL QSPI update +
        +
      • Update HAL_QSPI_Abort() and HAL_QSPI_Abort_IT() APIs to check on QSPI BUSY flag status before executing the abort procedure.
      • +
    • +
    • HAL/LL RTC cleanup +
        +
      • Use bits definitions from CMSIS Device header file instead of hard-coded values.
      • +
      • Wrap comments to be 80-character long and correct typos.
      • +
      • Move constants RTC_IT_TAMP. from hal_rtc.h to hal_rtc_ex.h.
      • +
      • Gather all instructions related to exiting the “init†mode into new function RTC_ExitInitMode().
      • +
      • Add new macro assert_param(IS_RTC_TAMPER_FILTER_CONFIG_CORRECT(sTamper->Filter, sTamper->Trigger)) to check tamper filtering is disabled in case tamper events are triggered on signal edges.
      • +
      • Rework functions HAL_RTCEx_SetTamper() and HAL_RTCEx_SetTamper_IT() to: +
          +
        • Write in TAFCR register in one single access instead of two.
        • +
        • Avoid modifying user structure sTamper.
        • +
      • +
      • Remove functions LL_RTC_EnablePushPullMode() and LL_RTC_DisablePushPullMode() as related to non-supported features.
      • +
      • Remove any reference to non-supported features (e.g., LL_RTC_ISR_TAMP3F).
      • +
      • Remove useless conditional defines as corresponding features are supported by all part-numbers (e.g., #if defined(RTC_TAFCR_TAMPPRCH)).
      • +
    • +
    • HAL USB OTG update +
        +
      • Fix USB_FlushRxFifo() and USB_FlushTxFifo() APIs by adding check on AHB master IDLE state before flushing the USB FIFO
      • +
      • Fix to avoid resetting host channel direction during channel halt
      • +
      • Fix to report correct received amount of data with USB DMA enabled
      • +
      • Fix to avoid compiler optimization on count variable used for USB HAL timeout loop check
      • +
      • Add missing registered callbacks check for HAL_HCD_HC_NotifyURBChange_Callback()
      • +
      • Add new API HAL_PCD_SetTestMode() APIs to handle USB device high speed Test modes
      • +
      • Setting SNAK for EPs not required during device reset
      • +
      • Update USB IRQ handler to enable EP OUT disable
      • +
      • Add support of USB IN/OUT Iso incomplete
      • +
      • Fix USB BCD data contact timeout
      • +
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • HAL update +
      +
    • HAL EXTI update +
        +
      • Update HAL_EXTI_GetConfigLine() API to set default configuration value of Trigger and GPIOSel before checking each corresponding registers.
      • +
    • +
    • HAL GPIO update +
        +
      • Update HAL_GPIO_Init() API to avoid the configuration of PUPDR register when Analog mode is selected.
      • +
    • +
    • HAL DMA update +
        +
      • Update HAL_DMA_IRQHandler() API to set the DMA state before unlocking access to the DMA handle.
      • +
    • +
    • LL ADC update +
        +
      • Update LL_ADC_DeInit() API to clear missing SQR3 register.
      • +
    • +
    • HAL CAN update +
        +
      • Update HAL_CAN_Init() API to be aligned with reference manual and to avoid timeout error.
      • +
    • +
    • HAL/LL RTC_BKP update +
        +
      • Update __HAL_RTC_…(__HANDLE__, …) macros to access registers through (__HANDLE__)->Instance pointer and avoid “unused variable†warnings.
      • +
      • Correct month management in IS_LL_RTC_MONTH() macro.
      • +
    • +
    • HAL RNG update +
        +
      • Update timeout mechanism to avoid false timeout detection in case of preemption.
      • +
    • +
    • HAL QSPI update +
        +
      • ES0305 workaround disabled for STM32412xx devices.
      • +
    • +
    • HAL I2C update +
        +
      • Update HAL_I2C_Mem_Write_DMA() and HAL_I2C_Mem_Read_DMA() APIs to initialize Devaddress, Memaddress and EventCount parameters.
      • +
      • Update to prevent several calls of Start bit: +
          +
        • Update I2C_MemoryTransmit_TXE_BTF() API to increment EventCount.
        • +
      • +
      • Update to avoid I2C interrupt in endless loop: +
          +
        • Update HAL_I2C_Master_Transmit_IT(), HAL_I2C_Master_Receive_IT(), HAL_I2C_Master_Transmit_DMA() and HAL_I2C_Master_Receive_DMA() APIs to unlock the I2C peripheral before generating the start.
        • +
      • +
      • Update to use the right macro to clear I2C ADDR flag inside I2C_Slave_ADDR() API as it’s indicated in the reference manual.
      • +
      • Update I2C_IsAcknowledgeFailed() API to avoid I2C in busy state if NACK received after transmitting register address.
      • +
      • Update HAL_I2C_EV_IRQHandler() and I2C_MasterTransmit_BTF() APIs to correctly manage memory transfers: +
          +
        • Add check on memory mode before calling callbacks procedures.
        • +
      • +
    • +
    • LL USART update +
        +
      • Handling of UART concurrent register access in case of race condition between Tx and Rx transfers (HAL UART and LL LPUART)
      • +
    • +
    • HAL SMBUS update +
        +
      • Updated HAL_SMBUS_ER_IRQHandler() API to return the correct error code “SMBUS_FLAG_PECERR†in case of packet error occurs.
      • +
    • +
    • HAL/LL SPI update +
        +
      • Updated to fix MISRA-C 2012 Rule-13.2.
      • +
      • Update LL_SPI_TransmitData8() API to avoid casting the result to 8 bits.
      • +
    • +
    • HAL UART update +
        +
      • Fix wrong comment related to RX pin configuration within the description section
      • +
      • Correction on UART ReceptionType management in case of ReceptionToIdle API are called from RxEvent callback
      • +
      • Handling of UART concurrent register access in case of race condition between Tx and Rx transfers (HAL UART and LL LPUART) +
          +
        • Update CAN Initialization sequence to set “request initialization†bit before exit from sleep mode.
        • +
      • +
    • +
    • HAL USB update +
        +
      • HAL PCD: add fix transfer complete for IN Interrupt transaction in single buffer mode
      • +
      • Race condition in USB PCD control endpoint receive ISR.
      • +
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • HAL +
      +
    • HAL/LL USART update +
        +
      • Fix typo in USART_Receive_IT() and USART_TransmitReceive_IT() APIs to avoid possible compilation issues if the UART driver files are not included.
      • +
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • General updates to fix known defects and enhancements implementation
  • +
  • Added new HAL FMPSMBUS extended driver to support FMPSMBUS fast Mode Plus.
  • +
  • Removed “register†keyword to be compliant with new C++ rules: +
      +
    • The register storage class specifier was deprecated in C++11 and removed in C++17.
    • +
  • +
  • HAL +
      +
    • HAL update
    • +
    • General updates to fix known defects and enhancements implementation.
    • +
    • Added new defines for ARM compiler V6: +
        +
      • __weak
      • +
      • __packed
      • +
      • __NOINLINE
      • +
    • +
    • Updated HAL TimeBase TIM, RTC alarm and RTC WakeUp templates for more robustness +
        +
      • Updated Hal_Init_Tick() API to properly store the priority when using the non-default time base.
      • +
    • +
    • Updated PPP_MODULE_ENABLED for FMPSMBUS.
    • +
    • HAL/LL ADC update +
        +
      • Updated to add include of the LL ADC driver.
      • +
      • Updated the following APIs to set status HAL_ADC_STATE_ERROR_INTERNAL and error code HAL_ADC_ERROR_INTERNAL when error occurs: +
          +
        • HAL_ADC_Start()
        • +
        • HAL_ADC_Start_IT()
        • +
        • HAL_ADC_Start_DMA()
        • +
        • HAL_ADCEx_InjectedStart()
        • +
        • HAL_ADCEx_InjectedStart_IT()
        • +
        • HAL_ADCEx_MultiModeStart_DMA()
        • +
      • +
      • Updated HAL_ADC_Stop_DMA() API to check if DMA state is Busy before calling HAL_DMA_Abort() API to avoid DMA internal error.
      • +
      • Updated IS_ADC_CHANNEL to support temperature sensor for: +
          +
        • STM32F411xE
        • +
        • STM32F413xx
        • +
        • STM32F423xx
        • +
      • +
      • Fixed wrong defined values for: +
          +
        • LL_ADC_MULTI_REG_DMA_LIMIT_3
        • +
        • LL_ADC_MULTI_REG_DMA_UNLMT_3
        • +
      • +
      • Added __LL_ADC_CALC_VREFANALOG_VOLTAGE() macro to evaluate analog reference voltage.
      • +
      • Removed __LL_ADC_CALC_TEMPERATURE() macro for STM32F4x9 devices as the TS_CAL2 is not available.
      • +
    • +
    • HAL/LL DAC update +
        +
      • Added restruction on DAC Channel 2 defines and parameters.
      • +
      • HAL_DAC_MSPINIT_CB_ID and HAL_DAC_MSPDEINIT_CB_ID used instead of HAL_DAC_MSP_INIT_CB_ID and HAL_DAC_MSP_DEINIT_CB_ID.
      • +
      • Updated to support dual mode: +
          +
        • Added two new APIs: +
            +
          • HAL_DACEx_DualStart()
          • +
          • HAL_DACEx_DualStop()
          • +
        • +
      • +
      • Added position bit definition to be used instead of __DAC_MASK_SHIFT macro +
          +
        • __DAC_MASK_SHIFT macro has been removed.
        • +
      • +
      • Updated HAL_DAC_Start_DMA() API to return HAL_ERROR when error occurs.
      • +
      • Updated HAL_DAC_Stop_DMA() API to not return HAL_ERROR when DAC is already disabled.
      • +
    • +
    • HAL CEC update +
        +
      • Updated HAL_CEC_IRQHandler() API to avoid appending an extra byte to the end of a message.
      • +
    • +
    • HAL/LL GPIO update +
        +
      • Updated IS_GPIO_AF() to add missing values for STM32F401xC and STM32F401xE devices: +
          +
        • GPIO_AF3_TIM9
        • +
        • GPIO_AF3_TIM10
        • +
        • GPIO_AF3_TIM11
        • +
      • +
      • Updated LL/HAL GPIO_TogglePin() APIs to allow multi Pin’s toggling.
      • +
      • Updated HAL_GPIO_Init() API to avoid the configuration of PUPDR register when Analog mode is selected.
      • +
    • +
    • HAL/LL RCC update +
        +
      • Updated HAL_RCC_OscConfig() API to add missing checks and to don’t return HAL_ERROR if request repeats the current PLL configuration.
      • +
      • Updated IS_RCC_PLLN_VALUE(VALUE) macro in case of STM32F411xE device in order to be aligned with reference manual.
      • +
    • +
    • HAL SD update +
        +
      • Update function SD_FindSCR() to resolve issue of FIFO blocking when reading.
      • +
      • Update read/write functions in DMA mode in order to force the DMA direction, updated functions: +
          +
        • HAL_SD_ReadBlocks_DMA()
        • +
        • HAL_SD_WriteBlocks_DMA()
        • +
      • +
      • Add the block size settings in the initialization functions and remove it from read/write transactions to avoid repeated and inefficient reconfiguration, updated functions: +
          +
        • HAL_SD_InitCard()
        • +
        • HAL_SD_GetCardStatus()
        • +
        • HAL_SD_ConfigWideBusOperation()
        • +
        • HAL_SD_ReadBlocks()
        • +
        • HAL_SD_WriteBlocks()
        • +
        • HAL_SD_ReadBlocks_IT()
        • +
        • HAL_SD_WriteBlocks_IT()
        • +
        • HAL_SD_ReadBlocks_DMA()
        • +
        • HAL_SD_WriteBlocks_DMA()
        • +
      • +
    • +
    • HAL MMC update +
        +
      • Add the block size settings in the initialization function and remove it from read/write transactions to avoid repeated and inefficient reconfiguration, updated functions: +
          +
        • HAL_MMC_InitCard()
        • +
        • HAL_MMC_ReadBlocks()
        • +
        • HAL_MMC_WriteBlocks()
        • +
        • HAL_MMC_ReadBlocks_IT()
        • +
        • HAL_MMC_WriteBlocks_IT()
        • +
        • HAL_MMC_ReadBlocks_DMA()
        • +
        • HAL_MMC_WriteBlocks_DMA()
        • +
      • +
      • Update read/write functions in DMA mode in order to force the DMA direction, updated functions: +
          +
        • HAL_MMC_ReadBlocks_DMA()
        • +
        • HAL_MMC_WriteBlocks_DMA()
        • +
      • +
      • Deploy new functions MMC_ReadExtCSD() and SDMMC_CmdSendEXTCSD () that read and check the sectors number of the device in order to resolve the issue of wrongly reading big memory size.
      • +
    • +
    • HAL NAND update +
        +
      • Update functions HAL_NAND_Read_SpareArea_16b() and HAL_NAND_Write_SpareArea_16b() to fix column address calculation issue.
      • +
    • +
    • LL SDMMC update +
        +
      • Update the definition of SDMMC_DATATIMEOUT constant in order to allow the user to redefine it in his proper application.
      • +
      • Remove ‘register’ storage class specifier from LL SDMMC driver.
      • +
      • Deploy new functions MMC_ReadExtCSD() and SDMMC_CmdSendEXTCSD () that read and check the sectors number of the device in order to resolve the issue of wrongly reading big memory size.
      • +
    • +
    • HAL SMBUS update +
        +
      • Support for Fast Mode Plus to be SMBUS rev 3 compliant.
      • +
      • Added HAL_FMPSMBUSEx_EnableFastModePlus() and HAL_FMPSMBUSEx_DisableFastModePlus() APIs to manage Fm+.
      • +
      • Updated SMBUS_MasterTransmit_BTF() , SMBUS_MasterTransmit_TXE() and SMBUS_MasterReceive_BTF() APIs to allow stop generation when CurrentXferOptions is different from SMBUS_FIRST_FRAME and SMBUS_NEXT_FRAME.
      • +
      • Updated SMBUS_ITError() API to correct the twice call of HAL_SMBUS_ErrorCallback.
      • +
    • +
    • HAL SPI update +
        +
      • Updated HAL_SPI_Init() API +
          +
        • To avoid setting the BaudRatePrescaler in case of Slave Motorola Mode.
        • +
        • Use the bit-mask for SPI configuration.
        • +
      • +
      • Updated Transmit/Receive processes in half-duplex mode +
          +
        • Disable the SPI instance before setting BDIOE bit.
        • +
      • +
      • Fixed wrong timeout management
      • +
      • Calculate Timeout based on a software loop to avoid blocking issue if Systick is disabled.
      • +
    • +
    • HAL SPDIFRX update +
        +
      • Remove ‘register’ storage class specifier from HAL SPDIFRX driver.
      • +
    • +
    • HAL I2S update +
        +
      • Updated I2SEx APIs to correctly support circular transfers +
          +
        • Updated I2SEx_TxRxDMACplt() API to manage DMA circular mode.
        • +
      • +
      • Updated HAL_I2SEx_TransmitReceive_DMA() API to set hdmatx (transfer callback and half) to NULL.
      • +
    • +
    • HAL SAI update +
        +
      • Updated to avoid the incorrect left/right synchronization. +
          +
        • Updated HAL_SAI_Transmit_DMA() API to follow the sequence described in the reference manual for slave transmitter mode.
        • +
      • +
      • Updated HAL_SAI_Init() API to correct the formula in case of SPDIF is wrong.
      • +
    • +
    • HAL CRYP update +
        +
      • Updated HAL_CRYP_SetConfig() and HAL_CRYP_GetConfig() APIs to set/get the continent of KeyIVConfigSkip correctly.
      • +
    • +
    • HAL EXTI update +
        +
      • __EXTI_LINE__ is now used instead of __LINE__ which is a standard C macro.
      • +
    • +
    • HAL DCMI +
        +
      • Support of HAL callback registration feature for DCMI extended driver.
      • +
    • +
    • HAL/LL TIM update +
        +
      • Updated HAL_TIMEx_OnePulseN_Start() and HAL_TIMEx_OnePulseN_Stop() APIs (pooling and IT mode) to take into consideration all OutputChannel parameters.
      • +
      • Corrected reversed description of TIM_LL_EC_ONEPULSEMODE One Pulse Mode.
      • +
      • Updated LL_TIM_GetCounterMode() API to return the correct counter mode.
      • +
    • +
    • HAL/LL SMARTCARD update +
        +
      • Fixed invalid initialization of SMARTCARD configuration by removing FIFO mode configuration as it is not member of SMARTCARD_InitTypeDef Structure.
      • +
      • Fixed typos in SMARTCARD State definition description
      • +
    • +
    • HAL/LL IRDA update +
        +
      • Fixed typos in IRDA State definition description
      • +
    • +
    • LL USART update +
        +
      • Remove useless check on maximum BRR value by removing IS_LL_USART_BRR_MAX() macro.
      • +
      • Update USART polling and interruption processes to fix issues related to accesses out of user specified buffer.
      • +
    • +
    • HAL USB update +
        +
      • Enhanced USB OTG host HAL with USB DMA is enabled: +
          +
        • fixed ping and data toggle issue,
        • +
        • reworked Channel error report management
        • +
      • +
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • General updates to fix known defects.
  • +
  • HAL/LL I2C update +
      +
    • Update to fix hardfault issue with HAL_I2C_Mem_Write_DMA() API: +
        +
      • Abort the right ongoing DMA transfer when memory write access request operation failed: fix typo “hdmarx†replaced by “hdmatxâ€
      • +
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • General updates to fix known defects and enhancements implementation
  • +
  • HAL/LL I2C update +
      +
    • Update HAL_I2C_ER_IRQHandler() API to fix acknowledge failure issue with I2C memory IT processes +
        +
      • Add stop condition generation when NACK occurs.
      • +
    • +
    • Update I2C_DMAXferCplt(), I2C_DMAError() and I2C_DMAAbort() APIs to fix hardfault issue when hdmatx and hdmarx parameters in i2c handle aren’t initialized (NULL pointer). +
        +
      • Add additional check on hi2c->hdmtx and hi2c->hdmarx before resetting DMA Tx/Rx complete callbacks
      • +
    • +
    • Update Sequential transfer APIs to adjust xfermode condition. +
        +
      • Replace hi2c->XferCount < MAX_NBYTE_SIZE by hi2c->XferCount <= MAX_NBYTE_SIZE which corresponds to a case without reload
      • +
    • +
  • +
  • HAL/LL USB update +
      +
    • Bug fix: USB_ReadPMA() and USB_WritePMA() by ensuring 16-bits access to USB PMA memory
    • +
    • Bug fix: correct USB RX count calculation
    • +
    • Fix USB Bulk transfer double buffer mode
    • +
    • Remove register keyword from USB defined macros as no more supported by C++ compiler
    • +
    • Minor rework on USBD_Start() and USBD_Stop() APIs: stopping device will be handled by HAL_PCD_DeInit() API.
    • +
    • Remove non used API for USB device mode.
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Add new HAL FMPSMBUS and LL FMPI2C drivers
  • +
  • General updates to fix known defects and enhancements implementation
  • +
  • Update HAL CRYP driver to support block by block decryption without reinitializes the IV and KEY for each call.
  • +
  • Improve code quality by fixing MisraC-2012 violations
  • +
  • HAL/LL USB update +
      +
    • Add handling USB host babble error interrupt
    • +
    • Fix Enabling ULPI interface for platforms that integrates USB HS PHY
    • +
    • Fix Host data toggling for IN Iso transfers
    • +
    • Ensure to disable USB EP during endpoint deactivation
    • +
  • +
  • HAL CRYP update +
      +
    • Update HAL CRYP driver to support block by block decryption without initializing the IV and KEY at each call. +
        +
      • Add new CRYP Handler parameters: “KeyIVConfig†and “SizesSumâ€
      • +
      • Add new CRYP init parameter: "KeyIVConfigSkip
      • +
    • +
  • +
  • HAL I2S update +
      +
    • Update HAL_I2S_DMAStop() API to be more safe +
        +
      • Add a check on BSY, TXE and RXNE flags before disabling the I2S
      • +
    • +
    • Update HAL_I2S_DMAStop() API to fix multi-call transfer issue(to avoid re-initializing the I2S for the next transfer). +
        +
      • Add __HAL_I2SEXT_FLUSH_RX_DR() and __HAL_I2S_FLUSH_RX_DR() macros to flush the remaining data inside DR registers.
      • +
      • Add new ErrorCode define: HAL_I2S_ERROR_BUSY_LINE_RX
      • +
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • General updates to fix known defects and enhancements implementation
  • +
  • HAL Generic update +
      +
    • HAL_SetTickFreq(): update to restore the previous tick frequency when HAL_InitTick() configuration failed.
    • +
  • +
  • HAL/LL GPIO update +
      +
    • Update GPIO initialization sequence to avoid unwanted pulse on GPIO Pin’s
    • +
  • +
  • HAL EXTI update +
      +
    • General update to enhance HAL EXTI driver robustness +
        +
      • Add additional assert check on EXTI config lines
      • +
      • Update to compute EXTI line mask before read/write access to EXTI registers
      • +
    • +
    • Update EXTI callbacks management to be compliant with reference manual: only one PR register for rising and falling interrupts. +
        +
      • Update parameters in EXTI_HandleTypeDef structure: merge HAL EXTI RisingCallback and FallingCallback in only one PendingCallback
      • +
      • Remove HAL_EXTI_RISING_CB_ID and HAL_EXTI_FALLING_CB_ID values from EXTI_CallbackIDTypeDef enumeration.
      • +
    • +
    • Update HAL_EXTI_IRQHandler() API to serve interrupts correctly. +
        +
      • Update to compute EXTI line mask before handle EXTI interrupt.
      • +
    • +
    • Update to support GPIO port interrupts: +
        +
      • Add new “GPIOSel†parameter in EXTI_ConfigTypeDef structure
      • +
    • +
  • +
  • HAL/LL RCC update +
      +
    • Update HAL_RCCEx_PeriphCLKConfig() API to support PLLI2S configuration for STM32F42xxx and STM32F43xxx devices
    • +
    • Update the HAL_RCC_ClockConfig() and HAL_RCC_DeInit() API to don’t overwrite the custom tick priority
    • +
    • Fix LL_RCC_DeInit() failure detected with gcc compiler and high optimization level is selected(-03)
    • +
    • Update HAL_RCC_OscConfig() API to don’t return HAL_ERROR if request repeats the current PLL configuration
    • +
  • +
  • HAL ADC update +
      +
    • Update LL_ADC_REG_Init() to fix wrong ADC CR1 register configuration +
        +
      • The ADC sequencer length is part of ADC SQR1 register not of ADC CR1 register
      • +
    • +
  • +
  • HAL CRYP update +
      +
    • Update HAL_CRYP_Encrypt() and HAL_CRYP_Decrypt() APIs to take into consideration the datatype fed to the DIN register (1-, 8-, 16-, or 32-bit data) when padding the last block of the payload, in case the size of this last block is less than 128 bits.
    • +
  • +
  • HAL RNG update +
      +
    • Update HAL_RNG_IRQHandler() API to fix error code management issue: error code is assigned “HAL_RNG_ERROR_CLOCK†in case of clock error and “HAL_RNG_ERROR_SEED†in case of seed error, not the opposite.
    • +
  • +
  • HAL DFSDM update +
      +
    • Update DFSDM_GetChannelFromInstance() API to remove unreachable check condition
    • +
  • +
  • HAL DMA update +
      +
    • Update HAL_DMA_Start_IT() API to omit the FIFO error
    • +
  • +
  • HAL FLASH update +
      +
    • Update FLASH_Program_DoubleWord() API to fix with EWARM high level optimization issue
    • +
  • +
  • HAL QSPI update +
      +
    • Remove Lock mechanism from HAL_QSPI_Init() and HAL_QSPI_DeInit() APIs
    • +
  • +
  • HAL HASH update +
      +
    • Null pointer on handler “hhash†is now checked before accessing structure member “hhash->Init.DataType†in the following API: +
        +
      • HAL_HASH_Init()
      • +
    • +
    • Following interrupt-based APIs have been added. Interrupt mode could allow the MCU to enter “Sleep†mode while a data block is being processed. Please refer to the “##### How to use this driver #####†section for details about their use. +
        +
      • HAL_HASH_SHA1_Accmlt_IT()
      • +
      • HAL_HASH_MD5_Accmlt_IT()
      • +
      • HAL_HASHEx_SHA224_Accmlt_IT()
      • +
      • HAL_HASHEx_SHA256_Accmlt_IT()
      • +
    • +
    • Following aliases have been added (just for clarity sake) as they shall be used at the end of the computation of a multi-buffers message and not at the start: +
        +
      • HAL_HASH_SHA1_Accmlt_End() to be used instead of HAL_HASH_SHA1_Start()
      • +
      • HAL_HASH_MD5_Accmlt_End() to be used instead of HAL_HASH_MD5_Start()
      • +
      • HAL_HASH_SHA1_Accmlt_End_IT() to be used instead of HAL_HASH_SHA1_Start_IT()
      • +
      • HAL_HASH_MD5_Accmlt_End_IT() to be used instead of HAL_HASH_MD5_Start_IT()
      • +
      • HAL_HASHEx_SHA224_Accmlt_End() to be used instead of HAL_HASHEx_SHA224_Start()
      • +
      • HAL_HASHEx_SHA256_Accmlt_End() to be used instead of HAL_HASHEx_SHA256_Start()
      • +
      • HAL_HASHEx_SHA224_Accmlt_End_IT() to be used instead of HAL_HASHEx_SHA224_Start_IT()
      • +
      • HAL_HASHEx_SHA256_Accmlt_End_IT() to be used instead of HAL_HASHEx_SHA256_Start_IT()
      • +
    • +
    • MISRAC-2012 rule R.5.1 (identifiers shall be distinct in the first 31 characters) constrained the naming of the above listed aliases (e.g. HAL_HASHEx_SHA256_Accmlt_End() could not be named HAL_HASHEx_SHA256_Accumulate_End(). Otherwise the name would have conflicted with HAL_HASHEx_SHA256_Accumulate_End_IT()). In order to have aligned names following APIs have been renamed: +
        +
      • HAL_HASH_MD5_Accumulate() renamed HAL_HASH_MD5_Accmlt()
      • +
      • HAL_HASH_SHA1_Accumulate() renamed HAL_HASH_SHA1_Accmlt()
      • +
      • HAL_HASHEx_SHA224_Accumulate() renamed HAL_HASHEx_SHA224_Accmlt()
      • +
      • HAL_HASHEx_SHA256_Accumulate() renamed HAL_HASHEx_SHA256_Accmlt()
      • +
    • +
    • HASH handler state is no more reset to HAL_HASH_STATE_READY once DMA has been started in the following APIs: +
        +
      • HAL_HASH_MD5_Start_DMA()
      • +
      • HAL_HMAC_MD5_Start_DMA()
      • +
      • HAL_HASH_SHA1_Start_DMA()
      • +
      • HAL_HMAC_SHA1_Start_DMA()
      • +
    • +
    • HASH phase state is now set to HAL_HASH_PHASE_READY once the digest has been read in the following APIs: +
        +
      • HASH_IT()
      • +
      • HMAC_Processing()
      • +
      • HASH_Start()
      • +
      • HASH_Finish()
      • +
    • +
    • Case of a large buffer scattered around in memory each piece of which is not necessarily a multiple of 4 bytes in length. +
        +
      • In section “##### How to use this driver #####â€, sub-section "*** Remarks on message length ***" added to provide recommendations to follow in such case.
      • +
      • No modification of the driver as the root-cause is at design-level.
      • +
    • +
  • +
  • HAL CAN update +
      +
    • HAL_CAN_GetRxMessage() update to get the correct value for the RTR (type of frame for the message that will be transmitted) field in the CAN_RxHeaderTypeDef structure.
    • +
  • +
  • HAL DCMI update +
      +
    • Add new HAL_DCMI_ConfigSyncUnmask() API to set embedded synchronization delimiters unmasks.
    • +
  • +
  • HAL RTC update +
      +
    • Following IRQ handlers’ implementation has been aligned with the STM32Cube firmware specification (in case of interrupt lines shared by multiple events, first check the IT enable bit is set then check the IT flag is set too): +
        +
      • HAL_RTC_AlarmIRQHandler()
      • +
      • HAL_RTCEx_WakeUpTimerIRQHandler()
      • +
      • HAL_RTCEx_TamperTimeStampIRQHandler()
      • +
    • +
  • +
  • HAL WWDG update +
      +
    • In “##### WWDG Specific features #####†descriptive comment section: +
        +
      • Maximal prescaler value has been corrected (8 instead of 128).
      • +
      • Maximal APB frequency has been corrected (42MHz instead of 56MHz) and possible timeout values updated.
      • +
    • +
  • +
  • HAL DMA2D update +
      +
    • Add the following API’s to Start DMA2D CLUT Loading. +
        +
      • HAL_DMA2D_CLUTStartLoad() Start DMA2D CLUT Loading.
      • +
      • HAL_DMA2D_CLUTStartLoad_IT() Start DMA2D CLUT Loading with interrupt enabled.
      • +
    • +
    • The following old wrong services will be kept in the HAL DCMI driver for legacy purpose and a specific Note is added: +
        +
      • HAL_DMA2D_CLUTLoad() can be replaced with HAL_DMA2D_CLUTStartLoad()
      • +
      • HAL_DMA2D_CLUTLoad_IT() can be replaced with HAL_DMA2D_CLUTStartLoad_IT()
      • +
      • HAL_DMA2D_ConfigCLUT() can be omitted as the config can be performed using the HAL_DMA2D_CLUTStartLoad() API.
      • +
    • +
  • +
  • HAL SDMMC update +
      +
    • Fix typo in “FileFormatGroup†parameter in the HAL_MMC_CardCSDTypeDef and HAL_SD_CardCSDTypeDef structures
    • +
    • Fix an improve handle state and error management
    • +
    • Rename the defined MMC card capacity type to be more meaningful: +
        +
      • Update MMC_HIGH_VOLTAGE_CARD to MMC LOW_CAPACITY_CARD
      • +
      • Update MMC_DUAL_VOLTAGE_CRAD to MMC_HIGH_CAPACITY_CARD
      • +
    • +
    • Fix management of peripheral flags depending on commands or data transfers +
        +
      • Add new defines “SDIO_STATIC_CMD_FLAGS†and “SDIO_STATIC_DATA_FLAGSâ€
      • +
      • Updates HAL SD and HAL MMC drivers to manage the new SDIO static flags.
      • +
    • +
    • Due to limitation SDIO hardware flow control indicated in Errata Sheet: +
        +
      • In 4-bits bus wide mode, do not use the HAL_SD_WriteBlocks_IT() or HAL_SD_WriteBlocks() APIs otherwise underrun will occur and it isn’t possible to activate the flow control.
      • +
      • Use DMA mode when using 4-bits bus wide mode or decrease the SDIO_CK frequency.
      • +
    • +
  • +
  • HAL UART update +
      +
    • Update UART polling processes to handle efficiently the Lock mechanism +
        +
      • Move the process unlock at the top of the HAL_UART_Receive() and HAL_UART_Transmit() API.
      • +
    • +
    • Fix baudrate calculation error for clock higher than 172Mhz +
        +
      • Add a forced cast on UART_DIV_SAMPLING8() and UART_DIV_SAMPLING16() macros.
      • +
      • Remove useless parenthesis from UART_DIVFRAQ_SAMPLING8(), UART_DIVFRAQ_SAMPLING16(), UART_BRR_SAMPLING8() and UART_BRR_SAMPLING16() macros to solve some MISRA warnings.
      • +
    • +
    • Update UART interruption handler to manage correctly the overrun interrupt +
        +
      • Add in the HAL_UART_IRQHandler() API a check on USART_CR1_RXNEIE bit when an overrun interrupt occurs.
      • +
    • +
    • Fix baudrate calculation error UART9 and UART10 +
        +
      • In UART_SetConfig() API fix UART9 and UART10 clock source when computing baudrate values by adding a check on these instances and setting clock sourcePCLK2 instead of PCLK1.
      • +
    • +
    • Update UART_SetConfig() API +
        +
      • Split HAL_RCC_GetPCLK1Freq() and HAL_RCC_GetPCLK2Freq() macros from the UART_BRR_SAMPLING8() and UART_BRR_SAMPLING8() macros
      • +
    • +
  • +
  • HAL USART update +
      +
    • Fix baudrate calculation error for clock higher than 172Mhz +
        +
      • Add a forced cast on USART_DIV() macro.
      • +
      • Remove useless parenthesis from USART_DIVFRAQ() macro to solve some MISRA warnings.
      • +
    • +
    • Update USART interruption handler to manage correctly the overrun interrupt +
        +
      • Add in the HAL_USART_IRQHandler() API a check on USART_CR1_RXNEIE bit when an overrun interrupt occurs.
      • +
    • +
    • Fix baudrate calculation error UART9 and UART10 +
        +
      • In USART_SetConfig() API fix UART9 and UART10 clock source when computing baudrate values by adding a check on these instances and setting clock sourcePCLK2 instead of PCLK1.
      • +
    • +
    • Update USART_SetConfig() API +
        +
      • Split HAL_RCC_GetPCLK1Freq() and HAL_RCC_GetPCLK2Freq() macros from the USART_BRR() macro
      • +
    • +
  • +
  • HAL IRDA update +
      +
    • Fix baudrate calculation error for clock higher than 172Mhz +
        +
      • Add a forced cast on IRDA_DIV() macro.
      • +
      • Remove useless parenthesis from IRDA_DIVFRAQ() macro to solve some MISRA warnings.
      • +
    • +
    • Update IRDA interruption handler to manage correctly the overrun interrupt +
        +
      • Add in the HAL_IRDA_IRQHandler() API a check on USART_CR1_RXNEIE bit when an overrun interrupt occurs.
      • +
    • +
    • Fix baudrate calculation error UART9 and UART10 +
        +
      • In IRDA_SetConfig() API fix UART9 and UART10 clock source when computing baudrate values by adding a check on these instances and setting clock sourcePCLK2 instead of PCLK1.
      • +
    • +
    • Update IRDA_SetConfig() API +
        +
      • Split HAL_RCC_GetPCLK1Freq() and HAL_RCC_GetPCLK2Freq() macros from the IRDA_BRR() macro
      • +
    • +
  • +
  • HAL SMARTCARD update +
      +
    • Fix baudrate calculation error for clock higher than 172Mhz +
        +
      • Add a forced cast on SMARTCARD_DIV() macro.
      • +
      • Remove useless parenthesis from SMARTCARD_DIVFRAQ() macro to solve some MISRA warnings.
      • +
    • +
    • Update SMARTCARD interruption handler to manage correctly the overrun interrupti +
        +
      • Add in the HAL_SMARTCARD_IRQHandler() API a check on USART_CR1_RXNEIE bit when an overrun interrupt occurs.
      • +
    • +
    • Update SMARTCARD_SetConfig() API +
        +
      • Split HAL_RCC_GetPCLK1Freq() and HAL_RCC_GetPCLK2Freq() macros from the SMARTCARD_BRR() macro
      • +
    • +
  • +
  • HAL TIM update +
      +
    • Add new macros to enable and disable the fast mode when using the one pulse mode to output a waveform with a minimum delay +
        +
      • __HAL_TIM_ENABLE_OCxFAST() and __HAL_TIM_DISABLE_OCxFAST().
      • +
    • +
    • Update Encoder interface mode to keep TIM_CCER_CCxNP bits low +
        +
      • Add TIM_ENCODERINPUTPOLARITY_RISING and TIM_ENCODERINPUTPOLARITY_FALLING definitions to determine encoder input polarity.
      • +
      • Add IS_TIM_ENCODERINPUT_POLARITY() macro to check the encoder input polarity.
      • +
      • Update HAL_TIM_Encoder_Init() API +
          +
        • Replace IS_TIM_IC_POLARITY() macro by IS_TIM_ENCODERINPUT_POLARITY() macro.
        • +
      • +
    • +
    • Update TIM remapping input configuration in HAL_TIMEx_RemapConfig() API +
        +
      • Remove redundant check on LPTIM_OR_TIM5_ITR1_RMP bit and replace it by check on LPTIM_OR_TIM9_ITR1_RMP bit.
      • +
    • +
    • Update HAL_TIMEx_MasterConfigSynchronization() API to avoid functional errors and assert fails when using some TIM instances as input trigger. +
        +
      • Replace IS_TIM_SYNCHRO_INSTANCE() macro by IS_TIM_MASTER_INSTANCE() macro.
      • +
      • Add IS_TIM_SLAVE_INSTANCE() macro to check on TIM_SMCR_MSM bit.
      • +
    • +
    • Add lacking TIM input remapping definition +
        +
      • Add LL_TIM_TIM11_TI1_RMP_SPDIFRX and LL_TIM_TIM2_ITR1_RMP_ETH_PTP.
      • +
      • Add lacking definition for linked LPTIM_TIM input trigger remapping +
          +
        • Add following definitions : LL_TIM_TIM9_ITR1_RMP_TIM3_TRGO, LL_TIM_TIM9_ITR1_RMP_LPTIM, LL_TIM_TIM5_ITR1_RMP_TIM3_TRGO, LL_TIM_TIM5_ITR1_RMP_LPTIM, LL_TIM_TIM1_ITR2_RMP_TIM3_TRGO and LL_TIM_TIM1_ITR2_RMP_LPTIM.
        • +
        • Add a new mechanism in LL_TIM_SetRemap() API to remap TIM1, TIM9, and TIM5 input triggers mapped on LPTIM register.
        • +
      • +
    • +
  • +
  • HAL LPTIM update +
      +
    • Add a polling mechanism to check on LPTIM_FLAG_XXOK flags in different API +
        +
      • Add LPTIM_WaitForFlag() API to wait for flag set.
      • +
      • Perform new checks on HAL_LPTIM_STATE_TIMEOUT.
      • +
    • +
    • Add lacking definitions of LPTIM input trigger remapping and its related API +
        +
      • LL_LPTIM_INPUT1_SRC_PAD_AF, LL_LPTIM_INPUT1_SRC_PAD_PA4, LL_LPTIM_INPUT1_SRC_PAD_PB9 and LL_LPTIM_INPUT1_SRC_TIM_DAC.
      • +
      • Add a new API LL_LPTIM_SetInput1Src() to access to the LPTIM_OR register and remap the LPTIM input trigger.
      • +
    • +
    • Perform a new check on indirect EXTI23 line associated to the LPTIM wake up timer +
        +
      • Condition the use of the LPTIM Wake-up Timer associated EXTI line configuration’s macros by EXTI_IMR_MR23 bit in different API : +
          +
        • __HAL_LPTIM_WAKEUPTIMER_EXTI_ENABLE/DDISABLE_FALLING_EDGE()
        • +
        • __HAL_LPTIM_WAKEUPTIMER_EXTI_DISABLE_FALLING_EDGE()
        • +
        • __HAL_LPTIM_WAKEUPTIMER_EXTI_ENABLE_RISING_EDGE()
        • +
        • __HAL_LPTIM_WAKEUPTIMER_EXTI_DISABLE_RISING_EDGE()
        • +
        • __HAL_LPTIM_WAKEUPTIMER_EXTI_ENABLE_RISING_FALLING_EDGE()
        • +
        • __HAL_LPTIM_WAKEUPTIMER_EXTI_DISABLE_RISING_FALLING_EDGE()
        • +
        • __HAL_LPTIM_WAKEUPTIMER_EXTI_GET_FLAG()
        • +
        • __HAL_LPTIM_WAKEUPTIMER_EXTI_CLEAR_FLAG()
        • +
        • __HAL_LPTIM_WAKEUPTIMER_EXTI_GENERATE_SWIT()
        • +
      • +
      • Update HAL_LPTIM_TimeOut_Start_IT(), HAL_LPTIM_TimeOut_Stop_IT(), HAL_LPTIM_Counter_Start_IT() and HAL_LPTIM_Counter_Stop_IT() API by adding Enable/Disable rising edge trigger on the LPTIM Wake-up Timer Exti line.
      • +
      • Add __HAL_LPTIM_WAKEUPTIMER_EXTI_CLEAR_FLAG() in the end of the HAL_LPTIM_IRQHandler() API conditioned by EXTI_IMR_MR23 bit.
      • +
    • +
  • +
  • HAL I2C update +
      +
    • Update HAL_I2C_EV_IRQHandler() API to fix I2C send break issue +
        +
      • Add additional check on hi2c->hdmatx, hdmatx->XferCpltCallback, hi2c->hdmarx, hdmarx->XferCpltCallback in I2C_Master_SB() API to avoid enabling DMA request when IT mode is used.
      • +
    • +
    • Update HAL_I2C_ER_IRQHandler() API to fix acknowledge failure issue with I2C memory IT processes +
        +
      • Add stop condition generation when NACK occurs.
      • +
    • +
    • Update HAL_I2C_Init() API to force software reset before setting new I2C configuration
    • +
    • Update HAL I2C processes to report ErrorCode when wrong I2C start condition occurs +
        +
      • Add new ErrorCode define: HAL_I2C_WRONG_START
      • +
      • Set ErrorCode parameter in I2C handle to HAL_I2C_WRONG_START
      • +
    • +
    • Update I2C_DMAXferCplt(), I2C_DMAError() and I2C_DMAAbort() APIs to fix hardfault issue when hdmatx and hdmarx parameters in i2c handle aren’t initialized (NULL pointer). +
        +
      • Add additional check on hi2c->hdmtx and hi2c->hdmarx before resetting DMA Tx/Rx complete callbacks
      • +
    • +
  • +
  • HAL FMPI2C update +
      +
    • Fix HAL FMPI2C slave interrupt handling issue with I2C sequential transfers. +
        +
      • Update FMPI2C_Slave_ISR_IT() and FMPI2C_Slave_ISR_DMA() APIs to check on STOP condition and handle it before clearing the ADDR flag
      • +
    • +
  • +
  • HAL NAND update +
      +
    • Update HAL_NAND_Write_Page_8b(), HAL_NAND_Write_Page_16b() and HAL_NAND_Write_SpareArea_16b() to manage correctly the time out condition.
    • +
  • +
  • HAL SAI update +
      +
    • Optimize SAI_DMATxCplt() and SAI_DMARxCplt() APIs to check on “Mode†parameter instead of CIRC bit in the CR register.
    • +
    • Remove unused SAI_FIFO_SIZE define
    • +
    • Update HAL_SAI_Receive_DMA() programming sequence to be inline with reference manual
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • General updates to fix known defects and enhancements implementation
  • +
  • HAL I2C update +
      +
    • Fix I2C send break issue in IT processes +
        +
      • Add additional check on hi2c->hdmatx and hi2c->hdmarx to avoid the DMA request enable when IT mode is used.
      • +
    • +
  • +
  • HAL SPI update +
      +
    • Update to implement Erratasheet: BSY bit may stay high at the end of a data transfer in Slave mode
    • +
  • +
  • LL LPTIM update +
      +
    • Fix compilation errors with LL_LPTIM_WriteReg() and LL_LPTIM_ReadReg() macros
    • +
  • +
  • HAL SDMMC update +
      +
    • Fix preprocessing compilation issue with SDIO STA STBITERR interrupt
    • +
  • +
  • HAL/LL USB update +
      +
    • Updated USB_WritePacket(), USB_ReadPacket() APIs to prevent compilation warning with GCC GNU v8.2.0
    • +
    • Rework USB_EPStartXfer() API to enable the USB endpoint before unmasking the TX FiFo empty interrupt in case DMA is not used
    • +
    • USB HAL_HCD_Init() and HAL_PCD_Init() APIs updated to avoid enabling USB DMA feature for OTG FS instance, USB DMA feature is available only on OTG HS Instance
    • +
    • Remove duplicated line in hal_hcd.c header file comment section
    • +
    • Rework USB HAL driver to use instance PCD_SPEED_xxx, HCD_SPEED_xx speeds instead of OTG register Core speed definition during the instance initialization
    • +
    • Software Quality improvement with a fix of CodeSonar warning on PCD_Port_IRQHandler() and HCD_Port_IRQHandler() interrupt handlers
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • General updates to fix known defects and enhancements implementation
  • +
  • General updates to fix CodeSonar compilation warnings
  • +
  • General updates to fix SW4STM32 compilation errors under Linux
  • +
  • General updates to fix the user manual .chm files
  • +
  • Add support of HAL callback registration feature
  • +
  • Add new HAL EXTI driver
  • +
  • Add new HAL SMBUS driver
  • +
  • The following changes done on the HAL drivers require an update on the application code based on older HAL versions +
      +
    • Rework of HAL CRYP driver (compatibility break) +
        +
      • HAL CRYP driver has been redesigned with new API’s, to bypass limitations on data Encryption/Decryption management present with previous HAL CRYP driver version.
      • +
      • The new HAL CRYP driver is the recommended version. It is located as usual in Drivers/STM32F4xx_HAL_Driver/Src and Drivers/STM32f4xx_HAL_Driver/Inc folders. It can be enabled through switch HAL_CRYP_MODULE_ENABLED in stm32f4xx_hal_conf.h
      • +
      • The legacy HAL CRYP driver is no longer supported.
      • +
    • +
    • Add new AutoReloadPreload field in TIM_Base_InitTypeDef structure to allow the possibilities to enable or disable the TIM Auto Reload Preload.
    • +
  • +
  • HAL/LL Generic update +
      +
    • Add support of HAL callback registration feature +
        +
      • The feature disabled by default is available for the following HAL drivers: +
          +
        • ADC, CAN, CEC, CRYP, DAC, DCMI, DFSDM, DMA2D, DSI, ETH, HASH, HCD, I2C, FMPI2C, SMBUS, UART, USART, IRDA, SMARTCARD, LPTIM, LTDC, MMC, NAND, NOR, PCCARD, PCD, QSPI, RNG, RTC, SAI, SD, SDRAM, SRAM, SPDIFRX, SPI, I2S, TIM, and WWDG
        • +
      • +
      • The feature may be enabled individually per HAL PPP driver by setting the corresponding definition USE_HAL_PPP_REGISTER_CALLBACKS to 1U in stm32f4xx_hal_conf.h project configuration file (template file stm32f4xx_hal_conf_template.h available from Drivers/STM32F4xx_HAL_Driver/Inc)
      • +
      • Once enabled , the user application may resort to HAL_PPP_RegisterCallback() to register specific callback function(s) and unregister it(them) with HAL_PPP_UnRegisterCallback().
      • +
    • +
    • General updates to fix MISRA 2012 compilation errors +
        +
      • Replace HAL_GetUID() API by HAL_GetUIDw0(), HAL_GetUIDw1() and HAL_GetUIDw2()
      • +
      • HAL_IS_BIT_SET()/HAL_IS_BIT_CLR() macros implementation update
      • +
      • “stdio.h†include updated with “stddef.hâ€
      • +
    • +
  • +
  • HAL GPIO update +
      +
    • Add missing define for SPI3 alternate function “GPIO_AF5_SPI3†for STM32F401VE devices
    • +
    • Remove “GPIO_AF9_TIM14†from defined alternate function list for STM32F401xx devices
    • +
    • HAL_GPIO_TogglePin() reentrancy robustness improvement
    • +
    • HAL_GPIO_DeInit() API update to avoid potential pending interrupt after call
    • +
    • Update GPIO_GET_INDEX() API for more compliance with STM32F412Vx/STM32F412Rx/STM32F412Cx devices
    • +
    • Update GPIO_BRR registers with Reference Manual regarding registers and bit definition values
    • +
  • +
  • HAL CRYP update +
      +
    • The CRYP_InitTypeDef is no more supported, changed by CRYP_ConfigTypedef to allow changing parameters using HAL_CRYP_setConfig() API without reinitialize the CRYP IP using the HAL_CRYP_Init() API
    • +
    • New parameters added in the CRYP_ConfigTypeDef structure: B0 and DataWidthUnit
    • +
    • Input data size parameter is added in the CRYP_HandleTypeDef structure
    • +
    • Add new APIs to manage the CRYP configuration: +
        +
      • HAL_CRYP_SetConfig()
      • +
      • HAL_CRYP_GetConfig()
      • +
    • +
    • Add new APIs to manage the Key derivation: +
        +
      • HAL_CRYPEx_EnableAutoKeyDerivation()
      • +
      • HAL_CRYPEx_DisableAutoKeyDerivation()
      • +
    • +
    • Add new APIs to encrypt and decrypt data: +
        +
      • HAL_CRYP_Encypt()
      • +
      • HAL_CRYP_Decypt()
      • +
      • HAL_CRYP_Encypt_IT()
      • +
      • HAL_CRYP_Decypt_IT()
      • +
      • HAL_CRYP_Encypt_DMA()
      • +
      • HAL_CRYP_Decypt_DMA()
      • +
    • +
    • Add new APIs to generate TAG: +
        +
      • HAL_CRYPEx_AES__GCM___GenerateAuthTAG()
      • +
      • HAL_CRYPEx_AES__CCM___Generago teAuthTAG()
      • +
    • +
  • +
  • HAL LPTIM update +
      +
    • Remove useless LPTIM Wakeup EXTI related macros from HAL_LPTIM_TimeOut_Start_IT() API
    • +
  • +
  • HAL I2C update +
      +
    • I2C API changes for MISRA-C 2012 compliance: +
        +
      • Rename HAL_I2C_Master_Sequential_Transmit_IT() to HAL_I2C_Master_Seq_Transmit_IT()
      • +
      • Rename HAL_I2C_Master_Sequentiel_Receive_IT() to HAL_I2C_Master_Seq_Receive_IT()
      • +
      • Rename HAL_I2C_Slave_Sequentiel_Transmit_IT() to HAL_I2C_Slave_Seq_Transmit_IT()
      • +
      • Rename HAL_I2C_Slave_Sequentiel_Receive_DMA() to HAL_I2C_Slave_Seq_Receive_DMA()
      • +
    • +
    • SMBUS defined flags are removed as not used by the HAL I2C driver +
        +
      • I2C_FLAG_SMBALERT
      • +
      • I2C_FLAG_TIMEOUT
      • +
      • I2C_FLAG_PECERR
      • +
      • I2C_FLAG_SMBHOST
      • +
      • I2C_FLAG_SMBDEFAULT
      • +
    • +
    • Add support of I2C repeated start feature in DMA Mode: +
        +
      • With the following new API’s +
          +
        • HAL_I2C_Master_Seq_Transmit_DMA()
        • +
        • HAL_I2C_Master_Seq_Receive_DMA()
        • +
        • HAL_I2C_Slave_Seq_Transmit_DMA()
        • +
        • HAL_I2C_Slave_Seq_Receive_DMA()
        • +
      • +
    • +
    • Add new I2C transfer options to easy manage the sequential transfers +
        +
      • I2C_FIRST_AND_NEXT_FRAME
      • +
      • I2C_LAST_FRAME_NO_STOP
      • +
      • I2C_OTHER_FRAME
      • +
      • I2C_OTHER_AND_LAST_FRAME
      • +
    • +
  • +
  • HAL FMPI2C update +
      +
    • I2C API changes for MISRA-C 2012 compliance: +
        +
      • Rename HAL_FMPI2C_Master_Sequential_Transmit_IT() to HAL_FMPI2C_Master_Seq_Transmit_IT()
      • +
      • Rename HAL_FMPI2C_Master_Sequentiel_Receive_IT() to HAL_FMPI2C_Master_Seq_Receive_IT()
      • +
      • Rename HAL_FMPI2C_Master_Sequentiel_Transmit_DMA() to HAL_FMPI2C_Master_Seq_Transmit_DMA()
      • +
      • Rename HAL_FMPI2C_Master_Sequentiel_Receive_DMA() to HAL_FMPI2C_Master_Seq_Receive_DMA()
      • +
    • +
    • Rename FMPI2C_CR1_DFN to FMPI2C_CR1_DNF for more compliance with Reference Manual regarding registers and bit definition naming
    • +
    • Add support of I2C repeated start feature in DMA Mode: +
        +
      • With the following new API’s +
          +
        • HAL_FMPI2C_Master_Seq_Transmit_DMA()
        • +
        • HAL_FMPI2C_Master_Seq_Receive_DMA()
        • +
        • HAL_FMPI2C_Slave_Seq_Transmit_DMA()
        • +
        • HAL_FMPI2C_Slave_Seq_Receive_DMA()
        • +
      • +
    • +
  • +
  • HAL FLASH update +
      +
    • Update the FLASH_OB_GetRDP() API to return the correct RDP level
    • +
  • +
  • HAL RCC update +
      +
    • Remove GPIOD CLK macros for STM32F412Cx devices (X = D)
    • +
    • Remove GPIOE CLK macros for STM32F412Rx\412Cx devices: (X = E)
    • +
    • Remove GPIOF/G CLK macros for STM32F412Vx\412Rx\412Cx devices (X= F or G) +
        +
      • __HAL_RCC_GPIOX_CLK_ENABLE()
      • +
      • __HAL_RCC_GPIO__X___CLK_DISABLE()
      • +
      • __HAL_RCC_GPIO__X___IS_CLK_ENABLED()
      • +
      • __HAL_RCC_GPIO__X___IS_CLK_DISABLED()
      • +
      • __HAL_RCC_GPIO__X___FORCE_RESET()
      • +
    • +
  • +
  • HAL RNG update +
      +
    • Update to manage RNG error code: +
        +
      • Add ErrorCode parameter in HAL RNG Handler structure
      • +
    • +
  • +
  • LL ADC update +
      +
    • Add __LL_ADC_CALC_TEMPERATURE() helper macro to calculate the temperature (unit: degree Celsius) from ADC conversion data of internal temperature sensor.
    • +
    • Fix ADC channels configuration issues on STM32F413xx/423xx devices +
        +
      • To allow possibility to switch between VBAT and TEMPERATURE channels configurations
      • +
      • HAL_ADC_Start(), HAL_ADC_Start_IT() and HAL_ADC_Start_DMA() update to prevention from starting ADC2 or ADC3 once multimode is enabled
      • +
    • +
  • +
  • HAL DFSDM update +
      +
    • General updates to be compliant with DFSDM bits naming used in CMSIS files.
    • +
  • +
  • HAL CAN update +
      +
    • Update possible values list for FilterActivation parameter in CAN_FilterTypeDef structure +
        +
      • CAN_FILTER_ENABLE instead of ENABLE
      • +
      • CAN_FILTER_DISABLE instead of DISABLE
      • +
    • +
  • +
  • HAL CEC update +
      +
    • Update HAL CEC State management method: +
        +
      • Remove HAL_CEC_StateTypeDef structure parameters
      • +
      • Add new defines for CEC states
      • +
    • +
  • +
  • HAL DMA update +
      +
    • Add clean of callbacks in HAL_DMA_DeInit() API
    • +
  • +
  • HAL DMA2D update +
      +
    • Remove unused DMA2D_ColorTypeDef structure to be compliant with MISRAC 2012 Rule 2.3
    • +
    • General update to use dedicated defines for DMA2D_BACKGROUND_LAYER and DMA2D_FOREGROUND_LAYER instead of numerical values: 0/1.
    • +
  • +
  • HAL DSI update +
      +
    • Fix read multibyte issue: remove extra call to HAL_UNLOCK from DSI_ShortWrite() API.
    • +
  • +
  • HAL/LL RTC update +
      +
    • HAL/ LL drivers optimization +
        +
      • HAL driver: remove unused variables
      • +
      • LL driver: getter APIs optimization
      • +
    • +
  • +
  • HAL PWR update +
      +
    • Remove the following API’s as feature not supported by STM32F469xx/479xx devices +
        +
      • HAL_PWREx_EnableWakeUpPinPolarityRisingEdge()
      • +
      • HAL_PWREx_EnableWakeUpPinPolarityRisingEdge()
      • +
    • +
  • +
  • HAL SPI update +
      +
    • Update HAL_SPI_StateTypeDef structure to add new state: HAL_SPI_STATE_ABORT
    • +
  • +
  • HAL/LL TIM update +
      +
    • Add new AutoReloadPreload field in TIM_Base_InitTypeDef structure +
        +
      • Refer to the TIM examples to identify the changes
      • +
    • +
    • Move the following TIM structures from stm32f4xx_hal_tim_ex.h into stm32f4xx_hal_tim.h +
        +
      • TIM_MasterConfigTypeDef
      • +
      • TIM_BreakDeadTimeConfigTypeDef
      • +
    • +
    • Add new TIM Callbacks API’s: +
        +
      • HAL_TIM_PeriodElapsedHalfCpltCallback()
      • +
      • HAL_TIM_IC_CaptureHalfCpltCallback()
      • +
      • HAL_TIM_PWM_PulseFinishedHalfCpltCallback()
      • +
      • HAL_TIM_TriggerHalfCpltCallback()
      • +
    • +
    • TIM API changes for MISRA-C 2012 compliance: +
        +
      • Rename HAL_TIM_SlaveConfigSynchronization to HAL_TIM_SlaveConfigSynchro
      • +
      • Rename HAL_TIM_SlaveConfigSynchronization_IT to HAL_TIM_SlaveConfigSynchro_IT
      • +
      • Rename HAL_TIMEx_ConfigCommutationEvent to HAL_TIMEx_ConfigCommutEvent
      • +
      • Rename HAL_TIMEx_ConfigCommutationEvent_IT to HAL_TIMEx_ConfigCommutEvent_IT
      • +
      • Rename HAL_TIMEx_ConfigCommutationEvent_DMA to HAL_TIMEx_ConfigCommutEvent_DMA
      • +
      • Rename HAL_TIMEx_CommutationCallback to HAL_TIMEx_CommutCallback
      • +
      • Rename HAL_TIMEx_DMACommutationCplt to TIMEx_DMACommutationCplt
      • +
    • +
  • +
  • HAL/LL USB update +
      +
    • Rework USB interrupt handler and improve HS DMA support in Device mode
    • +
    • Fix BCD handling fr OTG instance in device mode
    • +
    • cleanup reference to low speed in device mode
    • +
    • allow writing TX FIFO in case of transfer length is equal to available space in the TX FIFO
    • +
    • Fix Toggle OUT interrupt channel in host mode
    • +
    • Update USB OTG max number of endpoints (6 FS and 9 HS instead of 5 and 8)
    • +
    • Update USB OTG IP to enable internal transceiver when starting USB device after committee BCD negotiation
    • +
  • +
  • LL IWDG update +
      +
    • Update LL inline macros to use IWDGx parameter instead of IWDG instance defined in CMSIS device
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • General updates to fix known defects and enhancements implementation
  • +
  • HAL update +
      +
    • Update UNUSED() macro implementation to avoid GCC warning +
        +
      • The warning is detected when the UNUSED() macro is called from C++ file
      • +
    • +
    • Update to make RAMFUNC define as generic type instead of HAL_StatusTypdef type.
    • +
  • +
  • HAL FLASH update +
      +
    • Update the prototypes of the following APIs after change on RAMFUNC defines +
        +
      • HAL_FLASHEx_StopFlashInterfaceClk()
      • +
      • HAL_FLASHEx_StartFlashInterfaceClk()
      • +
      • HAL_FLASHEx_EnableFlashSleepMode()
      • +
      • HAL_FLASHEx_DisableFlashSleepMode()
      • +
    • +
  • +
  • HAL SAI update +
      +
    • Update HAL_SAI_DMAStop() and HAL_SAI_Abort() process to fix the lock/unlock audio issue
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • General updates to fix known defects and enhancements implementation
  • +
  • The following changes done on the HAL drivers require an update on the application code based on older HAL versions +
      +
    • Rework of HAL CAN driver (compatibility break) +
        +
      • A new HAL CAN driver has been redesigned with new APIs, to bypass limitations on CAN Tx/Rx FIFO management present with previous HAL CAN driver version.
      • +
      • The new HAL CAN driver is the recommended version. It is located as usual in Drivers/STM32F4xx_HAL_Driver/Src and Drivers/STM32f4xx_HAL_Driver/Inc folders. It can be enabled through switch HAL_CAN_MODULE_ENABLED in stm32f4xx_hal_conf.h
      • +
      • The legacy HAL CAN driver is also present in the release in Drivers/STM32F4xx_HAL_Driver/Src/Legacy and Drivers/STM32F4xx_HAL_Driver/Inc/Legacy folders for software compatibility reasons. Its usage is not recommended as deprecated. It can however be enabled through switch HAL_CAN_LEGACY_MODULE_ENABLED in stm32f4xx_hal_conf.h
      • +
    • +
  • +
  • HAL update +
      +
    • Update HAL driver to allow user to change systick period to 1ms, 10 ms or 100 ms : +
        +
      • Add the following API’s : +
          +
        • HAL_GetTickPrio(): Returns a tick priority.
        • +
        • HAL_SetTickFreq(): Sets new tick frequency.
        • +
        • HAL_GetTickFreq(): Returns tick frequency.
        • +
      • +
      • Add HAL_TickFreqTypeDef enumeration for the different Tick Frequencies: 10 Hz, 100 Hz and 1KHz (default).
      • +
    • +
  • +
  • HAL CAN update +
      +
    • Fields of CAN_InitTypeDef structure are reworked: +
        +
      • SJW to SyncJumpWidth, BS1 to TimeSeg1, BS2 to TimeSeg2, TTCM to TimeTriggeredMode, ABOM to AutoBusOff, AWUM to AutoWakeUp, NART to AutoRetransmission (inversed), RFLM to ReceiveFifoLocked and TXFP to TransmitFifoPriority
      • +
    • +
    • HAL_CAN_Init() is split into both HAL_CAN_Init() and HAL_CAN_Start() API’s
    • +
    • HAL_CAN_Transmit() is replaced by HAL_CAN_AddTxMessage() to place Tx Request, then HAL_CAN_GetTxMailboxesFreeLevel() for polling until completion.
    • +
    • HAL_CAN_Transmit_IT() is replaced by HAL_CAN_ActivateNotification() to enable transmit IT, then HAL_CAN_AddTxMessage() for place Tx request.
    • +
    • HAL_CAN_Receive() is replaced by HAL_CAN_GetRxFifoFillLevel() for polling until reception, then HAL_CAN_GetRxMessage() to get Rx message.
    • +
    • HAL_CAN_Receive_IT() is replaced by HAL_CAN_ActivateNotification() to enable receive IT, then HAL_CAN_GetRxMessage() in the receivecallback to get Rx message
    • +
    • HAL_CAN_Slepp() is renamed as HAL_CAN_RequestSleep()
    • +
    • HAL_CAN_TxCpltCallback() is split into HAL_CAN_TxMailbox0CompleteCallback(), HAL_CAN_TxMailbox1CompleteCallback() and HAL_CAN_TxMailbox2CompleteCallback().
    • +
    • HAL_CAN_RxCpltCallback is split into HAL_CAN_RxFifo0MsgPendingCallback() and HAL_CAN_RxFifo1MsgPendingCallback().
    • +
    • More complete “How to use the new driver†is detailed in the driver header section itself.
    • +
  • +
  • HAL FMPI2C update +
      +
    • Add new option FMPI2C_LAST_FRAME_NO_STOP for the sequential transfer management +
        +
      • This option allows to manage a restart condition after several call of the same master sequential interface.
      • +
    • +
  • +
  • HAL RCC update +
      +
    • Add new HAL macros +
        +
      • __HAL_RCC_GET_RTC_SOURCE() allowing to get the RTC clock source
      • +
      • __HAL_RCC_GET_RTC_HSE_PRESCALER() allowing to get the HSE clock divider for RTC peripheral
      • +
    • +
    • Ensure reset of CIR and CSR registers when issuing HAL_RCC_DeInit()/LL_RCC_DeInit functions
    • +
    • Update HAL_RCC_OscConfig() to keep backup domain enabled when configuring respectively LSE and RTC clock source
    • +
    • Add new HAL interfaces allowing to control the activation or deactivation of PLLI2S and PLLSAI: +
        +
      • HAL_RCCEx_EnablePLLI2S()
      • +
      • HAL_RCCEx_DisablePLLI2S()
      • +
      • HAL_RCCEx_EnablePLLSAI()
      • +
      • HAL_RCCEx_DisablePLLSAI()
      • +
    • +
  • +
  • LL RCC update +
      +
    • Add new LL RCC macro +
        +
      • LL_RCC_PLL_SetMainSource() allowing to configure PLL main clock source
      • +
    • +
  • +
  • LL FMC / LL FSMC update +
      +
    • Add clear of the PTYP bit to select the PCARD mode in FMC_PCCARD_Init() / FSMC_PCCARD_Init()
      +
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • General updates to fix known defects and enhancements implementation
  • +
  • Fix compilation warning with GCC compiler
  • +
  • Remove Date and version from header files
  • +
  • Update HAL drivers to refer to the new CMSIS bit position defines instead of usage the POSITION_VAL() macro
  • +
  • HAL Generic update +
      +
    • stm32f4xx_hal_def.h file changes: +
        +
      • Update __weak and __packed defined values for ARM compiler
      • +
      • Update __ALIGN_BEGIN and __ALIGN_END defined values for ARM compiler
      • +
    • +
    • stm32f4xx_ll_system.h file: add LL_SYSCFG_REMAP_SDRAM define
    • +
  • +
  • HAL ADC update +
      +
    • Fix wrong definition of ADC channel temperature sensor for STM32F413xx and STM32F423xx devices.
    • +
  • +
  • HAL DMA update +
      +
    • Update values for the following defines: DMA_FLAG_FEIF0_4 and DMA_FLAG_DMEIF0_4
    • +
  • +
  • HAL DSI update +
      +
    • Fix Extra warning with SW4STM32 compiler
    • +
    • Fix DSI display issue when using EWARM w/ high level optimization
    • +
    • Fix MISRAC errors
    • +
  • +
  • HAL FLASH update +
      +
    • HAL_FLASH_Unlock() update to return state error when the FLASH is already unlocked
    • +
  • +
  • HAL FMPI2C update +
      +
    • Update Interface APIs headers to remove confusing message about device address
    • +
    • Update FMPI2C_WaitOnRXNEFlagUntilTimeout() to resolve a race condition between STOPF and RXNE Flags
    • +
    • Update FMPI2C_TransferConfig() to fix wrong bit management.
    • +
    • Update code comments to use DMA stream instead of DMA channel
    • +
  • +
  • HAL PWR update +
      +
    • HAL_PWR_EnableWakeUpPin() update description to add support of PWR_WAKEUP_PIN2 and PWR_WAKEUP_PIN3
    • +
  • +
  • HAL NOR update +
      +
    • Add the support of STM32F412Rx devices
    • +
  • +
  • HAL I2C update +
      +
    • Update Interface APIs headers to remove confusing message about device address
    • +
    • Update I2C_MasterReceive_RXNE() and I2C_MasterReceive_BTF() static APIs to fix bad Handling of NACK in I2C master receive process.
    • +
  • +
  • HAL RCC update +
      +
    • Update HAL_RCC_GetOscConfig() API to: +
        +
      • set PLLR in the RCC_OscInitStruct
      • +
      • check on null pointer
      • +
    • +
    • Update HAL_RCC_ClockConfig() API to: +
        +
      • check on null pointer
      • +
      • optimize code size by updating the handling method of the SWS bits
      • +
      • update to use __HAL_FLASH_GET_LATENCY() flash macro instead of using direct register access to LATENCY bits in FLASH ACR register.
      • +
    • +
    • Update HAL_RCC_DeInit() and LL_RCC_DeInit() APIs to +
        +
      • Be able to return HAL/LL status
      • +
      • Add checks for HSI, PLL and PLLI2S ready before modifying RCC CFGR registers
      • +
      • Clear all interrupt flags
      • +
      • Initialize systick interrupt period
      • +
    • +
    • Update HAL_RCC_GetSysClockFreq() to avoid risk of rounding error which may leads to a wrong returned value.
    • +
  • +
  • HAL RNG update +
      +
    • HAL_RNG_Init() remove Lock()/Unlock()
    • +
  • +
  • HAL MMC update +
      +
    • HAL_MMC_Erase() API: add missing () to fix compilation warning detected with SW4STM32 when extra feature is enabled.
    • +
  • +
  • HAL RTC update +
      +
    • HAL_RTC_Init() API: update to force the wait for synchro before setting TAFCR register when BYPSHAD bit in CR register is 0.
    • +
  • +
  • HAL SAI update +
      +
    • Update HAL_SAI_DMAStop() API to flush fifo after disabling SAI
    • +
  • +
  • HAL I2S update +
      +
    • Update I2S DMA fullduplex process to handle I2S Rx and Tx DMA Half transfer complete callback
    • +
  • +
  • HAL TIM update +
      +
    • Update HAL_TIMEx_OCN_xxxx() and HAL_TIMEx_PWMN_xxx() API description to remove support of TIM_CHANNEL_4
    • +
  • +
  • LL DMA update +
      +
    • Update to clear DMA flags using WRITE_REG() instead SET_REG() API to avoid read access to the IFCR register that is write only.
    • +
  • +
  • LL RTC update +
      +
    • Fix warning with static analyzer
    • +
  • +
  • LL USART update +
      +
    • Add assert macros to check USART BaudRate register
    • +
  • +
  • LL I2C update +
      +
    • Rename IS_I2C_CLOCK_SPEED() and IS_I2C_DUTY_CYCLE() respectively to IS_LL_I2C_CLOCK_SPEED() and IS_LL_I2C_DUTY_CYCLE() to avoid incompatible macros redefinition.
    • +
  • +
  • LL TIM update +
      +
    • Update LL_TIM_EnableUpdateEvent() API to clear UDIS bit in TIM CR1 register instead of setting it.
    • +
    • Update LL_TIM_DisableUpdateEvent() API to set UDIS bit in TIM CR1 register instead of clearing it.
    • +
  • +
  • LL USART update +
      +
    • Fix MISRA error w/ IS_LL_USART_BRR() macro
    • +
    • Fix wrong check when UART10 instance is used
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Update CHM UserManuals to support LL drivers
  • +
  • General updates to fix known defects and enhancements implementation
  • +
  • HAL CAN update +
      +
    • Add management of overrun error.
    • +
    • Allow possibility to receive messages from the 2 RX FIFOs in parallel via interrupt.
    • +
    • Fix message lost issue with specific sequence of transmit requests.
    • +
    • Handle transmission failure with error callback, when NART is enabled.
    • +
    • Add __HAL_CAN_CANCEL_TRANSMIT() call to abort transmission when timeout is reached
    • +
  • +
  • HAL PWR update +
      +
    • HAL_PWREx_EnterUnderDriveSTOPMode() API: remove check on UDRDY flag
    • +
  • +
  • LL ADC update +
      +
    • Fix wrong ADC group injected sequence configuration +
        +
      • LL_ADC_INJ_SetSequencerRanks() and LL_ADC_INJ_GetSequencerRanks() API’s update to take in consideration the ADC number of conversions
      • +
      • Update the defined values for ADC group injected seqencer ranks
      • +
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Add Low Layer drivers allowing performance and footprint optimization +
      +
    • Low Layer drivers APIs provide register level programming: require deep knowledge of peripherals described in STM32F4xx Reference Manuals
    • +
    • Low Layer drivers are available for: ADC, Cortex, CRC, DAC, DMA, DMA2D, EXTI, GPIO, I2C, IWDG, LPTIM, PWR, RCC, RNG, RTC, SPI, TIM, USART, WWDG peripherals and additional Low Level Bus, System and Utilities APIs.
    • +
    • Low Layer drivers APIs are implemented as static inline function in new Inc/stm32f4xx_ll_ppp.h files for PPP peripherals, there is no configuration file and each stm32f4xx_ll_ppp.h file must be included in user code.
    • +
  • +
  • General updates to fix known defects and enhancements implementation
  • +
  • Fix extra warnings with GCC compiler
  • +
  • HAL drivers clean up: remove double casting ‘uint32_t’ and ‘U’
  • +
  • Add new HAL MMC driver
  • +
  • The following changes done on the HAL drivers require an update on the application code based on older HAL versions
  • +
  • HAL SD update +
      +
    • Overall rework of the driver for a more efficient implementation +
        +
      • Modify initialization API and structures
      • +
      • Modify Read / Write sequences: separate transfer process and SD Cards state management
      • +
      • Adding interrupt mode for Read / Write operations
      • +
      • Update the HAL_SD_IRQHandler function by optimizing the management of interrupt errors
      • +
    • +
    • Refer to the following example to identify the changes: BSP example and USB_Device/MSC_Standalone application
    • +
  • +
  • HAL NAND update +
      +
    • Modify NAND_AddressTypeDef, NAND_DeviceConfigTypeDef and NAND_HandleTypeDef structures fields
    • +
    • Add new HAL_NAND_ConfigDevice API
    • +
  • +
  • HAL DFSDM update +
      +
    • Add support of Multichannel Delay feature +
        +
      • Add HAL_DFSDM_ConfigMultiChannelDelay API
      • +
      • The following APIs are moved to internal static functions: HAL_DFSDM_ClockIn_SourceSelection, HAL_DFSDM_ClockOut_SourceSelection, HAL_DFSDM_DataInX_SourceSelection (X=0,2,4,6), HAL_DFSDM_BitStreamClkDistribution_Config
      • +
    • +
  • +
  • HAL I2S update +
      +
    • Add specific callback API to manage I2S full duplex end of transfer process: +
        +
      • HAL_I2S_TxCpltCallback() and HAL_I2S_RxCpltCallback() API’s will be replaced with only HAL_I2SEx_TxRxCpltCallback() API.
      • +
    • +
  • +
  • HAL update +
      +
    • Modify default HAL_Delay implementation to guarantee minimum delay
    • +
  • +
  • HAL Cortex update +
      +
    • Move HAL_MPU_Disable() and HAL_MPU_Enable() from stm32f4xx_hal_cortex.h to stm32f4xx_hal_cortex.c
    • +
    • Clear the whole MPU control register in HAL_MPU_Disable() API
    • +
  • +
  • HAL FLASH update +
      +
    • IS_FLASH_ADDRESS() macro update to support OTP range
    • +
    • FLASH_Program_DoubleWord(): Replace 64-bit accesses with 2 double-words operations
    • +
  • +
  • LL GPIO update +
      +
    • Update IS_GPIO_PIN() macro implementation to be more safe
    • +
  • +
  • LL RCC update +
      +
    • Update IS_RCC_PLLQ_VALUE() macro implementation: the minimum accepted value is 2 instead of 4
    • +
    • Rename RCC_LPTIM1CLKSOURCE_PCLK define to RCC_LPTIM1CLKSOURCE_PCLK1
    • +
    • Fix compilation issue w/ __HAL_RCC_USB_OTG_FS_IS_CLK_ENABLED() and __HAL_RCC_USB_OTG_FS_IS_CLK_DISABLED() macros for STM32F401xx devices
    • +
    • Add the following is clock enabled macros for STM32F401xx devices +
        +
      • __HAL_RCC_SDIO_IS_CLK_ENABLED()
      • +
      • __HAL_RCC_SPI4_IS_CLK_ENABLED()
      • +
      • __HAL_RCC_TIM10_IS_CLK_ENABLED()
      • +
    • +
    • Add the following is clock enabled macros for STM32F410xx devices +
        +
      • __HAL_RCC_CRC_IS_CLK_ENABLED()
      • +
      • __HAL_RCC_RNG_IS_CLK_ENABLED()
      • +
    • +
    • Update HAL_RCC_DeInit() to reset the RCC clock configuration to the default reset state.
    • +
    • Remove macros to configure BKPSRAM from STM32F401xx devices
    • +
    • Update to refer to AHBPrescTable[] and APBPrescTable[] tables defined in system_stm32f4xx.c file instead of APBAHBPrescTable[] table.
    • +
  • +
  • HAL FMPI2C update +
      +
    • Add FMPI2C_FIRST_AND_NEXT_FRAME define in Sequential Transfer Options
    • +
  • +
  • HAL ADC update +
      +
    • HAL_ADCEx_InjectedConfigChannel(): update the external trigger injected condition
    • +
  • +
  • HAL DMA update +
      +
    • HAL_DMA_Init(): update to check compatibility between FIFO threshold level and size of the memory burst
    • +
  • +
  • HAL QSPI update +
      +
    • QSPI_HandleTypeDef structure: Update transfer parameters on uint32_t instead of uint16_t
    • +
  • +
  • HAL UART/USART/IrDA/SMARTCARD update +
      +
    • DMA Receive process; the code has been updated to clear the USART OVR flag before enabling DMA receive request.
    • +
    • UART_SetConfig() update to manage correctly USART6 instance that is not available on STM32F410Tx devices
    • +
  • +
  • HAL CAN update +
      +
    • Remove Lock mechanism from HAL_CAN_Transmit_IT() and HAL_CAN_Receive_IT() processes
    • +
  • +
  • HAL TIM update +
      +
    • Add __HAL_TIM_MOE_DISABLE_UNCONDITIONALLY() macro to disable Master output without check on TIM channel state.
    • +
    • Update HAL_TIMEx_ConfigBreakDeadTime() to fix TIM BDTR register corruption.
    • +
  • +
  • HAL I2C update +
      +
    • Update HAL_I2C_Master_Transmit() and HAL_I2C_Slave_Transmit() to avoid sending extra bytes at the end of the transmit processes
    • +
    • Update HAL_I2C_Mem_Read() API to fix wrong check on misused parameter “Sizeâ€
    • +
    • Update I2C_MasterReceive_RXNE() and I2C_MasterReceive_BTF() static APIs to enhance Master sequential reception process.
    • +
  • +
  • HAL SPI update +
      +
    • Add transfer abort APIs and associated callbacks in interrupt mode +
        +
      • HAL_SPI_Abort()
      • +
      • HAL_SPI_Abort_IT()
      • +
      • HAL_SPI_AbortCpltCallback()
      • +
    • +
  • +
  • HAL I2S update +
      +
    • Add specific callback API to manage I2S full duplex end of transfer process: +
        +
      • HAL_I2S_TxCpltCallback() and HAL_I2S_RxCpltCallback() API’s will be replaced with only HAL_I2SEx_TxRxCpltCallback() API.
      • +
    • +
    • Update I2S Transmit/Receive polling process to manage Overrun and Underrun errors
    • +
    • Move the I2S clock input frequency calculation to HAL RCC driver.
    • +
    • Update the HAL I2SEx driver to keep only full duplex feature.
    • +
    • HAL_I2S_Init() API updated to +
        +
      • Fix wrong I2S clock calculation when PCM mode is used.
      • +
      • Return state HAL_I2S_ERROR_PRESCALER when the I2S clock is wrongly configured
      • +
    • +
  • +
  • HAL LTDC update +
      +
    • Optimize HAL_LTDC_IRQHandler() function by using direct register read
    • +
    • Rename the following API’s +
        +
      • HAL_LTDC_Relaod() by HAL_LTDC_Reload()
      • +
      • HAL_LTDC_StructInitFromVideoConfig() by HAL_LTDCEx_StructInitFromVideoConfig()
      • +
      • HAL_LTDC_StructInitFromAdaptedCommandConfig() by HAL_LTDCEx_StructInitFromAdaptedCommandConfig()
      • +
    • +
    • Add new defines for LTDC layers (LTDC_LAYER_1 / LTDC_LAYER_2)
    • +
    • Remove unused asserts
    • +
  • +
  • HAL USB PCD update +
      +
    • Flush all TX FIFOs on USB Reset
    • +
    • Remove Lock mechanism from HAL_PCD_EP_Transmit() and HAL_PCD_EP_Receive() API’s
    • +
  • +
  • LL USB update +
      +
    • Enable DMA Burst mode for USB OTG HS
    • +
    • Fix SD card detection issue
    • +
  • +
  • LL SDMMC update +
      +
    • Add new SDMMC_CmdSDEraseStartAdd, SDMMC_CmdSDEraseEndAdd, SDMMC_CmdOpCondition and SDMMC_CmdSwitch functions
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Add support of STM32F413xx and STM32F423xx devices
  • +
  • General updates to fix known defects and enhancements implementation
  • +
  • HAL CAN update +
      +
    • Update to add the support of 3 CAN management
    • +
  • +
  • HAL CRYP update +
      +
    • Update to add the support of AES features
    • +
  • +
  • HAL DFSDM update +
      +
    • Add definitions for new external trigger filters
    • +
    • Add definition for new Channels 4, 5, 6 and 7
    • +
    • Add functions and API for Filter state configuration and management
    • +
    • Add new functions: +
        +
      • HAL_DFSDM_BitstreamClock_Start()
      • +
      • HAL_DFSDM_BitstreamClock_Stop()
      • +
      • HAL_DFSDM_BitStreamClkDistribution_Config()
      • +
    • +
  • +
  • HAL DMA +
      +
    • Add the support of DMA Channels from 8 to 15
    • +
    • Update HAL_DMA_DeInit() function with the check on DMA stream instance
    • +
  • +
  • HAL DSI update +
      +
    • Update HAL_DSI_ConfigHostTimeouts() and HAL_DSI_Init() functions to avoid scratch in DSI_CCR register
    • +
  • +
  • HAL FLASH update +
      +
    • Enhance FLASH_WaitForLastOperation() function implementation
    • +
    • Update __HAL_FLASH_GET_FLAG() macro implementation
    • +
  • +
  • HAL GPIO update +
      +
    • Add specific alternate functions definitions
    • +
  • +
  • HAL I2C update +
      +
    • Update I2C_DMAError() function implementation to ignore DMA FIFO error
    • +
  • +
  • HAL I2S update +
      +
    • Enhance HAL_I2S_Init() implementation to test on PCM_SHORT and PCM_LONG standards
    • +
  • +
  • HAL IRDA update +
      +
    • Add new functions and call backs for Transfer Abort +
        +
      • HAL_IRDA_Abort()
      • +
      • HAL_IRDA_AbortTransmit()
      • +
      • HAL_IRDA_AbortReceive()
      • +
      • HAL_IRDA_Abort_IT()
      • +
      • HAL_IRDA_AbortTransmit_IT()
      • +
      • HAL_IRDA_AbortReceive_IT()
      • +
      • HAL_IRDA_AbortCpltCallback()
      • +
      • HAL_IRDA_AbortTransmitCpltCallback()
      • +
      • HAL_IRDA_AbortReceiveCpltCallback()
      • +
    • +
  • +
  • HAL PCD update +
      +
    • Update HAL_PCD_GetRxCount() function implementation
    • +
  • +
  • HAL RCC update +
      +
    • Update __HAL_RCC_HSE_CONFIG() macro implementation
    • +
    • Update __HAL_RCC_LSE_CONFIG() macro implementation
    • +
  • +
  • HAL SMARTCARD update +
      +
    • Add new functions and call backs for Transfer Abort +
        +
      • HAL_ SMARTCARD_Abort()
      • +
      • HAL_ SMARTCARD_AbortTransmit()
      • +
      • HAL_ SMARTCARD_AbortReceive()
      • +
      • HAL_ SMARTCARD_Abort_IT()
      • +
      • HAL_ SMARTCARD_AbortTransmit_IT()
      • +
      • HAL_ SMARTCARD_AbortReceive_IT()
      • +
      • HAL_ SMARTCARD_AbortCpltCallback()
      • +
      • HAL_ SMARTCARD_AbortTransmitCpltCallback()
      • +
      • HAL_ SMARTCARD_AbortReceiveCpltCallback()
      • +
    • +
  • +
  • HAL TIM update +
      +
    • Update HAL_TIMEx_RemapConfig() function to manage TIM internal trigger remap: LPTIM or TIM3_TRGO
    • +
  • +
  • HAL UART update +
      +
    • Add Transfer abort functions and callbacks
    • +
  • +
  • HAL USART update +
      +
    • Add Transfer abort functions and callbacks
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • HAL I2C update +
      +
    • Fix wrong behavior in consecutive transfers in case of single byte transmission (Master/Memory Receive interfaces)
    • +
    • Update HAL_I2C_Master_Transmit_DMA() / HAL_I2C_Master_Receive_DMA()/ HAL_I2C_Slave_Transmit_DMA() and HAL_I2C_Slave_Receive_DMA() to manage addressing phase through interruption instead of polling
    • +
    • Add a check on I2C handle state at start of all I2C API’s to ensure that I2C is ready
    • +
    • Update I2C API’s (Polling, IT and DMA interfaces) to manage I2C XferSize and XferCount handle parameters instead of API size parameter to help user to get information of counter in case of error.
    • +
    • Update Abort functionality to manage DMA use case
    • +
  • +
  • HAL FMPI2C update +
      +
    • Update to disable Own Address before setting the new Own Address configuration: +
        +
      • Update HAL_FMPI2C_Init() to disable FMPI2C_OARx_EN bit before any configuration in OARx registers
      • +
    • +
  • +
  • HAL CAN update +
      +
    • Update CAN receive processes to set CAN RxMsg FIFONumber parameter
    • +
  • +
  • HAL UART update +
      +
    • Update UART handle TxXferCount and RxXferCount parameters as volatile to avoid eventual issue with High Speed optimization
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • HAL GPIO update +
      +
    • HAL_GPIO_Init()/HAL_GPIO_DeInit() API’s: update GPIO_GET_INDEX() macro implementation to support all GPIO’s
    • +
  • +
  • HAL SPI update +
      +
    • Fix regression issue: restore HAL_SPI_DMAPause() and HAL_SPI_DMAResume() API’s
    • +
  • +
  • HAL RCC update +
      +
    • Fix FSMC macros compilation warnings with STM32F412Rx devices
    • +
  • +
  • HAL DMA update +
      +
    • HAL_DMA_PollFortransfer() API clean up
    • +
  • +
  • HAL PPP update(PPP refers to IRDA, UART, USART and SMARTCARD) +
      +
    • Update HAL_PPP_IRQHandler() to add a check on interrupt source before managing the error
    • +
  • +
  • HAL QSPI update +
      +
    • Implement workaround to fix the limitation pronounced in the Errata sheet 2.1.8 section: In some specific cases, DMA2 data corruption occurs when managing AHB and APB2 peripherals in a concurrent way
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Add support of STM32F412cx, STM32F412rx, STM32F412vx and STM32F412zx devices
  • +
  • General updates to fix known defects and enhancements implementation
  • +
  • Add new HAL driver for DFSDM peripheral
  • +
  • Enhance HAL delay and time base implementation: +
      +
    • Add new drivers stm32f4xx_hal_timebase_rtc_alarm_template.c and stm32f4xx_hal_timebase_rtc_wakeup_template.c which override the native HAL time base functions (defined as weak) to either use the RTC as time base tick source. For more details about the usage of these drivers, please refer to HAL_TimeBase_RTC examples and FreeRTOS-based applications
    • +
  • +
  • The following changes done on the HAL drivers require an update on the application code based on HAL V1.4.4 +
      +
    • HAL UART, USART, IRDA, SMARTCARD, SPI, I2C,FMPI2C, QSPI (referenced as PPP here below) drivers +
        +
      • Add PPP error management during DMA process. This requires the following updates on user application: +
          +
        • Configure and enable the PPP IRQ in HAL_PPP_MspInit() function
        • +
        • In stm32f4xx_it.c file, PPP_IRQHandler() function: add a call to HAL_PPP_IRQHandler() function
        • +
        • Add and customize the Error Callback API: HAL_PPP_ErrorCallback()
        • +
      • +
    • +
    • HAL I2C, FMPI2C (referenced as PPP here below) drivers: +
        +
      • Update to avoid waiting on STOPF/BTF/AF flag under DMA ISR by using the PPP end of transfer interrupt in the DMA transfer process. This requires the following updates on user application: +
          +
        • Configure and enable the PPP IRQ in HAL_PPP_MspInit() function
        • +
        • In stm32f4xx_it.c file, PPP_IRQHandler() function: add a call to HAL_PPP_IRQHandler() function
        • +
      • +
    • +
    • HAL I2C driver: +
        +
      • I2C transfer processes IT update: NACK during addressing phase is managed through I2C Error interrupt instead of HAL state
      • +
    • +
    • HAL IWDG driver: rework overall driver for better implementation +
        +
      • Remove HAL_IWDG_Start(), HAL_IWDG_MspInit() and HAL_IWDG_GetState() APIs
      • +
    • +
    • HAL WWDG driver: rework overall driver for better implementation +
        +
      • Remove HAL_WWDG_Start(), HAL_WWDG_Start_IT(), HAL_WWDG_MspDeInit() and HAL_WWDG_GetState() APIs
      • +
      • Update the HAL_WWDG_Refresh(WWDG_HandleTypeDef *hwwdg, uint32_t counter) function and API by removing the “counter†parameter
      • +
    • +
    • HAL QSPI driver: Enhance the DMA transmit process by using PPP TC interrupt instead of waiting on TC flag under DMA ISR. This requires the following updates on user application: +
        +
      • Configure and enable the QSPI IRQ in HAL_QSPI_MspInit() function
      • +
      • In stm32f4xx_it.c file, QSPI_IRQHandler() function: add a call to HAL_QSPI_IRQHandler() function
      • +
    • +
    • HAL CEC driver: Overall driver rework with compatibility break versus previous HAL version +
        +
      • Remove HAL CEC polling Process functions: HAL_CEC_Transmit() and HAL_CEC_Receive()
      • +
      • Remove HAL CEC receive interrupt process function HAL_CEC_Receive_IT() and enable the “receive†mode during the Init phase
      • +
      • Rename HAL_CEC_GetReceivedFrameSize() function to HAL_CEC_GetLastReceivedFrameSize()
      • +
      • Add new HAL APIs: HAL_CEC_SetDeviceAddress() and HAL_CEC_ChangeRxBuffer()
      • +
      • Remove the ‘InitiatorAddress’ field from the CEC_InitTypeDef structure and manage it as a parameter in the HAL_CEC_Transmit_IT() function
      • +
      • Add new parameter ‘RxFrameSize’ in HAL_CEC_RxCpltCallback() function
      • +
      • Move CEC Rx buffer pointer from CEC_HandleTypeDef structure to CEC_InitTypeDef structure
      • +
    • +
  • +
  • HAL RCC update +
      +
    • Update HAL_RCC_ClockConfig() function to adjust the SystemCoreClock
    • +
    • Rename macros and Literals: +
        +
      • RCC_PERIPHCLK_CK48 by RCC_PERIPHCLK_CLK48
      • +
      • IS_RCC_CK48CLKSOURCE by IS_RCC_CLK48CLKSOURCE
      • +
      • RCC_CK48CLKSOURCE_PLLSAIP by RCC_CLK48CLKSOURCE_PLLSAIP
      • +
      • RCC_SDIOCLKSOURCE_CK48 by RCC_SDIOCLKSOURCE_CLK48
      • +
      • RCC_CK48CLKSOURCE_PLLQ by RCC_CLK48CLKSOURCE_PLLQ
      • +
    • +
    • Update HAL_RCCEx_GetPeriphCLKConfig() and HAL_RCCEx_PeriphCLKConfig() functions to support TIM Prescaler for STM32F411xx devices
    • +
    • HAL_RCCEx_PeriphCLKConfig() API: update to fix the RTC clock configuration issue
    • +
  • +
  • HAL CEC update +
      +
    • Overall driver rework with break of compatibility with HAL V1.4.4 +
        +
      • Remove the HAL CEC polling Process: HAL_CEC_Transmit() and HAL_CEC_Receive()
      • +
      • Remove the HAL CEC receive interrupt process (HAL_CEC_Receive_IT()) and manage the “Receive†mode enable within the Init phase
      • +
      • Rename HAL_CEC_GetReceivedFrameSize() function to HAL_CEC_GetLastReceivedFrameSize() function
      • +
      • Add new HAL APIs: HAL_CEC_SetDeviceAddress() and HAL_CEC_ChangeRxBuffer()
      • +
      • Remove the ‘InitiatorAddress’ field from the CEC_InitTypeDef structure and manage it as a parameter in the HAL_CEC_Transmit_IT() function
      • +
      • Add new parameter ‘RxFrameSize’ in HAL_CEC_RxCpltCallback() function
      • +
      • Move CEC Rx buffer pointer from CEC_HandleTypeDef structure to CEC_InitTypeDef structure
      • +
    • +
    • Update driver to implement the new CEC state machine: +
        +
      • Add new “rxState†field in CEC_HandleTypeDef structure to provide the CEC state information related to Rx Operations
      • +
      • Rename “state†field in CEC_HandleTypeDef structure to “gstateâ€: CEC state information related to global Handle management and Tx Operations
      • +
      • Update CEC process to manage the new CEC states.
      • +
      • Update __HAL_CEC_RESET_HANDLE_STATE() macro to handle the new CEC state parameters (gState, rxState)
      • +
    • +
  • +
  • HAL UART, USART, SMARTCARD and IRDA (referenced as PPP here below) update +
      +
    • Update Polling management: +
        +
      • The user Timeout value must be estimated for the overall process duration: the Timeout measurement is cumulative
      • +
    • +
    • Update DMA process: +
        +
      • Update the management of PPP peripheral errors during DMA process. This requires the following updates in user application: +
          +
        • Configure and enable the PPP IRQ in HAL_PPP_MspInit() function
        • +
        • In stm32f4xx_it.c file, PPP_IRQHandler() function: add a call to HAL_PPP_IRQHandler() function
        • +
        • Add and customize the Error Callback API: HAL_PPP_ErrorCallback()
        • +
      • +
    • +
  • +
  • HAL FMC update +
      +
    • Update FMC_NORSRAM_Init() to remove the Burst access mode configuration
    • +
    • Update FMC_SDRAM_Timing_Init() to fix initialization issue when configuring 2 SDRAM banks
    • +
  • +
  • HAL HCD update +
      +
    • Update HCD_Port_IRQHandler() to unmask disconnect IT only when the port is disabled
    • +
  • +
  • HAL I2C/FMPI2C update +
      +
    • Update Polling management: +
        +
      • The Timeout value must be estimated for the overall process duration: the Timeout measurement is cumulative
      • +
    • +
    • Add the management of Abort service: Abort DMA transfer through interrupt +
        +
      • In the case of Master Abort IT transfer usage: +
          +
        • Add new user HAL_I2C_AbortCpltCallback() to inform user of the end of abort process
        • +
        • A new abort state is defined in the HAL_I2C_StateTypeDef structure
        • +
      • +
    • +
    • Add the management of I2C peripheral errors, ACK failure and STOP condition detection during DMA process. This requires the following updates on user application: +
        +
      • Configure and enable the I2C IRQ in HAL_I2C_MspInit() function
      • +
      • In stm32f4xx_it.c file, I2C_IRQHandler() function: add a call to HAL_I2C_IRQHandler() function
      • +
      • Add and customize the Error Callback API: HAL_I2C_ErrorCallback()
      • +
      • Refer to the I2C_EEPROM or I2C_TwoBoards_ComDMA project examples usage of the API
      • +
    • +
    • NACK error during addressing phase is returned through interrupt instead of previously through I2C transfer API’s
    • +
    • I2C addressing phase is updated to be managed using interrupt instead of polling (Only for HAL I2C driver) +
        +
      • Add new static functions to manage I2C SB, ADDR and ADD10 flags
      • +
    • +
  • +
  • HAL SPI update +
      +
    • Overall driver optimization to improve performance in polling/interrupt mode to reach maximum peripheral frequency +
        +
      • Polling mode: +
          +
        • Replace the use of SPI_WaitOnFlagUnitTimeout() function by “if†statement to check on RXNE/TXE flags while transferring data
        • +
      • +
      • Interrupt mode: +
          +
        • Minimize access on SPI registers
        • +
      • +
      • All modes: +
          +
        • Add the USE_SPI_CRC switch to minimize the number of statements when CRC calculation is disabled
        • +
        • Update timeout management to check on global processes
        • +
        • Update error code management in all processes
        • +
      • +
      • Update DMA process: +
          +
        • Add the management of SPI peripheral errors during DMA process. This requires the following updates in the user application:
        • +
        • Configure and enable the SPI IRQ in HAL_SPI_MspInit() function
        • +
        • In stm32f4xx_it.c file, SPI_IRQHandler() function: add a call to HAL_SPI_IRQHandler() function
        • +
        • Add and customize the Error Callback API: HAL_SPI_ErrorCallback()
        • +
        • Refer to the following example which describe the changes: SPI_FullDuplex_ComDMA
        • +
      • +
      • Fix regression in polling mode: +
          +
        • Add preparing data to transmit in case of slave mode in HAL_SPI_TransmitReceive() and HAL_SPI_Transmit()
        • +
        • Add to manage properly the overrun flag at the end of a HAL_SPI_TransmitReceive()
        • +
      • +
      • Fix regression in interrupt mode: +
          +
        • Add a wait on TXE flag in SPI_CloseTx_ISR() and in SPI_CloseTxRx_ISR()
        • +
        • Add to manage properly the overrun flag in SPI_CloseRxTx_ISR() and SPI_CloseRx_ISR()
        • +
      • +
    • +
  • +
  • HAL DMA2D update +
      +
    • Update the HAL_DMA2D_DeInit() function to: +
        +
      • Abort transfer in case of ongoing DMA2D transfer
      • +
      • Reset DMA2D control registers
      • +
    • +
    • Update HAL_DMA2D_Abort() to disable DMA2D interrupts after stopping transfer
    • +
    • Optimize HAL_DMA2D_IRQHandler() by reading status registers only once
    • +
    • Update HAL_DMA2D_ProgramLineEvent() function to: +
        +
      • Return HAL error state in case of wrong line value
      • +
      • Enable line interrupt after setting the line watermark configuration
      • +
    • +
    • Add new HAL_DMA2D_CLUTLoad() and HAL_DMA2D_CLUTLoad_IT() functions to start DMA2D CLUT loading +
        +
      • HAL_DMA2D_CLUTLoading_Abort() function to abort the DMA2D CLUT loading
      • +
      • HAL_DMA2D_CLUTLoading_Suspend() function to suspend the DMA2D CLUT loading
      • +
      • HAL_DMA2D_CLUTLoading_Resume() function to resume the DMA2D CLUT loading
      • +
    • +
    • Add new DMA2D dead time management: +
        +
      • HAL_DMA2D_EnableDeadTime() function to enable DMA2D dead time feature
      • +
      • HAL_DMA2D_DisableDeadTime() function to disable DMA2D dead time feature
      • +
      • HAL_DMA2D_ConfigDeadTime() function to configure dead time
      • +
    • +
    • Update the name of DMA2D Input/Output color mode defines to be more clear for user (DMA2D_INPUT_XXX for input layers Colors, DMA2D_OUTPUT_XXX for output framebuffer Colors)
    • +
  • +
  • HAL LTDC update +
      +
    • Update HAL_LTDC_IRQHandler() to manage the case of reload interrupt
    • +
    • Add new callback API HAL_LTDC_ReloadEventCallback()
    • +
    • Add HAL_LTDC_Reload() to configure LTDC reload feature
    • +
    • Add new No Reload LTDC variant APIs +
        +
      • HAL_LTDC_ConfigLayer_NoReload() to configure the LTDC Layer according to the specified without reloading
      • +
      • HAL_LTDC_SetWindowSize_NoReload() to set the LTDC window size without reloading
      • +
      • HAL_LTDC_SetWindowPosition_NoReload() to set the LTDC window position without reloading
      • +
      • HAL_LTDC_SetPixelFormat_NoReload() to reconfigure the pixel format without reloading
      • +
      • HAL_LTDC_SetAlpha_NoReload() to reconfigure the layer alpha value without reloading
      • +
      • HAL_LTDC_SetAddress_NoReload() to reconfigure the frame buffer Address without reloading
      • +
      • HAL_LTDC_SetPitch_NoReload() to reconfigure the pitch for specific cases
      • +
      • HAL_LTDC_ConfigColorKeying_NoReload() to configure the color keying without reloading
      • +
      • HAL_LTDC_EnableColorKeying_NoReload() to enable the color keying without reloading
      • +
      • HAL_LTDC_DisableColorKeying_NoReload() to disable the color keying without reloading
      • +
      • HAL_LTDC_EnableCLUT_NoReload() to enable the color lookup table without reloading
      • +
      • HAL_LTDC_DisableCLUT_NoReload() to disable the color lookup table without reloading
      • +
      • Note: Variant functions with “_NoReload†post fix allows to set the LTDC configuration/settings without immediate reload. This is useful in case when the program requires to modify several LTDC settings (on one or both layers) then applying (reload) these settings in one shot by calling the function “HAL_LTDC_Reloadâ€
      • +
    • +
  • +
  • HAL RTC update +
      +
    • Add new timeout implementation based on cpu cycles for ALRAWF, ALRBWF and WUTWF flags
    • +
  • +
  • HAL SAI update +
      +
    • Update SAI state in case of TIMEOUT error within the HAL_SAI_Transmit() / HAL_SAI_Receive()
    • +
    • Update HAL_SAI_IRQHandler: +
        +
      • Add error management in case DMA errors through XferAbortCallback() and HAL_DMA_Abort_IT()
      • +
      • Add error management in case of IT
      • +
    • +
    • Move SAI_BlockSynchroConfig() and SAI_GetInputClock() functions to stm32f4xx_hal_sai.c/.h files (extension files are kept empty for projects compatibility reason)
    • +
  • +
  • HAL DCMI update +
      +
    • Rename DCMI_DMAConvCplt to DCMI_DMAXferCplt
    • +
    • Update HAL_DCMI_Start_DMA() function to Enable the DCMI peripheral
    • +
    • Add new timeout implementation based on cpu cycles for DCMI stop
    • +
    • Add HAL_DCMI_Suspend() function to suspend DCMI capture
    • +
    • Add HAL_DCMI_Resume() function to resume capture after DCMI suspend
    • +
    • Update lock mechanism for DCMI process
    • +
    • Update HAL_DCMI_IRQHandler() function to: +
        +
      • Add error management in case DMA errors through XferAbortCallback() and HAL_DMA_Abort_IT()
      • +
      • Optimize code by using direct register read
      • +
    • +
  • +
  • HAL DMA update +
      +
    • Add new APIs HAL_DMA_RegisterCallback() and HAL_DMA_UnRegisterCallback to register/unregister the different callbacks identified by the enum typedef HAL_DMA_CallbackIDTypeDef
    • +
    • Add new API HAL_DMA_Abort_IT() to abort DMA transfer under interrupt context +
        +
      • The new registered Abort callback is called when DMA transfer abortion is completed
      • +
    • +
    • Add the check of compatibility between FIFO threshold level and size of the memory burst in the HAL_DMA_Init() API
    • +
    • Add new Error Codes: HAL_DMA_ERROR_PARAM, HAL_DMA_ERROR_NO_XFER and HAL_DMA_ERROR_NOT_SUPPORTED
    • +
    • Remove all DMA states related to MEM0/MEM1 in HAL_DMA_StateTypeDef
    • +
  • +
  • HAL IWDG update +
      +
    • Overall rework of the driver for a more efficient implementation +
        +
      • Remove the following APIs: +
          +
        • HAL_IWDG_Start()
        • +
        • HAL_IWDG_MspInit()
        • +
        • HAL_IWDG_GetState()
        • +
      • +
    • +
    • Update implementation: +
        +
      • HAL_IWDG_Init(): this function insures the configuration and the start of the IWDG counter
      • +
      • HAL_IWDG_Refresh(): this function insures the reload of the IWDG counter
      • +
    • +
    • Refer to the following example to identify the changes: IWDG_Example
    • +
  • +
  • HAL LPTIM update +
      +
    • Update HAL_LPTIM_TimeOut_Start_IT() and HAL_LPTIM_Counter_Start_IT( ) APIs to configure WakeUp Timer EXTI interrupt to be able to wakeup MCU from low power mode by pressing the EXTI line.
    • +
    • Update HAL_LPTIM_TimeOut_Stop_IT() and HAL_LPTIM_Counter_Stop_IT( ) APIs to disable WakeUp Timer EXTI interrupt.
    • +
  • +
  • HAL NOR update +
      +
    • Update NOR_ADDR_SHIFT macro implementation
    • +
  • +
  • HAL PCD update +
      +
    • Update HAL_PCD_IRQHandler() to get HCLK frequency before setting TRDT value
    • +
  • +
  • HAL QSPI update +
      +
    • Update to manage QSPI error management during DMA process
    • +
    • Improve the DMA transmit process by using QSPI TC interrupt instead of waiting loop on TC flag under DMA ISR
    • +
    • These two improvements require the following updates on user application: +
        +
      • Configure and enable the QSPI IRQ in HAL_QSPI_MspInit() function
      • +
      • In stm32f4xx_it.c file, QSPI_IRQHandler() function: add a call to HAL_QSPI_IRQHandler() function
      • +
      • Add and customize the Error Callback API: HAL_QSPI_ErrorCallback()
      • +
    • +
    • Add the management of non-blocking transfer abort service: HAL_QSPI_Abort_IT(). In this case the user must: +
        +
      • Add new callback HAL_QSPI_AbortCpltCallback() to inform user at the end of abort process
      • +
      • A new value of State in the HAL_QSPI_StateTypeDef provides the current state during the abort phase
      • +
    • +
    • Polling management update: +
        +
      • The Timeout value user must be estimated for the overall process duration: the Timeout measurement is cumulative.
      • +
    • +
    • Refer to the following examples, which describe the changes: +
        +
      • QSPI_ReadWrite_DMA
      • +
      • QSPI_MemoryMapped
      • +
      • QSPI_ExecuteInPlace
      • +
    • +
    • Add two new APIs for the QSPI fifo threshold: +
        +
      • HAL_QSPI_SetFifoThreshold(): configure the FIFO threshold of the QSPI
      • +
      • HAL_QSPI_GetFifoThreshold(): give the current FIFO threshold
      • +
    • +
    • Fix wrong data size management in HAL_QSPI_Receive_DMA()
    • +
  • +
  • HAL ADC update +
      +
    • Add new __HAL_ADC_PATH_INTERNAL_VBAT_DISABLE() macro for STM32F42x and STM32F43x devices to provide the possibility to convert VrefInt channel when both VrefInt and Vbat channels are selected.
    • +
  • +
  • HAL SPDIFRX update +
      +
    • Overall driver update for wait on flag management optimization
    • +
  • +
  • HAL WWDG update +
      +
    • Overall rework of the driver for more efficient implementation +
        +
      • Remove the following APIs: +
          +
        • HAL_WWDG_Start()
        • +
        • HAL_WWDG_Start_IT()
        • +
        • HAL_WWDG_MspDeInit()
        • +
        • HAL_WWDG_GetState()
        • +
      • +
      • Update implementation: +
          +
        • HAL_WWDG_Init() +
            +
          • A new parameter in the Init Structure: EWIMode
          • +
        • +
        • HAL_WWDG_MspInit()
        • +
        • HAL_WWDG_Refresh() +
            +
          • This function insures the reload of the counter
          • +
          • The “counter†parameter has been removed
          • +
        • +
        • HAL_WWDG_IRQHandler()
        • +
        • HAL_WWDG_EarlyWakeupCallback() is the new prototype of HAL_WWDG_WakeUpCallback()
        • +
      • +
    • +
    • Refer to the following example to identify the changes: WWDG_Example
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • HAL Generic update +
      +
    • stm32f4xx_hal_conf_template.h +
        +
      • Optimize HSE Startup Timeout value from 5000ms to 100 ms
      • +
      • Add new define LSE_STARTUP_TIMEOUT
      • +
      • Add new define USE_SPI_CRC for code cleanup when the CRC calculation is disabled.
      • +
    • +
    • Update HAL drivers to support MISRA C 2004 rule 10.6
    • +
    • Add new template driver to configure timebase using TIMER : +
        +
      • stm32f4xx_hal_timebase_tim_template.c
      • +
    • +
  • +
  • HAL CAN update +
      +
    • Update HAL_CAN_Transmit() and HAL_CAN_Transmit_IT() functions to unlock process when all Mailboxes are busy
    • +
  • +
  • HAL DSI update +
      +
    • Update HAL_DSI_SetPHYTimings() functions to use the correct mask
    • +
  • +
  • HAL UART update +
      +
    • Several update on HAL UART driver to implement the new UART state machine: +
        +
      • Add new field in UART_HandleTypeDef structure: “rxStateâ€, UART state information related to Rx Operations
      • +
      • Rename “state†field in UART_HandleTypeDef structure by “gstateâ€: UART state information related to global Handle management and Tx Operations
      • +
      • Update UART process to manage the new UART states.
      • +
      • Update __HAL_UART_RESET_HANDLE_STATE() macro to handle the new UART state parameters (gState, rxState)
      • +
    • +
    • Update UART_BRR_SAMPLING16() and UART_BRR_SAMPLING8() Macros to fix wrong baudrate calculation.
    • +
  • +
  • HAL IRDA update +
      +
    • Several update on HAL IRDA driver to implement the new UART state machine: +
        +
      • Add new field in IRDA_HandleTypeDef structure: “rxStateâ€, IRDA state information related to Rx Operations
      • +
      • Rename “state†field in UART_HandleTypeDef structure by “gstateâ€: IRDA state information related to global Handle management and Tx Operations
      • +
      • Update IRDA process to manage the new UART states.
      • +
      • Update __HAL_IRDA_RESET_HANDLE_STATE() macro to handle the new IRDA state parameters (gState, rxState)
      • +
    • +
    • Removal of IRDA_TIMEOUT_VALUE define
    • +
    • Update IRDA_BRR() Macro to fix wrong baudrate calculation
    • +
  • +
  • HAL SMARTCARD update +
      +
    • Several update on HAL SMARTCARD driver to implement the new UART state machine: +
        +
      • Add new field in SMARTCARD_HandleTypeDef structure: “rxStateâ€, SMARTCARDstate information related to Rx Operations
      • +
      • Rename “state†field in UART_HandleTypeDef structure by “gstateâ€: SMARTCARDstate information related to global Handle management and Tx Operations
      • +
      • Update SMARTCARD process to manage the new UART states.
      • +
      • Update __HAL_SMARTCARD_RESET_HANDLE_STATE() macro to handle the new SMARTCARD state parameters (gState, rxState)
      • +
    • +
    • Update SMARTCARD_BRR() macro to fix wrong baudrate calculation
    • +
  • +
  • HAL RCC update +
      +
    • Add new default define value for HSI calibration “RCC_HSICALIBRATION_DEFAULTâ€
    • +
    • Optimize Internal oscillators and PLL startup timeout
    • +
    • Update to avoid the disable for HSE/LSE oscillators before setting the new RCC HSE/LSE configuration and add the following notes in HAL_RCC_OscConfig() API description:

      +
                         * @note   Transitions LSE Bypass to LSE On and LSE On to LSE Bypass are not
      +                   *             supported by this API. User should request a transition to LSE Off
      +                   *             first and then LSE On or LSE Bypass.
      +                   * @note   Transition HSE Bypass to HSE On and HSE On to HSE Bypass are not
      +                   *             supported by this API. User should request a transition to HSE Off
      +                   *             first and then HSE On or HSE Bypass.
    • +
    • Optimize the HAL_RCC_ClockConfig() API implementation.
    • +
  • +
  • HAL DMA2D update +
      +
    • Update HAL_DMA2D_Abort() Function to end current DMA2D transfer properly
    • +
    • Update HAL_DMA2D_PollForTransfer() function to add poll for background CLUT loading (layer 0 and layer 1).
    • +
    • Update HAL_DMA2D_PollForTransfer() to set the corresponding ErrorCode in case of error occurrence
    • +
    • Update HAL_DMA2D_ConfigCLUT() function to fix wrong CLUT size and color mode settings
    • +
    • Removal of useless macro __HAL_DMA2D_DISABLE()
    • +
    • Update HAL_DMA2D_Suspend() to manage correctly the case where no transfer is on going
    • +
    • Update HAL_DMA2D_Resume() to manage correctly the case where no transfer is on going
    • +
    • Update HAL_DMA2D_Start_IT() to enable all required interrupts before enabling the transfer.
    • +
    • Add HAL_DMA2D_CLUTLoad_IT() Function to allow loading a CLUT with interruption model.
    • +
    • Update HAL_DMA2D_IRQHandler() to manage the following cases :

      +
        +
      • CLUT transfer complete
      • +
      • CLUT access error
      • +
      • Transfer watermark reached
      • +
    • +
    • Add new Callback APIs: +
        +
      • HAL_DMA2D_LineEventCallback() to signal a transfer watermark reached event
      • +
      • HAL_DMA2D_CLUTLoadingCpltCallback() to signal a CLUT loading complete event
      • +
    • +
    • Miscellaneous Improvement: +
        +
      • Add “HAL_DMA2D_ERROR_CAE†new define for CLUT Access error management.
      • +
      • Add “assert_param†used for parameters check is now done on the top of the exported functions : before locking the process using __HAL_LOCK
      • +
    • +
  • +
  • HAL I2C update +
      +
    • Add support of I2C repeated start feature: +
        +
      • With the following new API’s +
          +
        • HAL_I2C_Master_Sequential_Transmit_IT()
        • +
        • HAL_I2C_Master_Sequential_Receive_IT()
        • +
        • HAL_I2C_Master_Abort_IT()
        • +
        • HAL_I2C_Slave_Sequential_Transmit_IT()
        • +
        • HAL_I2C_Slave_Sequential_Receive_IT()
        • +
        • HAL_I2C_EnableListen_IT()
        • +
        • HAL_I2C_DisableListen_IT()
        • +
      • +
    • +
    • Add new user callbacks: +
        +
      • HAL_I2C_ListenCpltCallback()
      • +
      • HAL_I2C_AddrCallback()
      • +
    • +
    • Update to generate STOP condition when a acknowledge failure error is detected
    • +
    • Several update on HAL I2C driver to implement the new I2C state machine: +
        +
      • Add new API to get the I2C mode: HAL_I2C_GetMode()
      • +
      • Update I2C process to manage the new I2C states.
      • +
    • +
    • Fix wrong behaviour in single byte transmission
    • +
    • Update I2C_WaitOnFlagUntilTimeout() to manage the NACK feature.
    • +
    • Update I2C transmission process to support the case data size equal 0
    • +
  • +
  • HAL FMPI2C update +
      +
    • Add support of FMPI2C repeated start feature: +
        +
      • With the following new API’s +
          +
        • HAL_FMPI2C_Master_Sequential_Transmit_IT()
        • +
        • HAL_FMPI2C_Master_Sequential_Receive_IT()
        • +
        • HAL_FMPI2C_Master_Abort_IT()
        • +
        • HAL_FMPI2C_Slave_Sequential_Transmit_IT()
        • +
        • HAL_FMPI2C_Slave_Sequential_Receive_IT()
        • +
        • HAL_FMPI2C_EnableListen_IT()
        • +
        • HAL_FMPI2C_DisableListen_IT()
        • +
      • +
      • Add new user callbacks: +
          +
        • HAL_FMPI2C_ListenCpltCallback()
        • +
        • HAL_FMPI2C_AddrCallback()
        • +
      • +
      • Several update on HAL I2C driver to implement the new I2C state machine: +
          +
        • Add new API to get the FMPI2C mode: HAL_FMPI2C_GetMode()
        • +
        • Update FMPI2C process to manage the new FMPI2C states.
        • +
      • +
    • +
  • +
  • HAL SPI update +
      +
    • Major Update to improve performance in polling/interrupt mode to reach max frequency: +
        +
      • Polling mode : +
          +
        • Replace use of SPI_WaitOnFlagUnitTimeout() funnction by “if†statement to check on RXNE/TXE flags while transferring data.
        • +
        • Use API data pointer instead of SPI handle data pointer.
        • +
        • Use a Goto implementation instead of “if..else†statements.
        • +
      • +
      • Interrupt mode +
          +
        • Minimize access on SPI registers.
        • +
        • Split the SPI modes into dedicated static functions to minimize checking statements under HAL_IRQHandler(): +
            +
          • 1lines/2lines modes
          • +
          • 8 bit/ 16 bits data formats
          • +
          • CRC calculation enabled/disabled.
          • +
        • +
        • Remove waiting loop under ISR when closing the communication.
        • +
      • +
      • All modes: +
          +
        • Adding switch USE_SPI_CRC to minimize number of statements when CRC calculation is disabled.
        • +
        • Update Timeout management to check on global process.
        • +
        • Update Error code management in all processes.
        • +
      • +
    • +
    • Add note to the max frequencies reached in all modes.
    • +
    • Add note about Master Receive mode restrictions : +
        +
      • Master Receive mode restriction: +
        +(#) In Master unidirectional receive-only mode (MSTR =1, BIDIMODE=0, RXONLY=0) or bidirectional receive mode (MSTR=1, BIDIMODE=1, BIDIOE=0), to ensure that the SPI does not initiate a new transfer the following procedure has to be respected:
        +(##) HAL_SPI_DeInit()
        +(##) HAL_SPI_Init() +
      • +
    • +
  • +
  • HAL SAI update +
      +
    • Update for proper management of the external synchronization input selection +
        +
      • update of HAL_SAI_Init () function
      • +
      • update definition of SAI_Block_SyncExt and SAI_Block_Synchronization groups
      • +
    • +
    • Update SAI_SLOTACTIVE_X defines values
    • +
    • Update HAL_SAI_Init() function for proper companding mode management
    • +
    • Update SAI_Transmit_ITxxBit() functions to add the check on transfer counter before writing new data to SAIx_DR registers
    • +
    • Update SAI_FillFifo() function to avoid issue when the number of data to transmit is smaller than the FIFO size
    • +
    • Update HAL_SAI_EnableRxMuteMode() function for proper mute management
    • +
    • Update SAI_InitPCM() function to support 24bits configuration
    • +
  • +
  • HAL ETH update +
      +
    • Removal of ETH MAC debug register defines
    • +
  • +
  • HAL FLASH update +
      +
    • Update FLASH_MassErase() function to apply correctly voltage range parameter
    • +
  • +
  • HAL I2S update +
      +
    • Update I2S_DMATxCplt() and I2S_DMARxCplt() to manage properly FullDuplex mode without any risk of missing data.
    • +
  • +
  • LL FMC update +
      +
    • Update the FMC_NORSRAM_Init() function to use BurstAccessMode field properly
    • +
  • +
  • LL FSMC update +
      +
    • Update the FSMC_NORSRAM_Init() function to use BurstAccessMode field properly
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • HAL Generic update +
      +
    • Update HAL weak empty callbacks to prevent unused argument compilation warnings with some compilers by calling the following line: +
        +
      • UNUSED(hppp);
      • +
    • +
    • STM32Fxxx_User_Manual.chm files regenerated for HAL V1.4.3
    • +
  • +
  • HAL ETH update +
      +
    • Update HAL_ETH_Init() function to add timeout on the Software reset management
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • General updates to fix known defects and enhancements implementation
    +
  • +
  • One change done on the HAL CRYP requires an update on the application code based on HAL V1.4.1 +
      +
    • Update HAL_CRYP_DESECB_Decrypt() API to invert pPlainData and pCypherData parameters
      +
    • +
  • +
  • HAL generic update +
      +
    • Update HAL weak empty callbacks to prevent unused argument compilation warnings with some compilers by calling the following line: +
        +
      • UNUSED(hppp);
        +
      • +
    • +
  • +
  • HAL CORTEX update +
      +
    • Remove duplication for __HAL_CORTEX_SYSTICKCLK_CONFIG() macro
      +
    • +
  • +
  • HAL HASH update +
      +
    • Rename HAL_HASH_STATETypeDef to HAL_HASH_StateTypeDef
    • +
    • Rename HAL_HASH_PhaseTypeDef to HAL_HASH_PhaseTypeDef
    • +
  • +
  • HAL RCC update +
      +
    • Add new macros __HAL_RCC_PPP_IS_CLK_ENABLED() to check on Clock enable/disable status
    • +
    • Update __HAL_RCC_USB_OTG_FS_CLK_DISABLE() macro to remove the disable for the SYSCFG
    • +
    • Update HAL_RCC_MCOConfig() API to use new defines for the GPIO Speed
    • +
    • Generic update to improve the PLL VCO min value(100MHz): PLLN, PLLI2S and PLLSAI min value is 50 instead of 192
    • +
  • +
  • HAL FLASH update +
      +
    • __HAL_FLASH_INSTRUCTION_CACHE_RESET() macro: update to reset ICRST bit in the ACR register after setting it.
    • +
    • Update to support until 15 FLASH wait state (FLASH_LATENCY_15) for STM32F446xx devices
    • +
  • +
  • HAL CRYP update +
      +
    • Update HAL_CRYP_DESECB_Decrypt() API to fix the inverted pPlainData and pCypherData parameters issue
    • +
  • +
  • HAL I2S update +
      +
    • Update HAL_I2S_Init() API to call __HAL_RCC_I2S_CONFIG() macro when external I2S clock is selected
    • +
  • +
  • HAL LTDC update +
      +
    • Update HAL_LTDC_SetWindowPosition() API to configure Immediate reload register instead of vertical blanking reload register.
    • +
  • +
  • HAL TIM update +
      +
    • Update HAL_TIM_ConfigClockSource() API to check only the required parameters
    • +
  • +
  • HAL NAND update +
      +
    • Update HAL_NAND_Read_Page()/HAL_NAND_Write_Page()/HAL_NAND_Read_SpareArea() APIs to manage correctly the NAND Page access
    • +
  • +
  • HAL CAN update +
      +
    • Update to use “=†instead of “|=†to clear flags in the MSR, TSR, RF0R and RF1R registers
    • +
  • +
  • HAL HCD update +
      +
    • Fix typo in __HAL_USB_OTG_HS_WAKEUP_EXTI_ENABLE_RISING_FALLING_EDGE() macro implementation
    • +
  • +
  • HAL PCD update +
      +
    • Update HAL_PCD_IRQHandler() API to avoid issue when DMA mode enabled for Status Phase IN stage
    • +
  • +
  • LL FMC update +
      +
    • Update the FMC_NORSRAM_Extended_Timing_Init() API to remove the check on CLKDIvison and DataLatency parameters
    • +
    • Update the FMC_NORSRAM_Init() API to add a check on the PageSize parameter for STM32F42/43xx devices
    • +
  • +
  • LL FSMC update +
      +
    • Update the FSMC_NORSRAM_Extended_Timing_Init() API to remove the check on CLKDIvison and DataLatency parameters
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • HAL DSI update +
      +
    • Update TCCR register assigned value in HAL_DSI_ConfigHostTimeouts() function
    • +
    • Update WPCR register assigned value in HAL_DSI_Init(), HAL_DSI_SetSlewRateAndDelayTuning(), HAL_DSI_SetSlewRateAndDelayTuning(), HAL_DSI_SetLowPowerRXFilter() / HAL_DSI_SetSDD(), HAL_DSI_SetLanePinsConfiguration(), HAL_DSI_SetPHYTimings(), HAL_DSI_ForceTXStopMode(), HAL_DSI_ForceRXLowPower(), HAL_DSI_ForceDataLanesInRX(), HAL_DSI_SetPullDown() and HAL_DSI_SetContentionDetectionOff() functions
    • +
    • Update DSI_HS_PM_ENABLE define value
    • +
    • Implement workaround for the hardware limitation: “The time to activate the clock between HS transmissions is not calculated correctlyâ€
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Add support of STM32F469xx, STM32F479xx, STM32F410Cx, STM32F410Rx and STM32F410Tx devices
  • +
  • General updates to fix known defects and enhancements implementation
  • +
  • Add new HAL drivers for DSI and LPTIM peripherals
  • +
  • HAL ADC update +
      +
    • Rename ADC_CLOCKPRESCALER_PCLK_DIV2 define to ADC_CLOCK_SYNC_PCLK_DIV2
    • +
    • Rename ADC_CLOCKPRESCALER_PCLK_DIV4 define to ADC_CLOCK_SYNC_PCLK_DIV4
    • +
    • Rename ADC_CLOCKPRESCALER_PCLK_DIV6 define to ADC_CLOCK_SYNC_PCLK_DIV6
    • +
    • Rename ADC_CLOCKPRESCALER_PCLK_DIV8 define to ADC_CLOCK_SYNC_PCLK_DIV8
    • +
  • +
  • HAL CORTEX update +
      +
    • Add specific API for MPU management +
        +
      • add MPU_Region_InitTypeDef structure
      • +
      • add new function HAL_MPU_ConfigRegion()
      • +
    • +
  • +
  • HAL DMA update +
      +
    • Overall driver update for code optimization +
        +
      • add StreamBaseAddress and StreamIndex new fields in the DMA_HandleTypeDef structure
      • +
      • add DMA_Base_Registers private structure
      • +
      • add static function DMA_CalcBaseAndBitshift()
      • +
      • update HAL_DMA_Init() function to use the new added static function
      • +
      • update HAL_DMA_DeInit() function to optimize clear flag operations
      • +
      • update HAL_DMA_Start_IT() function to optimize interrupts enable
      • +
      • update HAL_DMA_PollForTransfer() function to optimize check on flags
      • +
      • update HAL_DMA_IRQHandler() function to optimize interrupt flag management
      • +
    • +
  • +
  • HAL FLASH update +
      +
    • update HAL_FLASH_Program_IT() function by removing the pending flag clear
    • +
    • update HAL_FLASH_IRQHandler() function to improve erase operation procedure
    • +
    • update FLASH_WaitForLastOperation() function by checking on end of operation flag
    • +
  • +
  • HAL GPIO update +
      +
    • Rename GPIO_SPEED_LOW define to GPIO_SPEED_FREQ_LOW
    • +
    • Rename GPIO_SPEED_MEDIUM define to GPIO_SPEED_FREQ_MEDIUM
    • +
    • Rename GPIO_SPEED_FAST define to GPIO_SPEED_FREQ_HIGH
    • +
    • Rename GPIO_SPEED_HIGH define to GPIO_SPEED_FREQ_VERY_HIGH
    • +
  • +
  • HAL I2S update +
      +
    • Move I2S_Clock_Source defines to extension file to properly add the support of STM32F410xx devices
    • +
  • +
  • HAL LTDC update +
      +
    • rename HAL_LTDC_LineEvenCallback() function to HAL_LTDC_LineEventCallback()
    • +
    • add new function HAL_LTDC_SetPitch()
    • +
    • add new functions HAL_LTDC_StructInitFromVideoConfig() and HAL_LTDC_StructInitFromAdaptedCommandConfig() applicable only to STM32F469xx and STM32F479xx devices
    • +
  • +
  • HAL PWR update +
      +
    • move __HAL_PWR_VOLTAGESCALING_CONFIG() macro to extension file
    • +
    • move PWR_WAKEUP_PIN2 define to extension file
    • +
    • add PWR_WAKEUP_PIN3 define, applicable only to STM32F10xx devices
    • +
    • add new functions HAL_PWREx_EnableWakeUpPinPolarityRisingEdge() and HAL_PWREx_EnableWakeUpPinPolarityFallingEdge(), applicable only to STM32F469xx and STM32F479xx devices
    • +
  • +
  • HAL RTC update +
      +
    • Update HAL_RTCEx_SetWakeUpTimer() and HAL_RTCEx_SetWakeUpTimer_IT() functions to properly check on the WUTWF flag
    • +
  • +
  • HAL TIM update +
      +
    • add new defines TIM_SYSTEMBREAKINPUT_HARDFAULT, TIM_SYSTEMBREAKINPUT_PVD and TIM_SYSTEMBREAKINPUT_HARDFAULT_PVD, applicable only to STM32F410xx devices
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • General updates to fix known defects and enhancements implementation
  • +
  • One changes done on the HAL may require an update on the application code based on HAL V1.3.1 +
      +
    • HASH IT process: update to call the HAL_HASH_InCpltCallback() at the end of the complete buffer instead of every each 512 bits
    • +
  • +
  • HAL RCC update +
      +
    • HAL_RCCEx_PeriphCLKConfig() updates: +
        +
      • Update the LSE check condition after backup domain reset: update to check LSE ready flag when LSE oscillator is already enabled instead of check on LSE oscillator only when LSE is used as RTC clock source
      • +
      • Use the right macro to check the PLLI2SQ parameters
      • +
    • +
  • +
  • HAL RTC update +
      +
    • __HAL_RTC_TAMPER_TIMESTAMP_EXTI_GET_FLAG() macro: fix implementation issue
    • +
    • __HAL_RTC_ALARM_GET_IT(), __HAL_RTC_ALARM_CLEAR_FLAG(), __HAL_RTC_WAKEUPTIMER_CLEAR_FLAG(), __HAL_RTC_TIMESTAMP_CLEAR_FLAG() and __HAL_RTC_TAMPER_CLEAR_FLAG() macros implementation changed: remove unused cast
    • +
    • IS_RTC_TAMPER() macro: update to use literal instead of hardcoded value
    • +
    • Add new parameter SecondFraction in RTC_TimeTypeDef structure
    • +
    • HAL_RTC_GetTime() API update to support the new parameter SecondFraction
    • +
  • +
  • HAL ADC update +
      +
    • Add new literal: ADC_INJECTED_SOFTWARE_START to be used as possible value for the ExternalTrigInjecConvEdge parameter in the ADC_InitTypeDef structure to select the ADC software trigger mode.
    • +
  • +
  • HAL FLASH update +
      +
    • FLASH_OB_GetRDP() API update to return uint8_t instead of FlagStatus
    • +
    • __HAL_FLASH_GET_LATENCY() new macro add to get the flash latency
    • +
  • +
  • HAL SPI update +
      +
    • Fix the wrong definition of HAL_SPI_ERROR_FLAG literal
    • +
  • +
  • HAL I2S update +
      +
    • HAL_I2S_Transmit() API update to check on busy flag only for I2S slave mode
    • +
  • +
  • HAL CRC update +
      +
    • __HAL_CRC_SET_IDR() macro implementation change to use WRITE_REG() instead of MODIFY_REG()
    • +
  • +
  • HAL DMA2D update +
      +
    • HAL_DMA2D_ConfigLayer() API update to use “=†instead of “|=†to erase BGCOLR and FGCOLR registers before setting the new configuration
    • +
  • +
  • HAL HASH update +
      +
    • HAL_HASH_MODE_Start_IT() (MODE stands for MD5, SHA1, SHA224 and SHA36) updates: +
        +
      • Fix processing fail for small input buffers
      • +
      • Update to unlock the process and call return HAL_OK at the end of HASH processing to avoid incorrectly repeating software
      • +
      • Update to properly manage the HashITCounter
      • +
      • Update to call the HAL_HASH_InCpltCallback() at the end of the complete buffer instead of every each 512 bits
      • +
    • +
    • __HAL_HASH_GET_FLAG() update to check the right register when the DINNE flag is selected
    • +
    • HAL_HASH_SHA1_Accumulate() updates: +
        +
      • Add a call to the new IS_HASH_SHA1_BUFFER_SIZE() macro to check the size parameter.
      • +
      • Add the following note in API description +
          +
        • @note Input buffer size in bytes must be a multiple of 4 otherwise the digest computation is corrupted.
        • +
      • +
    • +
  • +
  • HAL RTC update +
      +
    • Update to define hardware independent literals names: +
        +
      • Rename RTC_TAMPERPIN_PC13 by RTC_TAMPERPIN_DEFAULT
      • +
      • Rename RTC_TAMPERPIN_PA0 by RTC_TAMPERPIN_POS1
      • +
      • Rename RTC_TAMPERPIN_PI8 by RTC_TAMPERPIN_POS1
      • +
      • Rename RTC_TIMESTAMPPIN_PC13 by RTC_TIMESTAMPPIN_DEFAULT
      • +
      • Rename RTC_TIMESTAMPPIN_PA0 by RTC_TIMESTAMPPIN_POS1
      • +
      • Rename RTC_TIMESTAMPPIN_PI8 by RTC_TIMESTAMPPIN_POS1
      • +
    • +
  • +
  • HAL ETH update +
      +
    • Remove duplicated IS_ETH_DUPLEX_MODE() and IS_ETH_RX_MODE() macros
    • +
    • Remove illegal space ETH_MAC_READCONTROLLER_FLUSHING macro
    • +
    • Update ETH_MAC_READCONTROLLER_XXX defined values (XXX can be IDLE, READING_DATA and READING_STATUS)
    • +
  • +
  • HAL PCD update +
      +
    • HAL_PCD_IRQHandler API: fix the bad Configuration of Turnaround Time
    • +
  • +
  • HAL HCD update +
      +
    • Update to use local variable in USB Host channel re-activation
    • +
  • +
  • LL FMC update +
      +
    • FMC_SDRAM_SendCommand() API: remove the following line: return HAL_ERROR;
    • +
  • +
  • LL USB update +
      +
    • USB_FlushTxFifo API: update to flush all Tx FIFO
    • +
    • Update to use local variable in USB Host channel re-activation
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • HAL PWR update +
      +
    • Fix compilation issue with STM32F417xx product: update STM32F17xx by STM32F417xx
    • +
  • +
  • HAL SPI update +
      +
    • Remove unused variable to avoid warning with TrueSTUDIO
    • +
  • +
  • HAL I2C update +
      +
    • I2C Polling/IT/DMA processes: move the wait loop on busy flag at the top of the processes, to ensure that software not perform any write access to I2C_CR1 register before hardware clearing STOP bit and to avoid also the waiting loop on BUSY flag under I2C/DMA ISR.
    • +
    • Update busy flag Timeout value
    • +
    • I2C Master Receive Processes update to disable ACK before generate the STOP
    • +
  • +
  • HAL DAC update +
      +
    • Fix V1.3.0 regression issue with DAC software trigger configuration
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Add support of STM32F446xx devices
  • +
  • General updates to fix known defects and enhancements implementation
  • +
  • Add new HAL drivers for CEC, QSPI, FMPI2C and SPDIFRX peripherals
  • +
  • Two changes done on the HAL requires an update on the application code based on HAL V1.2.0 +
      +
    • Overall SAI driver rework to have exhaustive support of the peripheral features: details are provided in HAL SAI update section below –> Compatibility with previous version is impacted
    • +
    • CRYP driver updated to support multi instance,so user must ensure that the new parameter Instance is initialized in his application(CRYPHandle.Instance = CRYP)
    • +
  • +
  • HAL Generic update +
      +
    • stm32f4xx_hal_def.h +
        +
      • Remove NULL definition and add include for stdio.h
      • +
    • +
    • stm32_hal_legacy.h +
        +
      • Update method to manage deference in alias implementation between all STM32 families
      • +
    • +
    • stm32f4xx_hal_ppp.c +
        +
      • HAL_PPP_Init(): update to force the HAL_PPP_STATE_RESET before calling the HAL_PPP_MspInit()
      • +
    • +
  • +
  • HAL RCC update +
      +
    • Add new function HAL_RCCEx_GetPeriphCLKFreq()
    • +
    • Move RCC_PLLInitTypeDef structure to extension file and add the new PLLR field specific to STM32F446xx devices
    • +
    • Move the following functions to extension file and add a __weak attribute in generic driver : this update is related to new system clock source (PLL/PLLR) added and only available for STM32F44xx devices +
        +
      • HAL_RCC_OscConfig()
      • +
      • HAL_RCC_GetSysClockFreq()
      • +
      • HAL_RCC_GetOscConfig()
      • +
    • +
    • Move the following macro to extension file as they have device dependent implementation +
        +
      • __HAL_RCC_PLL_CONFIG()
      • +
      • __HAL_RCC_PLLI2S_CONFIG()
      • +
      • __HAL_RCC_I2S_CONFIG()
      • +
    • +
    • Add new structure RCC_PLLI2SInitTypeDef containing new PLLI2S division factors used only w/ STM32F446xx devices
    • +
    • Add new structure RCC_PLLSAIInitTypeDef containing new PLLSAI division factors used only w/ STM32F446xx devices
    • +
    • Add new RCC_PeriphCLKInitTypeDef to support the peripheral source clock selection for (I2S, SAI, SDIO, FMPI2C, CEC, SPDIFRX and CLK48)
    • +
    • Update the HAL_RCCEx_PeriphCLKConfig() and HAL_RCCEx_GetPeriphCLKConfig() functions to support the new peripherals Clock source selection
    • +
    • Add __HAL_RCC_PLL_CONFIG() macro (the number of parameter and the implementation depend on the device part number)
    • +
    • Add __HAL_RCC_PLLI2S_CONFIG() macro(the number of parameter and the implementation depend on device part number)
    • +
    • Update __HAL_RCC_PLLSAI_CONFIG() macro to support new PLLSAI factors (PLLSAIM and PLLSAIP)
    • +
    • Add new macros for clock enable/Disable for the following peripherals (CEC, SPDIFRX, SAI2, QUADSPI)
    • +
    • Add the following new macros for clock source selection : +
        +
      • __HAL_RCC_SAI1_CONFIG() / __HAL_RCC_GET_SAI1_SOURCE()
      • +
      • __HAL_RCC_SAI2_CONFIG() / __HAL_RCC_GET_SAI2_SOURCE()
      • +
      • __HAL_RCC_I2S1_CONFIG() / __HAL_RCC_GET_I2S1_SOURCE()
      • +
      • __HAL_RCC_I2S2_CONFIG() / __HAL_RCC_GET_I2S2_SOURCE()
      • +
      • __HAL_RCC_CEC_CONFIG() / __HAL_RCC__GET_CEC_SOURCE()
      • +
      • __HAL_RCC_FMPI2C1_CONFIG() / __HAL_RCC_GET_FMPI2C1_SOURCE()
      • +
      • __HAL_RCC_SDIO_CONFIG() / __HAL_RCC_GET_SDIO_SOURCE()
      • +
      • __HAL_RCC_CLK48_CONFIG() / __HAL_RCC_GET_CLK48_SOURCE()
      • +
      • __HAL_RCC_SPDIFRXCLK_CONFIG() / __HAL_RCC_GET_SPDIFRX_SOURCE()
      • +
    • +
    • __HAL_RCC_PPP_CLK_ENABLE(): Implement workaround to cover RCC limitation regarding peripheral enable delay
    • +
    • HAL_RCC_OscConfig() fix issues: +
        +
      • Add a check on LSERDY flag when LSE_BYPASS is selected as new state for LSE oscillator.
      • +
    • +
    • Add new possible value RCC_PERIPHCLK_PLLI2S to be selected as PeriphClockSelection parameter in the RCC_PeriphCLKInitTypeDef structure to allow the possibility to output the PLLI2S on MCO without activating the I2S or the SAI.
    • +
    • __HAL_RCC_HSE_CONFIG() macro: add the comment below:
      +* @note Transition HSE Bypass to HSE On and HSE On to HSE Bypass are not supported by this macro.
      +* User should request a transition to HSE Off first and then HSE On or HSE Bypass.

    • +
    • __HAL_RCC_LSE_CONFIG() macro: add the comment below:
      +* @note Transition LSE Bypass to LSE On and LSE On to LSE Bypass are not supported by this macro.
      +* User should request a transition to LSE Off first and then LSE On or LSE Bypass.

    • +
    • Add the following new macros for PLL source and PLLM selection : +
        +
      • __HAL_RCC_PLL_PLLSOURCE_CONFIG()
      • +
      • __HAL_RCC_PLL_PLLM_CONFIG()
      • +
    • +
    • Macros rename: +
        +
      • HAL_RCC_OTGHS_FORCE_RESET() by HAL_RCC_USB_OTG_HS_FORCE_RESET()
      • +
      • HAL_RCC_OTGHS_RELEASE_RESET() by HAL_RCC_USB_OTG_HS_RELEASE_RESET()
      • +
      • HAL_RCC_OTGHS_CLK_SLEEP_ENABLE() by HAL_RCC_USB_OTG_HS_CLK_SLEEP_ENABLE()
      • +
      • HAL_RCC_OTGHS_CLK_SLEEP_DISABLE() by HAL_RCC_USB_OTG_HS_CLK_SLEEP_DISABLE()
      • +
      • HAL_RCC_OTGHSULPI_CLK_SLEEP_ENABLE() by HAL_RCC_USB_OTG_HS_ULPI_CLK_SLEEP_ENABLE()
      • +
      • HAL_RCC_OTGHSULPI_CLK_SLEEP_DISABLE() by HAL_RCC_USB_OTG_HS_ULPI_CLK_SLEEP_DISABLE()
      • +
    • +
    • Add __HAL_RCC_SYSCLK_CONFIG() new macro to configure the system clock source (SYSCLK)
    • +
    • __HAL_RCC_GET_SYSCLK_SOURCE() updates: +
        +
      • Add new RCC Literals: +
          +
        • RCC_SYSCLKSOURCE_STATUS_HSI
        • +
        • RCC_SYSCLKSOURCE_STATUS_HSE
        • +
        • RCC_SYSCLKSOURCE_STATUS_PLLCLK
        • +
        • RCC_SYSCLKSOURCE_STATUS_PLLRCLK
        • +
      • +
      • Update macro description to refer to the literals above
      • +
    • +
  • +
  • HAL PWR update +
      +
    • Add new define PWR_WAKEUP_PIN2
    • +
    • Add new API to Control/Get VOS bits of CR register +
        +
      • HAL_PWR_HAL_PWREx_ControlVoltageScaling()
      • +
      • HAL_PWREx_GetVoltageRange()
      • +
    • +
    • __HAL_PWR_ VOLTAGESCALING_CONFIG(): Implement workaround to cover VOS limitation delay when PLL is enabled after setting the VOS configuration
    • +
  • +
  • HAL GPIO update +
      +
    • Add the new Alternate functions literals related to remap for SPI, USART, I2C, SPDIFRX, CEC and QSPI
    • +
    • HAL_GPIO_DeInit(): Update to check if GPIO Pin x is already used in EXTI mode on another GPIO Port before De-Initialize the EXTI registers
    • +
  • +
  • HAL FLASH update +
      +
    • __HAL_FLASH_INSTRUCTION_CACHE_RESET() macro: update to reset ICRST bit in the ACR register after setting it.
    • +
    • __HAL_FLASH_DATA_CACHE_RESET() macro: update to reset DCRST bit in the ACR register after setting it.
    • +
  • +
  • HAL ADC update +
      +
    • Add new literal: ADC_SOFTWARE_START to be used as possible value for the ExternalTrigConv parameter in the ADC_InitTypeDef structure to select the ADC software trigger mode.
    • +
    • IS_ADC_CHANNEL() macro update to don’t assert stop the ADC_CHANNEL_TEMPSENSOR value
    • +
    • HAL_ADC_PollForConversion(): update to manage particular case when ADC configured in DMA mode and ADC sequencer with several ranks and polling for end of each conversion
    • +
    • HAL_ADC_Start()/HAL_ADC_Start_IT() /HAL_ADC_Start_DMA() update: +
        +
      • unlock the process before starting the ADC software conversion.
      • +
      • Optimize the ADC stabilization delays
      • +
    • +
    • __HAL_ADC_GET_IT_SOURCE() update macro implementation
    • +
    • Add more details in ‘How to use this driver’ section
    • +
  • +
  • HAL DAC update +
      +
    • Add new macro to check if the specified DAC interrupt source is enabled or disabled +
        +
      • __HAL_DAC_GET_IT_SOURCE()
      • +
    • +
    • HAL_DACEx_TriangleWaveGeneration() update to use DAC CR bit mask definition
    • +
    • HAL_DACEx_NoiseWaveGeneration() update to use DAC CR bit mask definition
    • +
  • +
  • HAL CAN update +
      +
    • CanTxMsgTypeDef structure: update to use uint8_t Data[8] instead of uint32_t Data[8]
    • +
    • CanRxMsgTypeDef structure: update to use uint8_t Data[8] instead of uint32_t Data[8]
    • +
  • +
  • HAL RTC update +
      +
    • Update to use CMSIS mask definition instead of hardcoded values (EXTI_IMR_IM17, EXTI_IMR_IM19..)
    • +
  • +
  • HAL LTDC update +
      +
    • LTDC_SetConfig() update to allow the drawing of partial bitmap in active layer.
    • +
  • +
  • HAL USART update +
      +
    • HAL_USART_Init() fix USART baud rate configuration issue: USART baud rate is twice Higher than expected
    • +
  • +
  • HAL SMARTCARD update +
      +
    • HAL_SMARTCARD_Transmit_IT() update to force the disable for the ERR interrupt to avoid the OVR interrupt
    • +
    • HAL_SMARTCARD_IRQHandler() update check condition for transmission end
    • +
    • Clean up: remove the following literals that aren’t used in smartcard mode +
        +
      • SMARTCARD_PARITY_NONE
      • +
      • SMARTCARD_WORDLENGTH_8B
      • +
      • SMARTCARD_STOPBITS_1
      • +
      • SMARTCADR_STOPBITS_2
      • +
    • +
  • +
  • HAL SPI update +
      +
    • HAL_SPI_Transmit_DMA()/HAL_SPI_Receive_DMA()/HAL_SPI_TarnsmitReceive_DMA() update to unlock the process before enabling the SPI peripheral
    • +
    • HAL_SPI_Transmit_DMA() update to manage correctly the DMA RX stream in SPI Full duplex mode
    • +
    • Section SPI_Exported_Functions_Group2 update to remove duplication in *.chm UM
    • +
  • +
  • HAL CRYP update +
      +
    • Update to manage multi instance: +
        +
      • Add new parameter Instance in the CRYP_HandleTypeDef Handle structure.
      • +
      • Add new parameter in all HAL CRYP macros +
          +
        • example: __HAL_CRYP_ENABLE() updated by __HAL_CRYP_ENABLE(__HANDLE__)
        • +
      • +
    • +
  • +
  • HAL DCMI update +
      +
    • Add an extension driver stm32f4xx_hal_dcmi_ex.c/h to manage the support of new Black and White feature
    • +
    • Add __weak attribute for HAL_DCMI_Init() function and add a new implementation in the extension driver to manage the black and white configuration only available in the STM32F446xx devices.
    • +
    • Move DCMI_InitTypeDef structure to extension driver and add the following new fields related to black and white feature: ByteSelectMode, ByteSelectStart, LineSelectMode and LineSelectStart
    • +
  • +
  • HAL PCD update +
      +
    • Add the support of LPM feature +
        +
      • add PCD_LPM_StateTypeDef enum
      • +
      • update PCD_HandleTypeDef structure to support the LPM feature
      • +
      • add new functions HAL_PCDEx_ActivateLPM(), HAL_PCDEx_DeActivateLPM() and HAL_PCDEx_LPM_Callback() in the stm32f4xx_hal_pcd_ex.h/.c files
      • +
    • +
  • +
  • HAL TIM update +
      +
    • Add TIM_TIM11_SPDIFRX define
    • +
  • +
  • HAL SAI update +
      +
    • Add stm32f4xx_hal_sai_ex.h/.c files for the SAI_BlockSynchroConfig() and the SAI_GetInputClock() management
    • +
    • Add new defines HAL_SAI_ERROR_AFSDET, HAL_SAI_ERROR_LFSDET, HAL_SAI_ERROR_CNREADY, HAL_SAI_ERROR_WCKCFG, HAL_SAI_ERROR_TIMEOUT in the SAI_Error_Code group
    • +
    • Add new defines SAI_SYNCEXT_DISABLE, SAI_SYNCEXT_IN_ENABLE, SAI_SYNCEXT_OUTBLOCKA_ENABLE, SAI_SYNCEXT_OUTBLOCKB_ENABLE for the SAI External synchronization
    • +
    • Add new defines SAI_I2S_STANDARD, SAI_I2S_MSBJUSTIFIED, SAI_I2S_LSBJUSTIFIED, SAI_PCM_LONG and SAI_PCM_SHORT for the SAI Supported protocol
    • +
    • Add new defines SAI_PROTOCOL_DATASIZE_16BIT, SAI_PROTOCOL_DATASIZE_16BITEXTENDED, SAI_PROTOCOL_DATASIZE_24BIT and SAI_PROTOCOL_DATASIZE_32BIT for SAI protocol data size
    • +
    • Add SAI Callback prototype definition
    • +
    • Update SAI_InitTypeDef structure by adding new fields: SynchroExt, Mckdiv, MonoStereoMode, CompandingMode, TriState
    • +
    • Update SAI_HandleTypeDef structure: +
        +
      • remove uint16_t pTxBuffPtr, pRxBuffPtr, TxXferSize, RxXferSize, TxXferCount and RxXferCount and replace them respectively by uint8_t *pBuffPtr, uint16_t XferSize and uint16_t XferCount
      • +
      • add mutecallback field
      • +
      • add struct __SAI_HandleTypeDef *hsai field
      • +
    • +
    • Remove SAI_CLKSOURCE_PLLR and SAI_CLOCK_PLLSRC defines
    • +
    • Add SAI_CLKSOURCE_NA define
    • +
    • Add SAI_AUDIO_FREQUENCY_MCKDIV define
    • +
    • Add SAI_SPDIF_PROTOCOL define
    • +
    • Add SAI_SYNCHRONOUS_EXT define
    • +
    • Add new functions HAL_SAI_InitProtocol(), HAL_SAI_Abort(), HAL_SAI_EnableTxMuteMode(), HAL_SAI_DisableTxMuteMode(), HAL_SAI_EnableRxMuteMode(), HAL_SAI_DisableRxMuteMode()
    • +
    • Update HAL_SAI_Transmit(), HAL_SAI_Receive(), HAL_SAI_Transmit_IT(), HAL_SAI_Receive_IT(), HAL_SAI_Transmit_DMA(), HAL_SAI_Receive_DMA() functions to use uint8_t pData instead of uint16_t pData –> This update is mainly impacting the compatibility with previous driver version.
    • +
  • +
  • HAL I2S update +
      +
    • Split the following functions between Generic and Extended API based on full duplex management and add the attribute __weak in the Generic API +
        +
      • HAL_I2S_Init(), HAL_I2S_DMAPause(), HAL_I2S_DMAStop(), HAL_I2S_DMAResume(), HAL_I2S_IRQHandle()
      • +
    • +
    • Move the following static functions from generic to extension driver +
        +
      • I2S_DMARxCplt() and I2S_DMATxCplt()
      • +
    • +
    • Remove static attribute from I2S_Transmit_IT() and I2S_Receive_IT() functions
    • +
    • Move I2SxEXT() macro to extension file
    • +
    • Add I2S_CLOCK_PLLR and I2S_CLOCK_PLLSRC defines for I2S clock source
    • +
    • Add new function I2S_GetInputClock()
    • +
  • +
  • HAL LL FMC update +
      +
    • Add WriteFifo and PageSize fields in the FMC_NORSRAM_InitTypeDef structure
    • +
    • Add FMC_PAGE_SIZE_NONE, FMC_PAGE_SIZE_128, FMC_PAGE_SIZE_256, FMC_PAGE_SIZE_1024, FMC_WRITE_FIFO_DISABLE, FMC_WRITE_FIFO_ENABLE defines
    • +
    • Update FMC_NORSRAM_Init(), FMC_NORSRAM_DeInit() and FMC_NORSRAM_Extended_Timing_Init() functions
    • +
  • +
  • HAL LL USB update +
      +
    • Update USB_OTG_CfgTypeDef structure to support LPM, lpm_enable field added
    • +
    • Update USB_HostInit() and USB_DevInit() functions to support the VBUS Sensing B activation
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Maintenance release to fix known defects and enhancements implementation
  • +
  • Macros and literals renaming to ensure compatibles across STM32 series, backward compatibility maintained thanks to new added file stm32_hal_legacy.h under /Inc/Legacy
  • +
  • Add *.chm UM for all drivers, a UM is provided for each superset RPN
  • +
  • Update drivers to be C++ compliant
  • +
  • Several update on source code formatting, for better UM generation (i.e. Doxygen tags updated)
  • +
  • Two changes done on the HAL requires an update on the application code based on HAL V1.1.0 +
      +
    • LSI_VALUE constant has been corrected in stm32f4xx_hal_conf.h file, its value changed from 40 KHz to 32 KHz
    • +
    • UART, USART, IRDA and SMARTCARD (referenced as PPP here below) drivers: in DMA transmit process, the code has been updated to avoid waiting on TC flag under DMA ISR, PPP TC interrupt is used instead. Below the update to be done on user application: +
        +
      • Configure and enable the USART IRQ in HAL_PPP_MspInit() function
      • +
      • In stm32f4xx_it.c file, PPP_IRQHandler() function: add a call to HAL_PPP_IRQHandler() function
      • +
    • +
  • +
  • HAL generic update +
      +
    • stm32f4xx_hal_def.h +
        +
      • Update NULL definition to fix C++ compilation issue
      • +
      • Add UNUSED() macro
      • +
      • Add a new define __NOINLINE to be used for the no inline code independent from tool chain
      • +
    • +
    • stm32f4xx_hal_conf_template.h +
        +
      • LSI_VALUE constant has been corrected, its value changed from 40 KHz to 32 KHz
      • +
    • +
    • Update all macros and literals naming to be uper case
    • +
    • ErrorCode parameter in PPP_HandleTypeDef structure updated to uint32_t instead of enum HAL_PPP_ErrorTypeDef
    • +
    • Remove the unused FLAG and IT assert macros
    • +
  • +
  • HAL ADC update +
      +
    • Fix temperature sensor channel configuration issue for STM32F427/437xx and STM32F429/439xx devices
    • +
  • +
  • HAL DAC update +
      +
    • HAL_DAC_ConfigChannel(): update the access to the DAC peripheral registers via the hdac handle instance
    • +
    • HAL_DAC_IRQHandler(): update to check on both DAC_FLAG_DMAUDR1 and DAC_FLAG_DMAUDR2
    • +
    • HAL_DACEx_NoiseWaveGenerate(): update to reset DAC CR register before setting the new DAC configuration
    • +
    • HAL_DACEx_TriangleWaveGenerate(): update to reset DAC CR register before setting the new DAC configuration
    • +
  • +
  • HAL CAN update +
      +
    • Unlock the CAN process when communication error occurred
    • +
  • +
  • HAL CORTEX update +
      +
    • Add new macro IS_NVIC_DEVICE_IRQ() to check on negative values of IRQn parameter
    • +
  • +
  • HAL CRYP update

    +
      +
    • HAL_CRYP_DESECB_Decrypt_DMA(): fix the inverted pPlainData and pCypherData parameters issue
    • +
    • CRYPEx_GCMCCM_SetInitVector(): remove the IVSize parameter as the key length 192bits and 256bits are not supported by this version
    • +
    • Add restriction for the CCM Encrypt/Decrypt API’s that only DataType equal to 8bits is supported
    • +
    • HAL_CRYPEx_AESGCM_Finish(): +
        +
      • Add restriction that the implementation is limited to 32bits inputs data length (Plain/Ciphertext, Header) compared with GCM stadards specifications (800-38D)
      • +
      • Update Size parameter on 32bits instead of 16bits
      • +
      • Fix issue with 16-bit Data Type: update to use intrinsic __ROR() instead of __REV16()
      • +
    • +
  • +
  • HAL DCMI update

    +
      +
    • HAL_DCMI_ConfigCROP(): Invert assert macros to check Y0 and Ysize parameters
    • +
  • +
  • HAL DMA update

    +
      +
    • HAL_DMA_Init(): Update to clear the DBM bit in the SxCR register before setting the new configuration
    • +
    • DMA_SetConfig(): add to clear the DBM bit in the SxCR register
    • +
  • +
  • HAL FLASH update

    +
      +
    • Add “HAL_†prefix in the defined values for the FLASH error code +
        +
      • Example: FLASH_ERROR_PGP renamed by HAL_FLASH_ERROR_PGP
      • +
    • +
    • Clear the Flash ErrorCode in the FLASH_WaitForLastOperation() function
    • +
    • Update FLASH_SetErrorCode() function to use “|=†operant to update the Flash ErrorCode parameter in the FLASH handle
    • +
    • IS_FLASH_ADDRESS(): Update the macro check using ‘<=’ condition instead of ‘<’
    • +
    • IS_OPTIONBYTE(): Update the macro check using ‘<=’ condition instead of ‘<’
    • +
    • Add “FLASH_†prefix in the defined values of FLASH Type Program parameter +
        +
      • Example: TYPEPROGRAM_BYTE renamed by FLASH_TYPEPROGRAM_BYTE
      • +
    • +
    • Add “FLASH_†prefix in the defined values of FLASH Type Erase parameter +
        +
      • Example: TYPEERASE_SECTORS renamed by FLASH_TYPEERASE_SECTORS
      • +
    • +
    • Add “FLASH_†prefix in the defined values of FLASH Voltage Range parameter +
        +
      • Example: VOLTAGE_RANGE_1 renamed by FLASH_VOLTAGE_RANGE_1
      • +
    • +
    • Add “OB_†prefix in the defined values of FLASH WRP State parameter +
        +
      • Example: WRPSTATE_ENABLE renamed by OB_WRPSTATE_ENABLE
      • +
    • +
    • Add “OB_†prefix in the defined values of the FLASH PCROP State parameter +
        +
      • PCROPSTATE_DISABLE updated by OB_PCROP_STATE_DISABLE
      • +
      • PCROPSTATE_ENABLE updated by OB_PCROP_STATE_ENABLE
      • +
    • +
    • Change “OBEX†prefix by “OPTIONBYTE†prefix in these defines: +
        +
      • OBEX_PCROP by OPTIONBYTE_PCROP
      • +
      • OBEX_BOOTCONFIG by OPTIONBYTE_BOOTCONFIG
      • +
    • +
  • +
  • HAL ETH update +
      +
    • Fix macros naming typo +
        +
      • Update __HAL_ETH_EXTI_SET_RISING_EGDE_TRIGGER() by __HAL_ETH_EXTI_SET_RISING_EDGE_TRIGGER()
      • +
      • Update __HAL_ETH_EXTI_SET_FALLING_EGDE_TRIGGER() by __HAL_ETH_EXTI_SET_FALLING_EDGE_TRIGGER()
      • +
    • +
  • +
  • HAL PWR update +
      +
    • Add new API to manage SLEEPONEXIT and SEVONPEND bits of SCR register +
        +
      • HAL_PWR_DisableSleepOnExit()
      • +
      • HAL_PWR_EnableSleepOnExit()
      • +
      • HAL_PWR_EnableSEVOnPend()
      • +
      • HAL_PWR_DisableSEVOnPend()
      • +
    • +
    • HAL_PWR_EnterSTOPMode() +
        +
      • Update to clear the CORTEX SLEEPDEEP bit of SCR register before entering in sleep mode
      • +
      • Update usage of __WFE() in low power entry function: if there is a pending event, calling __WFE() will not enter the CortexM4 core to sleep mode. The solution is to made the call below; the first __WFE() is always ignored and clears the event if one was already pending, the second is always applied
        +__SEV()
        +__WFE()
        +__WFE()
      • +
    • +
    • Add new PVD configuration modes +
        +
      • PWR_PVD_MODE_NORMAL
      • +
      • PWR_PVD_MODE_EVENT_RISING
      • +
      • PWR_PVD_MODE_EVENT_FALLING
      • +
      • PWR_PVD_MODE_EVENT_RISING_FALLING
      • +
    • +
    • Add new macros to manage PVD Trigger +
        +
      • __HAL_PWR_PVD_EXTI_ENABLE_RISING_EDGE()
      • +
      • __HAL_PWR_PVD_EXTI_DISABLE_RISING_EDGE(
      • +
      • __HAL_PWR_PVD_EXTI_ENABLE_FALLING_EDGE()
      • +
      • __HAL_PWR_PVD_EXTI_DISABLE_FALLING_EDGE()
      • +
      • __HAL_PWR_PVD_EXTI_ENABLE_RISING_FALLING_EDGE()
      • +
      • __HAL_PWR_PVD_EXTI_DISABLE_RISING_FALLING_EDGE()
      • +
    • +
    • PVD macros: +
        +
      • Remove the __EXTILINE__ parameter
      • +
      • Update to use prefix "__HAL_PWR_PVD_" instead of prefix "__HAL_PVD"
      • +
    • +
    • Rename HAL_PWR_PVDConfig() by HAL_PWR_ConfigPVD()
    • +
    • Rename HAL_PWREx_ActivateOverDrive() by HAL_PWREx_EnableOverDrive()
    • +
    • Rename HAL_PWREx_DeactivateOverDrive() by HAL_PWREx_DisableOverDrive()
    • +
  • +
  • HAL GPIO update +
      +
    • HAL_GPIO_Init()/HAL_GPIO_DeInit(): add a call to the CMSIS assert macro to check GPIO instance: IS_GPIO_ALL_INSTANCE()
    • +
    • HAL_GPIO_WritePin(): update to write in BSRR register
    • +
    • Rename GPIO_GET_SOURCE() by GET_GPIO_INDEX() and move this later to file stm32f4xx_hal_gpio_ex.h
    • +
    • Add new define for alternate function GPIO_AF5_SPI3 for STM32F429xx/439xx and STM32F427xx/437xx devices
    • +
  • +
  • HAL HASH update +
      +
    • HAL_HASH_MD5_Start_IT(): fix input address management issue
      +
    • +
  • +
  • HAL RCC update +
      +
    • Rename the following Macros +
        +
      • __PPP_CLK_ENABLE() by __HAL_RCC_PPP_CLK_ENABLE()
      • +
      • __PPP_CLK_DISABLE() by __HAL_RCC_PPP_CLK_DISABLE()
      • +
      • __PPP_FORCE_RESET() by __HAL_RCC_PPP_FORCE_RESET()
      • +
      • __PPP_RELEASE_RESET() by __HAL_RCC_PPP_RELEASE_RESET()
      • +
      • __PPP_CLK_SLEEP_ENABLE() by __HAL_RCC_PPP_CLK_SLEEP_ENABLE()
      • +
      • __PPP_CLK_SLEEP_DISABLE() by __HAL_RCC_PPP_CLK_SLEEP_DISABLE()
      • +
    • +
    • IS_RCC_PLLSAIN_VALUE() macro: update the check condition
    • +
    • Add description of RCC known Limitations
    • +
    • Rename HAL_RCC_CCSCallback() by HAL_RCC_CSSCallback()
    • +
    • HAL_RCC_OscConfig() fix issues: +
        +
      • Remove the disable of HSE oscillator when HSE_BYPASS is used as system clock source or as PPL clock source
      • +
      • Add a check on HSERDY flag when HSE_BYPASS is selected as new state for HSE oscillator.
      • +
    • +
    • Rename __HAL_RCC_I2SCLK() by __HAL_RCC_I2S_Config()
    • +
  • +
  • HAL I2S update

    +
      +
    • HAL_I2S_Init(): add check on I2S instance using CMSIS macro IS_I2S_ALL_INSTANCE()
    • +
    • HAL_I2S_IRQHandler() update for compliance w/ C++
    • +
    • Add use of tmpreg variable in __HAL_I2S_CLEAR_OVRFLAG() and __HAL_I2S_CLEAR_UDRFLAG() macro for compliance with C++
    • +
    • HAL_I2S_GetError(): update to return uint32_t instead of HAL_I2S_ErrorTypeDef enumeration
    • +
  • +
  • HAL I2C update

    +
      +
    • Update to clear the POS bit in the CR1 register at the end of HAL_I2C_Master_Read_IT() and HAL_I2C_Mem_Read_IT() process
    • +
    • Rename HAL_I2CEx_DigitalFilter_Config() by HAL_I2CEx_ConfigDigitalFilter()
    • +
    • Rename HAL_I2CEx_AnalogFilter_Config() by HAL_I2CEx_ConfigAnalogFilter()
    • +
    • Add use of tmpreg variable in __HAL_I2C_CLEAR_ADDRFLAG() and __HAL_I2C_CLEAR_STOPFLAG() macro for compliance with C++
    • +
  • +
  • HAL IrDA update +
      +
    • DMA transmit process; the code has been updated to avoid waiting on TC flag under DMA ISR, IrDA TC interrupt is used instead. Below the update to be done on user application: +
        +
      • Configure and enable the USART IRQ in HAL_IRDA_MspInit() function
      • +
      • In stm32f4xx_it.c file, UASRTx_IRQHandler() function: add a call to HAL_IRDA_IRQHandler() function
      • +
    • +
    • IT transmit process; the code has been updated to avoid waiting on TC flag under IRDA ISR, IrDA TC interrupt is used instead. No impact on user application
    • +
    • Rename Macros: add prefix "__HAL" +
        +
      • __IRDA_ENABLE() by __HAL_IRDA_ENABLE()
      • +
      • __IRDA_DISABLE() by __HAL_IRDA_DISABLE()
      • +
    • +
  • +
  • Add new user macros to manage the sample method feature +
      +
    • __HAL_IRDA_ONE_BIT_SAMPLE_ENABLE()
    • +
    • __HAL_IRDA_ONE_BIT_SAMPLE_DISABLE()
    • +
  • +
  • HAL_IRDA_Transmit_IT(): update to remove the enable of the parity error interrupt
  • +
  • Add use of tmpreg variable in __HAL_IRDA_CLEAR_PEFLAG() macro for compliance with C++
  • +
  • HAL_IRDA_Transmit_DMA() update to follow the right procedure “Transmission using DMA†in the reference manual +
      +
    • Add clear the TC flag in the SR register before enabling the DMA transmit request
    • +
  • +
  • HAL IWDG update +
      +
    • Rename the defined IWDG keys: +
        +
      • KR_KEY_RELOAD by IWDG_KEY_RELOAD
      • +
      • KR_KEY_ENABLE by IWDG_KEY_ENABLE
      • +
      • KR_KEY_EWA by IWDG_KEY_WRITE_ACCESS_ENABLE
      • +
      • KR_KEY_DWA by IWDG_KEY_WRITE_ACCESS_DISABLE
      • +
    • +
    • Add new macros __HAL_IWDG_RESET_HANDLE_STATE() and __HAL_IWDG_CLEAR_FLAG()
    • +
    • Update __HAL_IWDG_ENABLE_WRITE_ACCESS() and __HAL_IWDG_DISABLE_WRITE_ACCESS() as private macro
    • +
  • +
  • HAL SPI update

    +
      +
    • HAL_SPI_TransmitReceive_DMA() update to remove the DMA Tx Error Callback initialization when SPI RxOnly mode is selected
    • +
    • Add use of UNUSED(tmpreg) in __HAL_SPI_CLEAR_MODFFLAG(), __HAL_SPI_CLEAR_OVRFLAG(), __HAL_SPI_CLEAR_FREFLAG() to fix “Unused variable†warning with TrueSTUDIO.
    • +
    • Rename Literals: remove “D†from “DISABLED†and “ENABLED†+
        +
      • SPI_TIMODE_DISABLED by SPI_TIMODE_DISABLE
      • +
      • SPI_TIMODE_ENABLED by SPI_TIMODE_ENABLE
      • +
      • SPI_CRCCALCULATION_DISABLED by SPI_CRCCALCULATION_DISABLE
      • +
      • SPI_CRCCALCULATION_ENABLED by SPI_CRCCALCULATION_ENABLE
      • +
    • +
    • Add use of tmpreg variable in __HAL_SPI_CLEAR_MODFFLAG(), __HAL_SPI_CLEAR_FREFLAG() and __HAL_SPI_CLEAR_OVRFLAG() macros for compliance with C++
      +
    • +
  • +
  • HAL SDMMC update

    +
      +
    • IS_SDIO_ALL_INSTANCE() macro moved to CMSIS files
    • +
  • +
  • HAL LTDC update +
      +
    • HAL_LTDC_ConfigCLUT: optimize the function when pixel format is LTDC_PIXEL_FORMAT_AL44 +
        +
      • Update the size of color look up table to 16 instead of 256 when the pixel format is LTDC_PIXEL_FORMAT_AL44
      • +
    • +
  • +
  • HAL NAND update +
      +
    • Rename NAND Address structure to NAND_AddressTypeDef instead of NAND_AddressTypedef
    • +
    • Update the used algorithm of these functions +
        +
      • HAL_NAND_Read_Page()
      • +
      • HAL_NAND_Write_Page()
      • +
      • HAL_NAND_Read_SpareArea()
      • +
      • HAL_NAND_Write_SpareArea()
      • +
    • +
    • HAL_NAND_Write_Page(): move initialization of tickstart before while loop
    • +
    • HAL_NAND_Erase_Block(): add whait until NAND status is ready before exiting this function
    • +
  • +
  • HAL NOR update +
      +
    • Rename NOR Address structure to NOR_AddressTypeDef instead of NOR_AddressTypedef
    • +
    • NOR Status literals renamed +
        +
      • NOR_SUCCESS by HAL_NOR_STATUS_SUCCESS
      • +
      • NOR_ONGOING by HAL_NOR_STATUS_ONGOING
      • +
      • NOR_ERROR by HAL_NOR_STATUS_ERROR
      • +
      • NOR_TIMEOUT by HAL_NOR_STATUS_TIMEOUT
      • +
    • +
    • HAL_NOR_GetStatus() update to fix Timeout issue and exit from waiting loop when timeout occurred
    • +
  • +
  • HAL PCCARD update +
      +
    • Rename PCCARD Address structure to HAL_PCCARD_StatusTypeDef instead of CF_StatusTypedef
    • +
    • PCCARD Status literals renamed +
        +
      • CF_SUCCESS by HAL_PCCARD_STATUS_SUCCESS
      • +
      • CF_ONGOING by HAL_PCCARD_STATUS_ONGOING
      • +
      • CF_ERROR by HAL_PCCARD_STATUS_ERROR
      • +
      • CF_TIMEOUT by HAL_PCCARD_STATUS_TIMEOUT
      • +
    • +
    • Update “CF†by “PCCARD†in functions, literals and macros
    • +
  • +
  • HAL PCD update +
      +
    • Rename functions +
        +
      • HAL_PCD_ActiveRemoteWakeup() by HAL_PCD_ActivateRemoteWakeup()
      • +
      • HAL_PCD_DeActiveRemoteWakeup() by HAL_PCD_DeActivateRemoteWakeup()
      • +
    • +
    • Rename literals +
        +
      • USB_FS_EXTI_TRIGGER_RISING_EDGE by USB_OTG_FS_WAKEUP_EXTI_RISING_EDGE
      • +
      • USB_FS_EXTI_TRIGGER_FALLING_EDGE by USB_OTG_FS_WAKEUP_EXTI_FALLING_EDGE
      • +
      • USB_FS_EXTI_TRIGGER_BOTH_EDGE() by USB_OTG_FS_WAKEUP_EXTI_RISING_FALLING_EDGE
      • +
      • USB_HS_EXTI_TRIGGER_RISING_EDGE by USB_OTG_HS_WAKEUP_EXTI_RISING_EDGE
      • +
      • USB_HS_EXTI_TRIGGER_FALLING_EDGE by USB_OTG_HS_WAKEUP_EXTI_FALLING_EDGE
      • +
      • USB_HS_EXTI_TRIGGER_BOTH_EDGE by USB_OTG_HS_WAKEUP_EXTI_RISING_FALLING_EDGE
      • +
      • USB_HS_EXTI_LINE_WAKEUP by USB_OTG_HS_EXTI_LINE_WAKEUP
      • +
      • USB_FS_EXTI_LINE_WAKEUP by USB_OTG_FS_EXTI_LINE_WAKEUP
      • +
    • +
  • +
  • Rename USB EXTI macros (FS, HS referenced as SUBBLOCK here below) +
      +
    • __HAL_USB_SUBBLOCK_EXTI_ENABLE_IT() by __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_ENABLE_IT()
      +
    • +
    • __HAL_USB_SUBBLOCK_EXTI_DISABLE_IT() by __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_DISABLE_IT()
    • +
    • __HAL_USB_SUBBLOCK_EXTI_GET_FLAG() by __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_GET_FLAG()
    • +
    • __HAL_USB_SUBBLOCK_EXTI_CLEAR_FLAG() by __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_CLEAR_FLAG()
    • +
    • __HAL_USB_SUBBLOCK_EXTI_SET_RISING_EGDE_TRIGGER() by __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_ENABLE_RISING_EDGE()
    • +
    • __HAL_USB_SUBBLOCK_EXTI_SET_FALLING_EGDE_TRIGGER() by __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_ENABLE_FALLING_EDGE()
    • +
    • __HAL_USB_SUBBLOCK_EXTI_SET_FALLINGRISING_TRIGGER() by __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_ENABLE_RISING_FALLING_EDGE()
    • +
    • __HAL_USB_SUBBLOCK_EXTI_GENERATE_SWIT() by __HAL_USB_OTG_SUBBLOCK_WAKEUP_EXTI_GENERATE_SWIT()
      +
    • +
  • +
  • HAL RNG update +
      +
    • Add new functions +
        +
      • HAL_RNG_GenerateRandomNumber(): to generate a 32-bits random number, return random value in argument and return HAL status.
      • +
      • HAL_RNG_GenerateRandomNumber_IT(): to start generation of the 32-bits random number, user should call the HAL_RNG_ReadLastRandomNumber() function under the HAL_RNG_ReadyCallback() to get the generated random value.
      • +
      • HAL_RNG_ReadLastRandomNumber(): to return the last random value stored in the RNG handle
      • +
    • +
    • HAL_RNG_GetRandomNumber(): return value update (obsolete), replaced by HAL_RNG_GenerateRandomNumber()
    • +
    • HAL_RNG_GetRandomNumber_IT(): wrong implementation (obsolete), replaced by HAL_RNG_GenerateRandomNumber_IT()
    • +
    • __HAL_RNG_CLEAR_FLAG() macro (obsolete), replaced by new __HAL_RNG_CLEAR_IT() macro
    • +
    • Add new define for RNG ready interrupt: RNG_IT_DRDY
    • +
  • +
  • HAL RTC update +
      +
    • HAL_RTC_GetTime() and HAL_RTC_GetDate(): add the comment below
      +* @note You must call HAL_RTC_GetDate() after HAL_RTC_GetTime() to unlock the values
      +* in the higher-order calendar shadow registers to ensure consistency between the time and date values.
      +* Reading RTC current time locks the values in calendar shadow registers until Current date is read.

    • +
    • Rename literals: add prefix "__HAL" +
        +
      • FORMAT_BIN by HAL_FORMAT_BIN
      • +
      • FORMAT_BCD by HAL_FORMAT_BCD
      • +
    • +
    • Rename macros (ALARM, WAKEUPTIMER and TIMESTAMP referenced as SUBBLOCK here below) +
        +
      • __HAL_RTC_EXTI_ENABLE_IT() by __HAL_RTC_SUBBLOCK_EXTI_ENABLE_IT()
      • +
      • __HAL_RTC_EXTI_DISABLE_IT() by __HAL_RTC__SUBBLOCK_EXTI_DISABLE_IT()
      • +
      • __HAL_RTC_EXTI_CLEAR_FLAG() by __HAL_RTC_SUBBLOCK_EXTI_CLEAR_FLAG()
      • +
      • __HAL_RTC_EXTI_GENERATE_SWIT() by __HAL_RTC_SUBBLOCK_EXTI_GENERATE_SWIT()
      • +
    • +
    • Add new macros (ALARM, WAKEUPTIMER and TAMPER_TIMESTAMP referenced as SUBBLOCK here below) +
        +
      • __HAL_RTC_SUBBLOCK_GET_IT_SOURCE()
      • +
      • __HAL_RTC_SUBBLOCK_EXTI_ENABLE_EVENT()
      • +
      • __HAL_RTC_SUBBLOCK_EXTI_DISABLE_EVENT()
      • +
      • __HAL_RTC_SUBBLOCK_EXTI_ENABLE_FALLING_EDGE()
      • +
      • __HAL_RTC_SUBBLOCK_EXTI_DISABLE_FALLING_EDGE()
      • +
      • __HAL_RTC_SUBBLOCK_EXTI_ENABLE_RISING_EDGE()
      • +
      • __HAL_RTC_SUBBLOCK_EXTI_DISABLE_RISING_EDGE()
      • +
      • __HAL_RTC_SUBBLOCK_EXTI_ENABLE_RISING_FALLING_EDGE()
      • +
      • __HAL_RTC_SUBBLOCK_EXTI_DISABLE_RISING_FALLING_EDGE()
      • +
      • __HAL_RTC_SUBBLOCK_EXTI_GET_FLAG()
      • +
    • +
  • +
  • HAL SAI update +
      +
    • Update SAI_STREOMODE by SAI_STEREOMODE
    • +
    • Update FIFO status Level defines in upper case
    • +
    • Rename literals: remove “D†from “DISABLED†and “ENABLED†+
        +
      • SAI_OUTPUTDRIVE_DISABLED by SAI_OUTPUTDRIVE_DISABLE
      • +
      • SAI_OUTPUTDRIVE_ENABLED by SAI_OUTPUTDRIVE_ENABLE
      • +
      • SAI_MASTERDIVIDER_ENABLED by SAI_MASTERDIVIDER_ENABLE
      • +
      • SAI_MASTERDIVIDER_DISABLED by SAI_MASTERDIVIDER_DISABLE
      • +
    • +
  • +
  • HAL SD update +
      +
    • Rename SD_CMD_SD_APP_STAUS by SD_CMD_SD_APP_STATUS
    • +
    • SD_PowerON() updated to add 1ms required power up waiting time before starting the SD initialization sequence
    • +
    • SD_DMA_RxCplt()/SD_DMA_TxCplt(): add a call to HAL_DMA_Abort()
    • +
    • HAL_SD_ReadBlocks() update to set the defined DATA_BLOCK_SIZE as SDIO DataBlockSize parameter
    • +
    • HAL_SD_ReadBlocks_DMA()/HAL_SD_WriteBlocks_DMA() update to call the HAL_DMA_Start_IT() function with DMA Datalength set to BlockSize/4 as the DMA is configured in word
    • +
  • +
  • HAL SMARTCARD update +
      +
    • DMA transmit process; the code has been updated to avoid waiting on TC flag under DMA ISR, SMARTCARD TC interrupt is used instead. Below the update to be done on user application: +
        +
      • Configure and enable the USART IRQ in HAL_SAMRTCARD_MspInit() function
      • +
      • In stm32f4xx_it.c file, UASRTx_IRQHandler() function: add a call to HAL_SMARTCARD_IRQHandler() function
      • +
    • +
    • IT transmit process; the code has been updated to avoid waiting on TC flag under SMARTCARD ISR, SMARTCARD TC interrupt is used instead. No impact on user application
    • +
    • Rename macros: add prefix "__HAL" +
        +
      • __SMARTCARD_ENABLE() by __HAL_SMARTCARD_ENABLE()
      • +
      • __SMARTCARD_DISABLE() by __HAL_SMARTCARD_DISABLE()
      • +
      • __SMARTCARD_ENABLE_IT() by __HAL_SMARTCARD_ENABLE_IT()
      • +
      • __SMARTCARD_DISABLE_IT() by __HAL_SMARTCARD_DISABLE_IT()
      • +
      • __SMARTCARD_DMA_REQUEST_ENABLE() by __HAL_SMARTCARD_DMA_REQUEST_ENABLE()
      • +
      • __SMARTCARD_DMA_REQUEST_DISABLE() by __HAL_SMARTCARD_DMA_REQUEST_DISABLE()
      • +
    • +
    • Rename literals: remove “D†from “DISABLED†and “ENABLED†+
        +
      • SMARTCARD_NACK_ENABLED by SMARTCARD_NACK_ENABLE
      • +
      • SMARTCARD_NACK_DISABLED by SMARTCARD_NACK_DISABLE
      • +
    • +
    • Add new user macros to manage the sample method feature +
        +
      • __HAL_SMARTCARD_ONE_BIT_SAMPLE_ENABLE()
      • +
      • __HAL_SMARTCARD_ONE_BIT_SAMPLE_DISABLE()
      • +
    • +
    • Add use of tmpreg variable in __HAL_SMARTCARD_CLEAR_PEFLAG() macro for compliance with C++
    • +
    • HAL_SMARTCARD_Transmit_DMA() update to follow the right procedure “Transmission using DMA†in the reference manual +
        +
      • Add clear the TC flag in the SR register before enabling the DMA transmit request
      • +
    • +
  • +
  • HAL TIM update +
      +
    • Add TIM_CHANNEL_ALL as possible value for all Encoder Start/Stop APIs Description
    • +
    • HAL_TIM_OC_ConfigChannel() remove call to IS_TIM_FAST_STATE() assert macro
    • +
    • HAL_TIM_PWM_ConfigChannel() add a call to IS_TIM_FAST_STATE() assert macro to check the OCFastMode parameter
    • +
    • HAL_TIM_DMADelayPulseCplt() Update to set the TIM Channel before to call HAL_TIM_PWM_PulseFinishedCallback()
    • +
    • HAL_TIM_DMACaptureCplt() update to set the TIM Channel before to call HAL_TIM_IC_CaptureCallback()
    • +
    • TIM_ICx_ConfigChannel() update to fix Timer CCMR1 register corruption when setting ICFilter parameter
    • +
    • HAL_TIM_DMABurst_WriteStop()/HAL_TIM_DMABurst_ReadStop() update to abort the DMA transfer for the specific TIM channel
    • +
    • Add new function for TIM Slave configuration in IT mode: HAL_TIM_SlaveConfigSynchronization_IT()
    • +
    • HAL_TIMEx_ConfigBreakDeadTime() add an assert check on Break & DeadTime parameters values
    • +
    • HAL_TIMEx_OCN_Start_IT() add the enable of Break Interrupt for all output modes
    • +
    • Add new macros to ENABLE/DISABLE URS bit in TIM CR1 register: +
        +
      • __HAL_TIM_URS_ENABLE()
      • +
      • __HAL_TIM_URS_DISABLE()
      • +
    • +
    • Add new macro for TIM Edge modification: __HAL_TIM_SET_CAPTUREPOLARITY()
    • +
  • +
  • HAL UART update +
      +
    • Add IS_LIN_WORD_LENGTH() and IS_LIN_OVERSAMPLING() macros: to check respectively WordLength and OverSampling parameters in LIN mode
    • +
    • DMA transmit process; the code has been updated to avoid waiting on TC flag under DMA ISR, UART TC interrupt is used instead. Below the update to be done on user application: +
        +
      • Configure and enable the USART IRQ in HAL_UART_MspInit() function
      • +
      • In stm32f4xx_it.c file, USARTx_IRQHandler() function: add a call to HAL_UART_IRQHandler() function
      • +
    • +
    • IT transmit process; the code has been updated to avoid waiting on TC flag under UART ISR, UART TC interrupt is used instead. No impact on user application
    • +
    • Rename macros: +
        +
      • __HAL_UART_ONEBIT_ENABLE() by __HAL_UART_ONE_BIT_SAMPLE_ENABLE()
      • +
      • __HAL_UART_ONEBIT_DISABLE() by __HAL_UART_ONE_BIT_SAMPLE_DISABLE()
      • +
    • +
    • Rename literals: +
        +
      • UART_WAKEUPMETHODE_IDLELINE by UART_WAKEUPMETHOD_IDLELINE
      • +
      • UART_WAKEUPMETHODE_ADDRESSMARK by UART_WAKEUPMETHOD_ADDRESSMARK
      • +
    • +
    • Add use of tmpreg variable in __HAL_UART_CLEAR_PEFLAG() macro for compliance with C++
    • +
    • HAL_UART_Transmit_DMA() update to follow the right procedure “Transmission using DMA†in the reference manual +
        +
      • Add clear the TC flag in the SR register before enabling the DMA transmit request
      • +
    • +
  • +
  • HAL USART update +
      +
    • DMA transmit process; the code has been updated to avoid waiting on TC flag under DMA ISR, USART TC interrupt is used instead. Below the update to be done on user application: +
        +
      • Configure and enable the USART IRQ in HAL_USART_MspInit() function
      • +
      • In stm32f4xx_it.c file, USARTx_IRQHandler() function: add a call to HAL_USART_IRQHandler() function
      • +
    • +
    • IT transmit process; the code has been updated to avoid waiting on TC flag under USART ISR, USART TC interrupt is used instead. No impact on user application
    • +
    • HAL_USART_Init() update to enable the USART oversampling by 8 by default in order to reach max USART frequencies
    • +
    • USART_DMAReceiveCplt() update to set the new USART state after checking on the old state
    • +
    • HAL_USART_Transmit_DMA()/HAL_USART_TransmitReceive_DMA() update to follow the right procedure “Transmission using DMA†in the reference manual +
        +
      • Add clear the TC flag in the SR register before enabling the DMA transmit request
      • +
    • +
    • Rename macros: +
        +
      • __USART_ENABLE() by __HAL_USART_ENABLE()
      • +
      • __USART_DISABLE() by __HAL_USART_DISABLE()
      • +
      • __USART_ENABLE_IT() by __HAL_USART_ENABLE_IT()
      • +
      • __USART_DISABLE_IT() by __HAL_USART_DISABLE_IT()
      • +
    • +
    • Rename literals: remove “D†from “DISABLED†and “ENABLED†+
        +
      • USART_CLOCK_DISABLED by USART_CLOCK_DISABLE
      • +
      • USART_CLOCK_ENABLED by USART_CLOCK_ENABLE
      • +
      • USARTNACK_ENABLED by USART_NACK_ENABLE
      • +
      • USARTNACK_DISABLED by USART_NACK_DISABLE
      • +
    • +
    • Add new user macros to manage the sample method feature +
        +
      • __HAL_USART_ONE_BIT_SAMPLE_ENABLE()
      • +
      • __HAL_USART_ONE_BIT_SAMPLE_DISABLE()
      • +
    • +
    • Add use of tmpreg variable in __HAL_USART_CLEAR_PEFLAG() macro for compliance with C++
    • +
  • +
  • HAL WWDG update +
      +
    • Add new parameter in __HAL_WWDG_ENABLE_IT() macro
    • +
    • Add new macros to manage WWDG IT & correction: +
        +
      • __HAL_WWDG_DISABLE()
      • +
      • __HAL_WWDG_DISABLE_IT()
      • +
      • __HAL_WWDG_GET_IT()
      • +
      • __HAL_WWDG_GET_IT_SOURCE()
      • +
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Add support of STM32F411xE devices
  • +
  • HAL generic update +
      +
    • Enhance HAL delay and time base implementation +
        +
      • Systick timer is used by default as source of time base, but user can eventually implement his proper time base source (a general purpose timer for example or other time source)
      • +
      • Functions affecting time base configurations are declared as __Weak to make override possible in case of other implementations in user file, for more details please refer to HAL_TimeBase example
      • +
    • +
    • Fix flag clear procedure: use atomic write operation “=†instead of ready-modify-write operation “|=†or “&=â€
    • +
    • Fix on Timeout management, Timeout value set to 0 passed to API automatically exits the function after checking the flag without any wait
    • +
    • Common update for the following communication peripherals: SPI, UART, USART and IRDA +
        +
      • Add DMA circular mode support
      • +
      • Remove lock from recursive process
      • +
    • +
    • Add new macro __HAL_RESET_HANDLE_STATE to reset a given handle state
    • +
    • Add a new attribute for functions executed from internal SRAM and depending from Compiler implementation
    • +
    • When USE_RTOS == 1 (in stm32l0xx_hal_conf.h), the __HAL_LOCK() is not defined instead of being defined empty
    • +
    • Miscellaneous comments and formatting update
    • +
    • stm32f4xx_hal_conf_template.h +
        +
      • Add a new define for LSI default value LSI_VALUE
      • +
      • Add a new define for LSE default value LSE_VALUE
      • +
      • Add a new define for Tick interrupt priority TICK_INT_PRIORITY (needed for the enhanced time base implementation)
      • +
      • Important Note: aliases has been added for any API naming change, to keep compatibility with previous version
      • +
    • +
  • +
  • HAL GPIO update +
      +
    • Add a new macro __HAL_GPIO_EXTI_GENERATE_SWIT() to manage the generation of software interrupt on selected EXTI line
    • +
    • HAL_GPIO_Init(): use temporary variable when modifying the registers, to avoid unexpected transition in the GPIO pin configuration
    • +
    • Remove IS_GET_GPIO_PIN macro
    • +
    • Add a new function HAL_GPIO_LockPin()
    • +
    • Private Macro __HAL_GET_GPIO_SOURCE renamed into GET_GPIO_SOURCE
    • +
    • Add the support of STM32F411xx devices : add the new Alternate functions values related to new remap added for SPI, USART, I2C
    • +
    • Update the following HAL GPIO macros description: rename EXTI_Linex by GPIO_PIN_x +
        +
      • __HAL_GPIO_EXTI_CLEAR_IT()
      • +
      • __HAL_GPIO_EXTI_GET_IT()
      • +
      • __HAL_GPIO_EXTI_CLEAR_FLAG()
      • +
      • __HAL_GPIO_EXTI_GET_FLAG()
      • +
    • +
  • +
  • HAL DMA update

    +
      +
    • Fix in HAL_DMA_PollForTransfer() to: +
        +
      • set DMA error code in case of HAL_ERROR status
      • +
      • set HAL Unlock before DMA state update
      • +
    • +
  • +
  • HAL DMA2D update

    +
      +
    • Add configuration of source address in case of A8 or A4 M2M_PFC DMA2D mode
    • +
  • +
  • HAL FLASH update +
      +
    • Functions reorganization update, depending on the features supported by each STM32F4 device
    • +
    • Add new driver (stm32f4xx_hal_flash_ramfunc.h/.c) to manage function executed from RAM, these functions are available only for STM32F411xx Devices +
        +
      • FLASH_StopFlashInterfaceClk() : Stop the flash interface while System Run
      • +
      • FLASH_StartFlashInterfaceClk() : Stop the flash interface while System Run
      • +
      • FLASH_EnableFlashSleepMode() : Enable the flash sleep while System Run
      • +
      • FLASH_DisableFlashSleepMode() : Disable the flash sleep while System Run
      • +
    • +
  • +
  • HAL PWR update +
      +
    • HAL_PWR_PVDConfig(): add clear of the EXTI trigger before new configuration
    • +
    • Fix in HAL_PWR_EnterSTANDBYMode() to not clear Wakeup flag (WUF), which need to be cleared at application level before to call this function
    • +
    • HAL_PWR_EnterSLEEPMode() +
        +
      • Remove disable and enable of SysTick Timer
      • +
      • Update usage of __WFE() in low power entry function: if there is a pending event, calling __WFE() will not enter the CortexM4 core to sleep mode. The solution is to made the call below; the first __WFE() is always ignored and clears the event if one was already pending, the second is always applied
        +__SEV()
        +__WFE()
        +__WFE()
      • +
    • +
    • Add new macro for software event generation __HAL_PVD_EXTI_GENERATE_SWIT()
    • +
    • Remove the following defines form Generic driver and add them under extension driver because they are only used within extension functions. +
        +
      • CR_FPDS_BB: used within HAL_PWREx_EnableFlashPowerDown() function
      • +
      • CSR_BRE_BB: used within HAL_PWREx_EnableBkUpReg() function
      • +
    • +
    • Add the support of STM32F411xx devices add the define STM32F411xE +
        +
      • For STM32F401xC, STM32F401xE and STM32F411xE devices add the following functions used to enable or disable the low voltage mode for regulators +
          +
        • HAL_PWREx_EnableMainRegulatorLowVoltage()
        • +
        • HAL_PWREx_DisableMainRegulatorLowVoltage()
        • +
        • HAL_PWREx_EnableLowRegulatorLowVoltage()
        • +
        • HAL_PWREx_DisableLowRegulatorLowVoltage()
        • +
      • +
      • For STM32F42xxx/43xxx devices, add a new function for Under Driver management as the macro already added for this mode is not sufficient: HAL_PWREx_EnterUnderDriveSTOPMode()
      • +
    • +
  • +
  • HAL RCC update +
      +
    • In HAL_RCC_ClockConfig() function: update the AHB clock divider before clock switch to new source
    • +
    • Allow to calibrate the HSI when it is used as system clock source
    • +
    • Rename the following macros +
        +
      • __OTGFS_FORCE_RESET () by __USB_OTG_FS_FORCE_RESET()
      • +
      • __OTGFS_RELEASE_RESET () by __USB_OTG_FS_RELEASE_RESET()
      • +
      • __OTGFS_CLK_SLEEP_ENABLE () by __USB_OTG_FS_CLK_SLEEP_ENABLE()
      • +
      • __OTGFS_CLK_SLEEP_DISABLE () by __USB_OTG_FS_CLK_SLEEP_DISABLE()
      • +
    • +
    • Add new field PLLI2SM in RCC_PLLI2SInitTypeDef structure, this division factor is added for PLLI2S VCO input clock only STM32F411xE devices => the FW compatibility is broken vs. STM32F401xx devices
    • +
    • Update HAL_RCCEx_PeriphCLKConfig() and HAL_RCCEx_GetPeriphCLKConfig() functions to support the new PLLI2SM
    • +
    • Add new function to manage the new LSE mode : HAL_RCCEx_SelectLSEMode()
    • +
    • Reorganize the macros depending from Part number used and make them more clear
    • +
  • +
  • HAL UART update

    +
      +
    • Add new macros to control CTS and RTS
    • +
    • Add specific macros to manage the flags cleared only by a software sequence +
        +
      • __HAL_UART_CLEAR_PEFLAG()
      • +
      • __HAL_UART_CLEAR_FEFLAG()
      • +
      • __HAL_UART_CLEAR_NEFLAG()
      • +
      • __HAL_UART_CLEAR_OREFLAG()
      • +
      • __HAL_UART_CLEAR_IDLEFLAG()
      • +
    • +
    • Add several enhancements without affecting the driver functionalities +
        +
      • Remove the check on RXNE set after reading the Data in the DR register
      • +
      • Update the transmit processes to use TXE instead of TC
      • +
      • Update HAL_UART_Transmit_IT() to enable UART_IT_TXE instead of UART_IT_TC
      • +
    • +
  • +
  • HAL USART update

    +
      +
    • Add specific macros to manage the flags cleared only by a software sequence +
        +
      • __HAL_USART_CLEAR_PEFLAG()
      • +
      • __HAL_USART_CLEAR_FEFLAG()
      • +
      • __HAL_USART_CLEAR_NEFLAG()
      • +
      • __HAL_USART_CLEAR_OREFLAG()
      • +
      • __HAL_USART_CLEAR_IDLEFLAG()
      • +
    • +
    • Update HAL_USART_Transmit_IT() to enable USART_IT_TXE instead of USART_IT_TC
    • +
  • +
  • HAL IRDA update

    +
      +
    • Add specific macros to manage the flags cleared only by a software sequence __HAL_IRDA_CLEAR_PEFLAG() __HAL_ IRDA _CLEAR_FEFLAG() __HAL_ IRDA _CLEAR_NEFLAG() __HAL_ IRDA _CLEAR_OREFLAG() __HAL_ IRDA _CLEAR_IDLEFLAG()
    • +
    • Add several enhancements without affecting the driver functionalities +
        +
      • Remove the check on RXNE set after reading the Data in the DR register
      • +
      • Update HAL_IRDA_Transmit_IT() to enable IRDA_IT_TXE instead of IRDA_IT_TC
      • +
    • +
    • Add the following APIs used within DMA process +
        +
      • HAL_StatusTypeDef HAL_IRDA_DMAPause(IRDA_HandleTypeDef *hirda);
      • +
      • HAL_StatusTypeDef HAL_IRDA_DMAResume(IRDA_HandleTypeDef *hirda);
      • +
      • HAL_StatusTypeDef HAL_IRDA_DMAStop(IRDA_HandleTypeDef *hirda);
      • +
      • void HAL_IRDA_TxHalfCpltCallback(IRDA_HandleTypeDef *hirda);
      • +
      • void HAL_IRDA_RxHalfCpltCallback(IRDA_HandleTypeDef *hirda);
      • +
    • +
  • +
  • HAL SMARTCARD update

    +
      +
    • Add specific macros to manage the flags cleared only by a software sequence +
        +
      • __HAL_SMARTCARD_CLEAR_PEFLAG()
      • +
      • __HAL_SMARTCARD_CLEAR_FEFLAG()
      • +
      • __HAL_SMARTCARD_CLEAR_NEFLAG()
      • +
      • __HAL_SMARTCARD_CLEAR_OREFLAG()
      • +
      • __HAL_SMARTCARD_CLEAR_IDLEFLAG()
      • +
    • +
    • Add several enhancements without affecting the driver functionalities +
        +
      • Add a new state HAL_SMARTCARD_STATE_BUSY_TX_RX and all processes has been updated accordingly
      • +
      • Update HAL_SMARTCARD_Transmit_IT() to enable SMARTCARD_IT_TXE instead of SMARTCARD_IT_TC
      • +
    • +
  • +
  • HAL SPI update +
      +
    • Bugs fix +
        +
      • SPI interface is used in synchronous polling mode: at high clock rates like SPI prescaler 2 and 4, calling HAL_SPI_TransmitReceive() returns with error HAL_TIMEOUT
      • +
      • HAL_SPI_TransmitReceive_DMA() does not clean up the TX DMA, so any subsequent SPI calls return the DMA error
      • +
      • HAL_SPI_Transmit_DMA() is failing when data size is equal to 1 byte
      • +
    • +
    • Add the following APIs used within the DMA process +
        +
      • HAL_StatusTypeDef HAL_SPI_DMAPause(SPI_HandleTypeDef *hspi);
      • +
      • HAL_StatusTypeDef HAL_SPI_DMAResume(SPI_HandleTypeDef *hspi);
      • +
      • HAL_StatusTypeDef HAL_SPI_DMAStop(SPI_HandleTypeDef *hspi);
      • +
      • void HAL_SPI_TxHalfCpltCallback(SPI_HandleTypeDef *hspi);
      • +
      • void HAL_SPI_RxHalfCpltCallback(SPI_HandleTypeDef *hspi);
      • +
      • void HAL_SPI_TxRxHalfCpltCallback(SPI_HandleTypeDef *hspi);
      • +
    • +
  • +
  • HAL RNG update +
      +
    • Add a conditional define to make this driver visible for all STM32F4xx devices except STM32F401xx and STM32F411xx Devices.
    • +
  • +
  • HAL CRC update +
      +
    • These macros are added to read/write the CRC IDR register: __HAL_CRC_SET_IDR() and __HAL_CRC_GET_IDR()
    • +
  • +
  • HAL DAC update +
      +
    • Enhance the DMA channel configuration when used with DAC
    • +
  • +
  • HAL TIM update +
      +
    • HAL_TIM_IRQHandler(): update to check the input capture channel 3 and 4 in CCMR2 instead of CCMR1
    • +
    • __HAL_TIM_PRESCALER() updated to use ‘=’ instead of ‘|=’
    • +
    • Add the following macro in TIM HAL driver +
        +
      • __HAL_TIM_GetCompare()
      • +
      • __HAL_TIM_GetCounter()
      • +
      • __HAL_TIM_GetAutoreload()
      • +
      • __HAL_TIM_GetClockDivision()
      • +
      • __HAL_TIM_GetICPrescaler()
      • +
    • +
  • +
  • HAL SDMMC update +
      +
    • Use of CMSIS constants instead of magic values
    • +
    • Miscellaneous update in functions internal coding
    • +
  • +
  • HAL NAND update +
      +
    • Fix issue of macros returning wrong address for NAND blocks
    • +
    • Fix issue for read/write NAND page/spare area
    • +
  • +
  • HAL NOR update +
      +
    • Add the NOR address bank macro used within the API
    • +
    • Update NOR API implementation to avoid the use of NOR address bank hard coded
    • +
  • +
  • HAL HCD update +
      +
    • HCD_StateTypeDef structure members renamed
    • +
    • These macro are renamed +
        +
      • __HAL_GET_FLAG(__HANDLE__, __INTERRUPT__) by __HAL_HCD_GET_FLAG(__HANDLE__, __INTERRUPT__)
      • +
      • __HAL_CLEAR_FLAG(__HANDLE__, __INTERRUPT__) by __HAL_HCD_CLEAR_FLAG(__HANDLE__, __INTERRUPT__)
      • +
      • __HAL_IS_INVALID_INTERRUPT(__HANDLE__) by __HAL_HCD_IS_INVALID_INTERRUPT(__HANDLE__)
      • +
    • +
  • +
  • HAL PCD update +
      +
    • HAL_PCD_SetTxFiFo() and HAL_PCD_SetRxFiFo() renamed into HAL_PCDEx_SetTxFiFo() and HAL_PCDEx_SetRxFiFo() and moved to the extension files stm32f4xx_hal_pcd_ex.h/.c
    • +
    • PCD_StateTypeDef structure members renamed
    • +
    • Fix incorrect masking of TxFIFOEmpty
    • +
    • stm32f4xx_ll_usb.c: fix issue in HS mode
    • +
    • New macros added +
        +
      • __HAL_PCD_IS_PHY_SUSPENDED()
      • +
      • __HAL_USB_HS_EXTI_GENERATE_SWIT()
      • +
      • __HAL_USB_FS_EXTI_GENERATE_SWIT()
      • +
    • +
    • These macro are renamed +
        +
      • __HAL_GET_FLAG(__HANDLE__, __INTERRUPT__) by __HAL_PCD_GET_FLAG(__HANDLE__, __INTERRUPT__)
      • +
      • __HAL_CLEAR_FLAG(__HANDLE__, __INTERRUPT__) by __HAL_PCD_CLEAR_FLAG(__HANDLE__, __INTERRUPT__)
      • +
      • __HAL_IS_INVALID_INTERRUPT(__HANDLE__) by __HAL_PCD_IS_INVALID_INTERRUPT(__HANDLE__)
      • +
      • __HAL_PCD_UNGATE_CLOCK(__HANDLE__) by __HAL_PCD_UNGATE_PHYCLOCK(__HANDLE__)
      • +
      • __HAL_PCD_GATE_CLOCK(__HANDLE___) by __HAL_PCD_GATE_PHYCLOCK(__HANDLE__)
      • +
    • +
  • +
  • HAL ETH update +
      +
    • Update HAL_ETH_GetReceivedFrame_IT() function to return HAL_ERROR if the received packet is not complete
    • +
    • Use HAL_Delay() instead of counting loop
    • +
    • __HAL_ETH_MAC_CLEAR_FLAG() macro is removed: the MACSR register is read only
    • +
    • Add the following macros used to Wake up the device from STOP mode by Ethernet event : +
        +
      • __HAL_ETH_EXTI_ENABLE_IT()
      • +
      • __HAL_ETH_EXTI_DISABLE_IT()
      • +
      • __HAL_ETH_EXTI_GET_FLAG()
      • +
      • __HAL_ETH_EXTI_CLEAR_FLAG()
      • +
      • __HAL_ETH_EXTI_SET_RISING_EGDE_TRIGGER()
      • +
      • __HAL_ETH_EXTI_SET_FALLING_EGDE_TRIGGER()
      • +
      • __HAL_ETH_EXTI_SET_FALLINGRISING_TRIGGER()
      • +
    • +
  • +
  • HAL WWDG update +
      +
    • Update macro parameters to use underscore: __XXX__
    • +
    • Use of CMSIS constants instead of magic values
    • +
    • Use MODIFY_REG macro in HAL_WWDG_Init()
    • +
    • Add IS_WWDG_ALL_INSTANCE in HAL_WWDG_Init() and HAL_WWDG_DeInit()
    • +
  • +
  • HAL IWDG update +
      +
    • Use WRITE_REG instead of SET_BIT for all IWDG macros
    • +
    • __HAL_IWDG_CLEAR_FLAG removed: no IWDG flag cleared by access to SR register
    • +
    • Use MODIFY_REG macro in HAL_IWDG_Init()
    • +
    • Add IS_IWDG_ALL_INSTANCE in HAL_IWDG_Init()Add the following macros used to Wake
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • First official release
  • +
+
+
+
+ +
+

For complete documentation on STM32F4xx, visit: [www.st.com/stm32f4]

+

This release note uses up to date web standards and, for this reason, should not be opened with Internet Explorer but preferably with popular browsers such as Google Chrome, Mozilla Firefox, Opera or Microsoft Edge.

+
+ + diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal.c index fb7811dd..de14a0e3 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal.c @@ -50,11 +50,11 @@ * @{ */ /** - * @brief STM32F4xx HAL Driver version number V1.8.1 + * @brief STM32F4xx HAL Driver version number V1.8.2 */ #define __STM32F4xx_HAL_VERSION_MAIN (0x01U) /*!< [31:24] main version */ #define __STM32F4xx_HAL_VERSION_SUB1 (0x08U) /*!< [23:16] sub1 version */ -#define __STM32F4xx_HAL_VERSION_SUB2 (0x01U) /*!< [15:8] sub2 version */ +#define __STM32F4xx_HAL_VERSION_SUB2 (0x02U) /*!< [15:8] sub2 version */ #define __STM32F4xx_HAL_VERSION_RC (0x00U) /*!< [7:0] release candidate */ #define __STM32F4xx_HAL_VERSION ((__STM32F4xx_HAL_VERSION_MAIN << 24U)\ |(__STM32F4xx_HAL_VERSION_SUB1 << 16U)\ @@ -368,7 +368,8 @@ HAL_StatusTypeDef HAL_SetTickFreq(HAL_TickFreqTypeDef Freq) /** * @brief Return tick frequency. - * @retval tick period in Hz + * @retval Tick frequency. + * Value of @ref HAL_TickFreqTypeDef. */ HAL_TickFreqTypeDef HAL_GetTickFreq(void) { diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_adc.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_adc.c index 128b2360..9ad943d8 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_adc.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_adc.c @@ -266,7 +266,7 @@ * @{ */ /* Private function prototypes -----------------------------------------------*/ -static void ADC_Init(ADC_HandleTypeDef* hadc); +static void ADC_Init(ADC_HandleTypeDef *hadc); static void ADC_DMAConvCplt(DMA_HandleTypeDef *hdma); static void ADC_DMAError(DMA_HandleTypeDef *hdma); static void ADC_DMAHalfConvCplt(DMA_HandleTypeDef *hdma); @@ -308,12 +308,12 @@ static void ADC_DMAHalfConvCplt(DMA_HandleTypeDef *hdma); * the configuration information for the specified ADC. * @retval HAL status */ -HAL_StatusTypeDef HAL_ADC_Init(ADC_HandleTypeDef* hadc) +HAL_StatusTypeDef HAL_ADC_Init(ADC_HandleTypeDef *hadc) { HAL_StatusTypeDef tmp_hal_status = HAL_OK; /* Check ADC handle */ - if(hadc == NULL) + if (hadc == NULL) { return HAL_ERROR; } @@ -331,12 +331,12 @@ HAL_StatusTypeDef HAL_ADC_Init(ADC_HandleTypeDef* hadc) assert_param(IS_ADC_EOCSelection(hadc->Init.EOCSelection)); assert_param(IS_FUNCTIONAL_STATE(hadc->Init.DiscontinuousConvMode)); - if(hadc->Init.ExternalTrigConv != ADC_SOFTWARE_START) + if (hadc->Init.ExternalTrigConv != ADC_SOFTWARE_START) { assert_param(IS_ADC_EXT_TRIG_EDGE(hadc->Init.ExternalTrigConvEdge)); } - if(hadc->State == HAL_ADC_STATE_RESET) + if (hadc->State == HAL_ADC_STATE_RESET) { #if (USE_HAL_ADC_REGISTER_CALLBACKS == 1) /* Init the ADC Callback settings */ @@ -402,12 +402,12 @@ HAL_StatusTypeDef HAL_ADC_Init(ADC_HandleTypeDef* hadc) * the configuration information for the specified ADC. * @retval HAL status */ -HAL_StatusTypeDef HAL_ADC_DeInit(ADC_HandleTypeDef* hadc) +HAL_StatusTypeDef HAL_ADC_DeInit(ADC_HandleTypeDef *hadc) { HAL_StatusTypeDef tmp_hal_status = HAL_OK; /* Check ADC handle */ - if(hadc == NULL) + if (hadc == NULL) { return HAL_ERROR; } @@ -424,19 +424,19 @@ HAL_StatusTypeDef HAL_ADC_DeInit(ADC_HandleTypeDef* hadc) /* Configuration of ADC parameters if previous preliminary actions are */ /* correctly completed. */ - if(HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_ADON)) + if (HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_ADON)) { #if (USE_HAL_ADC_REGISTER_CALLBACKS == 1) - if (hadc->MspDeInitCallback == NULL) - { - hadc->MspDeInitCallback = HAL_ADC_MspDeInit; /* Legacy weak MspDeInit */ - } + if (hadc->MspDeInitCallback == NULL) + { + hadc->MspDeInitCallback = HAL_ADC_MspDeInit; /* Legacy weak MspDeInit */ + } - /* DeInit the low level hardware: RCC clock, NVIC */ - hadc->MspDeInitCallback(hadc); + /* DeInit the low level hardware: RCC clock, NVIC */ + hadc->MspDeInitCallback(hadc); #else - /* DeInit the low level hardware: RCC clock, NVIC */ - HAL_ADC_MspDeInit(hadc); + /* DeInit the low level hardware: RCC clock, NVIC */ + HAL_ADC_MspDeInit(hadc); #endif /* USE_HAL_ADC_REGISTER_CALLBACKS */ /* Set ADC error code to none */ @@ -659,7 +659,7 @@ HAL_StatusTypeDef HAL_ADC_UnRegisterCallback(ADC_HandleTypeDef *hadc, HAL_ADC_Ca * the configuration information for the specified ADC. * @retval None */ -__weak void HAL_ADC_MspInit(ADC_HandleTypeDef* hadc) +__weak void HAL_ADC_MspInit(ADC_HandleTypeDef *hadc) { /* Prevent unused argument(s) compilation warning */ UNUSED(hadc); @@ -674,7 +674,7 @@ __weak void HAL_ADC_MspInit(ADC_HandleTypeDef* hadc) * the configuration information for the specified ADC. * @retval None */ -__weak void HAL_ADC_MspDeInit(ADC_HandleTypeDef* hadc) +__weak void HAL_ADC_MspDeInit(ADC_HandleTypeDef *hadc) { /* Prevent unused argument(s) compilation warning */ UNUSED(hadc); @@ -713,7 +713,7 @@ __weak void HAL_ADC_MspDeInit(ADC_HandleTypeDef* hadc) * the configuration information for the specified ADC. * @retval HAL status */ -HAL_StatusTypeDef HAL_ADC_Start(ADC_HandleTypeDef* hadc) +HAL_StatusTypeDef HAL_ADC_Start(ADC_HandleTypeDef *hadc) { __IO uint32_t counter = 0U; ADC_Common_TypeDef *tmpADC_Common; @@ -728,7 +728,7 @@ HAL_StatusTypeDef HAL_ADC_Start(ADC_HandleTypeDef* hadc) /* Enable the ADC peripheral */ /* Check if ADC peripheral is disabled in order to enable it and wait during Tstab time the ADC's stabilization */ - if((hadc->Instance->CR2 & ADC_CR2_ADON) != ADC_CR2_ADON) + if ((hadc->Instance->CR2 & ADC_CR2_ADON) != ADC_CR2_ADON) { /* Enable the Peripheral */ __HAL_ADC_ENABLE(hadc); @@ -736,14 +736,14 @@ HAL_StatusTypeDef HAL_ADC_Start(ADC_HandleTypeDef* hadc) /* Delay for ADC stabilization time */ /* Compute number of CPU cycles to wait for */ counter = (ADC_STAB_DELAY_US * (SystemCoreClock / 1000000U)); - while(counter != 0U) + while (counter != 0U) { counter--; } } /* Start conversion if ADC is effectively enabled */ - if(HAL_IS_BIT_SET(hadc->Instance->CR2, ADC_CR2_ADON)) + if (HAL_IS_BIT_SET(hadc->Instance->CR2, ADC_CR2_ADON)) { /* Set ADC state */ /* - Clear state bitfield related to regular group conversion results */ @@ -786,15 +786,15 @@ HAL_StatusTypeDef HAL_ADC_Start(ADC_HandleTypeDef* hadc) __HAL_ADC_CLEAR_FLAG(hadc, ADC_FLAG_EOC | ADC_FLAG_OVR); /* Check if Multimode enabled */ - if(HAL_IS_BIT_CLR(tmpADC_Common->CCR, ADC_CCR_MULTI)) + if (HAL_IS_BIT_CLR(tmpADC_Common->CCR, ADC_CCR_MULTI)) { #if defined(ADC2) && defined(ADC3) - if((hadc->Instance == ADC1) || ((hadc->Instance == ADC2) && ((ADC->CCR & ADC_CCR_MULTI_Msk) < ADC_CCR_MULTI_0)) \ - || ((hadc->Instance == ADC3) && ((ADC->CCR & ADC_CCR_MULTI_Msk) < ADC_CCR_MULTI_4))) + if ((hadc->Instance == ADC1) || ((hadc->Instance == ADC2) && ((ADC->CCR & ADC_CCR_MULTI_Msk) < ADC_CCR_MULTI_0)) \ + || ((hadc->Instance == ADC3) && ((ADC->CCR & ADC_CCR_MULTI_Msk) < ADC_CCR_MULTI_4))) { #endif /* ADC2 || ADC3 */ /* if no external trigger present enable software conversion of regular channels */ - if((hadc->Instance->CR2 & ADC_CR2_EXTEN) == RESET) + if ((hadc->Instance->CR2 & ADC_CR2_EXTEN) == RESET) { /* Enable the selected ADC software conversion for regular group */ hadc->Instance->CR2 |= (uint32_t)ADC_CR2_SWSTART; @@ -806,10 +806,10 @@ HAL_StatusTypeDef HAL_ADC_Start(ADC_HandleTypeDef* hadc) else { /* if instance of handle correspond to ADC1 and no external trigger present enable software conversion of regular channels */ - if((hadc->Instance == ADC1) && ((hadc->Instance->CR2 & ADC_CR2_EXTEN) == RESET)) + if ((hadc->Instance == ADC1) && ((hadc->Instance->CR2 & ADC_CR2_EXTEN) == RESET)) { /* Enable the selected ADC software conversion for regular group */ - hadc->Instance->CR2 |= (uint32_t)ADC_CR2_SWSTART; + hadc->Instance->CR2 |= (uint32_t)ADC_CR2_SWSTART; } } } @@ -836,7 +836,7 @@ HAL_StatusTypeDef HAL_ADC_Start(ADC_HandleTypeDef* hadc) * * @retval HAL status. */ -HAL_StatusTypeDef HAL_ADC_Stop(ADC_HandleTypeDef* hadc) +HAL_StatusTypeDef HAL_ADC_Stop(ADC_HandleTypeDef *hadc) { /* Check the parameters */ assert_param(IS_ADC_ALL_INSTANCE(hadc->Instance)); @@ -849,7 +849,7 @@ HAL_StatusTypeDef HAL_ADC_Stop(ADC_HandleTypeDef* hadc) __HAL_ADC_DISABLE(hadc); /* Check if ADC is effectively disabled */ - if(HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_ADON)) + if (HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_ADON)) { /* Set ADC state */ ADC_STATE_CLR_SET(hadc->State, @@ -879,7 +879,7 @@ HAL_StatusTypeDef HAL_ADC_Stop(ADC_HandleTypeDef* hadc) * @param Timeout Timeout value in millisecond. * @retval HAL status */ -HAL_StatusTypeDef HAL_ADC_PollForConversion(ADC_HandleTypeDef* hadc, uint32_t Timeout) +HAL_StatusTypeDef HAL_ADC_PollForConversion(ADC_HandleTypeDef *hadc, uint32_t Timeout) { uint32_t tickstart = 0U; @@ -890,7 +890,7 @@ HAL_StatusTypeDef HAL_ADC_PollForConversion(ADC_HandleTypeDef* hadc, uint32_t Ti /* For code simplicity sake, this particular case is generalized to */ /* ADC configured in DMA mode and polling for end of each conversion. */ if (HAL_IS_BIT_SET(hadc->Instance->CR2, ADC_CR2_EOCS) && - HAL_IS_BIT_SET(hadc->Instance->CR2, ADC_CR2_DMA) ) + HAL_IS_BIT_SET(hadc->Instance->CR2, ADC_CR2_DMA)) { /* Update ADC state machine to error */ SET_BIT(hadc->State, HAL_ADC_STATE_ERROR_CONFIG); @@ -905,15 +905,15 @@ HAL_StatusTypeDef HAL_ADC_PollForConversion(ADC_HandleTypeDef* hadc, uint32_t Ti tickstart = HAL_GetTick(); /* Check End of conversion flag */ - while(!(__HAL_ADC_GET_FLAG(hadc, ADC_FLAG_EOC))) + while (!(__HAL_ADC_GET_FLAG(hadc, ADC_FLAG_EOC))) { /* Check if timeout is disabled (set to infinite wait) */ - if(Timeout != HAL_MAX_DELAY) + if (Timeout != HAL_MAX_DELAY) { - if((Timeout == 0U) || ((HAL_GetTick() - tickstart ) > Timeout)) + if ((Timeout == 0U) || ((HAL_GetTick() - tickstart) > Timeout)) { /* New check to avoid false timeout detection in case of preemption */ - if(!(__HAL_ADC_GET_FLAG(hadc, ADC_FLAG_EOC))) + if (!(__HAL_ADC_GET_FLAG(hadc, ADC_FLAG_EOC))) { /* Update ADC state machine to timeout */ SET_BIT(hadc->State, HAL_ADC_STATE_TIMEOUT); @@ -939,10 +939,10 @@ HAL_StatusTypeDef HAL_ADC_PollForConversion(ADC_HandleTypeDef* hadc, uint32_t Ti /* The test of scan sequence on going is done either with scan */ /* sequence disabled or with end of conversion flag set to */ /* of end of sequence. */ - if(ADC_IS_SOFTWARE_START_REGULAR(hadc) && - (hadc->Init.ContinuousConvMode == DISABLE) && - (HAL_IS_BIT_CLR(hadc->Instance->SQR1, ADC_SQR1_L) || - HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_EOCS) ) ) + if (ADC_IS_SOFTWARE_START_REGULAR(hadc) && + (hadc->Init.ContinuousConvMode == DISABLE) && + (HAL_IS_BIT_CLR(hadc->Instance->SQR1, ADC_SQR1_L) || + HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_EOCS))) { /* Set ADC state */ CLEAR_BIT(hadc->State, HAL_ADC_STATE_REG_BUSY); @@ -968,7 +968,7 @@ HAL_StatusTypeDef HAL_ADC_PollForConversion(ADC_HandleTypeDef* hadc, uint32_t Ti * @param Timeout Timeout value in millisecond. * @retval HAL status */ -HAL_StatusTypeDef HAL_ADC_PollForEvent(ADC_HandleTypeDef* hadc, uint32_t EventType, uint32_t Timeout) +HAL_StatusTypeDef HAL_ADC_PollForEvent(ADC_HandleTypeDef *hadc, uint32_t EventType, uint32_t Timeout) { uint32_t tickstart = 0U; @@ -980,15 +980,15 @@ HAL_StatusTypeDef HAL_ADC_PollForEvent(ADC_HandleTypeDef* hadc, uint32_t EventTy tickstart = HAL_GetTick(); /* Check selected event flag */ - while(!(__HAL_ADC_GET_FLAG(hadc,EventType))) + while (!(__HAL_ADC_GET_FLAG(hadc, EventType))) { /* Check for the Timeout */ - if(Timeout != HAL_MAX_DELAY) + if (Timeout != HAL_MAX_DELAY) { - if((Timeout == 0U) || ((HAL_GetTick() - tickstart ) > Timeout)) + if ((Timeout == 0U) || ((HAL_GetTick() - tickstart) > Timeout)) { /* New check to avoid false timeout detection in case of preemption */ - if(!(__HAL_ADC_GET_FLAG(hadc,EventType))) + if (!(__HAL_ADC_GET_FLAG(hadc, EventType))) { /* Update ADC state machine to timeout */ SET_BIT(hadc->State, HAL_ADC_STATE_TIMEOUT); @@ -1003,7 +1003,7 @@ HAL_StatusTypeDef HAL_ADC_PollForEvent(ADC_HandleTypeDef* hadc, uint32_t EventTy } /* Analog watchdog (level out of window) event */ - if(EventType == ADC_AWD_EVENT) + if (EventType == ADC_AWD_EVENT) { /* Set ADC state */ SET_BIT(hadc->State, HAL_ADC_STATE_AWD1); @@ -1034,7 +1034,7 @@ HAL_StatusTypeDef HAL_ADC_PollForEvent(ADC_HandleTypeDef* hadc, uint32_t EventTy * the configuration information for the specified ADC. * @retval HAL status. */ -HAL_StatusTypeDef HAL_ADC_Start_IT(ADC_HandleTypeDef* hadc) +HAL_StatusTypeDef HAL_ADC_Start_IT(ADC_HandleTypeDef *hadc) { __IO uint32_t counter = 0U; ADC_Common_TypeDef *tmpADC_Common; @@ -1049,7 +1049,7 @@ HAL_StatusTypeDef HAL_ADC_Start_IT(ADC_HandleTypeDef* hadc) /* Enable the ADC peripheral */ /* Check if ADC peripheral is disabled in order to enable it and wait during Tstab time the ADC's stabilization */ - if((hadc->Instance->CR2 & ADC_CR2_ADON) != ADC_CR2_ADON) + if ((hadc->Instance->CR2 & ADC_CR2_ADON) != ADC_CR2_ADON) { /* Enable the Peripheral */ __HAL_ADC_ENABLE(hadc); @@ -1057,14 +1057,14 @@ HAL_StatusTypeDef HAL_ADC_Start_IT(ADC_HandleTypeDef* hadc) /* Delay for ADC stabilization time */ /* Compute number of CPU cycles to wait for */ counter = (ADC_STAB_DELAY_US * (SystemCoreClock / 1000000U)); - while(counter != 0U) + while (counter != 0U) { counter--; } } /* Start conversion if ADC is effectively enabled */ - if(HAL_IS_BIT_SET(hadc->Instance->CR2, ADC_CR2_ADON)) + if (HAL_IS_BIT_SET(hadc->Instance->CR2, ADC_CR2_ADON)) { /* Set ADC state */ /* - Clear state bitfield related to regular group conversion results */ @@ -1110,15 +1110,15 @@ HAL_StatusTypeDef HAL_ADC_Start_IT(ADC_HandleTypeDef* hadc) __HAL_ADC_ENABLE_IT(hadc, (ADC_IT_EOC | ADC_IT_OVR)); /* Check if Multimode enabled */ - if(HAL_IS_BIT_CLR(tmpADC_Common->CCR, ADC_CCR_MULTI)) + if (HAL_IS_BIT_CLR(tmpADC_Common->CCR, ADC_CCR_MULTI)) { #if defined(ADC2) && defined(ADC3) - if((hadc->Instance == ADC1) || ((hadc->Instance == ADC2) && ((ADC->CCR & ADC_CCR_MULTI_Msk) < ADC_CCR_MULTI_0)) \ - || ((hadc->Instance == ADC3) && ((ADC->CCR & ADC_CCR_MULTI_Msk) < ADC_CCR_MULTI_4))) + if ((hadc->Instance == ADC1) || ((hadc->Instance == ADC2) && ((ADC->CCR & ADC_CCR_MULTI_Msk) < ADC_CCR_MULTI_0)) \ + || ((hadc->Instance == ADC3) && ((ADC->CCR & ADC_CCR_MULTI_Msk) < ADC_CCR_MULTI_4))) { #endif /* ADC2 || ADC3 */ /* if no external trigger present enable software conversion of regular channels */ - if((hadc->Instance->CR2 & ADC_CR2_EXTEN) == RESET) + if ((hadc->Instance->CR2 & ADC_CR2_EXTEN) == RESET) { /* Enable the selected ADC software conversion for regular group */ hadc->Instance->CR2 |= (uint32_t)ADC_CR2_SWSTART; @@ -1130,10 +1130,10 @@ HAL_StatusTypeDef HAL_ADC_Start_IT(ADC_HandleTypeDef* hadc) else { /* if instance of handle correspond to ADC1 and no external trigger present enable software conversion of regular channels */ - if((hadc->Instance == ADC1) && ((hadc->Instance->CR2 & ADC_CR2_EXTEN) == RESET)) + if ((hadc->Instance == ADC1) && ((hadc->Instance->CR2 & ADC_CR2_EXTEN) == RESET)) { /* Enable the selected ADC software conversion for regular group */ - hadc->Instance->CR2 |= (uint32_t)ADC_CR2_SWSTART; + hadc->Instance->CR2 |= (uint32_t)ADC_CR2_SWSTART; } } } @@ -1159,7 +1159,7 @@ HAL_StatusTypeDef HAL_ADC_Start_IT(ADC_HandleTypeDef* hadc) * the configuration information for the specified ADC. * @retval HAL status. */ -HAL_StatusTypeDef HAL_ADC_Stop_IT(ADC_HandleTypeDef* hadc) +HAL_StatusTypeDef HAL_ADC_Stop_IT(ADC_HandleTypeDef *hadc) { /* Check the parameters */ assert_param(IS_ADC_ALL_INSTANCE(hadc->Instance)); @@ -1172,9 +1172,9 @@ HAL_StatusTypeDef HAL_ADC_Stop_IT(ADC_HandleTypeDef* hadc) __HAL_ADC_DISABLE(hadc); /* Check if ADC is effectively disabled */ - if(HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_ADON)) + if (HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_ADON)) { - /* Disable ADC end of conversion interrupt for regular group */ + /* Disable ADC end of conversion interrupt for regular group */ __HAL_ADC_DISABLE_IT(hadc, (ADC_IT_EOC | ADC_IT_OVR)); /* Set ADC state */ @@ -1196,7 +1196,7 @@ HAL_StatusTypeDef HAL_ADC_Stop_IT(ADC_HandleTypeDef* hadc) * the configuration information for the specified ADC. * @retval None */ -void HAL_ADC_IRQHandler(ADC_HandleTypeDef* hadc) +void HAL_ADC_IRQHandler(ADC_HandleTypeDef *hadc) { uint32_t tmp1 = 0U, tmp2 = 0U; @@ -1211,7 +1211,7 @@ void HAL_ADC_IRQHandler(ADC_HandleTypeDef* hadc) tmp1 = tmp_sr & ADC_FLAG_EOC; tmp2 = tmp_cr1 & ADC_IT_EOC; /* Check End of conversion flag for regular channels */ - if(tmp1 && tmp2) + if (tmp1 && tmp2) { /* Update state machine on conversion status if not in error state */ if (HAL_IS_BIT_CLR(hadc->State, HAL_ADC_STATE_ERROR_INTERNAL)) @@ -1226,10 +1226,10 @@ void HAL_ADC_IRQHandler(ADC_HandleTypeDef* hadc) /* The test of scan sequence on going is done either with scan */ /* sequence disabled or with end of conversion flag set to */ /* of end of sequence. */ - if(ADC_IS_SOFTWARE_START_REGULAR(hadc) && - (hadc->Init.ContinuousConvMode == DISABLE) && - (HAL_IS_BIT_CLR(hadc->Instance->SQR1, ADC_SQR1_L) || - HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_EOCS) ) ) + if (ADC_IS_SOFTWARE_START_REGULAR(hadc) && + (hadc->Init.ContinuousConvMode == DISABLE) && + (HAL_IS_BIT_CLR(hadc->Instance->SQR1, ADC_SQR1_L) || + HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_EOCS))) { /* Disable ADC end of single conversion interrupt on group regular */ /* Note: Overrun interrupt was enabled with EOC interrupt in */ @@ -1260,7 +1260,7 @@ void HAL_ADC_IRQHandler(ADC_HandleTypeDef* hadc) tmp1 = tmp_sr & ADC_FLAG_JEOC; tmp2 = tmp_cr1 & ADC_IT_JEOC; /* Check End of conversion flag for injected channels */ - if(tmp1 && tmp2) + if (tmp1 && tmp2) { /* Update state machine on conversion status if not in error state */ if (HAL_IS_BIT_CLR(hadc->State, HAL_ADC_STATE_ERROR_INTERNAL)) @@ -1273,12 +1273,12 @@ void HAL_ADC_IRQHandler(ADC_HandleTypeDef* hadc) /* by external trigger, scan sequence on going or by automatic injected */ /* conversion from group regular (same conditions as group regular */ /* interruption disabling above). */ - if(ADC_IS_SOFTWARE_START_INJECTED(hadc) && - (HAL_IS_BIT_CLR(hadc->Instance->JSQR, ADC_JSQR_JL) || - HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_EOCS) ) && - (HAL_IS_BIT_CLR(hadc->Instance->CR1, ADC_CR1_JAUTO) && - (ADC_IS_SOFTWARE_START_REGULAR(hadc) && - (hadc->Init.ContinuousConvMode == DISABLE) ) ) ) + if (ADC_IS_SOFTWARE_START_INJECTED(hadc) && + (HAL_IS_BIT_CLR(hadc->Instance->JSQR, ADC_JSQR_JL) || + HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_EOCS)) && + (HAL_IS_BIT_CLR(hadc->Instance->CR1, ADC_CR1_JAUTO) && + (ADC_IS_SOFTWARE_START_REGULAR(hadc) && + (hadc->Init.ContinuousConvMode == DISABLE)))) { /* Disable ADC end of single conversion interrupt on group injected */ __HAL_ADC_DISABLE_IT(hadc, ADC_IT_JEOC); @@ -1295,9 +1295,9 @@ void HAL_ADC_IRQHandler(ADC_HandleTypeDef* hadc) /* Conversion complete callback */ /* Conversion complete callback */ #if (USE_HAL_ADC_REGISTER_CALLBACKS == 1) - hadc->InjectedConvCpltCallback(hadc); + hadc->InjectedConvCpltCallback(hadc); #else - HAL_ADCEx_InjectedConvCpltCallback(hadc); + HAL_ADCEx_InjectedConvCpltCallback(hadc); #endif /* USE_HAL_ADC_REGISTER_CALLBACKS */ /* Clear injected group conversion flag */ @@ -1307,9 +1307,9 @@ void HAL_ADC_IRQHandler(ADC_HandleTypeDef* hadc) tmp1 = tmp_sr & ADC_FLAG_AWD; tmp2 = tmp_cr1 & ADC_IT_AWD; /* Check Analog watchdog flag */ - if(tmp1 && tmp2) + if (tmp1 && tmp2) { - if(__HAL_ADC_GET_FLAG(hadc, ADC_FLAG_AWD)) + if (__HAL_ADC_GET_FLAG(hadc, ADC_FLAG_AWD)) { /* Set ADC state */ SET_BIT(hadc->State, HAL_ADC_STATE_AWD1); @@ -1329,7 +1329,7 @@ void HAL_ADC_IRQHandler(ADC_HandleTypeDef* hadc) tmp1 = tmp_sr & ADC_FLAG_OVR; tmp2 = tmp_cr1 & ADC_IT_OVR; /* Check Overrun flag */ - if(tmp1 && tmp2) + if (tmp1 && tmp2) { /* Note: On STM32F4, ADC overrun can be set through other parameters */ /* refer to description of parameter "EOCSelection" for more */ @@ -1343,9 +1343,9 @@ void HAL_ADC_IRQHandler(ADC_HandleTypeDef* hadc) /* Error callback */ #if (USE_HAL_ADC_REGISTER_CALLBACKS == 1) - hadc->ErrorCallback(hadc); + hadc->ErrorCallback(hadc); #else - HAL_ADC_ErrorCallback(hadc); + HAL_ADC_ErrorCallback(hadc); #endif /* USE_HAL_ADC_REGISTER_CALLBACKS */ /* Clear the Overrun flag */ @@ -1361,7 +1361,7 @@ void HAL_ADC_IRQHandler(ADC_HandleTypeDef* hadc) * @param Length The length of data to be transferred from ADC peripheral to memory. * @retval HAL status */ -HAL_StatusTypeDef HAL_ADC_Start_DMA(ADC_HandleTypeDef* hadc, uint32_t* pData, uint32_t Length) +HAL_StatusTypeDef HAL_ADC_Start_DMA(ADC_HandleTypeDef *hadc, uint32_t *pData, uint32_t Length) { __IO uint32_t counter = 0U; ADC_Common_TypeDef *tmpADC_Common; @@ -1376,7 +1376,7 @@ HAL_StatusTypeDef HAL_ADC_Start_DMA(ADC_HandleTypeDef* hadc, uint32_t* pData, ui /* Enable the ADC peripheral */ /* Check if ADC peripheral is disabled in order to enable it and wait during Tstab time the ADC's stabilization */ - if((hadc->Instance->CR2 & ADC_CR2_ADON) != ADC_CR2_ADON) + if ((hadc->Instance->CR2 & ADC_CR2_ADON) != ADC_CR2_ADON) { /* Enable the Peripheral */ __HAL_ADC_ENABLE(hadc); @@ -1384,7 +1384,7 @@ HAL_StatusTypeDef HAL_ADC_Start_DMA(ADC_HandleTypeDef* hadc, uint32_t* pData, ui /* Delay for ADC stabilization time */ /* Compute number of CPU cycles to wait for */ counter = (ADC_STAB_DELAY_US * (SystemCoreClock / 1000000U)); - while(counter != 0U) + while (counter != 0U) { counter--; } @@ -1392,13 +1392,13 @@ HAL_StatusTypeDef HAL_ADC_Start_DMA(ADC_HandleTypeDef* hadc, uint32_t* pData, ui /* Check ADC DMA Mode */ /* - disable the DMA Mode if it is already enabled */ - if((hadc->Instance->CR2 & ADC_CR2_DMA) == ADC_CR2_DMA) + if ((hadc->Instance->CR2 & ADC_CR2_DMA) == ADC_CR2_DMA) { CLEAR_BIT(hadc->Instance->CR2, ADC_CR2_DMA); } /* Start conversion if ADC is effectively enabled */ - if(HAL_IS_BIT_SET(hadc->Instance->CR2, ADC_CR2_ADON)) + if (HAL_IS_BIT_SET(hadc->Instance->CR2, ADC_CR2_ADON)) { /* Set ADC state */ /* - Clear state bitfield related to regular group conversion results */ @@ -1463,15 +1463,15 @@ HAL_StatusTypeDef HAL_ADC_Start_DMA(ADC_HandleTypeDef* hadc, uint32_t* pData, ui HAL_DMA_Start_IT(hadc->DMA_Handle, (uint32_t)&hadc->Instance->DR, (uint32_t)pData, Length); /* Check if Multimode enabled */ - if(HAL_IS_BIT_CLR(tmpADC_Common->CCR, ADC_CCR_MULTI)) + if (HAL_IS_BIT_CLR(tmpADC_Common->CCR, ADC_CCR_MULTI)) { #if defined(ADC2) && defined(ADC3) - if((hadc->Instance == ADC1) || ((hadc->Instance == ADC2) && ((ADC->CCR & ADC_CCR_MULTI_Msk) < ADC_CCR_MULTI_0)) \ - || ((hadc->Instance == ADC3) && ((ADC->CCR & ADC_CCR_MULTI_Msk) < ADC_CCR_MULTI_4))) + if ((hadc->Instance == ADC1) || ((hadc->Instance == ADC2) && ((ADC->CCR & ADC_CCR_MULTI_Msk) < ADC_CCR_MULTI_0)) \ + || ((hadc->Instance == ADC3) && ((ADC->CCR & ADC_CCR_MULTI_Msk) < ADC_CCR_MULTI_4))) { #endif /* ADC2 || ADC3 */ /* if no external trigger present enable software conversion of regular channels */ - if((hadc->Instance->CR2 & ADC_CR2_EXTEN) == RESET) + if ((hadc->Instance->CR2 & ADC_CR2_EXTEN) == RESET) { /* Enable the selected ADC software conversion for regular group */ hadc->Instance->CR2 |= (uint32_t)ADC_CR2_SWSTART; @@ -1483,10 +1483,10 @@ HAL_StatusTypeDef HAL_ADC_Start_DMA(ADC_HandleTypeDef* hadc, uint32_t* pData, ui else { /* if instance of handle correspond to ADC1 and no external trigger present enable software conversion of regular channels */ - if((hadc->Instance == ADC1) && ((hadc->Instance->CR2 & ADC_CR2_EXTEN) == RESET)) + if ((hadc->Instance == ADC1) && ((hadc->Instance->CR2 & ADC_CR2_EXTEN) == RESET)) { /* Enable the selected ADC software conversion for regular group */ - hadc->Instance->CR2 |= (uint32_t)ADC_CR2_SWSTART; + hadc->Instance->CR2 |= (uint32_t)ADC_CR2_SWSTART; } } } @@ -1509,7 +1509,7 @@ HAL_StatusTypeDef HAL_ADC_Start_DMA(ADC_HandleTypeDef* hadc, uint32_t* pData, ui * the configuration information for the specified ADC. * @retval HAL status */ -HAL_StatusTypeDef HAL_ADC_Stop_DMA(ADC_HandleTypeDef* hadc) +HAL_StatusTypeDef HAL_ADC_Stop_DMA(ADC_HandleTypeDef *hadc) { HAL_StatusTypeDef tmp_hal_status = HAL_OK; @@ -1524,7 +1524,7 @@ HAL_StatusTypeDef HAL_ADC_Stop_DMA(ADC_HandleTypeDef* hadc) __HAL_ADC_DISABLE(hadc); /* Check if ADC is effectively disabled */ - if(HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_ADON)) + if (HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_ADON)) { /* Disable the selected ADC DMA mode */ hadc->Instance->CR2 &= ~ADC_CR2_DMA; @@ -1565,7 +1565,7 @@ HAL_StatusTypeDef HAL_ADC_Stop_DMA(ADC_HandleTypeDef* hadc) * the configuration information for the specified ADC. * @retval Converted value */ -uint32_t HAL_ADC_GetValue(ADC_HandleTypeDef* hadc) +uint32_t HAL_ADC_GetValue(ADC_HandleTypeDef *hadc) { /* Return the selected ADC converted value */ return hadc->Instance->DR; @@ -1577,7 +1577,7 @@ uint32_t HAL_ADC_GetValue(ADC_HandleTypeDef* hadc) * the configuration information for the specified ADC. * @retval None */ -__weak void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc) +__weak void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef *hadc) { /* Prevent unused argument(s) compilation warning */ UNUSED(hadc); @@ -1592,7 +1592,7 @@ __weak void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc) * the configuration information for the specified ADC. * @retval None */ -__weak void HAL_ADC_ConvHalfCpltCallback(ADC_HandleTypeDef* hadc) +__weak void HAL_ADC_ConvHalfCpltCallback(ADC_HandleTypeDef *hadc) { /* Prevent unused argument(s) compilation warning */ UNUSED(hadc); @@ -1607,7 +1607,7 @@ __weak void HAL_ADC_ConvHalfCpltCallback(ADC_HandleTypeDef* hadc) * the configuration information for the specified ADC. * @retval None */ -__weak void HAL_ADC_LevelOutOfWindowCallback(ADC_HandleTypeDef* hadc) +__weak void HAL_ADC_LevelOutOfWindowCallback(ADC_HandleTypeDef *hadc) { /* Prevent unused argument(s) compilation warning */ UNUSED(hadc); @@ -1642,7 +1642,7 @@ __weak void HAL_ADC_ErrorCallback(ADC_HandleTypeDef *hadc) */ /** @defgroup ADC_Exported_Functions_Group3 Peripheral Control functions - * @brief Peripheral Control functions + * @brief Peripheral Control functions * @verbatim =============================================================================== @@ -1658,15 +1658,15 @@ __weak void HAL_ADC_ErrorCallback(ADC_HandleTypeDef *hadc) * @{ */ - /** - * @brief Configures for the selected ADC regular channel its corresponding - * rank in the sequencer and its sample time. - * @param hadc pointer to a ADC_HandleTypeDef structure that contains - * the configuration information for the specified ADC. - * @param sConfig ADC configuration structure. - * @retval HAL status - */ -HAL_StatusTypeDef HAL_ADC_ConfigChannel(ADC_HandleTypeDef* hadc, ADC_ChannelConfTypeDef* sConfig) +/** +* @brief Configures for the selected ADC regular channel its corresponding +* rank in the sequencer and its sample time. +* @param hadc pointer to a ADC_HandleTypeDef structure that contains +* the configuration information for the specified ADC. +* @param sConfig ADC configuration structure. +* @retval HAL status +*/ +HAL_StatusTypeDef HAL_ADC_ConfigChannel(ADC_HandleTypeDef *hadc, ADC_ChannelConfTypeDef *sConfig) { __IO uint32_t counter = 0U; ADC_Common_TypeDef *tmpADC_Common; @@ -1725,10 +1725,10 @@ HAL_StatusTypeDef HAL_ADC_ConfigChannel(ADC_HandleTypeDef* hadc, ADC_ChannelConf hadc->Instance->SQR1 |= ADC_SQR1_RK(sConfig->Channel, sConfig->Rank); } - /* Pointer to the common control register to which is belonging hadc */ - /* (Depending on STM32F4 product, there may be up to 3 ADCs and 1 common */ - /* control register) */ - tmpADC_Common = ADC_COMMON_REGISTER(hadc); + /* Pointer to the common control register to which is belonging hadc */ + /* (Depending on STM32F4 product, there may be up to 3 ADCs and 1 common */ + /* control register) */ + tmpADC_Common = ADC_COMMON_REGISTER(hadc); /* if ADC1 Channel_18 is selected for VBAT Channel ennable VBATE */ if ((hadc->Instance == ADC1) && (sConfig->Channel == ADC_CHANNEL_VBAT)) @@ -1754,12 +1754,12 @@ HAL_StatusTypeDef HAL_ADC_ConfigChannel(ADC_HandleTypeDef* hadc, ADC_ChannelConf /* Enable the Temperature sensor and VREFINT channel*/ tmpADC_Common->CCR |= ADC_CCR_TSVREFE; - if(sConfig->Channel == ADC_CHANNEL_TEMPSENSOR) + if (sConfig->Channel == ADC_CHANNEL_TEMPSENSOR) { /* Delay for temperature sensor stabilization time */ /* Compute number of CPU cycles to wait for */ counter = (ADC_TEMPSENSOR_DELAY_US * (SystemCoreClock / 1000000U)); - while(counter != 0U) + while (counter != 0U) { counter--; } @@ -1789,7 +1789,7 @@ HAL_StatusTypeDef HAL_ADC_ConfigChannel(ADC_HandleTypeDef* hadc, ADC_ChannelConf * that contains the configuration information of ADC analog watchdog. * @retval HAL status */ -HAL_StatusTypeDef HAL_ADC_AnalogWDGConfig(ADC_HandleTypeDef* hadc, ADC_AnalogWDGConfTypeDef* AnalogWDGConfig) +HAL_StatusTypeDef HAL_ADC_AnalogWDGConfig(ADC_HandleTypeDef *hadc, ADC_AnalogWDGConfTypeDef *AnalogWDGConfig) { #ifdef USE_FULL_ASSERT uint32_t tmp = 0U; @@ -1809,7 +1809,7 @@ HAL_StatusTypeDef HAL_ADC_AnalogWDGConfig(ADC_HandleTypeDef* hadc, ADC_AnalogWDG /* Process locked */ __HAL_LOCK(hadc); - if(AnalogWDGConfig->ITMode == ENABLE) + if (AnalogWDGConfig->ITMode == ENABLE) { /* Enable the ADC Analog watchdog interrupt */ __HAL_ADC_ENABLE_IT(hadc, ADC_IT_AWD); @@ -1871,7 +1871,7 @@ HAL_StatusTypeDef HAL_ADC_AnalogWDGConfig(ADC_HandleTypeDef* hadc, ADC_AnalogWDG * the configuration information for the specified ADC. * @retval HAL state */ -uint32_t HAL_ADC_GetState(ADC_HandleTypeDef* hadc) +uint32_t HAL_ADC_GetState(ADC_HandleTypeDef *hadc) { /* Return ADC state */ return hadc->State; @@ -1903,7 +1903,7 @@ uint32_t HAL_ADC_GetError(ADC_HandleTypeDef *hadc) * the configuration information for the specified ADC. * @retval None */ -static void ADC_Init(ADC_HandleTypeDef* hadc) +static void ADC_Init(ADC_HandleTypeDef *hadc) { ADC_Common_TypeDef *tmpADC_Common; @@ -1934,7 +1934,7 @@ static void ADC_Init(ADC_HandleTypeDef* hadc) /* Note: This configuration keeps the hardware feature of parameter */ /* ExternalTrigConvEdge "trigger edge none" equivalent to */ /* software start. */ - if(hadc->Init.ExternalTrigConv != ADC_SOFTWARE_START) + if (hadc->Init.ExternalTrigConv != ADC_SOFTWARE_START) { /* Select external trigger to start conversion */ hadc->Instance->CR2 &= ~(ADC_CR2_EXTSEL); @@ -1955,7 +1955,7 @@ static void ADC_Init(ADC_HandleTypeDef* hadc) hadc->Instance->CR2 &= ~(ADC_CR2_CONT); hadc->Instance->CR2 |= ADC_CR2_CONTINUOUS((uint32_t)hadc->Init.ContinuousConvMode); - if(hadc->Init.DiscontinuousConvMode != DISABLE) + if (hadc->Init.DiscontinuousConvMode != DISABLE) { assert_param(IS_ADC_REGULAR_DISC_NUMBER(hadc->Init.NbrOfDiscConversion)); @@ -1994,7 +1994,7 @@ static void ADC_Init(ADC_HandleTypeDef* hadc) static void ADC_DMAConvCplt(DMA_HandleTypeDef *hdma) { /* Retrieve ADC handle corresponding to current DMA handle */ - ADC_HandleTypeDef* hadc = ( ADC_HandleTypeDef* )((DMA_HandleTypeDef* )hdma)->Parent; + ADC_HandleTypeDef *hadc = (ADC_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent; /* Update state machine on conversion status if not in error state */ if (HAL_IS_BIT_CLR(hadc->State, HAL_ADC_STATE_ERROR_INTERNAL | HAL_ADC_STATE_ERROR_DMA)) @@ -2008,10 +2008,10 @@ static void ADC_DMAConvCplt(DMA_HandleTypeDef *hdma) /* The test of scan sequence on going is done either with scan */ /* sequence disabled or with end of conversion flag set to */ /* of end of sequence. */ - if(ADC_IS_SOFTWARE_START_REGULAR(hadc) && - (hadc->Init.ContinuousConvMode == DISABLE) && - (HAL_IS_BIT_CLR(hadc->Instance->SQR1, ADC_SQR1_L) || - HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_EOCS) ) ) + if (ADC_IS_SOFTWARE_START_REGULAR(hadc) && + (hadc->Init.ContinuousConvMode == DISABLE) && + (HAL_IS_BIT_CLR(hadc->Instance->SQR1, ADC_SQR1_L) || + HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_EOCS))) { /* Disable ADC end of single conversion interrupt on group regular */ /* Note: Overrun interrupt was enabled with EOC interrupt in */ @@ -2046,8 +2046,8 @@ static void ADC_DMAConvCplt(DMA_HandleTypeDef *hdma) HAL_ADC_ErrorCallback(hadc); #endif /* USE_HAL_ADC_REGISTER_CALLBACKS */ } - else - { + else + { /* Call DMA error callback */ hadc->DMA_Handle->XferErrorCallback(hdma); } @@ -2062,8 +2062,8 @@ static void ADC_DMAConvCplt(DMA_HandleTypeDef *hdma) */ static void ADC_DMAHalfConvCplt(DMA_HandleTypeDef *hdma) { - ADC_HandleTypeDef* hadc = ( ADC_HandleTypeDef* )((DMA_HandleTypeDef* )hdma)->Parent; - /* Half conversion callback */ + ADC_HandleTypeDef *hadc = (ADC_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent; + /* Half conversion callback */ #if (USE_HAL_ADC_REGISTER_CALLBACKS == 1) hadc->ConvHalfCpltCallback(hadc); #else @@ -2079,11 +2079,11 @@ static void ADC_DMAHalfConvCplt(DMA_HandleTypeDef *hdma) */ static void ADC_DMAError(DMA_HandleTypeDef *hdma) { - ADC_HandleTypeDef* hadc = ( ADC_HandleTypeDef* )((DMA_HandleTypeDef* )hdma)->Parent; - hadc->State= HAL_ADC_STATE_ERROR_DMA; + ADC_HandleTypeDef *hadc = (ADC_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent; + hadc->State = HAL_ADC_STATE_ERROR_DMA; /* Set ADC error code to DMA error */ hadc->ErrorCode |= HAL_ADC_ERROR_DMA; - /* Error callback */ + /* Error callback */ #if (USE_HAL_ADC_REGISTER_CALLBACKS == 1) hadc->ErrorCallback(hadc); #else diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_adc_ex.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_adc_ex.c index cff0760c..7db19937 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_adc_ex.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_adc_ex.c @@ -143,7 +143,7 @@ static void ADC_MultiModeDMAHalfConvCplt(DMA_HandleTypeDef *hdma); * the configuration information for the specified ADC. * @retval HAL status */ -HAL_StatusTypeDef HAL_ADCEx_InjectedStart(ADC_HandleTypeDef* hadc) +HAL_StatusTypeDef HAL_ADCEx_InjectedStart(ADC_HandleTypeDef *hadc) { __IO uint32_t counter = 0U; uint32_t tmp1 = 0U, tmp2 = 0U; @@ -156,7 +156,7 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedStart(ADC_HandleTypeDef* hadc) /* Check if ADC peripheral is disabled in order to enable it and wait during Tstab time the ADC's stabilization */ - if((hadc->Instance->CR2 & ADC_CR2_ADON) != ADC_CR2_ADON) + if ((hadc->Instance->CR2 & ADC_CR2_ADON) != ADC_CR2_ADON) { /* Enable the Peripheral */ __HAL_ADC_ENABLE(hadc); @@ -164,14 +164,14 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedStart(ADC_HandleTypeDef* hadc) /* Delay for ADC stabilization time */ /* Compute number of CPU cycles to wait for */ counter = (ADC_STAB_DELAY_US * (SystemCoreClock / 1000000U)); - while(counter != 0U) + while (counter != 0U) { counter--; } } /* Start conversion if ADC is effectively enabled */ - if(HAL_IS_BIT_SET(hadc->Instance->CR2, ADC_CR2_ADON)) + if (HAL_IS_BIT_SET(hadc->Instance->CR2, ADC_CR2_ADON)) { /* Set ADC state */ /* - Clear state bitfield related to injected group conversion results */ @@ -205,11 +205,11 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedStart(ADC_HandleTypeDef* hadc) tmpADC_Common = ADC_COMMON_REGISTER(hadc); /* Check if Multimode enabled */ - if(HAL_IS_BIT_CLR(tmpADC_Common->CCR, ADC_CCR_MULTI)) + if (HAL_IS_BIT_CLR(tmpADC_Common->CCR, ADC_CCR_MULTI)) { tmp1 = HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_JEXTEN); tmp2 = HAL_IS_BIT_CLR(hadc->Instance->CR1, ADC_CR1_JAUTO); - if(tmp1 && tmp2) + if (tmp1 && tmp2) { /* Enable the selected ADC software conversion for injected group */ hadc->Instance->CR2 |= ADC_CR2_JSWSTART; @@ -219,7 +219,7 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedStart(ADC_HandleTypeDef* hadc) { tmp1 = HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_JEXTEN); tmp2 = HAL_IS_BIT_CLR(hadc->Instance->CR1, ADC_CR1_JAUTO); - if((hadc->Instance == ADC1) && tmp1 && tmp2) + if ((hadc->Instance == ADC1) && tmp1 && tmp2) { /* Enable the selected ADC software conversion for injected group */ hadc->Instance->CR2 |= ADC_CR2_JSWSTART; @@ -246,7 +246,7 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedStart(ADC_HandleTypeDef* hadc) * * @retval HAL status. */ -HAL_StatusTypeDef HAL_ADCEx_InjectedStart_IT(ADC_HandleTypeDef* hadc) +HAL_StatusTypeDef HAL_ADCEx_InjectedStart_IT(ADC_HandleTypeDef *hadc) { __IO uint32_t counter = 0U; uint32_t tmp1 = 0U, tmp2 = 0U; @@ -259,7 +259,7 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedStart_IT(ADC_HandleTypeDef* hadc) /* Check if ADC peripheral is disabled in order to enable it and wait during Tstab time the ADC's stabilization */ - if((hadc->Instance->CR2 & ADC_CR2_ADON) != ADC_CR2_ADON) + if ((hadc->Instance->CR2 & ADC_CR2_ADON) != ADC_CR2_ADON) { /* Enable the Peripheral */ __HAL_ADC_ENABLE(hadc); @@ -267,14 +267,14 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedStart_IT(ADC_HandleTypeDef* hadc) /* Delay for ADC stabilization time */ /* Compute number of CPU cycles to wait for */ counter = (ADC_STAB_DELAY_US * (SystemCoreClock / 1000000U)); - while(counter != 0U) + while (counter != 0U) { counter--; } } /* Start conversion if ADC is effectively enabled */ - if(HAL_IS_BIT_SET(hadc->Instance->CR2, ADC_CR2_ADON)) + if (HAL_IS_BIT_SET(hadc->Instance->CR2, ADC_CR2_ADON)) { /* Set ADC state */ /* - Clear state bitfield related to injected group conversion results */ @@ -311,11 +311,11 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedStart_IT(ADC_HandleTypeDef* hadc) tmpADC_Common = ADC_COMMON_REGISTER(hadc); /* Check if Multimode enabled */ - if(HAL_IS_BIT_CLR(tmpADC_Common->CCR, ADC_CCR_MULTI)) + if (HAL_IS_BIT_CLR(tmpADC_Common->CCR, ADC_CCR_MULTI)) { tmp1 = HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_JEXTEN); tmp2 = HAL_IS_BIT_CLR(hadc->Instance->CR1, ADC_CR1_JAUTO); - if(tmp1 && tmp2) + if (tmp1 && tmp2) { /* Enable the selected ADC software conversion for injected group */ hadc->Instance->CR2 |= ADC_CR2_JSWSTART; @@ -325,7 +325,7 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedStart_IT(ADC_HandleTypeDef* hadc) { tmp1 = HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_JEXTEN); tmp2 = HAL_IS_BIT_CLR(hadc->Instance->CR1, ADC_CR1_JAUTO); - if((hadc->Instance == ADC1) && tmp1 && tmp2) + if ((hadc->Instance == ADC1) && tmp1 && tmp2) { /* Enable the selected ADC software conversion for injected group */ hadc->Instance->CR2 |= ADC_CR2_JSWSTART; @@ -357,7 +357,7 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedStart_IT(ADC_HandleTypeDef* hadc) * @param hadc ADC handle * @retval None */ -HAL_StatusTypeDef HAL_ADCEx_InjectedStop(ADC_HandleTypeDef* hadc) +HAL_StatusTypeDef HAL_ADCEx_InjectedStop(ADC_HandleTypeDef *hadc) { HAL_StatusTypeDef tmp_hal_status = HAL_OK; @@ -373,15 +373,15 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedStop(ADC_HandleTypeDef* hadc) /* continue (injected and regular groups stop conversion and ADC disable */ /* are common) */ /* - In case of auto-injection mode, HAL_ADC_Stop must be used. */ - if(((hadc->State & HAL_ADC_STATE_REG_BUSY) == RESET) && - HAL_IS_BIT_CLR(hadc->Instance->CR1, ADC_CR1_JAUTO) ) + if (((hadc->State & HAL_ADC_STATE_REG_BUSY) == RESET) && + HAL_IS_BIT_CLR(hadc->Instance->CR1, ADC_CR1_JAUTO)) { /* Stop potential conversion on going, on regular and injected groups */ /* Disable ADC peripheral */ __HAL_ADC_DISABLE(hadc); /* Check if ADC is effectively disabled */ - if(HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_ADON)) + if (HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_ADON)) { /* Set ADC state */ ADC_STATE_CLR_SET(hadc->State, @@ -411,7 +411,7 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedStop(ADC_HandleTypeDef* hadc) * @param Timeout Timeout value in millisecond. * @retval HAL status */ -HAL_StatusTypeDef HAL_ADCEx_InjectedPollForConversion(ADC_HandleTypeDef* hadc, uint32_t Timeout) +HAL_StatusTypeDef HAL_ADCEx_InjectedPollForConversion(ADC_HandleTypeDef *hadc, uint32_t Timeout) { uint32_t tickstart = 0U; @@ -419,17 +419,17 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedPollForConversion(ADC_HandleTypeDef* hadc, u tickstart = HAL_GetTick(); /* Check End of conversion flag */ - while(!(__HAL_ADC_GET_FLAG(hadc, ADC_FLAG_JEOC))) + while (!(__HAL_ADC_GET_FLAG(hadc, ADC_FLAG_JEOC))) { /* Check for the Timeout */ - if(Timeout != HAL_MAX_DELAY) + if (Timeout != HAL_MAX_DELAY) { - if((Timeout == 0U)||((HAL_GetTick() - tickstart ) > Timeout)) + if ((Timeout == 0U) || ((HAL_GetTick() - tickstart) > Timeout)) { /* New check to avoid false timeout detection in case of preemption */ - if(!(__HAL_ADC_GET_FLAG(hadc, ADC_FLAG_JEOC))) + if (!(__HAL_ADC_GET_FLAG(hadc, ADC_FLAG_JEOC))) { - hadc->State= HAL_ADC_STATE_TIMEOUT; + hadc->State = HAL_ADC_STATE_TIMEOUT; /* Process unlocked */ __HAL_UNLOCK(hadc); return HAL_TIMEOUT; @@ -450,12 +450,12 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedPollForConversion(ADC_HandleTypeDef* hadc, u /* The test of scan sequence on going is done either with scan */ /* sequence disabled or with end of conversion flag set to */ /* of end of sequence. */ - if(ADC_IS_SOFTWARE_START_INJECTED(hadc) && - (HAL_IS_BIT_CLR(hadc->Instance->JSQR, ADC_JSQR_JL) || - HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_EOCS) ) && - (HAL_IS_BIT_CLR(hadc->Instance->CR1, ADC_CR1_JAUTO) && - (ADC_IS_SOFTWARE_START_REGULAR(hadc) && - (hadc->Init.ContinuousConvMode == DISABLE) ) ) ) + if (ADC_IS_SOFTWARE_START_INJECTED(hadc) && + (HAL_IS_BIT_CLR(hadc->Instance->JSQR, ADC_JSQR_JL) || + HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_EOCS)) && + (HAL_IS_BIT_CLR(hadc->Instance->CR1, ADC_CR1_JAUTO) && + (ADC_IS_SOFTWARE_START_REGULAR(hadc) && + (hadc->Init.ContinuousConvMode == DISABLE)))) { /* Set ADC state */ CLEAR_BIT(hadc->State, HAL_ADC_STATE_INJ_BUSY); @@ -482,7 +482,7 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedPollForConversion(ADC_HandleTypeDef* hadc, u * @param hadc ADC handle * @retval None */ -HAL_StatusTypeDef HAL_ADCEx_InjectedStop_IT(ADC_HandleTypeDef* hadc) +HAL_StatusTypeDef HAL_ADCEx_InjectedStop_IT(ADC_HandleTypeDef *hadc) { HAL_StatusTypeDef tmp_hal_status = HAL_OK; @@ -498,15 +498,15 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedStop_IT(ADC_HandleTypeDef* hadc) /* continue (injected and regular groups stop conversion and ADC disable */ /* are common) */ /* - In case of auto-injection mode, HAL_ADC_Stop must be used. */ - if(((hadc->State & HAL_ADC_STATE_REG_BUSY) == RESET) && - HAL_IS_BIT_CLR(hadc->Instance->CR1, ADC_CR1_JAUTO) ) + if (((hadc->State & HAL_ADC_STATE_REG_BUSY) == RESET) && + HAL_IS_BIT_CLR(hadc->Instance->CR1, ADC_CR1_JAUTO)) { /* Stop potential conversion on going, on regular and injected groups */ /* Disable ADC peripheral */ __HAL_ADC_DISABLE(hadc); /* Check if ADC is effectively disabled */ - if(HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_ADON)) + if (HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_ADON)) { /* Disable ADC end of conversion interrupt for injected channels */ __HAL_ADC_DISABLE_IT(hadc, ADC_IT_JEOC); @@ -544,7 +544,7 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedStop_IT(ADC_HandleTypeDef* hadc) * @arg ADC_INJECTED_RANK_4: Injected Channel4 selected * @retval None */ -uint32_t HAL_ADCEx_InjectedGetValue(ADC_HandleTypeDef* hadc, uint32_t InjectedRank) +uint32_t HAL_ADCEx_InjectedGetValue(ADC_HandleTypeDef *hadc, uint32_t InjectedRank) { __IO uint32_t tmp = 0U; @@ -556,7 +556,7 @@ uint32_t HAL_ADCEx_InjectedGetValue(ADC_HandleTypeDef* hadc, uint32_t InjectedRa __HAL_ADC_CLEAR_FLAG(hadc, ADC_FLAG_JEOC); /* Return the selected ADC converted value */ - switch(InjectedRank) + switch (InjectedRank) { case ADC_INJECTED_RANK_4: { @@ -579,7 +579,7 @@ uint32_t HAL_ADCEx_InjectedGetValue(ADC_HandleTypeDef* hadc, uint32_t InjectedRa } break; default: - break; + break; } return tmp; } @@ -595,7 +595,7 @@ uint32_t HAL_ADCEx_InjectedGetValue(ADC_HandleTypeDef* hadc, uint32_t InjectedRa * @param Length The length of data to be transferred from ADC peripheral to memory. * @retval HAL status */ -HAL_StatusTypeDef HAL_ADCEx_MultiModeStart_DMA(ADC_HandleTypeDef* hadc, uint32_t* pData, uint32_t Length) +HAL_StatusTypeDef HAL_ADCEx_MultiModeStart_DMA(ADC_HandleTypeDef *hadc, uint32_t *pData, uint32_t Length) { __IO uint32_t counter = 0U; ADC_Common_TypeDef *tmpADC_Common; @@ -610,7 +610,7 @@ HAL_StatusTypeDef HAL_ADCEx_MultiModeStart_DMA(ADC_HandleTypeDef* hadc, uint32_t /* Check if ADC peripheral is disabled in order to enable it and wait during Tstab time the ADC's stabilization */ - if((hadc->Instance->CR2 & ADC_CR2_ADON) != ADC_CR2_ADON) + if ((hadc->Instance->CR2 & ADC_CR2_ADON) != ADC_CR2_ADON) { /* Enable the Peripheral */ __HAL_ADC_ENABLE(hadc); @@ -618,14 +618,14 @@ HAL_StatusTypeDef HAL_ADCEx_MultiModeStart_DMA(ADC_HandleTypeDef* hadc, uint32_t /* Delay for temperature sensor stabilization time */ /* Compute number of CPU cycles to wait for */ counter = (ADC_STAB_DELAY_US * (SystemCoreClock / 1000000U)); - while(counter != 0U) + while (counter != 0U) { counter--; } } /* Start conversion if ADC is effectively enabled */ - if(HAL_IS_BIT_SET(hadc->Instance->CR2, ADC_CR2_ADON)) + if (HAL_IS_BIT_SET(hadc->Instance->CR2, ADC_CR2_ADON)) { /* Set ADC state */ /* - Clear state bitfield related to regular group conversion results */ @@ -697,7 +697,7 @@ HAL_StatusTypeDef HAL_ADCEx_MultiModeStart_DMA(ADC_HandleTypeDef* hadc, uint32_t HAL_DMA_Start_IT(hadc->DMA_Handle, (uint32_t)&tmpADC_Common->CDR, (uint32_t)pData, Length); /* if no external trigger present enable software conversion of regular channels */ - if((hadc->Instance->CR2 & ADC_CR2_EXTEN) == RESET) + if ((hadc->Instance->CR2 & ADC_CR2_EXTEN) == RESET) { /* Enable the selected ADC software conversion for regular group */ hadc->Instance->CR2 |= (uint32_t)ADC_CR2_SWSTART; @@ -722,7 +722,7 @@ HAL_StatusTypeDef HAL_ADCEx_MultiModeStart_DMA(ADC_HandleTypeDef* hadc, uint32_t * the configuration information for the specified ADC. * @retval HAL status */ -HAL_StatusTypeDef HAL_ADCEx_MultiModeStop_DMA(ADC_HandleTypeDef* hadc) +HAL_StatusTypeDef HAL_ADCEx_MultiModeStop_DMA(ADC_HandleTypeDef *hadc) { HAL_StatusTypeDef tmp_hal_status = HAL_OK; ADC_Common_TypeDef *tmpADC_Common; @@ -743,7 +743,7 @@ HAL_StatusTypeDef HAL_ADCEx_MultiModeStop_DMA(ADC_HandleTypeDef* hadc) tmpADC_Common = ADC_COMMON_REGISTER(hadc); /* Check if ADC is effectively disabled */ - if(HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_ADON)) + if (HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_ADON)) { /* Disable the selected ADC DMA mode for multimode */ tmpADC_Common->CCR &= ~ADC_CCR_DDS; @@ -775,7 +775,7 @@ HAL_StatusTypeDef HAL_ADCEx_MultiModeStop_DMA(ADC_HandleTypeDef* hadc) * the configuration information for the specified ADC. * @retval The converted data value. */ -uint32_t HAL_ADCEx_MultiModeGetValue(ADC_HandleTypeDef* hadc) +uint32_t HAL_ADCEx_MultiModeGetValue(ADC_HandleTypeDef *hadc) { ADC_Common_TypeDef *tmpADC_Common; @@ -794,7 +794,7 @@ uint32_t HAL_ADCEx_MultiModeGetValue(ADC_HandleTypeDef* hadc) * the configuration information for the specified ADC. * @retval None */ -__weak void HAL_ADCEx_InjectedConvCpltCallback(ADC_HandleTypeDef* hadc) +__weak void HAL_ADCEx_InjectedConvCpltCallback(ADC_HandleTypeDef *hadc) { /* Prevent unused argument(s) compilation warning */ UNUSED(hadc); @@ -811,7 +811,7 @@ __weak void HAL_ADCEx_InjectedConvCpltCallback(ADC_HandleTypeDef* hadc) * @param sConfigInjected ADC configuration structure for injected channel. * @retval None */ -HAL_StatusTypeDef HAL_ADCEx_InjectedConfigChannel(ADC_HandleTypeDef* hadc, ADC_InjectionConfTypeDef* sConfigInjected) +HAL_StatusTypeDef HAL_ADCEx_InjectedConfigChannel(ADC_HandleTypeDef *hadc, ADC_InjectionConfTypeDef *sConfigInjected) { #ifdef USE_FULL_ASSERT @@ -835,7 +835,7 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedConfigChannel(ADC_HandleTypeDef* hadc, ADC_I assert_param(IS_ADC_RANGE(tmp, sConfigInjected->InjectedOffset)); #endif /* USE_FULL_ASSERT */ - if(sConfigInjected->ExternalTrigInjecConv != ADC_INJECTED_SOFTWARE_START) + if (sConfigInjected->ExternalTrigInjecConv != ADC_INJECTED_SOFTWARE_START) { assert_param(IS_ADC_EXT_INJEC_TRIG_EDGE(sConfigInjected->ExternalTrigInjecConvEdge)); } @@ -868,17 +868,17 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedConfigChannel(ADC_HandleTypeDef* hadc, ADC_I /* Rank configuration */ /* Clear the old SQx bits for the selected rank */ - hadc->Instance->JSQR &= ~ADC_JSQR(ADC_JSQR_JSQ1, sConfigInjected->InjectedRank,sConfigInjected->InjectedNbrOfConversion); + hadc->Instance->JSQR &= ~ADC_JSQR(ADC_JSQR_JSQ1, sConfigInjected->InjectedRank, sConfigInjected->InjectedNbrOfConversion); /* Set the SQx bits for the selected rank */ - hadc->Instance->JSQR |= ADC_JSQR(sConfigInjected->InjectedChannel, sConfigInjected->InjectedRank,sConfigInjected->InjectedNbrOfConversion); + hadc->Instance->JSQR |= ADC_JSQR(sConfigInjected->InjectedChannel, sConfigInjected->InjectedRank, sConfigInjected->InjectedNbrOfConversion); /* Enable external trigger if trigger selection is different of software */ /* start. */ /* Note: This configuration keeps the hardware feature of parameter */ /* ExternalTrigConvEdge "trigger edge none" equivalent to */ /* software start. */ - if(sConfigInjected->ExternalTrigInjecConv != ADC_INJECTED_SOFTWARE_START) + if (sConfigInjected->ExternalTrigInjecConv != ADC_INJECTED_SOFTWARE_START) { /* Select external trigger to start conversion */ hadc->Instance->CR2 &= ~(ADC_CR2_JEXTSEL); @@ -917,7 +917,7 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedConfigChannel(ADC_HandleTypeDef* hadc, ADC_I hadc->Instance->CR1 &= ~(ADC_CR1_JDISCEN); } - switch(sConfigInjected->InjectedRank) + switch (sConfigInjected->InjectedRank) { case 1U: /* Set injected channel 1 offset */ @@ -944,7 +944,7 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedConfigChannel(ADC_HandleTypeDef* hadc, ADC_I /* Pointer to the common control register to which is belonging hadc */ /* (Depending on STM32F4 product, there may be up to 3 ADC and 1 common */ /* control register) */ - tmpADC_Common = ADC_COMMON_REGISTER(hadc); + tmpADC_Common = ADC_COMMON_REGISTER(hadc); /* if ADC1 Channel_18 is selected enable VBAT Channel */ if ((hadc->Instance == ADC1) && (sConfigInjected->InjectedChannel == ADC_CHANNEL_VBAT)) @@ -975,7 +975,7 @@ HAL_StatusTypeDef HAL_ADCEx_InjectedConfigChannel(ADC_HandleTypeDef* hadc, ADC_I * the configuration information for multimode. * @retval HAL status */ -HAL_StatusTypeDef HAL_ADCEx_MultiModeConfigChannel(ADC_HandleTypeDef* hadc, ADC_MultiModeTypeDef* multimode) +HAL_StatusTypeDef HAL_ADCEx_MultiModeConfigChannel(ADC_HandleTypeDef *hadc, ADC_MultiModeTypeDef *multimode) { ADC_Common_TypeDef *tmpADC_Common; @@ -1025,7 +1025,7 @@ HAL_StatusTypeDef HAL_ADCEx_MultiModeConfigChannel(ADC_HandleTypeDef* hadc, ADC_ static void ADC_MultiModeDMAConvCplt(DMA_HandleTypeDef *hdma) { /* Retrieve ADC handle corresponding to current DMA handle */ - ADC_HandleTypeDef* hadc = ( ADC_HandleTypeDef* )((DMA_HandleTypeDef* )hdma)->Parent; + ADC_HandleTypeDef *hadc = (ADC_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent; /* Update state machine on conversion status if not in error state */ if (HAL_IS_BIT_CLR(hadc->State, HAL_ADC_STATE_ERROR_INTERNAL | HAL_ADC_STATE_ERROR_DMA)) @@ -1039,10 +1039,10 @@ static void ADC_MultiModeDMAConvCplt(DMA_HandleTypeDef *hdma) /* The test of scan sequence on going is done either with scan */ /* sequence disabled or with end of conversion flag set to */ /* of end of sequence. */ - if(ADC_IS_SOFTWARE_START_REGULAR(hadc) && - (hadc->Init.ContinuousConvMode == DISABLE) && - (HAL_IS_BIT_CLR(hadc->Instance->SQR1, ADC_SQR1_L) || - HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_EOCS) ) ) + if (ADC_IS_SOFTWARE_START_REGULAR(hadc) && + (hadc->Init.ContinuousConvMode == DISABLE) && + (HAL_IS_BIT_CLR(hadc->Instance->SQR1, ADC_SQR1_L) || + HAL_IS_BIT_CLR(hadc->Instance->CR2, ADC_CR2_EOCS))) { /* Disable ADC end of single conversion interrupt on group regular */ /* Note: Overrun interrupt was enabled with EOC interrupt in */ @@ -1077,9 +1077,9 @@ static void ADC_MultiModeDMAConvCplt(DMA_HandleTypeDef *hdma) */ static void ADC_MultiModeDMAHalfConvCplt(DMA_HandleTypeDef *hdma) { - ADC_HandleTypeDef* hadc = ( ADC_HandleTypeDef* )((DMA_HandleTypeDef* )hdma)->Parent; - /* Conversion complete callback */ - HAL_ADC_ConvHalfCpltCallback(hadc); + ADC_HandleTypeDef *hadc = (ADC_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent; + /* Conversion complete callback */ + HAL_ADC_ConvHalfCpltCallback(hadc); } /** @@ -1090,11 +1090,11 @@ static void ADC_MultiModeDMAHalfConvCplt(DMA_HandleTypeDef *hdma) */ static void ADC_MultiModeDMAError(DMA_HandleTypeDef *hdma) { - ADC_HandleTypeDef* hadc = ( ADC_HandleTypeDef* )((DMA_HandleTypeDef* )hdma)->Parent; - hadc->State= HAL_ADC_STATE_ERROR_DMA; - /* Set ADC error code to DMA error */ - hadc->ErrorCode |= HAL_ADC_ERROR_DMA; - HAL_ADC_ErrorCallback(hadc); + ADC_HandleTypeDef *hadc = (ADC_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent; + hadc->State = HAL_ADC_STATE_ERROR_DMA; + /* Set ADC error code to DMA error */ + hadc->ErrorCode |= HAL_ADC_ERROR_DMA; + HAL_ADC_ErrorCallback(hadc); } /** diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_can.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_can.c index 4abdc60a..f9911163 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_can.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_can.c @@ -33,7 +33,7 @@ (++) Enable the CAN interface clock using __HAL_RCC_CANx_CLK_ENABLE() (++) Configure CAN pins (+++) Enable the clock for the CAN GPIOs - (+++) Configure CAN pins as alternate function open-drain + (+++) Configure CAN pins as alternate function (++) In case of using interrupts (e.g. HAL_CAN_ActivateNotification()) (+++) Configure the CAN interrupt priority using HAL_NVIC_SetPriority() @@ -226,8 +226,8 @@ #ifdef HAL_CAN_MODULE_ENABLED #ifdef HAL_CAN_LEGACY_MODULE_ENABLED - #error "The CAN driver cannot be used with its legacy, Please enable only one CAN module at once" -#endif +#error "The CAN driver cannot be used with its legacy, Please enable only one CAN module at once" +#endif /* HAL_CAN_LEGACY_MODULE_ENABLED */ /* Private typedef -----------------------------------------------------------*/ /* Private define ------------------------------------------------------------*/ @@ -235,6 +235,7 @@ * @{ */ #define CAN_TIMEOUT_VALUE 10U +#define CAN_WAKEUP_TIMEOUT_COUNTER 1000000U /** * @} */ @@ -248,8 +249,8 @@ */ /** @defgroup CAN_Exported_Functions_Group1 Initialization and de-initialization functions - * @brief Initialization and Configuration functions - * + * @brief Initialization and Configuration functions + * @verbatim ============================================================================== ##### Initialization and de-initialization functions ##### @@ -328,7 +329,7 @@ HAL_StatusTypeDef HAL_CAN_Init(CAN_HandleTypeDef *hcan) /* Init the low level hardware: CLOCK, NVIC */ HAL_CAN_MspInit(hcan); } -#endif /* (USE_HAL_CAN_REGISTER_CALLBACKS) */ +#endif /* USE_HAL_CAN_REGISTER_CALLBACKS */ /* Request initialisation */ SET_BIT(hcan->Instance->MCR, CAN_MCR_INRQ); @@ -482,7 +483,7 @@ HAL_StatusTypeDef HAL_CAN_DeInit(CAN_HandleTypeDef *hcan) #else /* DeInit the low level hardware: CLOCK, NVIC */ HAL_CAN_MspDeInit(hcan); -#endif /* (USE_HAL_CAN_REGISTER_CALLBACKS) */ +#endif /* USE_HAL_CAN_REGISTER_CALLBACKS */ /* Reset the CAN peripheral */ SET_BIT(hcan->Instance->MCR, CAN_MCR_RESET); @@ -555,7 +556,8 @@ __weak void HAL_CAN_MspDeInit(CAN_HandleTypeDef *hcan) * @param pCallback pointer to the Callback function * @retval HAL status */ -HAL_StatusTypeDef HAL_CAN_RegisterCallback(CAN_HandleTypeDef *hcan, HAL_CAN_CallbackIDTypeDef CallbackID, void (* pCallback)(CAN_HandleTypeDef *_hcan)) +HAL_StatusTypeDef HAL_CAN_RegisterCallback(CAN_HandleTypeDef *hcan, HAL_CAN_CallbackIDTypeDef CallbackID, + void (* pCallback)(CAN_HandleTypeDef *_hcan)) { HAL_StatusTypeDef status = HAL_OK; @@ -813,8 +815,8 @@ HAL_StatusTypeDef HAL_CAN_UnRegisterCallback(CAN_HandleTypeDef *hcan, HAL_CAN_Ca */ /** @defgroup CAN_Exported_Functions_Group2 Configuration functions - * @brief Configuration functions. - * + * @brief Configuration functions. + * @verbatim ============================================================================== ##### Configuration functions ##### @@ -835,7 +837,7 @@ HAL_StatusTypeDef HAL_CAN_UnRegisterCallback(CAN_HandleTypeDef *hcan, HAL_CAN_Ca * contains the filter configuration information. * @retval None */ -HAL_StatusTypeDef HAL_CAN_ConfigFilter(CAN_HandleTypeDef *hcan, CAN_FilterTypeDef *sFilterConfig) +HAL_StatusTypeDef HAL_CAN_ConfigFilter(CAN_HandleTypeDef *hcan, const CAN_FilterTypeDef *sFilterConfig) { uint32_t filternbrbitpos; CAN_TypeDef *can_ip = hcan->Instance; @@ -886,7 +888,7 @@ HAL_StatusTypeDef HAL_CAN_ConfigFilter(CAN_HandleTypeDef *hcan, CAN_FilterTypeDe /* Check the parameters */ assert_param(IS_CAN_FILTER_BANK_SINGLE(sFilterConfig->FilterBank)); -#endif +#endif /* CAN3 */ /* Initialisation mode for the filter */ SET_BIT(can_ip->FMR, CAN_FMR_FINIT); @@ -905,7 +907,7 @@ HAL_StatusTypeDef HAL_CAN_ConfigFilter(CAN_HandleTypeDef *hcan, CAN_FilterTypeDe CLEAR_BIT(can_ip->FMR, CAN_FMR_CAN2SB); SET_BIT(can_ip->FMR, sFilterConfig->SlaveStartFilterBank << CAN_FMR_CAN2SB_Pos); -#endif +#endif /* CAN3 */ /* Convert filter number into bit position */ filternbrbitpos = (uint32_t)1 << (sFilterConfig->FilterBank & 0x1FU); @@ -997,8 +999,8 @@ HAL_StatusTypeDef HAL_CAN_ConfigFilter(CAN_HandleTypeDef *hcan, CAN_FilterTypeDe */ /** @defgroup CAN_Exported_Functions_Group3 Control functions - * @brief Control functions - * + * @brief Control functions + * @verbatim ============================================================================== ##### Control functions ##### @@ -1170,7 +1172,6 @@ HAL_StatusTypeDef HAL_CAN_RequestSleep(CAN_HandleTypeDef *hcan) HAL_StatusTypeDef HAL_CAN_WakeUp(CAN_HandleTypeDef *hcan) { __IO uint32_t count = 0; - uint32_t timeout = 1000000U; HAL_CAN_StateTypeDef state = hcan->State; if ((state == HAL_CAN_STATE_READY) || @@ -1186,15 +1187,14 @@ HAL_StatusTypeDef HAL_CAN_WakeUp(CAN_HandleTypeDef *hcan) count++; /* Check if timeout is reached */ - if (count > timeout) + if (count > CAN_WAKEUP_TIMEOUT_COUNTER) { /* Update error code */ hcan->ErrorCode |= HAL_CAN_ERROR_TIMEOUT; return HAL_ERROR; } - } - while ((hcan->Instance->MSR & CAN_MSR_SLAK) != 0U); + } while ((hcan->Instance->MSR & CAN_MSR_SLAK) != 0U); /* Return function status */ return HAL_OK; @@ -1216,7 +1216,7 @@ HAL_StatusTypeDef HAL_CAN_WakeUp(CAN_HandleTypeDef *hcan) * - 0 : Sleep mode is not active. * - 1 : Sleep mode is active. */ -uint32_t HAL_CAN_IsSleepActive(CAN_HandleTypeDef *hcan) +uint32_t HAL_CAN_IsSleepActive(const CAN_HandleTypeDef *hcan) { uint32_t status = 0U; HAL_CAN_StateTypeDef state = hcan->State; @@ -1247,7 +1247,8 @@ uint32_t HAL_CAN_IsSleepActive(CAN_HandleTypeDef *hcan) * This parameter can be a value of @arg CAN_Tx_Mailboxes. * @retval HAL status */ -HAL_StatusTypeDef HAL_CAN_AddTxMessage(CAN_HandleTypeDef *hcan, CAN_TxHeaderTypeDef *pHeader, uint8_t aData[], uint32_t *pTxMailbox) +HAL_StatusTypeDef HAL_CAN_AddTxMessage(CAN_HandleTypeDef *hcan, const CAN_TxHeaderTypeDef *pHeader, + const uint8_t aData[], uint32_t *pTxMailbox) { uint32_t transmitmailbox; HAL_CAN_StateTypeDef state = hcan->State; @@ -1278,15 +1279,6 @@ HAL_StatusTypeDef HAL_CAN_AddTxMessage(CAN_HandleTypeDef *hcan, CAN_TxHeaderType /* Select an empty transmit mailbox */ transmitmailbox = (tsr & CAN_TSR_CODE) >> CAN_TSR_CODE_Pos; - /* Check transmit mailbox value */ - if (transmitmailbox > 2U) - { - /* Update error code */ - hcan->ErrorCode |= HAL_CAN_ERROR_INTERNAL; - - return HAL_ERROR; - } - /* Store the Tx mailbox */ *pTxMailbox = (uint32_t)1 << transmitmailbox; @@ -1404,7 +1396,7 @@ HAL_StatusTypeDef HAL_CAN_AbortTxRequest(CAN_HandleTypeDef *hcan, uint32_t TxMai * the configuration information for the specified CAN. * @retval Number of free Tx Mailboxes. */ -uint32_t HAL_CAN_GetTxMailboxesFreeLevel(CAN_HandleTypeDef *hcan) +uint32_t HAL_CAN_GetTxMailboxesFreeLevel(const CAN_HandleTypeDef *hcan) { uint32_t freelevel = 0U; HAL_CAN_StateTypeDef state = hcan->State; @@ -1447,7 +1439,7 @@ uint32_t HAL_CAN_GetTxMailboxesFreeLevel(CAN_HandleTypeDef *hcan) * - 1 : Pending transmission request on at least one of the selected * Tx Mailbox. */ -uint32_t HAL_CAN_IsTxMessagePending(CAN_HandleTypeDef *hcan, uint32_t TxMailboxes) +uint32_t HAL_CAN_IsTxMessagePending(const CAN_HandleTypeDef *hcan, uint32_t TxMailboxes) { uint32_t status = 0U; HAL_CAN_StateTypeDef state = hcan->State; @@ -1479,7 +1471,7 @@ uint32_t HAL_CAN_IsTxMessagePending(CAN_HandleTypeDef *hcan, uint32_t TxMailboxe * This parameter can be one value of @arg CAN_Tx_Mailboxes. * @retval Timestamp of message sent from Tx Mailbox. */ -uint32_t HAL_CAN_GetTxTimestamp(CAN_HandleTypeDef *hcan, uint32_t TxMailbox) +uint32_t HAL_CAN_GetTxTimestamp(const CAN_HandleTypeDef *hcan, uint32_t TxMailbox) { uint32_t timestamp = 0U; uint32_t transmitmailbox; @@ -1513,7 +1505,8 @@ uint32_t HAL_CAN_GetTxTimestamp(CAN_HandleTypeDef *hcan, uint32_t TxMailbox) * @param aData array where the payload of the Rx frame will be stored. * @retval HAL status */ -HAL_StatusTypeDef HAL_CAN_GetRxMessage(CAN_HandleTypeDef *hcan, uint32_t RxFifo, CAN_RxHeaderTypeDef *pHeader, uint8_t aData[]) +HAL_StatusTypeDef HAL_CAN_GetRxMessage(CAN_HandleTypeDef *hcan, uint32_t RxFifo, + CAN_RxHeaderTypeDef *pHeader, uint8_t aData[]) { HAL_CAN_StateTypeDef state = hcan->State; @@ -1554,10 +1547,19 @@ HAL_StatusTypeDef HAL_CAN_GetRxMessage(CAN_HandleTypeDef *hcan, uint32_t RxFifo, } else { - pHeader->ExtId = ((CAN_RI0R_EXID | CAN_RI0R_STID) & hcan->Instance->sFIFOMailBox[RxFifo].RIR) >> CAN_RI0R_EXID_Pos; + pHeader->ExtId = ((CAN_RI0R_EXID | CAN_RI0R_STID) & + hcan->Instance->sFIFOMailBox[RxFifo].RIR) >> CAN_RI0R_EXID_Pos; } pHeader->RTR = (CAN_RI0R_RTR & hcan->Instance->sFIFOMailBox[RxFifo].RIR); - pHeader->DLC = (CAN_RDT0R_DLC & hcan->Instance->sFIFOMailBox[RxFifo].RDTR) >> CAN_RDT0R_DLC_Pos; + if (((CAN_RDT0R_DLC & hcan->Instance->sFIFOMailBox[RxFifo].RDTR) >> CAN_RDT0R_DLC_Pos) >= 8U) + { + /* Truncate DLC to 8 if received field is over range */ + pHeader->DLC = 8U; + } + else + { + pHeader->DLC = (CAN_RDT0R_DLC & hcan->Instance->sFIFOMailBox[RxFifo].RDTR) >> CAN_RDT0R_DLC_Pos; + } pHeader->FilterMatchIndex = (CAN_RDT0R_FMI & hcan->Instance->sFIFOMailBox[RxFifo].RDTR) >> CAN_RDT0R_FMI_Pos; pHeader->Timestamp = (CAN_RDT0R_TIME & hcan->Instance->sFIFOMailBox[RxFifo].RDTR) >> CAN_RDT0R_TIME_Pos; @@ -1603,7 +1605,7 @@ HAL_StatusTypeDef HAL_CAN_GetRxMessage(CAN_HandleTypeDef *hcan, uint32_t RxFifo, * This parameter can be a value of @arg CAN_receive_FIFO_number. * @retval Number of messages available in Rx FIFO. */ -uint32_t HAL_CAN_GetRxFifoFillLevel(CAN_HandleTypeDef *hcan, uint32_t RxFifo) +uint32_t HAL_CAN_GetRxFifoFillLevel(const CAN_HandleTypeDef *hcan, uint32_t RxFifo) { uint32_t filllevel = 0U; HAL_CAN_StateTypeDef state = hcan->State; @@ -1633,8 +1635,8 @@ uint32_t HAL_CAN_GetRxFifoFillLevel(CAN_HandleTypeDef *hcan, uint32_t RxFifo) */ /** @defgroup CAN_Exported_Functions_Group4 Interrupts management - * @brief Interrupts management - * + * @brief Interrupts management + * @verbatim ============================================================================== ##### Interrupts management ##### @@ -2099,8 +2101,8 @@ void HAL_CAN_IRQHandler(CAN_HandleTypeDef *hcan) */ /** @defgroup CAN_Exported_Functions_Group5 Callback functions - * @brief CAN Callback functions - * + * @brief CAN Callback functions + * @verbatim ============================================================================== ##### Callback functions ##### @@ -2349,8 +2351,8 @@ __weak void HAL_CAN_ErrorCallback(CAN_HandleTypeDef *hcan) */ /** @defgroup CAN_Exported_Functions_Group6 Peripheral State and Error functions - * @brief CAN Peripheral State functions - * + * @brief CAN Peripheral State functions + * @verbatim ============================================================================== ##### Peripheral State and Error functions ##### @@ -2371,7 +2373,7 @@ __weak void HAL_CAN_ErrorCallback(CAN_HandleTypeDef *hcan) * the configuration information for the specified CAN. * @retval HAL state */ -HAL_CAN_StateTypeDef HAL_CAN_GetState(CAN_HandleTypeDef *hcan) +HAL_CAN_StateTypeDef HAL_CAN_GetState(const CAN_HandleTypeDef *hcan) { HAL_CAN_StateTypeDef state = hcan->State; @@ -2406,7 +2408,7 @@ HAL_CAN_StateTypeDef HAL_CAN_GetState(CAN_HandleTypeDef *hcan) * the configuration information for the specified CAN. * @retval CAN Error Code */ -uint32_t HAL_CAN_GetError(CAN_HandleTypeDef *hcan) +uint32_t HAL_CAN_GetError(const CAN_HandleTypeDef *hcan) { /* Return CAN error code */ return hcan->ErrorCode; diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_cec.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_cec.c index 56e6e848..6a39d16a 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_cec.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_cec.c @@ -233,7 +233,8 @@ HAL_StatusTypeDef HAL_CEC_Init(CEC_HandleTypeDef *hcec) /* Write to CEC Control Register */ hcec->Instance->CFGR = hcec->Init.SignalFreeTime | hcec->Init.Tolerance | hcec->Init.BRERxStop | \ - hcec->Init.BREErrorBitGen | hcec->Init.LBPEErrorBitGen | hcec->Init.BroadcastMsgNoErrorBitGen | \ + hcec->Init.BREErrorBitGen | hcec->Init.LBPEErrorBitGen | \ + hcec->Init.BroadcastMsgNoErrorBitGen | \ hcec->Init.SignalFreeTimeOption | ((uint32_t)(hcec->Init.OwnAddress) << 16U) | \ hcec->Init.ListenMode; @@ -412,10 +413,10 @@ __weak void HAL_CEC_MspDeInit(CEC_HandleTypeDef *hcec) * @param hcec CEC handle * @param CallbackID ID of the callback to be registered * This parameter can be one of the following values: - * @arg @ref HAL_CEC_TX_CPLT_CB_ID Tx Complete callback ID - * @arg @ref HAL_CEC_ERROR_CB_ID Error callback ID - * @arg @ref HAL_CEC_MSPINIT_CB_ID MspInit callback ID - * @arg @ref HAL_CEC_MSPDEINIT_CB_ID MspDeInit callback ID + * @arg HAL_CEC_TX_CPLT_CB_ID Tx Complete callback ID + * @arg HAL_CEC_ERROR_CB_ID Error callback ID + * @arg HAL_CEC_MSPINIT_CB_ID MspInit callback ID + * @arg HAL_CEC_MSPDEINIT_CB_ID MspDeInit callback ID * @param pCallback pointer to the Callback function * @retval HAL status */ @@ -501,10 +502,10 @@ HAL_StatusTypeDef HAL_CEC_RegisterCallback(CEC_HandleTypeDef *hcec, HAL_CEC_Call * @param hcec uart handle * @param CallbackID ID of the callback to be unregistered * This parameter can be one of the following values: - * @arg @ref HAL_CEC_TX_CPLT_CB_ID Tx Complete callback ID - * @arg @ref HAL_CEC_ERROR_CB_ID Error callback ID - * @arg @ref HAL_CEC_MSPINIT_CB_ID MspInit callback ID - * @arg @ref HAL_CEC_MSPDEINIT_CB_ID MspDeInit callback ID + * @arg HAL_CEC_TX_CPLT_CB_ID Tx Complete callback ID + * @arg HAL_CEC_ERROR_CB_ID Error callback ID + * @arg HAL_CEC_MSPINIT_CB_ID MspInit callback ID + * @arg HAL_CEC_MSPDEINIT_CB_ID MspDeInit callback ID * @retval status */ HAL_StatusTypeDef HAL_CEC_UnRegisterCallback(CEC_HandleTypeDef *hcec, HAL_CEC_CallbackIDTypeDef CallbackID) @@ -694,7 +695,7 @@ HAL_StatusTypeDef HAL_CEC_UnRegisterRxCpltCallback(CEC_HandleTypeDef *hcec) * @retval HAL status */ HAL_StatusTypeDef HAL_CEC_Transmit_IT(CEC_HandleTypeDef *hcec, uint8_t InitiatorAddress, uint8_t DestinationAddress, - uint8_t *pData, uint32_t Size) + const uint8_t *pData, uint32_t Size) { /* if the peripheral isn't already busy and if there is no previous transmission already pending due to arbitration lost */ @@ -749,7 +750,7 @@ HAL_StatusTypeDef HAL_CEC_Transmit_IT(CEC_HandleTypeDef *hcec, uint8_t Initiator * @param hcec CEC handle * @retval Frame size */ -uint32_t HAL_CEC_GetLastReceivedFrameSize(CEC_HandleTypeDef *hcec) +uint32_t HAL_CEC_GetLastReceivedFrameSize(const CEC_HandleTypeDef *hcec) { return hcec->RxXferSize; } @@ -775,13 +776,13 @@ void HAL_CEC_IRQHandler(CEC_HandleTypeDef *hcec) { /* save interrupts register for further error or interrupts handling purposes */ - uint32_t reg; - reg = hcec->Instance->ISR; + uint32_t itflag; + itflag = hcec->Instance->ISR; /* ----------------------------Arbitration Lost Management----------------------------------*/ /* CEC TX arbitration error interrupt occurred --------------------------------------*/ - if ((reg & CEC_FLAG_ARBLST) != 0U) + if (HAL_IS_BIT_SET(itflag, CEC_FLAG_ARBLST)) { hcec->ErrorCode = HAL_CEC_ERROR_ARBLST; __HAL_CEC_CLEAR_FLAG(hcec, CEC_FLAG_ARBLST); @@ -789,7 +790,7 @@ void HAL_CEC_IRQHandler(CEC_HandleTypeDef *hcec) /* ----------------------------Rx Management----------------------------------*/ /* CEC RX byte received interrupt ---------------------------------------------------*/ - if ((reg & CEC_FLAG_RXBR) != 0U) + if (HAL_IS_BIT_SET(itflag, CEC_FLAG_RXBR)) { /* reception is starting */ hcec->RxState = HAL_CEC_STATE_BUSY_RX; @@ -801,7 +802,7 @@ void HAL_CEC_IRQHandler(CEC_HandleTypeDef *hcec) } /* CEC RX end received interrupt ---------------------------------------------------*/ - if ((reg & CEC_FLAG_RXEND) != 0U) + if (HAL_IS_BIT_SET(itflag, CEC_FLAG_RXEND)) { /* clear IT */ __HAL_CEC_CLEAR_FLAG(hcec, CEC_FLAG_RXEND); @@ -820,7 +821,7 @@ void HAL_CEC_IRQHandler(CEC_HandleTypeDef *hcec) /* ----------------------------Tx Management----------------------------------*/ /* CEC TX byte request interrupt ------------------------------------------------*/ - if ((reg & CEC_FLAG_TXBR) != 0U) + if (HAL_IS_BIT_SET(itflag, CEC_FLAG_TXBR)) { --hcec->TxXferCount; if (hcec->TxXferCount == 0U) @@ -829,14 +830,14 @@ void HAL_CEC_IRQHandler(CEC_HandleTypeDef *hcec) __HAL_CEC_LAST_BYTE_TX_SET(hcec); } /* In all cases transmit the byte */ - hcec->Instance->TXDR = *hcec->pTxBuffPtr; + hcec->Instance->TXDR = (uint8_t) * hcec->pTxBuffPtr; hcec->pTxBuffPtr++; /* clear Tx-Byte request flag */ __HAL_CEC_CLEAR_FLAG(hcec, CEC_FLAG_TXBR); } /* CEC TX end interrupt ------------------------------------------------*/ - if ((reg & CEC_FLAG_TXEND) != 0U) + if (HAL_IS_BIT_SET(itflag, CEC_FLAG_TXEND)) { __HAL_CEC_CLEAR_FLAG(hcec, CEC_FLAG_TXEND); @@ -854,21 +855,21 @@ void HAL_CEC_IRQHandler(CEC_HandleTypeDef *hcec) } /* ----------------------------Rx/Tx Error Management----------------------------------*/ - if ((reg & (CEC_ISR_RXOVR | CEC_ISR_BRE | CEC_ISR_SBPE | CEC_ISR_LBPE | CEC_ISR_RXACKE | CEC_ISR_TXUDR | CEC_ISR_TXERR | - CEC_ISR_TXACKE)) != 0U) + if ((itflag & (CEC_ISR_RXOVR | CEC_ISR_BRE | CEC_ISR_SBPE | CEC_ISR_LBPE | CEC_ISR_RXACKE | CEC_ISR_TXUDR | + CEC_ISR_TXERR | CEC_ISR_TXACKE)) != 0U) { - hcec->ErrorCode = reg; + hcec->ErrorCode = itflag; __HAL_CEC_CLEAR_FLAG(hcec, HAL_CEC_ERROR_RXOVR | HAL_CEC_ERROR_BRE | CEC_FLAG_LBPE | CEC_FLAG_SBPE | HAL_CEC_ERROR_RXACKE | HAL_CEC_ERROR_TXUDR | HAL_CEC_ERROR_TXERR | HAL_CEC_ERROR_TXACKE); - if ((reg & (CEC_ISR_RXOVR | CEC_ISR_BRE | CEC_ISR_SBPE | CEC_ISR_LBPE | CEC_ISR_RXACKE)) != 0U) + if ((itflag & (CEC_ISR_RXOVR | CEC_ISR_BRE | CEC_ISR_SBPE | CEC_ISR_LBPE | CEC_ISR_RXACKE)) != 0U) { hcec->Init.RxBuffer -= hcec->RxXferSize; hcec->RxXferSize = 0U; hcec->RxState = HAL_CEC_STATE_READY; } - else if (((reg & CEC_ISR_ARBLST) == 0U) && ((reg & (CEC_ISR_TXUDR | CEC_ISR_TXERR | CEC_ISR_TXACKE)) != 0U)) + else if (((itflag & CEC_ISR_ARBLST) == 0U) && ((itflag & (CEC_ISR_TXUDR | CEC_ISR_TXERR | CEC_ISR_TXACKE)) != 0U)) { /* Set the CEC state ready to be able to start again the process */ hcec->gState = HAL_CEC_STATE_READY; @@ -957,9 +958,10 @@ __weak void HAL_CEC_ErrorCallback(CEC_HandleTypeDef *hcec) * the configuration information for the specified CEC module. * @retval HAL state */ -HAL_CEC_StateTypeDef HAL_CEC_GetState(CEC_HandleTypeDef *hcec) +HAL_CEC_StateTypeDef HAL_CEC_GetState(const CEC_HandleTypeDef *hcec) { - uint32_t temp1, temp2; + uint32_t temp1; + uint32_t temp2; temp1 = hcec->gState; temp2 = hcec->RxState; @@ -972,7 +974,7 @@ HAL_CEC_StateTypeDef HAL_CEC_GetState(CEC_HandleTypeDef *hcec) * the configuration information for the specified CEC. * @retval CEC Error Code */ -uint32_t HAL_CEC_GetError(CEC_HandleTypeDef *hcec) +uint32_t HAL_CEC_GetError(const CEC_HandleTypeDef *hcec) { return hcec->ErrorCode; } @@ -993,4 +995,3 @@ uint32_t HAL_CEC_GetError(CEC_HandleTypeDef *hcec) /** * @} */ - diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_cortex.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_cortex.c index 54d44b49..3de962f8 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_cortex.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_cortex.c @@ -335,6 +335,16 @@ void HAL_MPU_ConfigRegion(MPU_Region_InitTypeDef *MPU_Init) } #endif /* __MPU_PRESENT */ +/** + * @brief Clear pending events. + * @retval None + */ +void HAL_CORTEX_ClearEvent(void) +{ + __SEV(); + __WFE(); +} + /** * @brief Gets the priority grouping field from the NVIC Interrupt Controller. * @retval Priority grouping field (SCB->AIRCR [10:8] PRIGROUP field) diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_crc.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_crc.c index 2e86b2b6..9bd354ab 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_crc.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_crc.c @@ -147,7 +147,7 @@ HAL_StatusTypeDef HAL_CRC_DeInit(CRC_HandleTypeDef *hcrc) __HAL_CRC_DR_RESET(hcrc); /* Reset IDR register content */ - CLEAR_BIT(hcrc->Instance->IDR, CRC_IDR_IDR); + __HAL_CRC_SET_IDR(hcrc, 0); /* DeInit the low level hardware */ HAL_CRC_MspDeInit(hcrc); @@ -303,7 +303,7 @@ uint32_t HAL_CRC_Calculate(CRC_HandleTypeDef *hcrc, uint32_t pBuffer[], uint32_t * @param hcrc CRC handle * @retval HAL state */ -HAL_CRC_StateTypeDef HAL_CRC_GetState(CRC_HandleTypeDef *hcrc) +HAL_CRC_StateTypeDef HAL_CRC_GetState(const CRC_HandleTypeDef *hcrc) { /* Return CRC handle state */ return hcrc->State; diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_cryp.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_cryp.c index 65ad59df..bb41673b 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_cryp.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_cryp.c @@ -1011,7 +1011,7 @@ HAL_StatusTypeDef HAL_CRYP_Encrypt(CRYP_HandleTypeDef *hcryp, uint32_t *Input, u /* Set the phase */ hcryp->Phase = CRYP_PHASE_PROCESS; - /* Statrt DES/TDES encryption process */ + /* Start DES/TDES encryption process */ status = CRYP_TDES_Process(hcryp, Timeout); break; @@ -2533,15 +2533,17 @@ static HAL_StatusTypeDef CRYP_AES_Encrypt_IT(CRYP_HandleTypeDef *hcryp) /* Enable CRYP */ __HAL_CRYP_ENABLE(hcryp); - /* Write the input block in the IN FIFO */ - hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + hcryp->CrypInCount); + /* Increment the pointer before writing the input block in the IN FIFO to make sure that + when Computation Completed IRQ fires, the hcryp->CrypInCount has always a consistent value + and it is ready for the next operation. */ hcryp->CrypInCount++; - hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + hcryp->CrypInCount); + hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + (hcryp->CrypInCount - 1U)); hcryp->CrypInCount++; - hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + hcryp->CrypInCount); + hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + (hcryp->CrypInCount - 1U)); hcryp->CrypInCount++; - hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + hcryp->CrypInCount); + hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + (hcryp->CrypInCount - 1U)); hcryp->CrypInCount++; + hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + (hcryp->CrypInCount - 1U)); #else /* CRYP */ @@ -2780,7 +2782,8 @@ static HAL_StatusTypeDef CRYP_AES_Decrypt_IT(CRYP_HandleTypeDef *hcryp) __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); + } + while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); /* Clear CCF Flag */ __HAL_CRYP_CLEAR_FLAG(hcryp, CRYP_CCF_CLEAR); @@ -2822,7 +2825,8 @@ static HAL_StatusTypeDef CRYP_AES_Decrypt_IT(CRYP_HandleTypeDef *hcryp) __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_SET(hcryp->Instance->SR, CRYP_FLAG_BUSY)); + } + while (HAL_IS_BIT_SET(hcryp->Instance->SR, CRYP_FLAG_BUSY)); /* Turn back to ALGOMODE of the configuration */ MODIFY_REG(hcryp->Instance->CR, CRYP_CR_ALGOMODE, hcryp->Init.Algorithm); @@ -2867,15 +2871,17 @@ static HAL_StatusTypeDef CRYP_AES_Decrypt_IT(CRYP_HandleTypeDef *hcryp) /* Enable CRYP */ __HAL_CRYP_ENABLE(hcryp); - /* Write the input block in the IN FIFO */ - hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + hcryp->CrypInCount); + /* Increment the pointer before writing the input block in the IN FIFO to make sure that + when Computation Completed IRQ fires, the hcryp->CrypInCount has always a consistent value + and it is ready for the next operation. */ hcryp->CrypInCount++; - hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + hcryp->CrypInCount); + hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + (hcryp->CrypInCount - 1U)); hcryp->CrypInCount++; - hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + hcryp->CrypInCount); + hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + (hcryp->CrypInCount - 1U)); hcryp->CrypInCount++; - hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + hcryp->CrypInCount); + hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + (hcryp->CrypInCount - 1U)); hcryp->CrypInCount++; + hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + (hcryp->CrypInCount - 1U)); #else /* CRYP */ @@ -2961,7 +2967,8 @@ static HAL_StatusTypeDef CRYP_AES_Decrypt_DMA(CRYP_HandleTypeDef *hcryp) __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); + } + while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); /* Clear CCF Flag */ __HAL_CRYP_CLEAR_FLAG(hcryp, CRYP_CCF_CLEAR); @@ -3005,7 +3012,8 @@ static HAL_StatusTypeDef CRYP_AES_Decrypt_DMA(CRYP_HandleTypeDef *hcryp) __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_SET(hcryp->Instance->SR, CRYP_FLAG_BUSY)); + } + while (HAL_IS_BIT_SET(hcryp->Instance->SR, CRYP_FLAG_BUSY)); /* Turn back to ALGOMODE of the configuration */ MODIFY_REG(hcryp->Instance->CR, CRYP_CR_ALGOMODE, hcryp->Init.Algorithm); @@ -3943,7 +3951,8 @@ static HAL_StatusTypeDef CRYP_AESGCM_Process_IT(CRYP_HandleTypeDef *hcryp) __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while ((hcryp->Instance->CR & CRYP_CR_CRYPEN) == CRYP_CR_CRYPEN); + } + while ((hcryp->Instance->CR & CRYP_CR_CRYPEN) == CRYP_CR_CRYPEN); #else /* AES */ @@ -3980,7 +3989,8 @@ static HAL_StatusTypeDef CRYP_AESGCM_Process_IT(CRYP_HandleTypeDef *hcryp) __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); + } + while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); /* Clear CCF flag */ __HAL_CRYP_CLEAR_FLAG(hcryp, CRYP_CCF_CLEAR); @@ -4218,7 +4228,8 @@ static HAL_StatusTypeDef CRYP_AESGCM_Process_DMA(CRYP_HandleTypeDef *hcryp) __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while ((hcryp->Instance->CR & CRYP_CR_CRYPEN) == CRYP_CR_CRYPEN); + } + while ((hcryp->Instance->CR & CRYP_CR_CRYPEN) == CRYP_CR_CRYPEN); #else /* AES */ @@ -4255,7 +4266,8 @@ static HAL_StatusTypeDef CRYP_AESGCM_Process_DMA(CRYP_HandleTypeDef *hcryp) __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); + } + while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); /* Clear CCF flag */ __HAL_CRYP_CLEAR_FLAG(hcryp, CRYP_CCF_CLEAR); @@ -4855,7 +4867,8 @@ static HAL_StatusTypeDef CRYP_AESCCM_Process_IT(CRYP_HandleTypeDef *hcryp) __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while ((hcryp->Instance->CR & CRYP_CR_CRYPEN) == CRYP_CR_CRYPEN); + } + while ((hcryp->Instance->CR & CRYP_CR_CRYPEN) == CRYP_CR_CRYPEN); /* Select header phase */ CRYP_SET_PHASE(hcryp, CRYP_PHASE_HEADER); @@ -5016,7 +5029,8 @@ static HAL_StatusTypeDef CRYP_AESCCM_Process_DMA(CRYP_HandleTypeDef *hcryp) __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while ((hcryp->Instance->CR & CRYP_CR_CRYPEN) == CRYP_CR_CRYPEN); + } + while ((hcryp->Instance->CR & CRYP_CR_CRYPEN) == CRYP_CR_CRYPEN); #else /* AES */ @@ -5062,7 +5076,8 @@ static HAL_StatusTypeDef CRYP_AESCCM_Process_DMA(CRYP_HandleTypeDef *hcryp) __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); + } + while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); /* Clear CCF flag */ __HAL_CRYP_CLEAR_FLAG(hcryp, CRYP_CCF_CLEAR); @@ -5107,7 +5122,8 @@ static HAL_StatusTypeDef CRYP_AESCCM_Process_DMA(CRYP_HandleTypeDef *hcryp) __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); + } + while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); /* Clear CCF flag */ __HAL_CRYP_CLEAR_FLAG(hcryp, CRYP_CCF_CLEAR); } @@ -5144,7 +5160,8 @@ static HAL_StatusTypeDef CRYP_AESCCM_Process_DMA(CRYP_HandleTypeDef *hcryp) __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); + } + while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); /* Clear CCF flag */ __HAL_CRYP_CLEAR_FLAG(hcryp, CRYP_CCF_CLEAR); } @@ -5178,7 +5195,8 @@ static HAL_StatusTypeDef CRYP_AESCCM_Process_DMA(CRYP_HandleTypeDef *hcryp) __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); + } + while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); /* Clear CCF flag */ __HAL_CRYP_CLEAR_FLAG(hcryp, CRYP_CCF_CLEAR); } @@ -5247,7 +5265,8 @@ static HAL_StatusTypeDef CRYP_AESCCM_Process_DMA(CRYP_HandleTypeDef *hcryp) __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); + } + while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); /* Clear CCF flag */ __HAL_CRYP_CLEAR_FLAG(hcryp, CRYP_CCF_CLEAR); @@ -5320,7 +5339,7 @@ static HAL_StatusTypeDef CRYP_AESCCM_Process_DMA(CRYP_HandleTypeDef *hcryp) } /** - * @brief Sets the payload phase in iterrupt mode + * @brief Sets the payload phase in interrupt mode * @param hcryp: pointer to a CRYP_HandleTypeDef structure that contains * the configuration information for CRYP module * @retval state @@ -5545,16 +5564,16 @@ static void CRYP_GCMCCM_SetPayloadPhase_IT(CRYP_HandleTypeDef *hcryp) hcryp->Instance->DINR = *(uint32_t *)(hcryp->pCrypInBuffPtr + hcryp->CrypInCount); hcryp->CrypInCount++; if ((hcryp->CrypInCount == (hcryp->Size / 4U)) && ((hcryp->Size % 16U) == 0U)) - { - /* Call Input transfer complete callback */ + { + /* Call Input transfer complete callback */ #if (USE_HAL_CRYP_REGISTER_CALLBACKS == 1) - /*Call registered Input complete callback*/ - hcryp->InCpltCallback(hcryp); + /*Call registered Input complete callback*/ + hcryp->InCpltCallback(hcryp); #else - /*Call legacy weak Input complete callback*/ - HAL_CRYP_InCpltCallback(hcryp); + /*Call legacy weak Input complete callback*/ + HAL_CRYP_InCpltCallback(hcryp); #endif /* USE_HAL_CRYP_REGISTER_CALLBACKS */ - } + } } else /* Last block of payload < 128bit*/ { @@ -5966,7 +5985,8 @@ static HAL_StatusTypeDef CRYP_GCMCCM_SetHeaderPhase_DMA(CRYP_HandleTypeDef *hcry __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_CLR(hcryp->Instance->SR, CRYP_FLAG_IFEM)); + } + while (HAL_IS_BIT_CLR(hcryp->Instance->SR, CRYP_FLAG_IFEM)); } } else @@ -6001,7 +6021,8 @@ static HAL_StatusTypeDef CRYP_GCMCCM_SetHeaderPhase_DMA(CRYP_HandleTypeDef *hcry __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_CLR(hcryp->Instance->SR, CRYP_FLAG_IFEM)); + } + while (HAL_IS_BIT_CLR(hcryp->Instance->SR, CRYP_FLAG_IFEM)); } /* Last block optionally pad the data with zeros*/ for (loopcounter = 0U; (loopcounter < ((headersize_in_bytes / 4U) % 4U)); loopcounter++) @@ -6051,7 +6072,8 @@ static HAL_StatusTypeDef CRYP_GCMCCM_SetHeaderPhase_DMA(CRYP_HandleTypeDef *hcry __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_CLR(hcryp->Instance->SR, CRYP_FLAG_IFEM)); + } + while (HAL_IS_BIT_CLR(hcryp->Instance->SR, CRYP_FLAG_IFEM)); } /* Wait until the complete message has been processed */ count = CRYP_TIMEOUT_GCMCCMHEADERPHASE; @@ -6071,7 +6093,8 @@ static HAL_StatusTypeDef CRYP_GCMCCM_SetHeaderPhase_DMA(CRYP_HandleTypeDef *hcry __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_SET(hcryp->Instance->SR, CRYP_FLAG_BUSY)); + } + while (HAL_IS_BIT_SET(hcryp->Instance->SR, CRYP_FLAG_BUSY)); #else /* AES */ @@ -6119,7 +6142,8 @@ static HAL_StatusTypeDef CRYP_GCMCCM_SetHeaderPhase_DMA(CRYP_HandleTypeDef *hcry __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); + } + while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); /* Clear CCF flag */ __HAL_CRYP_CLEAR_FLAG(hcryp, CRYP_CCF_CLEAR); @@ -6158,13 +6182,14 @@ static HAL_StatusTypeDef CRYP_GCMCCM_SetHeaderPhase_DMA(CRYP_HandleTypeDef *hcry __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); + } + while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); /* Clear CCF flag */ __HAL_CRYP_CLEAR_FLAG(hcryp, CRYP_CCF_CLEAR); } /* Last block optionally pad the data with zeros*/ - for (loopcounter = 0U; (loopcounter < ((headersize_in_bytes /4U) % 4U)); loopcounter++) + for (loopcounter = 0U; (loopcounter < ((headersize_in_bytes / 4U) % 4U)); loopcounter++) { hcryp->Instance->DINR = *(uint32_t *)(hcryp->Init.Header + hcryp->CrypHeaderCount); hcryp->CrypHeaderCount++ ; @@ -6211,7 +6236,8 @@ static HAL_StatusTypeDef CRYP_GCMCCM_SetHeaderPhase_DMA(CRYP_HandleTypeDef *hcry __HAL_UNLOCK(hcryp); return HAL_ERROR; } - } while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); + } + while (HAL_IS_BIT_CLR(hcryp->Instance->SR, AES_SR_CCF)); /* Clear CCF flag */ __HAL_CRYP_CLEAR_FLAG(hcryp, CRYP_CCF_CLEAR); @@ -6329,10 +6355,10 @@ static void CRYP_GCMCCM_SetHeaderPhase_IT(CRYP_HandleTypeDef *hcryp) loopcounter++; hcryp->CrypHeaderCount++; /* Pad the data with zeros to have a complete block */ - while (loopcounter < 4U) - { - hcryp->Instance->DIN = 0x0U; - loopcounter++; + while (loopcounter < 4U) + { + hcryp->Instance->DIN = 0x0U; + loopcounter++; hcryp->CrypHeaderCount++; } } @@ -6463,10 +6489,10 @@ static void CRYP_GCMCCM_SetHeaderPhase_IT(CRYP_HandleTypeDef *hcryp) loopcounter++; hcryp->CrypHeaderCount++; /* Pad the data with zeros to have a complete block */ - while (loopcounter < 4U) - { - hcryp->Instance->DINR = 0x0U; - loopcounter++; + while (loopcounter < 4U) + { + hcryp->Instance->DINR = 0x0U; + loopcounter++; hcryp->CrypHeaderCount++; } } diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dac.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dac.c index f6883eea..8953638e 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dac.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dac.c @@ -168,7 +168,7 @@ and a pointer to the user callback function. Use function HAL_DAC_UnRegisterCallback() to reset a callback to the default - weak (surcharged) function. It allows to reset following callbacks: + weak (overridden) function. It allows to reset following callbacks: (+) ConvCpltCallbackCh1 : callback when a half transfer is completed on Ch1. (+) ConvHalfCpltCallbackCh1 : callback when a transfer is completed on Ch1. (+) ErrorCallbackCh1 : callback when an error occurs on Ch1. @@ -183,9 +183,9 @@ This function) takes as parameters the HAL peripheral handle and the Callback ID. By default, after the HAL_DAC_Init and if the state is HAL_DAC_STATE_RESET - all callbacks are reset to the corresponding legacy weak (surcharged) functions. + all callbacks are reset to the corresponding legacy weak (overridden) functions. Exception done for MspInit and MspDeInit callbacks that are respectively - reset to the legacy weak (surcharged) functions in the HAL_DAC_Init + reset to the legacy weak (overridden) functions in the HAL_DAC_Init and HAL_DAC_DeInit only when these callbacks are null (not registered beforehand). If not, MspInit or MspDeInit are not null, the HAL_DAC_Init and HAL_DAC_DeInit keep and use the user MspInit/MspDeInit callbacks (registered beforehand) @@ -200,7 +200,7 @@ When The compilation define USE_HAL_DAC_REGISTER_CALLBACKS is set to 0 or not defined, the callback registering feature is not available - and weak (surcharged) callbacks are used. + and weak (overridden) callbacks are used. *** DAC HAL driver macros list *** ============================================= @@ -270,7 +270,7 @@ */ HAL_StatusTypeDef HAL_DAC_Init(DAC_HandleTypeDef *hdac) { - /* Check DAC handle */ + /* Check the DAC peripheral handle */ if (hdac == NULL) { return HAL_ERROR; @@ -331,7 +331,7 @@ HAL_StatusTypeDef HAL_DAC_Init(DAC_HandleTypeDef *hdac) */ HAL_StatusTypeDef HAL_DAC_DeInit(DAC_HandleTypeDef *hdac) { - /* Check DAC handle */ + /* Check the DAC peripheral handle */ if (hdac == NULL) { return HAL_ERROR; @@ -434,6 +434,12 @@ __weak void HAL_DAC_MspDeInit(DAC_HandleTypeDef *hdac) */ HAL_StatusTypeDef HAL_DAC_Start(DAC_HandleTypeDef *hdac, uint32_t Channel) { + /* Check the DAC peripheral handle */ + if (hdac == NULL) + { + return HAL_ERROR; + } + /* Check the parameters */ assert_param(IS_DAC_CHANNEL(Channel)); @@ -489,6 +495,12 @@ HAL_StatusTypeDef HAL_DAC_Start(DAC_HandleTypeDef *hdac, uint32_t Channel) */ HAL_StatusTypeDef HAL_DAC_Stop(DAC_HandleTypeDef *hdac, uint32_t Channel) { + /* Check the DAC peripheral handle */ + if (hdac == NULL) + { + return HAL_ERROR; + } + /* Check the parameters */ assert_param(IS_DAC_CHANNEL(Channel)); @@ -519,11 +531,17 @@ HAL_StatusTypeDef HAL_DAC_Stop(DAC_HandleTypeDef *hdac, uint32_t Channel) * @arg DAC_ALIGN_12B_R: 12bit right data alignment selected * @retval HAL status */ -HAL_StatusTypeDef HAL_DAC_Start_DMA(DAC_HandleTypeDef *hdac, uint32_t Channel, uint32_t *pData, uint32_t Length, +HAL_StatusTypeDef HAL_DAC_Start_DMA(DAC_HandleTypeDef *hdac, uint32_t Channel, const uint32_t *pData, uint32_t Length, uint32_t Alignment) { - HAL_StatusTypeDef status = HAL_OK; - uint32_t tmpreg = 0U; + HAL_StatusTypeDef status = HAL_ERROR; + uint32_t tmpreg; + + /* Check the DAC peripheral handle */ + if (hdac == NULL) + { + return HAL_ERROR; + } /* Check the parameters */ assert_param(IS_DAC_CHANNEL(Channel)); @@ -560,12 +578,10 @@ HAL_StatusTypeDef HAL_DAC_Start_DMA(DAC_HandleTypeDef *hdac, uint32_t Channel, u /* Get DHR12L1 address */ tmpreg = (uint32_t)&hdac->Instance->DHR12L1; break; - case DAC_ALIGN_8B_R: + default: /* case DAC_ALIGN_8B_R */ /* Get DHR8R1 address */ tmpreg = (uint32_t)&hdac->Instance->DHR8R1; break; - default: - break; } } #if defined(DAC_CHANNEL2_SUPPORT) @@ -594,17 +610,13 @@ HAL_StatusTypeDef HAL_DAC_Start_DMA(DAC_HandleTypeDef *hdac, uint32_t Channel, u /* Get DHR12L2 address */ tmpreg = (uint32_t)&hdac->Instance->DHR12L2; break; - case DAC_ALIGN_8B_R: + default: /* case DAC_ALIGN_8B_R */ /* Get DHR8R2 address */ tmpreg = (uint32_t)&hdac->Instance->DHR8R2; break; - default: - break; } } #endif /* DAC_CHANNEL2_SUPPORT */ - - /* Enable the DMA Stream */ if (Channel == DAC_CHANNEL_1) { /* Enable the DAC DMA underrun interrupt */ @@ -653,6 +665,12 @@ HAL_StatusTypeDef HAL_DAC_Start_DMA(DAC_HandleTypeDef *hdac, uint32_t Channel, u */ HAL_StatusTypeDef HAL_DAC_Stop_DMA(DAC_HandleTypeDef *hdac, uint32_t Channel) { + /* Check the DAC peripheral handle */ + if (hdac == NULL) + { + return HAL_ERROR; + } + /* Check the parameters */ assert_param(IS_DAC_CHANNEL(Channel)); @@ -701,10 +719,13 @@ HAL_StatusTypeDef HAL_DAC_Stop_DMA(DAC_HandleTypeDef *hdac, uint32_t Channel) */ void HAL_DAC_IRQHandler(DAC_HandleTypeDef *hdac) { - if (__HAL_DAC_GET_IT_SOURCE(hdac, DAC_IT_DMAUDR1)) + uint32_t itsource = hdac->Instance->CR; + uint32_t itflag = hdac->Instance->SR; + + if ((itsource & DAC_IT_DMAUDR1) == DAC_IT_DMAUDR1) { /* Check underrun flag of DAC channel 1 */ - if (__HAL_DAC_GET_FLAG(hdac, DAC_FLAG_DMAUDR1)) + if ((itflag & DAC_FLAG_DMAUDR1) == DAC_FLAG_DMAUDR1) { /* Change DAC state to error state */ hdac->State = HAL_DAC_STATE_ERROR; @@ -716,7 +737,7 @@ void HAL_DAC_IRQHandler(DAC_HandleTypeDef *hdac) __HAL_DAC_CLEAR_FLAG(hdac, DAC_FLAG_DMAUDR1); /* Disable the selected DAC channel1 DMA request */ - CLEAR_BIT(hdac->Instance->CR, DAC_CR_DMAEN1); + __HAL_DAC_DISABLE_IT(hdac, DAC_CR_DMAEN1); /* Error callback */ #if (USE_HAL_DAC_REGISTER_CALLBACKS == 1) @@ -728,10 +749,10 @@ void HAL_DAC_IRQHandler(DAC_HandleTypeDef *hdac) } #if defined(DAC_CHANNEL2_SUPPORT) - if (__HAL_DAC_GET_IT_SOURCE(hdac, DAC_IT_DMAUDR2)) + if ((itsource & DAC_IT_DMAUDR2) == DAC_IT_DMAUDR2) { /* Check underrun flag of DAC channel 2 */ - if (__HAL_DAC_GET_FLAG(hdac, DAC_FLAG_DMAUDR2)) + if ((itflag & DAC_FLAG_DMAUDR2) == DAC_FLAG_DMAUDR2) { /* Change DAC state to error state */ hdac->State = HAL_DAC_STATE_ERROR; @@ -743,7 +764,7 @@ void HAL_DAC_IRQHandler(DAC_HandleTypeDef *hdac) __HAL_DAC_CLEAR_FLAG(hdac, DAC_FLAG_DMAUDR2); /* Disable the selected DAC channel2 DMA request */ - CLEAR_BIT(hdac->Instance->CR, DAC_CR_DMAEN2); + __HAL_DAC_DISABLE_IT(hdac, DAC_CR_DMAEN2); /* Error callback */ #if (USE_HAL_DAC_REGISTER_CALLBACKS == 1) @@ -776,6 +797,12 @@ HAL_StatusTypeDef HAL_DAC_SetValue(DAC_HandleTypeDef *hdac, uint32_t Channel, ui { __IO uint32_t tmp = 0UL; + /* Check the DAC peripheral handle */ + if (hdac == NULL) + { + return HAL_ERROR; + } + /* Check the parameters */ assert_param(IS_DAC_CHANNEL(Channel)); assert_param(IS_DAC_ALIGN(Alignment)); @@ -893,10 +920,13 @@ __weak void HAL_DAC_DMAUnderrunCallbackCh1(DAC_HandleTypeDef *hdac) * @arg DAC_CHANNEL_2: DAC Channel2 selected * @retval The selected DAC channel data output value. */ -uint32_t HAL_DAC_GetValue(DAC_HandleTypeDef *hdac, uint32_t Channel) +uint32_t HAL_DAC_GetValue(const DAC_HandleTypeDef *hdac, uint32_t Channel) { uint32_t result = 0; + /* Check the DAC peripheral handle */ + assert_param(hdac != NULL); + /* Check the parameters */ assert_param(IS_DAC_CHANNEL(Channel)); @@ -925,11 +955,19 @@ uint32_t HAL_DAC_GetValue(DAC_HandleTypeDef *hdac, uint32_t Channel) * @arg DAC_CHANNEL_2: DAC Channel2 selected * @retval HAL status */ -HAL_StatusTypeDef HAL_DAC_ConfigChannel(DAC_HandleTypeDef *hdac, DAC_ChannelConfTypeDef *sConfig, uint32_t Channel) +HAL_StatusTypeDef HAL_DAC_ConfigChannel(DAC_HandleTypeDef *hdac, + const DAC_ChannelConfTypeDef *sConfig, uint32_t Channel) { + HAL_StatusTypeDef status = HAL_OK; uint32_t tmpreg1; uint32_t tmpreg2; + /* Check the DAC peripheral handle and channel configuration struct */ + if ((hdac == NULL) || (sConfig == NULL)) + { + return HAL_ERROR; + } + /* Check the DAC parameters */ assert_param(IS_DAC_TRIGGER(sConfig->DAC_Trigger)); assert_param(IS_DAC_OUTPUT_BUFFER_STATE(sConfig->DAC_OutputBuffer)); @@ -944,7 +982,8 @@ HAL_StatusTypeDef HAL_DAC_ConfigChannel(DAC_HandleTypeDef *hdac, DAC_ChannelConf /* Get the DAC CR value */ tmpreg1 = hdac->Instance->CR; /* Clear BOFFx, TENx, TSELx, WAVEx and MAMPx bits */ - tmpreg1 &= ~(((uint32_t)(DAC_CR_MAMP1 | DAC_CR_WAVE1 | DAC_CR_TSEL1 | DAC_CR_TEN1 | DAC_CR_BOFF1)) << (Channel & 0x10UL)); + tmpreg1 &= ~(((uint32_t)(DAC_CR_MAMP1 | DAC_CR_WAVE1 | DAC_CR_TSEL1 | DAC_CR_TEN1 | DAC_CR_BOFF1)) + << (Channel & 0x10UL)); /* Configure for the selected DAC channel: buffer output, trigger */ /* Set TSELx and TENx bits according to DAC_Trigger value */ /* Set BOFFx bit according to DAC_OutputBuffer value */ @@ -963,7 +1002,7 @@ HAL_StatusTypeDef HAL_DAC_ConfigChannel(DAC_HandleTypeDef *hdac, DAC_ChannelConf __HAL_UNLOCK(hdac); /* Return function status */ - return HAL_OK; + return status; } /** @@ -992,7 +1031,7 @@ HAL_StatusTypeDef HAL_DAC_ConfigChannel(DAC_HandleTypeDef *hdac, DAC_ChannelConf * the configuration information for the specified DAC. * @retval HAL state */ -HAL_DAC_StateTypeDef HAL_DAC_GetState(DAC_HandleTypeDef *hdac) +HAL_DAC_StateTypeDef HAL_DAC_GetState(const DAC_HandleTypeDef *hdac) { /* Return DAC handle state */ return hdac->State; @@ -1005,7 +1044,7 @@ HAL_DAC_StateTypeDef HAL_DAC_GetState(DAC_HandleTypeDef *hdac) * the configuration information for the specified DAC. * @retval DAC Error Code */ -uint32_t HAL_DAC_GetError(DAC_HandleTypeDef *hdac) +uint32_t HAL_DAC_GetError(const DAC_HandleTypeDef *hdac) { return hdac->ErrorCode; } @@ -1028,7 +1067,9 @@ uint32_t HAL_DAC_GetError(DAC_HandleTypeDef *hdac) #if (USE_HAL_DAC_REGISTER_CALLBACKS == 1) /** * @brief Register a User DAC Callback - * To be used instead of the weak (surcharged) predefined callback + * To be used instead of the weak (overridden) predefined callback + * @note The HAL_DAC_RegisterCallback() may be called before HAL_DAC_Init() in HAL_DAC_STATE_RESET to register + * callbacks for HAL_DAC_MSPINIT_CB_ID and HAL_DAC_MSPDEINIT_CB_ID * @param hdac DAC handle * @param CallbackID ID of the callback to be registered * This parameter can be one of the following values: @@ -1052,6 +1093,12 @@ HAL_StatusTypeDef HAL_DAC_RegisterCallback(DAC_HandleTypeDef *hdac, HAL_DAC_Call { HAL_StatusTypeDef status = HAL_OK; + /* Check the DAC peripheral handle */ + if (hdac == NULL) + { + return HAL_ERROR; + } + if (pCallback == NULL) { /* Update the error code */ @@ -1059,9 +1106,6 @@ HAL_StatusTypeDef HAL_DAC_RegisterCallback(DAC_HandleTypeDef *hdac, HAL_DAC_Call return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(hdac); - if (hdac->State == HAL_DAC_STATE_READY) { switch (CallbackID) @@ -1132,14 +1176,14 @@ HAL_StatusTypeDef HAL_DAC_RegisterCallback(DAC_HandleTypeDef *hdac, HAL_DAC_Call status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hdac); return status; } /** * @brief Unregister a User DAC Callback - * DAC Callback is redirected to the weak (surcharged) predefined callback + * DAC Callback is redirected to the weak (overridden) predefined callback + * @note The HAL_DAC_UnRegisterCallback() may be called before HAL_DAC_Init() in HAL_DAC_STATE_RESET to un-register + * callbacks for HAL_DAC_MSPINIT_CB_ID and HAL_DAC_MSPDEINIT_CB_ID * @param hdac DAC handle * @param CallbackID ID of the callback to be unregistered * This parameter can be one of the following values: @@ -1160,8 +1204,11 @@ HAL_StatusTypeDef HAL_DAC_UnRegisterCallback(DAC_HandleTypeDef *hdac, HAL_DAC_Ca { HAL_StatusTypeDef status = HAL_OK; - /* Process locked */ - __HAL_LOCK(hdac); + /* Check the DAC peripheral handle */ + if (hdac == NULL) + { + return HAL_ERROR; + } if (hdac->State == HAL_DAC_STATE_READY) { @@ -1247,8 +1294,6 @@ HAL_StatusTypeDef HAL_DAC_UnRegisterCallback(DAC_HandleTypeDef *hdac, HAL_DAC_Ca status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hdac); return status; } #endif /* USE_HAL_DAC_REGISTER_CALLBACKS */ @@ -1334,8 +1379,6 @@ void DAC_DMAErrorCh1(DMA_HandleTypeDef *hdma) #endif /* DAC */ #endif /* HAL_DAC_MODULE_ENABLED */ - /** * @} */ - diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dac_ex.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dac_ex.c index 343dd986..6e5ab512 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dac_ex.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dac_ex.c @@ -23,15 +23,6 @@ ##### How to use this driver ##### ============================================================================== [..] - - *** Dual mode IO operation *** - ============================== - [..] - (+) When Dual mode is enabled (i.e. DAC Channel1 and Channel2 are used simultaneously) : - Use HAL_DACEx_DualGetValue() to get digital data to be converted and use - HAL_DACEx_DualSetValue() to set digital value to converted simultaneously in - Channel 1 and Channel 2. - *** Signal generation operation *** =================================== [..] @@ -61,6 +52,7 @@ /* Private typedef -----------------------------------------------------------*/ /* Private define ------------------------------------------------------------*/ + /* Private macro -------------------------------------------------------------*/ /* Private variables ---------------------------------------------------------*/ /* Private function prototypes -----------------------------------------------*/ @@ -100,6 +92,12 @@ HAL_StatusTypeDef HAL_DACEx_DualStart(DAC_HandleTypeDef *hdac) { uint32_t tmp_swtrig = 0UL; + /* Check the DAC peripheral handle */ + if (hdac == NULL) + { + return HAL_ERROR; + } + /* Process locked */ __HAL_LOCK(hdac); @@ -141,6 +139,12 @@ HAL_StatusTypeDef HAL_DACEx_DualStart(DAC_HandleTypeDef *hdac) */ HAL_StatusTypeDef HAL_DACEx_DualStop(DAC_HandleTypeDef *hdac) { + /* Check the DAC peripheral handle */ + if (hdac == NULL) + { + return HAL_ERROR; + } + /* Disable the Peripheral */ __HAL_DAC_DISABLE(hdac, DAC_CHANNEL_1); @@ -180,6 +184,12 @@ HAL_StatusTypeDef HAL_DACEx_DualStop(DAC_HandleTypeDef *hdac) */ HAL_StatusTypeDef HAL_DACEx_TriangleWaveGenerate(DAC_HandleTypeDef *hdac, uint32_t Channel, uint32_t Amplitude) { + /* Check the DAC peripheral handle */ + if (hdac == NULL) + { + return HAL_ERROR; + } + /* Check the parameters */ assert_param(IS_DAC_CHANNEL(Channel)); assert_param(IS_DAC_LFSR_UNMASK_TRIANGLE_AMPLITUDE(Amplitude)); @@ -230,6 +240,12 @@ HAL_StatusTypeDef HAL_DACEx_TriangleWaveGenerate(DAC_HandleTypeDef *hdac, uint32 */ HAL_StatusTypeDef HAL_DACEx_NoiseWaveGenerate(DAC_HandleTypeDef *hdac, uint32_t Channel, uint32_t Amplitude) { + /* Check the DAC peripheral handle */ + if (hdac == NULL) + { + return HAL_ERROR; + } + /* Check the parameters */ assert_param(IS_DAC_CHANNEL(Channel)); assert_param(IS_DAC_LFSR_UNMASK_TRIANGLE_AMPLITUDE(Amplitude)); @@ -275,6 +291,12 @@ HAL_StatusTypeDef HAL_DACEx_DualSetValue(DAC_HandleTypeDef *hdac, uint32_t Align uint32_t data; uint32_t tmp; + /* Check the DAC peripheral handle */ + if (hdac == NULL) + { + return HAL_ERROR; + } + /* Check the parameters */ assert_param(IS_DAC_ALIGN(Alignment)); assert_param(IS_DAC_DATA(Data1)); @@ -391,7 +413,7 @@ __weak void HAL_DACEx_DMAUnderrunCallbackCh2(DAC_HandleTypeDef *hdac) * the configuration information for the specified DAC. * @retval The selected DAC channel data output value. */ -uint32_t HAL_DACEx_DualGetValue(DAC_HandleTypeDef *hdac) +uint32_t HAL_DACEx_DualGetValue(const DAC_HandleTypeDef *hdac) { uint32_t tmp = 0UL; @@ -492,4 +514,3 @@ void DAC_DMAErrorCh2(DMA_HandleTypeDef *hdma) /** * @} */ - diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dfsdm.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dfsdm.c index 5279edf0..63126be3 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dfsdm.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dfsdm.c @@ -324,7 +324,7 @@ DFSDM_Channel_HandleTypeDef* a_dfsdm2ChannelHandle[DFSDM2_CHANNEL_NUMBER] = {NUL * @{ */ static uint32_t DFSDM_GetInjChannelsNbr(uint32_t Channels); -static uint32_t DFSDM_GetChannelFromInstance(DFSDM_Channel_TypeDef* Instance); +static uint32_t DFSDM_GetChannelFromInstance(const DFSDM_Channel_TypeDef *Instance); static void DFSDM_RegConvStart(DFSDM_Filter_HandleTypeDef *hdfsdm_filter); static void DFSDM_RegConvStop(DFSDM_Filter_HandleTypeDef* hdfsdm_filter); static void DFSDM_InjConvStart(DFSDM_Filter_HandleTypeDef* hdfsdm_filter); @@ -960,7 +960,7 @@ HAL_StatusTypeDef HAL_DFSDM_ChannelCkabStart(DFSDM_Channel_HandleTypeDef *hdfsdm * @param Timeout Timeout value in milliseconds. * @retval HAL status */ -HAL_StatusTypeDef HAL_DFSDM_ChannelPollForCkab(DFSDM_Channel_HandleTypeDef *hdfsdm_channel, +HAL_StatusTypeDef HAL_DFSDM_ChannelPollForCkab(const DFSDM_Channel_HandleTypeDef *hdfsdm_channel, uint32_t Timeout) { uint32_t tickstart; @@ -1329,7 +1329,7 @@ HAL_StatusTypeDef HAL_DFSDM_ChannelScdStart(DFSDM_Channel_HandleTypeDef *hdfsdm_ * @param Timeout Timeout value in milliseconds. * @retval HAL status */ -HAL_StatusTypeDef HAL_DFSDM_ChannelPollForScd(DFSDM_Channel_HandleTypeDef *hdfsdm_channel, +HAL_StatusTypeDef HAL_DFSDM_ChannelPollForScd(const DFSDM_Channel_HandleTypeDef *hdfsdm_channel, uint32_t Timeout) { uint32_t tickstart; @@ -1596,7 +1596,7 @@ HAL_StatusTypeDef HAL_DFSDM_ChannelScdStop_IT(DFSDM_Channel_HandleTypeDef *hdfsd * @param hdfsdm_channel DFSDM channel handle. * @retval Channel analog watchdog value. */ -int16_t HAL_DFSDM_ChannelGetAwdValue(DFSDM_Channel_HandleTypeDef *hdfsdm_channel) +int16_t HAL_DFSDM_ChannelGetAwdValue(const DFSDM_Channel_HandleTypeDef *hdfsdm_channel) { return (int16_t) hdfsdm_channel->Instance->CHWDATAR; } @@ -1655,7 +1655,7 @@ HAL_StatusTypeDef HAL_DFSDM_ChannelModifyOffset(DFSDM_Channel_HandleTypeDef *hdf * @param hdfsdm_channel DFSDM channel handle. * @retval DFSDM channel state. */ -HAL_DFSDM_Channel_StateTypeDef HAL_DFSDM_ChannelGetState(DFSDM_Channel_HandleTypeDef *hdfsdm_channel) +HAL_DFSDM_Channel_StateTypeDef HAL_DFSDM_ChannelGetState(const DFSDM_Channel_HandleTypeDef *hdfsdm_channel) { /* Return DFSDM channel handle state */ return hdfsdm_channel->State; @@ -2637,7 +2637,7 @@ HAL_StatusTypeDef HAL_DFSDM_FilterRegularStop_DMA(DFSDM_Filter_HandleTypeDef *hd * @param Channel Corresponding channel of regular conversion. * @retval Regular conversion value */ -int32_t HAL_DFSDM_FilterGetRegularValue(DFSDM_Filter_HandleTypeDef *hdfsdm_filter, +int32_t HAL_DFSDM_FilterGetRegularValue(const DFSDM_Filter_HandleTypeDef *hdfsdm_filter, uint32_t *Channel) { uint32_t reg = 0U; @@ -3051,7 +3051,7 @@ HAL_StatusTypeDef HAL_DFSDM_FilterInjectedStop_DMA(DFSDM_Filter_HandleTypeDef *h * @param Channel Corresponding channel of injected conversion. * @retval Injected conversion value */ -int32_t HAL_DFSDM_FilterGetInjectedValue(DFSDM_Filter_HandleTypeDef *hdfsdm_filter, +int32_t HAL_DFSDM_FilterGetInjectedValue(const DFSDM_Filter_HandleTypeDef *hdfsdm_filter, uint32_t *Channel) { uint32_t reg = 0U; @@ -3079,7 +3079,7 @@ int32_t HAL_DFSDM_FilterGetInjectedValue(DFSDM_Filter_HandleTypeDef *hdfsdm_filt * @retval HAL status */ HAL_StatusTypeDef HAL_DFSDM_FilterAwdStart_IT(DFSDM_Filter_HandleTypeDef *hdfsdm_filter, - DFSDM_Filter_AwdParamTypeDef *awdParam) + const DFSDM_Filter_AwdParamTypeDef *awdParam) { HAL_StatusTypeDef status = HAL_OK; @@ -3236,7 +3236,7 @@ HAL_StatusTypeDef HAL_DFSDM_FilterExdStop(DFSDM_Filter_HandleTypeDef *hdfsdm_fil * @retval Extreme detector maximum value * This value is between Min_Data = -8388608 and Max_Data = 8388607. */ -int32_t HAL_DFSDM_FilterGetExdMaxValue(DFSDM_Filter_HandleTypeDef *hdfsdm_filter, +int32_t HAL_DFSDM_FilterGetExdMaxValue(const DFSDM_Filter_HandleTypeDef *hdfsdm_filter, uint32_t *Channel) { uint32_t reg = 0U; @@ -3264,7 +3264,7 @@ int32_t HAL_DFSDM_FilterGetExdMaxValue(DFSDM_Filter_HandleTypeDef *hdfsdm_filter * @retval Extreme detector minimum value * This value is between Min_Data = -8388608 and Max_Data = 8388607. */ -int32_t HAL_DFSDM_FilterGetExdMinValue(DFSDM_Filter_HandleTypeDef *hdfsdm_filter, +int32_t HAL_DFSDM_FilterGetExdMinValue(const DFSDM_Filter_HandleTypeDef *hdfsdm_filter, uint32_t *Channel) { uint32_t reg = 0U; @@ -3291,7 +3291,7 @@ int32_t HAL_DFSDM_FilterGetExdMinValue(DFSDM_Filter_HandleTypeDef *hdfsdm_filter * @retval Conversion time value * @note To get time in second, this value has to be divided by DFSDM clock frequency. */ -uint32_t HAL_DFSDM_FilterGetConvTimeValue(DFSDM_Filter_HandleTypeDef *hdfsdm_filter) +uint32_t HAL_DFSDM_FilterGetConvTimeValue(const DFSDM_Filter_HandleTypeDef *hdfsdm_filter) { uint32_t reg = 0U; uint32_t value = 0U; @@ -3676,7 +3676,7 @@ __weak void HAL_DFSDM_FilterErrorCallback(DFSDM_Filter_HandleTypeDef *hdfsdm_fil * @param hdfsdm_filter DFSDM filter handle. * @retval DFSDM filter state. */ -HAL_DFSDM_Filter_StateTypeDef HAL_DFSDM_FilterGetState(DFSDM_Filter_HandleTypeDef *hdfsdm_filter) +HAL_DFSDM_Filter_StateTypeDef HAL_DFSDM_FilterGetState(const DFSDM_Filter_HandleTypeDef *hdfsdm_filter) { /* Return DFSDM filter handle state */ return hdfsdm_filter->State; @@ -3687,7 +3687,7 @@ HAL_DFSDM_Filter_StateTypeDef HAL_DFSDM_FilterGetState(DFSDM_Filter_HandleTypeDe * @param hdfsdm_filter DFSDM filter handle. * @retval DFSDM filter error code. */ -uint32_t HAL_DFSDM_FilterGetError(DFSDM_Filter_HandleTypeDef *hdfsdm_filter) +uint32_t HAL_DFSDM_FilterGetError(const DFSDM_Filter_HandleTypeDef *hdfsdm_filter) { return hdfsdm_filter->ErrorCode; } @@ -4183,7 +4183,7 @@ static uint32_t DFSDM_GetInjChannelsNbr(uint32_t Channels) * @param Instance DFSDM channel instance. * @retval Channel number. */ -static uint32_t DFSDM_GetChannelFromInstance(DFSDM_Channel_TypeDef* Instance) +static uint32_t DFSDM_GetChannelFromInstance(const DFSDM_Channel_TypeDef *Instance) { uint32_t channel; diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dma2d.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dma2d.c index 4cfac406..76f2b467 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dma2d.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dma2d.c @@ -118,7 +118,7 @@ and a pointer to the user callback function. (#) Use function @ref HAL_DMA2D_UnRegisterCallback() to reset a callback to the default - weak (surcharged) function. + weak (overridden) function. @ref HAL_DMA2D_UnRegisterCallback() takes as parameters the HAL peripheral handle, and the Callback ID. This function allows to reset following callbacks: @@ -130,16 +130,16 @@ (+) MspDeInitCallback : DMA2D MspDeInit. (#) By default, after the @ref HAL_DMA2D_Init and if the state is HAL_DMA2D_STATE_RESET - all callbacks are reset to the corresponding legacy weak (surcharged) functions: + all callbacks are reset to the corresponding legacy weak (overridden) functions: examples @ref HAL_DMA2D_LineEventCallback(), @ref HAL_DMA2D_CLUTLoadingCpltCallback() Exception done for MspInit and MspDeInit callbacks that are respectively - reset to the legacy weak (surcharged) functions in the @ref HAL_DMA2D_Init + reset to the legacy weak (overridden) functions in the @ref HAL_DMA2D_Init and @ref HAL_DMA2D_DeInit only when these callbacks are null (not registered beforehand) If not, MspInit or MspDeInit are not null, the @ref HAL_DMA2D_Init and @ref HAL_DMA2D_DeInit keep and use the user MspInit/MspDeInit callbacks (registered beforehand). Exception as well for Transfer Completion and Transfer Error callbacks that are not defined - as weak (surcharged) functions. They must be defined by the user to be resorted to. + as weak (overridden) functions. They must be defined by the user to be resorted to. Callbacks can be registered/unregistered in READY state only. Exception done for MspInit/MspDeInit callbacks that can be registered/unregistered @@ -151,7 +151,7 @@ When The compilation define USE_HAL_DMA2D_REGISTER_CALLBACKS is set to 0 or not defined, the callback registering feature is not available - and weak (surcharged) callbacks are used. + and weak (overridden) callbacks are used. [..] (@) You can refer to the DMA2D HAL driver header file for more useful macros @@ -422,7 +422,7 @@ __weak void HAL_DMA2D_MspDeInit(DMA2D_HandleTypeDef *hdma2d) #if (USE_HAL_DMA2D_REGISTER_CALLBACKS == 1) /** * @brief Register a User DMA2D Callback - * To be used instead of the weak (surcharged) predefined callback + * To be used instead of the weak (overridden) predefined callback * @param hdma2d DMA2D handle * @param CallbackID ID of the callback to be registered * This parameter can be one of the following values: @@ -521,7 +521,7 @@ HAL_StatusTypeDef HAL_DMA2D_RegisterCallback(DMA2D_HandleTypeDef *hdma2d, HAL_DM /** * @brief Unregister a DMA2D Callback - * DMA2D Callback is redirected to the weak (surcharged) predefined callback + * DMA2D Callback is redirected to the weak (overridden) predefined callback * @param hdma2d DMA2D handle * @param CallbackID ID of the callback to be unregistered * This parameter can be one of the following values: @@ -562,11 +562,11 @@ HAL_StatusTypeDef HAL_DMA2D_UnRegisterCallback(DMA2D_HandleTypeDef *hdma2d, HAL_ break; case HAL_DMA2D_MSPINIT_CB_ID : - hdma2d->MspInitCallback = HAL_DMA2D_MspInit; /* Legacy weak (surcharged) Msp Init */ + hdma2d->MspInitCallback = HAL_DMA2D_MspInit; /* Legacy weak (overridden) Msp Init */ break; case HAL_DMA2D_MSPDEINIT_CB_ID : - hdma2d->MspDeInitCallback = HAL_DMA2D_MspDeInit; /* Legacy weak (surcharged) Msp DeInit */ + hdma2d->MspDeInitCallback = HAL_DMA2D_MspDeInit; /* Legacy weak (overridden) Msp DeInit */ break; default : @@ -582,11 +582,11 @@ HAL_StatusTypeDef HAL_DMA2D_UnRegisterCallback(DMA2D_HandleTypeDef *hdma2d, HAL_ switch (CallbackID) { case HAL_DMA2D_MSPINIT_CB_ID : - hdma2d->MspInitCallback = HAL_DMA2D_MspInit; /* Legacy weak (surcharged) Msp Init */ + hdma2d->MspInitCallback = HAL_DMA2D_MspInit; /* Legacy weak (overridden) Msp Init */ break; case HAL_DMA2D_MSPDEINIT_CB_ID : - hdma2d->MspDeInitCallback = HAL_DMA2D_MspDeInit; /* Legacy weak (surcharged) Msp DeInit */ + hdma2d->MspDeInitCallback = HAL_DMA2D_MspDeInit; /* Legacy weak (overridden) Msp DeInit */ break; default : diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dsi.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dsi.c index bde68610..b193c24e 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dsi.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_dsi.c @@ -1835,6 +1835,95 @@ HAL_StatusTypeDef HAL_DSI_EnterULPMData(DSI_HandleTypeDef *hdsi) /* Process locked */ __HAL_LOCK(hdsi); + /* Verify the initial status of the DSI Host */ + + /* Verify that the clock lane and the digital section of the D-PHY are enabled */ + if ((hdsi->Instance->PCTLR & (DSI_PCTLR_CKE | DSI_PCTLR_DEN)) != (DSI_PCTLR_CKE | DSI_PCTLR_DEN)) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + + /* Verify that the D-PHY PLL and the reference bias are enabled */ + if ((hdsi->Instance->WRPCR & DSI_WRPCR_PLLEN) != DSI_WRPCR_PLLEN) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + else if ((hdsi->Instance->WRPCR & DSI_WRPCR_REGEN) != DSI_WRPCR_REGEN) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + else + { + /* Nothing to do */ + } + + /* Verify that there are no ULPS exit or request on data lanes */ + if ((hdsi->Instance->PUCR & (DSI_PUCR_UEDL | DSI_PUCR_URDL)) != 0U) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + + /* Verify that there are no Transmission trigger */ + if ((hdsi->Instance->PTTCR & DSI_PTTCR_TX_TRIG) != 0U) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + + /* Requires min of 400us delay before reading the PLLLS flag */ + /* 1ms delay is inserted that is the minimum HAL delay granularity */ + HAL_Delay(1); + + /* Verify that D-PHY PLL is locked */ + tickstart = HAL_GetTick(); + + while ((__HAL_DSI_GET_FLAG(hdsi, DSI_FLAG_PLLLS) == 0U)) + { + /* Check for the Timeout */ + if ((HAL_GetTick() - tickstart) > DSI_TIMEOUT_VALUE) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + + return HAL_TIMEOUT; + } + } + + /* Verify that all active lanes are in Stop state */ + if ((hdsi->Instance->PCONFR & DSI_PCONFR_NL) == DSI_ONE_DATA_LANE) + { + if ((hdsi->Instance->PSR & DSI_PSR_UAN0) != DSI_PSR_UAN0) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + } + else if ((hdsi->Instance->PCONFR & DSI_PCONFR_NL) == DSI_TWO_DATA_LANES) + { + if ((hdsi->Instance->PSR & (DSI_PSR_UAN0 | DSI_PSR_UAN1)) != (DSI_PSR_UAN0 | DSI_PSR_UAN1)) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + } + else + { + /* Process unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + /* ULPS Request on Data Lanes */ hdsi->Instance->PUCR |= DSI_PUCR_URDL; @@ -1898,6 +1987,58 @@ HAL_StatusTypeDef HAL_DSI_ExitULPMData(DSI_HandleTypeDef *hdsi) /* Process locked */ __HAL_LOCK(hdsi); + /* Verify that all active lanes are in ULPM */ + if ((hdsi->Instance->PCONFR & DSI_PCONFR_NL) == DSI_ONE_DATA_LANE) + { + if ((hdsi->Instance->PSR & DSI_PSR_UAN0) != 0U) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + + return HAL_ERROR; + } + } + else if ((hdsi->Instance->PCONFR & DSI_PCONFR_NL) == DSI_TWO_DATA_LANES) + { + if ((hdsi->Instance->PSR & (DSI_PSR_UAN0 | DSI_PSR_UAN1)) != 0U) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + + return HAL_ERROR; + } + } + else + { + /* Process unlocked */ + __HAL_UNLOCK(hdsi); + + return HAL_ERROR; + } + + /* Turn on the DSI PLL */ + __HAL_DSI_PLL_ENABLE(hdsi); + + /* Requires min of 400us delay before reading the PLLLS flag */ + /* 1ms delay is inserted that is the minimum HAL delay granularity */ + HAL_Delay(1); + + /* Get tick */ + tickstart = HAL_GetTick(); + + /* Wait for the lock of the PLL */ + while (__HAL_DSI_GET_FLAG(hdsi, DSI_FLAG_PLLLS) == 0U) + { + /* Check for the Timeout */ + if ((HAL_GetTick() - tickstart) > DSI_TIMEOUT_VALUE) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + + return HAL_TIMEOUT; + } + } + /* Exit ULPS on Data Lanes */ hdsi->Instance->PUCR |= DSI_PUCR_UEDL; @@ -1947,6 +2088,61 @@ HAL_StatusTypeDef HAL_DSI_ExitULPMData(DSI_HandleTypeDef *hdsi) /* De-assert the ULPM requests and the ULPM exit bits */ hdsi->Instance->PUCR = 0U; + /* Verify that D-PHY PLL is enabled */ + if ((hdsi->Instance->WRPCR & DSI_WRPCR_PLLEN) != DSI_WRPCR_PLLEN) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + + /* Verify that all active lanes are in Stop state */ + if ((hdsi->Instance->PCONFR & DSI_PCONFR_NL) == DSI_ONE_DATA_LANE) + { + if ((hdsi->Instance->PSR & DSI_PSR_UAN0) != DSI_PSR_UAN0) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + } + else if ((hdsi->Instance->PCONFR & DSI_PCONFR_NL) == DSI_TWO_DATA_LANES) + { + if ((hdsi->Instance->PSR & (DSI_PSR_UAN0 | DSI_PSR_UAN1)) != (DSI_PSR_UAN0 | DSI_PSR_UAN1)) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + } + else + { + /* Process unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + + /* Verify that D-PHY PLL is locked */ + /* Requires min of 400us delay before reading the PLLLS flag */ + /* 1ms delay is inserted that is the minimum HAL delay granularity */ + HAL_Delay(1); + + /* Get tick */ + tickstart = HAL_GetTick(); + + /* Wait for the lock of the PLL */ + while (__HAL_DSI_GET_FLAG(hdsi, DSI_FLAG_PLLLS) == 0U) + { + /* Check for the Timeout */ + if ((HAL_GetTick() - tickstart) > DSI_TIMEOUT_VALUE) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + + return HAL_TIMEOUT; + } + } + /* Process unlocked */ __HAL_UNLOCK(hdsi); @@ -1967,6 +2163,96 @@ HAL_StatusTypeDef HAL_DSI_EnterULPM(DSI_HandleTypeDef *hdsi) /* Process locked */ __HAL_LOCK(hdsi); + /* Verify the initial status of the DSI Host */ + + /* Verify that the clock lane and the digital section of the D-PHY are enabled */ + if ((hdsi->Instance->PCTLR & (DSI_PCTLR_CKE | DSI_PCTLR_DEN)) != (DSI_PCTLR_CKE | DSI_PCTLR_DEN)) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + + /* Verify that the D-PHY PLL and the reference bias are enabled */ + if ((hdsi->Instance->WRPCR & DSI_WRPCR_PLLEN) != DSI_WRPCR_PLLEN) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + else if ((hdsi->Instance->WRPCR & DSI_WRPCR_REGEN) != DSI_WRPCR_REGEN) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + else + { + /* Nothing to do */ + } + + /* Verify that there are no ULPS exit or request on both data and clock lanes */ + if ((hdsi->Instance->PUCR & (DSI_PUCR_UEDL | DSI_PUCR_URDL | DSI_PUCR_UECL | DSI_PUCR_URCL)) != 0U) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + + /* Verify that there are no Transmission trigger */ + if ((hdsi->Instance->PTTCR & DSI_PTTCR_TX_TRIG) != 0U) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + + /* Requires min of 400us delay before reading the PLLLS flag */ + /* 1ms delay is inserted that is the minimum HAL delay granularity */ + HAL_Delay(1); + + /* Verify that D-PHY PLL is locked */ + tickstart = HAL_GetTick(); + + while ((__HAL_DSI_GET_FLAG(hdsi, DSI_FLAG_PLLLS) == 0U)) + { + /* Check for the Timeout */ + if ((HAL_GetTick() - tickstart) > DSI_TIMEOUT_VALUE) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + + return HAL_TIMEOUT; + } + } + + /* Verify that all active lanes are in Stop state */ + if ((hdsi->Instance->PCONFR & DSI_PCONFR_NL) == DSI_ONE_DATA_LANE) + { + if ((hdsi->Instance->PSR & (DSI_PSR_UAN0 | DSI_PSR_PSS0)) != (DSI_PSR_UAN0 | DSI_PSR_PSS0)) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + } + else if ((hdsi->Instance->PCONFR & DSI_PCONFR_NL) == DSI_TWO_DATA_LANES) + { + if ((hdsi->Instance->PSR & (DSI_PSR_UAN0 | DSI_PSR_PSS0 | DSI_PSR_PSS1 | \ + DSI_PSR_UAN1)) != (DSI_PSR_UAN0 | DSI_PSR_PSS0 | DSI_PSR_PSS1 | DSI_PSR_UAN1)) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + } + else + { + /* Process unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + /* Clock lane configuration: no more HS request */ hdsi->Instance->CLCR &= ~DSI_CLCR_DPCC; @@ -1979,7 +2265,7 @@ HAL_StatusTypeDef HAL_DSI_EnterULPM(DSI_HandleTypeDef *hdsi) /* Get tick */ tickstart = HAL_GetTick(); - /* Wait until all active lanes exit ULPM */ + /* Wait until all active lanes enter ULPM */ if ((hdsi->Instance->PCONFR & DSI_PCONFR_NL) == DSI_ONE_DATA_LANE) { while ((hdsi->Instance->PSR & (DSI_PSR_UAN0 | DSI_PSR_UANC)) != 0U) @@ -2039,9 +2325,44 @@ HAL_StatusTypeDef HAL_DSI_ExitULPM(DSI_HandleTypeDef *hdsi) /* Process locked */ __HAL_LOCK(hdsi); + /* Verify that all active lanes are in ULPM */ + if ((hdsi->Instance->PCONFR & DSI_PCONFR_NL) == DSI_ONE_DATA_LANE) + { + if ((hdsi->Instance->PSR & (DSI_PSR_RUE0 | DSI_PSR_UAN0 | DSI_PSR_PSS0 | \ + DSI_PSR_UANC | DSI_PSR_PSSC | DSI_PSR_PD)) != 0U) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + + return HAL_ERROR; + } + } + else if ((hdsi->Instance->PCONFR & DSI_PCONFR_NL) == DSI_TWO_DATA_LANES) + { + if ((hdsi->Instance->PSR & (DSI_PSR_RUE0 | DSI_PSR_UAN0 | DSI_PSR_PSS0 | DSI_PSR_UAN1 | \ + DSI_PSR_PSS1 | DSI_PSR_UANC | DSI_PSR_PSSC | DSI_PSR_PD)) != 0U) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + + return HAL_ERROR; + } + } + else + { + /* Process unlocked */ + __HAL_UNLOCK(hdsi); + + return HAL_ERROR; + } + /* Turn on the DSI PLL */ __HAL_DSI_PLL_ENABLE(hdsi); + /* Requires min of 400us delay before reading the PLLLS flag */ + /* 1ms delay is inserted that is the minimum HAL delay granularity */ + HAL_Delay(1); + /* Get tick */ tickstart = HAL_GetTick(); @@ -2114,6 +2435,62 @@ HAL_StatusTypeDef HAL_DSI_ExitULPM(DSI_HandleTypeDef *hdsi) /* Restore clock lane configuration to HS */ hdsi->Instance->CLCR |= DSI_CLCR_DPCC; + /* Verify that D-PHY PLL is enabled */ + if ((hdsi->Instance->WRPCR & DSI_WRPCR_PLLEN) != DSI_WRPCR_PLLEN) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + + /* Verify that all active lanes are in Stop state */ + if ((hdsi->Instance->PCONFR & DSI_PCONFR_NL) == DSI_ONE_DATA_LANE) + { + if ((hdsi->Instance->PSR & (DSI_PSR_UAN0 | DSI_PSR_PSS0)) != (DSI_PSR_UAN0 | DSI_PSR_PSS0)) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + } + else if ((hdsi->Instance->PCONFR & DSI_PCONFR_NL) == DSI_TWO_DATA_LANES) + { + if ((hdsi->Instance->PSR & (DSI_PSR_UAN0 | DSI_PSR_PSS0 | DSI_PSR_PSS1 | \ + DSI_PSR_UAN1)) != (DSI_PSR_UAN0 | DSI_PSR_PSS0 | DSI_PSR_PSS1 | DSI_PSR_UAN1)) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + } + else + { + /* Process unlocked */ + __HAL_UNLOCK(hdsi); + return HAL_ERROR; + } + + /* Verify that D-PHY PLL is locked */ + /* Requires min of 400us delay before reading the PLLLS flag */ + /* 1ms delay is inserted that is the minimum HAL delay granularity */ + HAL_Delay(1); + + /* Get tick */ + tickstart = HAL_GetTick(); + + /* Wait for the lock of the PLL */ + while (__HAL_DSI_GET_FLAG(hdsi, DSI_FLAG_PLLLS) == 0U) + { + /* Check for the Timeout */ + if ((HAL_GetTick() - tickstart) > DSI_TIMEOUT_VALUE) + { + /* Process Unlocked */ + __HAL_UNLOCK(hdsi); + + return HAL_TIMEOUT; + } + } + /* Process unlocked */ __HAL_UNLOCK(hdsi); diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_eth.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_eth.c index 634da3fd..ff0cfec1 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_eth.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_eth.c @@ -501,7 +501,6 @@ HAL_StatusTypeDef HAL_ETH_RegisterCallback(ETH_HandleTypeDef *heth, HAL_ETH_Call { /* Update the error code */ heth->ErrorCode |= HAL_ETH_ERROR_INVALID_CALLBACK; - return HAL_ERROR; } @@ -579,7 +578,7 @@ HAL_StatusTypeDef HAL_ETH_RegisterCallback(ETH_HandleTypeDef *heth, HAL_ETH_Call /** * @brief Unregister an ETH Callback - * ETH callabck is redirected to the weak predefined callback + * ETH callback is redirected to the weak predefined callback * @param heth eth handle * @param CallbackID ID of the callback to be unregistered * This parameter can be one of the following values: @@ -702,7 +701,7 @@ HAL_StatusTypeDef HAL_ETH_Start(ETH_HandleTypeDef *heth) { heth->gState = HAL_ETH_STATE_BUSY; - /* Set nombre of descriptors to build */ + /* Set number of descriptors to build */ heth->RxDescList.RxBuildDescCnt = ETH_RX_DESC_CNT; /* Build all descriptors */ @@ -772,7 +771,7 @@ HAL_StatusTypeDef HAL_ETH_Start_IT(ETH_HandleTypeDef *heth) SET_BIT(heth->Instance->MMCTIMR, ETH_MMCTIMR_TGFM | ETH_MMCTIMR_TGFMSCM | \ ETH_MMCTIMR_TGFSCM); - /* Set nombre of descriptors to build */ + /* Set number of descriptors to build */ heth->RxDescList.RxBuildDescCnt = ETH_RX_DESC_CNT; /* Build all descriptors */ @@ -836,6 +835,7 @@ HAL_StatusTypeDef HAL_ETH_Stop(ETH_HandleTypeDef *heth) { /* Set the ETH peripheral state to BUSY */ heth->gState = HAL_ETH_STATE_BUSY; + /* Disable the DMA transmission */ CLEAR_BIT(heth->Instance->DMAOMR, ETH_DMAOMR_ST); @@ -903,6 +903,7 @@ HAL_StatusTypeDef HAL_ETH_Stop_IT(ETH_HandleTypeDef *heth) /* Disable the MAC reception */ CLEAR_BIT(heth->Instance->MACCR, ETH_MACCR_RE); + /* Wait until the write operation will be taken into account : at least four TX_CLK/RX_CLK clock cycles */ tmpreg1 = (heth->Instance)->MACCR; @@ -1085,7 +1086,6 @@ HAL_StatusTypeDef HAL_ETH_ReadData(ETH_HandleTypeDef *heth, void **pAppBuff) uint32_t bufflength; uint8_t rxdataready = 0U; - if (pAppBuff == NULL) { heth->ErrorCode |= HAL_ETH_ERROR_PARAM; @@ -1108,9 +1108,9 @@ HAL_StatusTypeDef HAL_ETH_ReadData(ETH_HandleTypeDef *heth, void **pAppBuff) if (READ_BIT(dmarxdesc->DESC0, ETH_DMARXDESC_LS) != (uint32_t)RESET) { /* Get timestamp high */ - heth->RxDescList.TimeStamp.TimeStampHigh = dmarxdesc->DESC6; + heth->RxDescList.TimeStamp.TimeStampHigh = dmarxdesc->DESC7; /* Get timestamp low */ - heth->RxDescList.TimeStamp.TimeStampLow = dmarxdesc->DESC7; + heth->RxDescList.TimeStamp.TimeStampLow = dmarxdesc->DESC6; } if ((READ_BIT(dmarxdesc->DESC0, ETH_DMARXDESC_FS) != (uint32_t)RESET) || (heth->RxDescList.pRxStart != NULL)) { @@ -1193,6 +1193,7 @@ HAL_StatusTypeDef HAL_ETH_ReadData(ETH_HandleTypeDef *heth, void **pAppBuff) */ static void ETH_UpdateDescriptor(ETH_HandleTypeDef *heth) { + uint32_t tailidx; uint32_t descidx; uint32_t desccount; ETH_DMADescTypeDef *dmarxdesc; @@ -1238,12 +1239,6 @@ static void ETH_UpdateDescriptor(ETH_HandleTypeDef *heth) WRITE_REG(dmarxdesc->DESC1, ETH_RX_BUF_SIZE | ETH_DMARXDESC_RCH); } - /* Before transferring the ownership to DMA, make sure that the RX descriptors bits writing - is fully performed. - The __DMB() instruction is added to avoid any potential compiler optimization that - may lead to abnormal behavior. */ - __DMB(); - SET_BIT(dmarxdesc->DESC0, ETH_DMARXDESC_OWN); /* Increment current rx descriptor index */ @@ -1256,8 +1251,14 @@ static void ETH_UpdateDescriptor(ETH_HandleTypeDef *heth) if (heth->RxDescList.RxBuildDescCnt != desccount) { + /* Set the tail pointer index */ + tailidx = (descidx + 1U) % ETH_RX_DESC_CNT; + + /* DMB instruction to avoid race condition */ + __DMB(); + /* Set the Tail pointer address */ - WRITE_REG(heth->Instance->DMARPDR, 0); + WRITE_REG(heth->Instance->DMARPDR, ((uint32_t)(heth->Init.RxDesc + (tailidx)))); heth->RxDescList.RxBuildDescIdx = descidx; heth->RxDescList.RxBuildDescCnt = desccount; @@ -1317,7 +1318,7 @@ __weak void HAL_ETH_RxAllocateCallback(uint8_t **buff) /** * @brief Rx Link callback. * @param pStart: pointer to packet start - * @param pStart: pointer to packet end + * @param pEnd: pointer to packet end * @param buff: pointer to received data * @param Length: received data length * @retval None @@ -1904,14 +1905,12 @@ void HAL_ETH_IRQHandler(ETH_HandleTypeDef *heth) } } - /* ETH DMA Error */ if (__HAL_ETH_DMA_GET_IT(heth, ETH_DMASR_AIS)) { if (__HAL_ETH_DMA_GET_IT_SOURCE(heth, ETH_DMAIER_AISE)) { heth->ErrorCode |= HAL_ETH_ERROR_DMA; - /* if fatal bus error occurred */ if (__HAL_ETH_DMA_GET_IT(heth, ETH_DMASR_FBES)) { @@ -2116,7 +2115,7 @@ HAL_StatusTypeDef HAL_ETH_ReadPHYRegister(ETH_HandleTypeDef *heth, uint32_t PHYA * @param RegValue: the value to write * @retval HAL status */ -HAL_StatusTypeDef HAL_ETH_WritePHYRegister(ETH_HandleTypeDef *heth, uint32_t PHYAddr, uint32_t PHYReg, +HAL_StatusTypeDef HAL_ETH_WritePHYRegister(const ETH_HandleTypeDef *heth, uint32_t PHYAddr, uint32_t PHYReg, uint32_t RegValue) { uint32_t tmpreg1; @@ -2254,6 +2253,7 @@ HAL_StatusTypeDef HAL_ETH_GetDMAConfig(ETH_HandleTypeDef *heth, ETH_DMAConfigTyp ETH_DMAOMR_FUGF) >> 6) > 0U) ? ENABLE : DISABLE; dmaconf->ReceiveThresholdControl = READ_BIT(heth->Instance->DMAOMR, ETH_DMAOMR_RTC); dmaconf->SecondFrameOperate = ((READ_BIT(heth->Instance->DMAOMR, ETH_DMAOMR_OSF) >> 2) > 0U) ? ENABLE : DISABLE; + return HAL_OK; } @@ -2369,7 +2369,7 @@ void HAL_ETH_SetMDIOClockRange(ETH_HandleTypeDef *heth) * the configuration of the ETH MAC filters. * @retval HAL status */ -HAL_StatusTypeDef HAL_ETH_SetMACFilterConfig(ETH_HandleTypeDef *heth, ETH_MACFilterConfigTypeDef *pFilterConfig) +HAL_StatusTypeDef HAL_ETH_SetMACFilterConfig(ETH_HandleTypeDef *heth, const ETH_MACFilterConfigTypeDef *pFilterConfig) { uint32_t filterconfig; uint32_t tmpreg1; @@ -2447,7 +2447,8 @@ HAL_StatusTypeDef HAL_ETH_GetMACFilterConfig(ETH_HandleTypeDef *heth, ETH_MACFil * @param pMACAddr: Pointer to MAC address buffer data (6 bytes) * @retval HAL status */ -HAL_StatusTypeDef HAL_ETH_SetSourceMACAddrMatch(ETH_HandleTypeDef *heth, uint32_t AddrNbr, uint8_t *pMACAddr) +HAL_StatusTypeDef HAL_ETH_SetSourceMACAddrMatch(const ETH_HandleTypeDef *heth, uint32_t AddrNbr, + const uint8_t *pMACAddr) { uint32_t macaddrlr; uint32_t macaddrhr; @@ -2546,7 +2547,7 @@ void HAL_ETH_SetRxVLANIdentifier(ETH_HandleTypeDef *heth, uint32_t ComparisonBit * that contains the Power Down configuration * @retval None. */ -void HAL_ETH_EnterPowerDownMode(ETH_HandleTypeDef *heth, ETH_PowerDownConfigTypeDef *pPowerDownConfig) +void HAL_ETH_EnterPowerDownMode(ETH_HandleTypeDef *heth, const ETH_PowerDownConfigTypeDef *pPowerDownConfig) { uint32_t powerdownconfig; @@ -2650,7 +2651,7 @@ HAL_StatusTypeDef HAL_ETH_SetWakeUpFilter(ETH_HandleTypeDef *heth, uint32_t *pFi * the configuration information for ETHERNET module * @retval HAL state */ -HAL_ETH_StateTypeDef HAL_ETH_GetState(ETH_HandleTypeDef *heth) +HAL_ETH_StateTypeDef HAL_ETH_GetState(const ETH_HandleTypeDef *heth) { return heth->gState; } @@ -2661,7 +2662,7 @@ HAL_ETH_StateTypeDef HAL_ETH_GetState(ETH_HandleTypeDef *heth) * the configuration information for ETHERNET module * @retval ETH Error Code */ -uint32_t HAL_ETH_GetError(ETH_HandleTypeDef *heth) +uint32_t HAL_ETH_GetError(const ETH_HandleTypeDef *heth) { return heth->ErrorCode; } @@ -2672,7 +2673,7 @@ uint32_t HAL_ETH_GetError(ETH_HandleTypeDef *heth) * the configuration information for ETHERNET module * @retval ETH DMA Error Code */ -uint32_t HAL_ETH_GetDMAError(ETH_HandleTypeDef *heth) +uint32_t HAL_ETH_GetDMAError(const ETH_HandleTypeDef *heth) { return heth->DMAErrorCode; } @@ -2683,7 +2684,7 @@ uint32_t HAL_ETH_GetDMAError(ETH_HandleTypeDef *heth) * the configuration information for ETHERNET module * @retval ETH MAC Error Code */ -uint32_t HAL_ETH_GetMACError(ETH_HandleTypeDef *heth) +uint32_t HAL_ETH_GetMACError(const ETH_HandleTypeDef *heth) { return heth->MACErrorCode; } @@ -2694,7 +2695,7 @@ uint32_t HAL_ETH_GetMACError(ETH_HandleTypeDef *heth) * the configuration information for ETHERNET module * @retval ETH MAC WakeUp event source */ -uint32_t HAL_ETH_GetMACWakeUpSource(ETH_HandleTypeDef *heth) +uint32_t HAL_ETH_GetMACWakeUpSource(const ETH_HandleTypeDef *heth) { return heth->MACWakeUpEvent; } @@ -2941,10 +2942,10 @@ static void ETH_DMATxDescListInit(ETH_HandleTypeDef *heth) { dmatxdesc = heth->Init.TxDesc + i; - WRITE_REG(dmatxdesc->DESC0, 0x0); - WRITE_REG(dmatxdesc->DESC1, 0x0); - WRITE_REG(dmatxdesc->DESC2, 0x0); - WRITE_REG(dmatxdesc->DESC3, 0x0); + WRITE_REG(dmatxdesc->DESC0, 0x0U); + WRITE_REG(dmatxdesc->DESC1, 0x0U); + WRITE_REG(dmatxdesc->DESC2, 0x0U); + WRITE_REG(dmatxdesc->DESC3, 0x0U); WRITE_REG(heth->TxDescList.TxDesc[i], (uint32_t)dmatxdesc); @@ -2986,12 +2987,12 @@ static void ETH_DMARxDescListInit(ETH_HandleTypeDef *heth) { dmarxdesc = heth->Init.RxDesc + i; - WRITE_REG(dmarxdesc->DESC0, 0x0); - WRITE_REG(dmarxdesc->DESC1, 0x0); - WRITE_REG(dmarxdesc->DESC2, 0x0); - WRITE_REG(dmarxdesc->DESC3, 0x0); - WRITE_REG(dmarxdesc->BackupAddr0, 0x0); - WRITE_REG(dmarxdesc->BackupAddr1, 0x0); + WRITE_REG(dmarxdesc->DESC0, 0x0U); + WRITE_REG(dmarxdesc->DESC1, 0x0U); + WRITE_REG(dmarxdesc->DESC2, 0x0U); + WRITE_REG(dmarxdesc->DESC3, 0x0U); + WRITE_REG(dmarxdesc->BackupAddr0, 0x0U); + WRITE_REG(dmarxdesc->BackupAddr1, 0x0U); /* Set Own bit of the Rx descriptor Status */ dmarxdesc->DESC0 = ETH_DMARXDESC_OWN; @@ -3015,11 +3016,11 @@ static void ETH_DMARxDescListInit(ETH_HandleTypeDef *heth) } } - WRITE_REG(heth->RxDescList.RxDescIdx, 0); - WRITE_REG(heth->RxDescList.RxDescCnt, 0); - WRITE_REG(heth->RxDescList.RxBuildDescIdx, 0); - WRITE_REG(heth->RxDescList.RxBuildDescCnt, 0); - WRITE_REG(heth->RxDescList.ItMode, 0); + WRITE_REG(heth->RxDescList.RxDescIdx, 0U); + WRITE_REG(heth->RxDescList.RxDescCnt, 0U); + WRITE_REG(heth->RxDescList.RxBuildDescIdx, 0U); + WRITE_REG(heth->RxDescList.RxBuildDescCnt, 0U); + WRITE_REG(heth->RxDescList.ItMode, 0U); /* Set Receive Descriptor List Address */ WRITE_REG(heth->Instance->DMARDLAR, (uint32_t) heth->Init.RxDesc); @@ -3170,7 +3171,6 @@ static uint32_t ETH_Prepare_Tx_Descriptors(ETH_HandleTypeDef *heth, ETH_TxPacket dmatxdesclist->PacketAddress[descidx] = dmatxdesclist->CurrentPacketAddress; dmatxdesclist->CurTxDesc = descidx; - /* disable the interrupt */ __disable_irq(); @@ -3217,4 +3217,3 @@ static void ETH_InitCallbacksToDefault(ETH_HandleTypeDef *heth) /** * @} */ - diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_exti.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_exti.c index 04b5215f..89166e26 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_exti.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_exti.c @@ -64,7 +64,7 @@ (++) Provide exiting handle as parameter. (++) Provide pointer on EXTI_ConfigTypeDef structure as second parameter. - (#) Clear Exti configuration of a dedicated line using HAL_EXTI_GetConfigLine(). + (#) Clear Exti configuration of a dedicated line using HAL_EXTI_ClearConfigLine(). (++) Provide exiting handle as parameter. (#) Register callback to treat Exti interrupts using HAL_EXTI_RegisterCallback(). @@ -75,7 +75,7 @@ (#) Get interrupt pending bit using HAL_EXTI_GetPending(). - (#) Clear interrupt pending bit using HAL_EXTI_GetPending(). + (#) Clear interrupt pending bit using HAL_EXTI_ClearPending(). (#) Generate software interrupt using HAL_EXTI_GenerateSWI(). @@ -300,8 +300,8 @@ HAL_StatusTypeDef HAL_EXTI_GetConfigLine(EXTI_HandleTypeDef *hexti, EXTI_ConfigT { assert_param(IS_EXTI_GPIO_PIN(linepos)); - regval = (SYSCFG->EXTICR[linepos >> 2u] << 16u ); - pExtiConfig->GPIOSel = ((regval << (SYSCFG_EXTICR1_EXTI1_Pos * (3uL - (linepos & 0x03u)))) >> 28u); + regval = SYSCFG->EXTICR[linepos >> 2u]; + pExtiConfig->GPIOSel = (regval >> (SYSCFG_EXTICR1_EXTI1_Pos * (linepos & 0x03u))) & SYSCFG_EXTICR1_EXTI0; } } diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_fmpi2c.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_fmpi2c.c index d079dd07..a84b29d3 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_fmpi2c.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_fmpi2c.c @@ -357,28 +357,28 @@ /* Private define for @ref PreviousState usage */ #define FMPI2C_STATE_MSK ((uint32_t)((uint32_t)((uint32_t)HAL_FMPI2C_STATE_BUSY_TX | \ - (uint32_t)HAL_FMPI2C_STATE_BUSY_RX) & \ - (uint32_t)(~((uint32_t)HAL_FMPI2C_STATE_READY)))) + (uint32_t)HAL_FMPI2C_STATE_BUSY_RX) & \ + (uint32_t)(~((uint32_t)HAL_FMPI2C_STATE_READY)))) /*!< Mask State define, keep only RX and TX bits */ #define FMPI2C_STATE_NONE ((uint32_t)(HAL_FMPI2C_MODE_NONE)) /*!< Default Value */ #define FMPI2C_STATE_MASTER_BUSY_TX ((uint32_t)(((uint32_t)HAL_FMPI2C_STATE_BUSY_TX & FMPI2C_STATE_MSK) | \ - (uint32_t)HAL_FMPI2C_MODE_MASTER)) + (uint32_t)HAL_FMPI2C_MODE_MASTER)) /*!< Master Busy TX, combinaison of State LSB and Mode enum */ #define FMPI2C_STATE_MASTER_BUSY_RX ((uint32_t)(((uint32_t)HAL_FMPI2C_STATE_BUSY_RX & FMPI2C_STATE_MSK) | \ - (uint32_t)HAL_FMPI2C_MODE_MASTER)) + (uint32_t)HAL_FMPI2C_MODE_MASTER)) /*!< Master Busy RX, combinaison of State LSB and Mode enum */ #define FMPI2C_STATE_SLAVE_BUSY_TX ((uint32_t)(((uint32_t)HAL_FMPI2C_STATE_BUSY_TX & FMPI2C_STATE_MSK) | \ - (uint32_t)HAL_FMPI2C_MODE_SLAVE)) + (uint32_t)HAL_FMPI2C_MODE_SLAVE)) /*!< Slave Busy TX, combinaison of State LSB and Mode enum */ #define FMPI2C_STATE_SLAVE_BUSY_RX ((uint32_t)(((uint32_t)HAL_FMPI2C_STATE_BUSY_RX & FMPI2C_STATE_MSK) | \ - (uint32_t)HAL_FMPI2C_MODE_SLAVE)) + (uint32_t)HAL_FMPI2C_MODE_SLAVE)) /*!< Slave Busy RX, combinaison of State LSB and Mode enum */ #define FMPI2C_STATE_MEM_BUSY_TX ((uint32_t)(((uint32_t)HAL_FMPI2C_STATE_BUSY_TX & FMPI2C_STATE_MSK) | \ - (uint32_t)HAL_FMPI2C_MODE_MEM)) + (uint32_t)HAL_FMPI2C_MODE_MEM)) /*!< Memory Busy TX, combinaison of State LSB and Mode enum */ #define FMPI2C_STATE_MEM_BUSY_RX ((uint32_t)(((uint32_t)HAL_FMPI2C_STATE_BUSY_RX & FMPI2C_STATE_MSK) | \ - (uint32_t)HAL_FMPI2C_MODE_MEM)) + (uint32_t)HAL_FMPI2C_MODE_MEM)) /*!< Memory Busy RX, combinaison of State LSB and Mode enum */ @@ -401,7 +401,16 @@ * @} */ -/* Private macro -------------------------------------------------------------*/ +/* Private macros ------------------------------------------------------------*/ +/** @addtogroup FMPI2C_Private_Macro + * @{ + */ +/* Macro to get remaining data to transfer on DMA side */ +#define FMPI2C_GET_DMA_REMAIN_DATA(__HANDLE__) __HAL_DMA_GET_COUNTER(__HANDLE__) +/** + * @} + */ + /* Private variables ---------------------------------------------------------*/ /* Private function prototypes -----------------------------------------------*/ @@ -416,6 +425,7 @@ static void FMPI2C_DMASlaveReceiveCplt(DMA_HandleTypeDef *hdma); static void FMPI2C_DMAError(DMA_HandleTypeDef *hdma); static void FMPI2C_DMAAbort(DMA_HandleTypeDef *hdma); + /* Private functions to handle IT transfer */ static void FMPI2C_ITAddrCplt(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags); static void FMPI2C_ITMasterSeqCplt(FMPI2C_HandleTypeDef *hfmpi2c); @@ -427,33 +437,37 @@ static void FMPI2C_ITError(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ErrorCode); /* Private functions to handle IT transfer */ static HAL_StatusTypeDef FMPI2C_RequestMemoryWrite(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, - uint16_t MemAddress, uint16_t MemAddSize, uint32_t Timeout, - uint32_t Tickstart); + uint16_t MemAddress, uint16_t MemAddSize, uint32_t Timeout, + uint32_t Tickstart); static HAL_StatusTypeDef FMPI2C_RequestMemoryRead(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, - uint16_t MemAddress, uint16_t MemAddSize, uint32_t Timeout, - uint32_t Tickstart); + uint16_t MemAddress, uint16_t MemAddSize, uint32_t Timeout, + uint32_t Tickstart); /* Private functions for FMPI2C transfer IRQ handler */ static HAL_StatusTypeDef FMPI2C_Master_ISR_IT(struct __FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags, + uint32_t ITSources); +static HAL_StatusTypeDef FMPI2C_Mem_ISR_IT(struct __FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags, uint32_t ITSources); static HAL_StatusTypeDef FMPI2C_Slave_ISR_IT(struct __FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags, - uint32_t ITSources); + uint32_t ITSources); static HAL_StatusTypeDef FMPI2C_Master_ISR_DMA(struct __FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags, + uint32_t ITSources); +static HAL_StatusTypeDef FMPI2C_Mem_ISR_DMA(struct __FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags, uint32_t ITSources); static HAL_StatusTypeDef FMPI2C_Slave_ISR_DMA(struct __FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags, - uint32_t ITSources); + uint32_t ITSources); /* Private functions to handle flags during polling transfer */ static HAL_StatusTypeDef FMPI2C_WaitOnFlagUntilTimeout(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t Flag, FlagStatus Status, - uint32_t Timeout, uint32_t Tickstart); + uint32_t Timeout, uint32_t Tickstart); static HAL_StatusTypeDef FMPI2C_WaitOnTXISFlagUntilTimeout(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t Timeout, - uint32_t Tickstart); + uint32_t Tickstart); static HAL_StatusTypeDef FMPI2C_WaitOnRXNEFlagUntilTimeout(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t Timeout, - uint32_t Tickstart); + uint32_t Tickstart); static HAL_StatusTypeDef FMPI2C_WaitOnSTOPFlagUntilTimeout(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t Timeout, - uint32_t Tickstart); + uint32_t Tickstart); static HAL_StatusTypeDef FMPI2C_IsErrorOccurred(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t Timeout, - uint32_t Tickstart); + uint32_t Tickstart); /* Private functions to centralize the enable/disable of Interrupts */ static void FMPI2C_Enable_IRQ(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t InterruptRequest); @@ -467,7 +481,7 @@ static void FMPI2C_Flush_TXDR(FMPI2C_HandleTypeDef *hfmpi2c); /* Private function to handle start, restart or stop a transfer */ static void FMPI2C_TransferConfig(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t Size, uint32_t Mode, - uint32_t Request); + uint32_t Request); /* Private function to Convert Specific options */ static void FMPI2C_ConvertOtherXferOptions(FMPI2C_HandleTypeDef *hfmpi2c); @@ -595,7 +609,12 @@ HAL_StatusTypeDef HAL_FMPI2C_Init(FMPI2C_HandleTypeDef *hfmpi2c) /* Configure FMPI2Cx: Addressing Master mode */ if (hfmpi2c->Init.AddressingMode == FMPI2C_ADDRESSINGMODE_10BIT) { - hfmpi2c->Instance->CR2 = (FMPI2C_CR2_ADD10); + SET_BIT(hfmpi2c->Instance->CR2, FMPI2C_CR2_ADD10); + } + else + { + /* Clear the FMPI2C ADD10 bit */ + CLEAR_BIT(hfmpi2c->Instance->CR2, FMPI2C_CR2_ADD10); } /* Enable the AUTOEND by default, and enable NACK (should be disable only during Slave process */ hfmpi2c->Instance->CR2 |= (FMPI2C_CR2_AUTOEND | FMPI2C_CR2_NACK); @@ -606,7 +625,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Init(FMPI2C_HandleTypeDef *hfmpi2c) /* Configure FMPI2Cx: Dual mode and Own Address2 */ hfmpi2c->Instance->OAR2 = (hfmpi2c->Init.DualAddressMode | hfmpi2c->Init.OwnAddress2 | \ - (hfmpi2c->Init.OwnAddress2Masks << 8)); + (hfmpi2c->Init.OwnAddress2Masks << 8)); /*---------------------------- FMPI2Cx CR1 Configuration ----------------------*/ /* Configure FMPI2Cx: Generalcall and NoStretch mode */ @@ -705,6 +724,8 @@ __weak void HAL_FMPI2C_MspDeInit(FMPI2C_HandleTypeDef *hfmpi2c) /** * @brief Register a User FMPI2C Callback * To be used instead of the weak predefined callback + * @note The HAL_FMPI2C_RegisterCallback() may be called before HAL_FMPI2C_Init() in HAL_FMPI2C_STATE_RESET + * to register callbacks for HAL_FMPI2C_MSPINIT_CB_ID and HAL_FMPI2C_MSPDEINIT_CB_ID. * @param hfmpi2c Pointer to a FMPI2C_HandleTypeDef structure that contains * the configuration information for the specified FMPI2C. * @param CallbackID ID of the callback to be registered @@ -724,7 +745,7 @@ __weak void HAL_FMPI2C_MspDeInit(FMPI2C_HandleTypeDef *hfmpi2c) * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_RegisterCallback(FMPI2C_HandleTypeDef *hfmpi2c, HAL_FMPI2C_CallbackIDTypeDef CallbackID, - pFMPI2C_CallbackTypeDef pCallback) + pFMPI2C_CallbackTypeDef pCallback) { HAL_StatusTypeDef status = HAL_OK; @@ -735,8 +756,6 @@ HAL_StatusTypeDef HAL_FMPI2C_RegisterCallback(FMPI2C_HandleTypeDef *hfmpi2c, HAL return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(hfmpi2c); if (HAL_FMPI2C_STATE_READY == hfmpi2c->State) { @@ -825,14 +844,14 @@ HAL_StatusTypeDef HAL_FMPI2C_RegisterCallback(FMPI2C_HandleTypeDef *hfmpi2c, HAL status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hfmpi2c); return status; } /** * @brief Unregister an FMPI2C Callback * FMPI2C callback is redirected to the weak predefined callback + * @note The HAL_FMPI2C_UnRegisterCallback() may be called before HAL_FMPI2C_Init() in HAL_FMPI2C_STATE_RESET + * to un-register callbacks for HAL_FMPI2C_MSPINIT_CB_ID and HAL_FMPI2C_MSPDEINIT_CB_ID. * @param hfmpi2c Pointer to a FMPI2C_HandleTypeDef structure that contains * the configuration information for the specified FMPI2C. * @param CallbackID ID of the callback to be unregistered @@ -855,9 +874,6 @@ HAL_StatusTypeDef HAL_FMPI2C_UnRegisterCallback(FMPI2C_HandleTypeDef *hfmpi2c, H { HAL_StatusTypeDef status = HAL_OK; - /* Process locked */ - __HAL_LOCK(hfmpi2c); - if (HAL_FMPI2C_STATE_READY == hfmpi2c->State) { switch (CallbackID) @@ -945,8 +961,6 @@ HAL_StatusTypeDef HAL_FMPI2C_UnRegisterCallback(FMPI2C_HandleTypeDef *hfmpi2c, H status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hfmpi2c); return status; } @@ -969,8 +983,6 @@ HAL_StatusTypeDef HAL_FMPI2C_RegisterAddrCallback(FMPI2C_HandleTypeDef *hfmpi2c, return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(hfmpi2c); if (HAL_FMPI2C_STATE_READY == hfmpi2c->State) { @@ -985,8 +997,6 @@ HAL_StatusTypeDef HAL_FMPI2C_RegisterAddrCallback(FMPI2C_HandleTypeDef *hfmpi2c, status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hfmpi2c); return status; } @@ -1001,9 +1011,6 @@ HAL_StatusTypeDef HAL_FMPI2C_UnRegisterAddrCallback(FMPI2C_HandleTypeDef *hfmpi2 { HAL_StatusTypeDef status = HAL_OK; - /* Process locked */ - __HAL_LOCK(hfmpi2c); - if (HAL_FMPI2C_STATE_READY == hfmpi2c->State) { hfmpi2c->AddrCallback = HAL_FMPI2C_AddrCallback; /* Legacy weak AddrCallback */ @@ -1017,8 +1024,6 @@ HAL_StatusTypeDef HAL_FMPI2C_UnRegisterAddrCallback(FMPI2C_HandleTypeDef *hfmpi2 status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hfmpi2c); return status; } @@ -1113,9 +1118,10 @@ HAL_StatusTypeDef HAL_FMPI2C_UnRegisterAddrCallback(FMPI2C_HandleTypeDef *hfmpi2 * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size, uint32_t Timeout) + uint16_t Size, uint32_t Timeout) { uint32_t tickstart; + uint32_t xfermode; if (hfmpi2c->State == HAL_FMPI2C_STATE_READY) { @@ -1139,19 +1145,40 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit(FMPI2C_HandleTypeDef *hfmpi2c, uint hfmpi2c->XferCount = Size; hfmpi2c->XferISR = NULL; - /* Send Slave Address */ - /* Set NBYTES to write and reload if hfmpi2c->XferCount > MAX_NBYTE_SIZE and generate RESTART */ if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) { hfmpi2c->XferSize = MAX_NBYTE_SIZE; - FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_RELOAD_MODE, - FMPI2C_GENERATE_START_WRITE); + xfermode = FMPI2C_RELOAD_MODE; } else { hfmpi2c->XferSize = hfmpi2c->XferCount; - FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_AUTOEND_MODE, - FMPI2C_GENERATE_START_WRITE); + xfermode = FMPI2C_AUTOEND_MODE; + } + + if (hfmpi2c->XferSize > 0U) + { + /* Preload TX register */ + /* Write data to TXDR */ + hfmpi2c->Instance->TXDR = *hfmpi2c->pBuffPtr; + + /* Increment Buffer pointer */ + hfmpi2c->pBuffPtr++; + + hfmpi2c->XferCount--; + hfmpi2c->XferSize--; + + /* Send Slave Address */ + /* Set NBYTES to write and reload if hfmpi2c->XferCount > MAX_NBYTE_SIZE and generate RESTART */ + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)(hfmpi2c->XferSize + 1U), xfermode, + FMPI2C_GENERATE_START_WRITE); + } + else + { + /* Send Slave Address */ + /* Set NBYTES to write and reload if hfmpi2c->XferCount > MAX_NBYTE_SIZE and generate RESTART */ + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, xfermode, + FMPI2C_GENERATE_START_WRITE); } while (hfmpi2c->XferCount > 0U) @@ -1182,13 +1209,13 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit(FMPI2C_HandleTypeDef *hfmpi2c, uint { hfmpi2c->XferSize = MAX_NBYTE_SIZE; FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_RELOAD_MODE, - FMPI2C_NO_STARTSTOP); + FMPI2C_NO_STARTSTOP); } else { hfmpi2c->XferSize = hfmpi2c->XferCount; FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_AUTOEND_MODE, - FMPI2C_NO_STARTSTOP); + FMPI2C_NO_STARTSTOP); } } } @@ -1232,7 +1259,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit(FMPI2C_HandleTypeDef *hfmpi2c, uint * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Master_Receive(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size, uint32_t Timeout) + uint16_t Size, uint32_t Timeout) { uint32_t tickstart; @@ -1262,15 +1289,15 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Receive(FMPI2C_HandleTypeDef *hfmpi2c, uint1 /* Set NBYTES to write and reload if hfmpi2c->XferCount > MAX_NBYTE_SIZE and generate RESTART */ if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) { - hfmpi2c->XferSize = MAX_NBYTE_SIZE; + hfmpi2c->XferSize = 1U; FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_RELOAD_MODE, - FMPI2C_GENERATE_START_READ); + FMPI2C_GENERATE_START_READ); } else { hfmpi2c->XferSize = hfmpi2c->XferCount; FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_AUTOEND_MODE, - FMPI2C_GENERATE_START_READ); + FMPI2C_GENERATE_START_READ); } while (hfmpi2c->XferCount > 0U) @@ -1302,13 +1329,13 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Receive(FMPI2C_HandleTypeDef *hfmpi2c, uint1 { hfmpi2c->XferSize = MAX_NBYTE_SIZE; FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_RELOAD_MODE, - FMPI2C_NO_STARTSTOP); + FMPI2C_NO_STARTSTOP); } else { hfmpi2c->XferSize = hfmpi2c->XferCount; FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_AUTOEND_MODE, - FMPI2C_NO_STARTSTOP); + FMPI2C_NO_STARTSTOP); } } } @@ -1350,9 +1377,11 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Receive(FMPI2C_HandleTypeDef *hfmpi2c, uint1 * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Slave_Transmit(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size, - uint32_t Timeout) + uint32_t Timeout) { uint32_t tickstart; + uint16_t tmpXferCount; + HAL_StatusTypeDef error; if (hfmpi2c->State == HAL_FMPI2C_STATE_READY) { @@ -1387,6 +1416,19 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Transmit(FMPI2C_HandleTypeDef *hfmpi2c, uint8 return HAL_ERROR; } + /* Preload TX data if no stretch enable */ + if (hfmpi2c->Init.NoStretchMode == FMPI2C_NOSTRETCH_ENABLE) + { + /* Preload TX register */ + /* Write data to TXDR */ + hfmpi2c->Instance->TXDR = *hfmpi2c->pBuffPtr; + + /* Increment Buffer pointer */ + hfmpi2c->pBuffPtr++; + + hfmpi2c->XferCount--; + } + /* Clear ADDR flag */ __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_ADDR); @@ -1432,26 +1474,48 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Transmit(FMPI2C_HandleTypeDef *hfmpi2c, uint8 hfmpi2c->XferCount--; } - /* Wait until STOP flag is set */ - if (FMPI2C_WaitOnSTOPFlagUntilTimeout(hfmpi2c, Timeout, tickstart) != HAL_OK) + /* Wait until AF flag is set */ + error = FMPI2C_WaitOnFlagUntilTimeout(hfmpi2c, FMPI2C_FLAG_AF, RESET, Timeout, tickstart); + + if (error != HAL_OK) { - /* Disable Address Acknowledge */ - hfmpi2c->Instance->CR2 |= FMPI2C_CR2_NACK; + /* Check that FMPI2C transfer finished */ + /* if yes, normal use case, a NACK is sent by the MASTER when Transfer is finished */ + /* Mean XferCount == 0 */ - if (hfmpi2c->ErrorCode == HAL_FMPI2C_ERROR_AF) + tmpXferCount = hfmpi2c->XferCount; + if ((hfmpi2c->ErrorCode == HAL_FMPI2C_ERROR_AF) && (tmpXferCount == 0U)) { - /* Normal use case for Transmitter mode */ - /* A NACK is generated to confirm the end of transfer */ + /* Reset ErrorCode to NONE */ hfmpi2c->ErrorCode = HAL_FMPI2C_ERROR_NONE; } else { + /* Disable Address Acknowledge */ + hfmpi2c->Instance->CR2 |= FMPI2C_CR2_NACK; return HAL_ERROR; } } + else + { + /* Flush TX register */ + FMPI2C_Flush_TXDR(hfmpi2c); - /* Clear STOP flag */ - __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_STOPF); + /* Clear AF flag */ + __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_AF); + + /* Wait until STOP flag is set */ + if (FMPI2C_WaitOnSTOPFlagUntilTimeout(hfmpi2c, Timeout, tickstart) != HAL_OK) + { + /* Disable Address Acknowledge */ + hfmpi2c->Instance->CR2 |= FMPI2C_CR2_NACK; + + return HAL_ERROR; + } + + /* Clear STOP flag */ + __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_STOPF); + } /* Wait until BUSY flag is reset */ if (FMPI2C_WaitOnFlagUntilTimeout(hfmpi2c, FMPI2C_FLAG_BUSY, SET, Timeout, tickstart) != HAL_OK) @@ -1488,7 +1552,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Transmit(FMPI2C_HandleTypeDef *hfmpi2c, uint8 * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Slave_Receive(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size, - uint32_t Timeout) + uint32_t Timeout) { uint32_t tickstart; @@ -1512,6 +1576,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Receive(FMPI2C_HandleTypeDef *hfmpi2c, uint8_ /* Prepare transfer parameters */ hfmpi2c->pBuffPtr = pData; hfmpi2c->XferCount = Size; + hfmpi2c->XferSize = hfmpi2c->XferCount; hfmpi2c->XferISR = NULL; /* Enable Address Acknowledge */ @@ -1554,6 +1619,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Receive(FMPI2C_HandleTypeDef *hfmpi2c, uint8_ hfmpi2c->pBuffPtr++; hfmpi2c->XferCount--; + hfmpi2c->XferSize--; } return HAL_ERROR; @@ -1566,6 +1632,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Receive(FMPI2C_HandleTypeDef *hfmpi2c, uint8_ hfmpi2c->pBuffPtr++; hfmpi2c->XferCount--; + hfmpi2c->XferSize--; } /* Wait until STOP flag is set */ @@ -1615,7 +1682,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Receive(FMPI2C_HandleTypeDef *hfmpi2c, uint8_ * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size) + uint16_t Size) { uint32_t xfermode; @@ -1652,7 +1719,26 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2c, u /* Send Slave Address */ /* Set NBYTES to write and reload if hfmpi2c->XferCount > MAX_NBYTE_SIZE */ - FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, xfermode, FMPI2C_GENERATE_START_WRITE); + if (hfmpi2c->XferSize > 0U) + { + /* Preload TX register */ + /* Write data to TXDR */ + hfmpi2c->Instance->TXDR = *hfmpi2c->pBuffPtr; + + /* Increment Buffer pointer */ + hfmpi2c->pBuffPtr++; + + hfmpi2c->XferCount--; + hfmpi2c->XferSize--; + + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)(hfmpi2c->XferSize + 1U), xfermode, + FMPI2C_GENERATE_START_WRITE); + } + else + { + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, xfermode, + FMPI2C_GENERATE_START_WRITE); + } /* Process Unlocked */ __HAL_UNLOCK(hfmpi2c); @@ -1686,7 +1772,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2c, u * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Master_Receive_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size) + uint16_t Size) { uint32_t xfermode; @@ -1712,7 +1798,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Receive_IT(FMPI2C_HandleTypeDef *hfmpi2c, ui if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) { - hfmpi2c->XferSize = MAX_NBYTE_SIZE; + hfmpi2c->XferSize = 1U; xfermode = FMPI2C_RELOAD_MODE; } else @@ -1775,6 +1861,20 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2c, ui hfmpi2c->XferOptions = FMPI2C_NO_OPTION_FRAME; hfmpi2c->XferISR = FMPI2C_Slave_ISR_IT; + /* Preload TX data if no stretch enable */ + if (hfmpi2c->Init.NoStretchMode == FMPI2C_NOSTRETCH_ENABLE) + { + /* Preload TX register */ + /* Write data to TXDR */ + hfmpi2c->Instance->TXDR = *hfmpi2c->pBuffPtr; + + /* Increment Buffer pointer */ + hfmpi2c->pBuffPtr++; + + hfmpi2c->XferCount--; + hfmpi2c->XferSize--; + } + /* Process Unlocked */ __HAL_UNLOCK(hfmpi2c); @@ -1857,10 +1957,11 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Receive_IT(FMPI2C_HandleTypeDef *hfmpi2c, uin * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size) + uint16_t Size) { uint32_t xfermode; HAL_StatusTypeDef dmaxferstatus; + uint32_t sizetoxfer = 0U; if (hfmpi2c->State == HAL_FMPI2C_STATE_READY) { @@ -1893,6 +1994,20 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2c, xfermode = FMPI2C_AUTOEND_MODE; } + if (hfmpi2c->XferSize > 0U) + { + /* Preload TX register */ + /* Write data to TXDR */ + hfmpi2c->Instance->TXDR = *hfmpi2c->pBuffPtr; + + /* Increment Buffer pointer */ + hfmpi2c->pBuffPtr++; + + sizetoxfer = hfmpi2c->XferSize; + hfmpi2c->XferCount--; + hfmpi2c->XferSize--; + } + if (hfmpi2c->XferSize > 0U) { if (hfmpi2c->hdmatx != NULL) @@ -1908,8 +2023,8 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2c, hfmpi2c->hdmatx->XferAbortCallback = NULL; /* Enable the DMA stream */ - dmaxferstatus = HAL_DMA_Start_IT(hfmpi2c->hdmatx, (uint32_t)pData, (uint32_t)&hfmpi2c->Instance->TXDR, - hfmpi2c->XferSize); + dmaxferstatus = HAL_DMA_Start_IT(hfmpi2c->hdmatx, (uint32_t)hfmpi2c->pBuffPtr, + (uint32_t)&hfmpi2c->Instance->TXDR, hfmpi2c->XferSize); } else { @@ -1930,7 +2045,8 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2c, { /* Send Slave Address */ /* Set NBYTES to write and reload if hfmpi2c->XferCount > MAX_NBYTE_SIZE and generate RESTART */ - FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, xfermode, FMPI2C_GENERATE_START_WRITE); + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)(hfmpi2c->XferSize + 1U), + xfermode, FMPI2C_GENERATE_START_WRITE); /* Update XferCount value */ hfmpi2c->XferCount -= hfmpi2c->XferSize; @@ -1969,8 +2085,8 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2c, /* Send Slave Address */ /* Set NBYTES to write and generate START condition */ - FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_AUTOEND_MODE, - FMPI2C_GENERATE_START_WRITE); + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)sizetoxfer, FMPI2C_AUTOEND_MODE, + FMPI2C_GENERATE_START_WRITE); /* Process Unlocked */ __HAL_UNLOCK(hfmpi2c); @@ -2004,7 +2120,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2c, * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Master_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size) + uint16_t Size) { uint32_t xfermode; HAL_StatusTypeDef dmaxferstatus; @@ -2031,7 +2147,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2c, u if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) { - hfmpi2c->XferSize = MAX_NBYTE_SIZE; + hfmpi2c->XferSize = 1U; xfermode = FMPI2C_RELOAD_MODE; } else @@ -2117,7 +2233,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2c, u /* Send Slave Address */ /* Set NBYTES to read and generate START condition */ FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_AUTOEND_MODE, - FMPI2C_GENERATE_START_READ); + FMPI2C_GENERATE_START_READ); /* Process Unlocked */ __HAL_UNLOCK(hfmpi2c); @@ -2125,11 +2241,11 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2c, u /* Note : The FMPI2C interrupts must be enabled after unlocking current process to avoid the risk of FMPI2C interrupt handle execution before current process unlock */ - /* Enable ERR, TC, STOP, NACK, TXI interrupt */ + /* Enable ERR, TC, STOP, NACK, RXI interrupt */ /* possible to enable all of these */ /* FMPI2C_IT_ERRI | FMPI2C_IT_TCI | FMPI2C_IT_STOPI | FMPI2C_IT_NACKI | FMPI2C_IT_ADDRI | FMPI2C_IT_RXI | FMPI2C_IT_TXI */ - FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_TX_IT); + FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_RX_IT); } return HAL_OK; @@ -2173,67 +2289,99 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2c, u hfmpi2c->XferOptions = FMPI2C_NO_OPTION_FRAME; hfmpi2c->XferISR = FMPI2C_Slave_ISR_DMA; - if (hfmpi2c->hdmatx != NULL) + /* Preload TX data if no stretch enable */ + if (hfmpi2c->Init.NoStretchMode == FMPI2C_NOSTRETCH_ENABLE) { - /* Set the FMPI2C DMA transfer complete callback */ - hfmpi2c->hdmatx->XferCpltCallback = FMPI2C_DMASlaveTransmitCplt; - - /* Set the DMA error callback */ - hfmpi2c->hdmatx->XferErrorCallback = FMPI2C_DMAError; + /* Preload TX register */ + /* Write data to TXDR */ + hfmpi2c->Instance->TXDR = *hfmpi2c->pBuffPtr; - /* Set the unused DMA callbacks to NULL */ - hfmpi2c->hdmatx->XferHalfCpltCallback = NULL; - hfmpi2c->hdmatx->XferAbortCallback = NULL; + /* Increment Buffer pointer */ + hfmpi2c->pBuffPtr++; - /* Enable the DMA stream */ - dmaxferstatus = HAL_DMA_Start_IT(hfmpi2c->hdmatx, (uint32_t)pData, (uint32_t)&hfmpi2c->Instance->TXDR, - hfmpi2c->XferSize); + hfmpi2c->XferCount--; + hfmpi2c->XferSize--; } - else + + if (hfmpi2c->XferCount != 0U) { - /* Update FMPI2C state */ - hfmpi2c->State = HAL_FMPI2C_STATE_LISTEN; - hfmpi2c->Mode = HAL_FMPI2C_MODE_NONE; + if (hfmpi2c->hdmatx != NULL) + { + /* Set the FMPI2C DMA transfer complete callback */ + hfmpi2c->hdmatx->XferCpltCallback = FMPI2C_DMASlaveTransmitCplt; - /* Update FMPI2C error code */ - hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_DMA_PARAM; + /* Set the DMA error callback */ + hfmpi2c->hdmatx->XferErrorCallback = FMPI2C_DMAError; - /* Process Unlocked */ - __HAL_UNLOCK(hfmpi2c); + /* Set the unused DMA callbacks to NULL */ + hfmpi2c->hdmatx->XferHalfCpltCallback = NULL; + hfmpi2c->hdmatx->XferAbortCallback = NULL; - return HAL_ERROR; - } + /* Enable the DMA stream */ + dmaxferstatus = HAL_DMA_Start_IT(hfmpi2c->hdmatx, + (uint32_t)hfmpi2c->pBuffPtr, (uint32_t)&hfmpi2c->Instance->TXDR, + hfmpi2c->XferSize); + } + else + { + /* Update FMPI2C state */ + hfmpi2c->State = HAL_FMPI2C_STATE_LISTEN; + hfmpi2c->Mode = HAL_FMPI2C_MODE_NONE; - if (dmaxferstatus == HAL_OK) - { - /* Enable Address Acknowledge */ - hfmpi2c->Instance->CR2 &= ~FMPI2C_CR2_NACK; + /* Update FMPI2C error code */ + hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_DMA_PARAM; - /* Process Unlocked */ - __HAL_UNLOCK(hfmpi2c); + /* Process Unlocked */ + __HAL_UNLOCK(hfmpi2c); - /* Note : The FMPI2C interrupts must be enabled after unlocking current process - to avoid the risk of FMPI2C interrupt handle execution before current - process unlock */ - /* Enable ERR, STOP, NACK, ADDR interrupts */ - FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_LISTEN_IT); + return HAL_ERROR; + } - /* Enable DMA Request */ - hfmpi2c->Instance->CR1 |= FMPI2C_CR1_TXDMAEN; + if (dmaxferstatus == HAL_OK) + { + /* Enable Address Acknowledge */ + hfmpi2c->Instance->CR2 &= ~FMPI2C_CR2_NACK; + + /* Process Unlocked */ + __HAL_UNLOCK(hfmpi2c); + + /* Note : The FMPI2C interrupts must be enabled after unlocking current process + to avoid the risk of FMPI2C interrupt handle execution before current + process unlock */ + /* Enable ERR, STOP, NACK, ADDR interrupts */ + FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_LISTEN_IT); + + /* Enable DMA Request */ + hfmpi2c->Instance->CR1 |= FMPI2C_CR1_TXDMAEN; + } + else + { + /* Update FMPI2C state */ + hfmpi2c->State = HAL_FMPI2C_STATE_LISTEN; + hfmpi2c->Mode = HAL_FMPI2C_MODE_NONE; + + /* Update FMPI2C error code */ + hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_DMA; + + /* Process Unlocked */ + __HAL_UNLOCK(hfmpi2c); + + return HAL_ERROR; + } } else { - /* Update FMPI2C state */ - hfmpi2c->State = HAL_FMPI2C_STATE_LISTEN; - hfmpi2c->Mode = HAL_FMPI2C_MODE_NONE; - - /* Update FMPI2C error code */ - hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_DMA; + /* Enable Address Acknowledge */ + hfmpi2c->Instance->CR2 &= ~FMPI2C_CR2_NACK; /* Process Unlocked */ __HAL_UNLOCK(hfmpi2c); - return HAL_ERROR; + /* Note : The FMPI2C interrupts must be enabled after unlocking current process + to avoid the risk of FMPI2C interrupt handle execution before current + process unlock */ + /* Enable ERR, STOP, NACK, ADDR interrupts */ + FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_LISTEN_IT); } return HAL_OK; @@ -2347,6 +2495,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2c, ui return HAL_BUSY; } } + /** * @brief Write an amount of data in blocking mode to a specific memory address * @param hfmpi2c Pointer to a FMPI2C_HandleTypeDef structure that contains @@ -2361,7 +2510,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2c, ui * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Mem_Write(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint16_t MemAddress, - uint16_t MemAddSize, uint8_t *pData, uint16_t Size, uint32_t Timeout) + uint16_t MemAddSize, uint8_t *pData, uint16_t Size, uint32_t Timeout) { uint32_t tickstart; @@ -2445,13 +2594,13 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Write(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t D { hfmpi2c->XferSize = MAX_NBYTE_SIZE; FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_RELOAD_MODE, - FMPI2C_NO_STARTSTOP); + FMPI2C_NO_STARTSTOP); } else { hfmpi2c->XferSize = hfmpi2c->XferCount; FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_AUTOEND_MODE, - FMPI2C_NO_STARTSTOP); + FMPI2C_NO_STARTSTOP); } } @@ -2498,7 +2647,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Write(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t D * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint16_t MemAddress, - uint16_t MemAddSize, uint8_t *pData, uint16_t Size, uint32_t Timeout) + uint16_t MemAddSize, uint8_t *pData, uint16_t Size, uint32_t Timeout) { uint32_t tickstart; @@ -2545,15 +2694,15 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t De /* Set NBYTES to write and reload if hfmpi2c->XferCount > MAX_NBYTE_SIZE and generate RESTART */ if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) { - hfmpi2c->XferSize = MAX_NBYTE_SIZE; + hfmpi2c->XferSize = 1U; FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_RELOAD_MODE, - FMPI2C_GENERATE_START_READ); + FMPI2C_GENERATE_START_READ); } else { hfmpi2c->XferSize = hfmpi2c->XferCount; FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_AUTOEND_MODE, - FMPI2C_GENERATE_START_READ); + FMPI2C_GENERATE_START_READ); } do @@ -2583,15 +2732,15 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t De if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) { - hfmpi2c->XferSize = MAX_NBYTE_SIZE; + hfmpi2c->XferSize = 1U; FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t) hfmpi2c->XferSize, FMPI2C_RELOAD_MODE, - FMPI2C_NO_STARTSTOP); + FMPI2C_NO_STARTSTOP); } else { hfmpi2c->XferSize = hfmpi2c->XferCount; FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_AUTOEND_MODE, - FMPI2C_NO_STARTSTOP); + FMPI2C_NO_STARTSTOP); } } } while (hfmpi2c->XferCount > 0U); @@ -2635,11 +2784,8 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t De * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Mem_Write_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint16_t MemAddress, - uint16_t MemAddSize, uint8_t *pData, uint16_t Size) + uint16_t MemAddSize, uint8_t *pData, uint16_t Size) { - uint32_t tickstart; - uint32_t xfermode; - /* Check the parameters */ assert_param(IS_FMPI2C_MEMADD_SIZE(MemAddSize)); @@ -2659,41 +2805,38 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Write_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_ /* Process Locked */ __HAL_LOCK(hfmpi2c); - /* Init tickstart for timeout management*/ - tickstart = HAL_GetTick(); - hfmpi2c->State = HAL_FMPI2C_STATE_BUSY_TX; hfmpi2c->Mode = HAL_FMPI2C_MODE_MEM; hfmpi2c->ErrorCode = HAL_FMPI2C_ERROR_NONE; /* Prepare transfer parameters */ + hfmpi2c->XferSize = 0U; hfmpi2c->pBuffPtr = pData; hfmpi2c->XferCount = Size; hfmpi2c->XferOptions = FMPI2C_NO_OPTION_FRAME; - hfmpi2c->XferISR = FMPI2C_Master_ISR_IT; + hfmpi2c->XferISR = FMPI2C_Mem_ISR_IT; + hfmpi2c->Devaddress = DevAddress; - if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) + /* If Memory address size is 8Bit */ + if (MemAddSize == FMPI2C_MEMADD_SIZE_8BIT) { - hfmpi2c->XferSize = MAX_NBYTE_SIZE; - xfermode = FMPI2C_RELOAD_MODE; + /* Prefetch Memory Address */ + hfmpi2c->Instance->TXDR = FMPI2C_MEM_ADD_LSB(MemAddress); + + /* Reset Memaddress content */ + hfmpi2c->Memaddress = 0xFFFFFFFFU; } + /* If Memory address size is 16Bit */ else { - hfmpi2c->XferSize = hfmpi2c->XferCount; - xfermode = FMPI2C_AUTOEND_MODE; - } + /* Prefetch Memory Address (MSB part, LSB will be manage through interrupt) */ + hfmpi2c->Instance->TXDR = FMPI2C_MEM_ADD_MSB(MemAddress); + /* Prepare Memaddress buffer for LSB part */ + hfmpi2c->Memaddress = FMPI2C_MEM_ADD_LSB(MemAddress); + } /* Send Slave Address and Memory Address */ - if (FMPI2C_RequestMemoryWrite(hfmpi2c, DevAddress, MemAddress, MemAddSize, FMPI2C_TIMEOUT_FLAG, tickstart) - != HAL_OK) - { - /* Process Unlocked */ - __HAL_UNLOCK(hfmpi2c); - return HAL_ERROR; - } - - /* Set NBYTES to write and reload if hfmpi2c->XferCount > MAX_NBYTE_SIZE and generate RESTART */ - FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, xfermode, FMPI2C_NO_STARTSTOP); + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)MemAddSize, FMPI2C_RELOAD_MODE, FMPI2C_GENERATE_START_WRITE); /* Process Unlocked */ __HAL_UNLOCK(hfmpi2c); @@ -2729,11 +2872,8 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Write_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_ * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint16_t MemAddress, - uint16_t MemAddSize, uint8_t *pData, uint16_t Size) + uint16_t MemAddSize, uint8_t *pData, uint16_t Size) { - uint32_t tickstart; - uint32_t xfermode; - /* Check the parameters */ assert_param(IS_FMPI2C_MEMADD_SIZE(MemAddSize)); @@ -2753,9 +2893,6 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t /* Process Locked */ __HAL_LOCK(hfmpi2c); - /* Init tickstart for timeout management*/ - tickstart = HAL_GetTick(); - hfmpi2c->State = HAL_FMPI2C_STATE_BUSY_RX; hfmpi2c->Mode = HAL_FMPI2C_MODE_MEM; hfmpi2c->ErrorCode = HAL_FMPI2C_ERROR_NONE; @@ -2764,29 +2901,29 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t hfmpi2c->pBuffPtr = pData; hfmpi2c->XferCount = Size; hfmpi2c->XferOptions = FMPI2C_NO_OPTION_FRAME; - hfmpi2c->XferISR = FMPI2C_Master_ISR_IT; + hfmpi2c->XferISR = FMPI2C_Mem_ISR_IT; + hfmpi2c->Devaddress = DevAddress; - if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) + /* If Memory address size is 8Bit */ + if (MemAddSize == FMPI2C_MEMADD_SIZE_8BIT) { - hfmpi2c->XferSize = MAX_NBYTE_SIZE; - xfermode = FMPI2C_RELOAD_MODE; + /* Prefetch Memory Address */ + hfmpi2c->Instance->TXDR = FMPI2C_MEM_ADD_LSB(MemAddress); + + /* Reset Memaddress content */ + hfmpi2c->Memaddress = 0xFFFFFFFFU; } + /* If Memory address size is 16Bit */ else { - hfmpi2c->XferSize = hfmpi2c->XferCount; - xfermode = FMPI2C_AUTOEND_MODE; - } + /* Prefetch Memory Address (MSB part, LSB will be manage through interrupt) */ + hfmpi2c->Instance->TXDR = FMPI2C_MEM_ADD_MSB(MemAddress); - /* Send Slave Address and Memory Address */ - if (FMPI2C_RequestMemoryRead(hfmpi2c, DevAddress, MemAddress, MemAddSize, FMPI2C_TIMEOUT_FLAG, tickstart) != HAL_OK) - { - /* Process Unlocked */ - __HAL_UNLOCK(hfmpi2c); - return HAL_ERROR; + /* Prepare Memaddress buffer for LSB part */ + hfmpi2c->Memaddress = FMPI2C_MEM_ADD_LSB(MemAddress); } - - /* Set NBYTES to write and reload if hfmpi2c->XferCount > MAX_NBYTE_SIZE and generate RESTART */ - FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, xfermode, FMPI2C_GENERATE_START_READ); + /* Send Slave Address and Memory Address */ + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)MemAddSize, FMPI2C_SOFTEND_MODE, FMPI2C_GENERATE_START_WRITE); /* Process Unlocked */ __HAL_UNLOCK(hfmpi2c); @@ -2795,11 +2932,11 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t to avoid the risk of FMPI2C interrupt handle execution before current process unlock */ - /* Enable ERR, TC, STOP, NACK, RXI interrupt */ + /* Enable ERR, TC, STOP, NACK, TXI interrupt */ /* possible to enable all of these */ /* FMPI2C_IT_ERRI | FMPI2C_IT_TCI | FMPI2C_IT_STOPI | FMPI2C_IT_NACKI | FMPI2C_IT_ADDRI | FMPI2C_IT_RXI | FMPI2C_IT_TXI */ - FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_RX_IT); + FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_TX_IT); return HAL_OK; } @@ -2808,6 +2945,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t return HAL_BUSY; } } + /** * @brief Write an amount of data in non-blocking mode with DMA to a specific memory address * @param hfmpi2c Pointer to a FMPI2C_HandleTypeDef structure that contains @@ -2821,10 +2959,8 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Mem_Write_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint16_t MemAddress, - uint16_t MemAddSize, uint8_t *pData, uint16_t Size) + uint16_t MemAddSize, uint8_t *pData, uint16_t Size) { - uint32_t tickstart; - uint32_t xfermode; HAL_StatusTypeDef dmaxferstatus; /* Check the parameters */ @@ -2846,9 +2982,6 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Write_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16 /* Process Locked */ __HAL_LOCK(hfmpi2c); - /* Init tickstart for timeout management*/ - tickstart = HAL_GetTick(); - hfmpi2c->State = HAL_FMPI2C_STATE_BUSY_TX; hfmpi2c->Mode = HAL_FMPI2C_MODE_MEM; hfmpi2c->ErrorCode = HAL_FMPI2C_ERROR_NONE; @@ -2857,28 +2990,36 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Write_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16 hfmpi2c->pBuffPtr = pData; hfmpi2c->XferCount = Size; hfmpi2c->XferOptions = FMPI2C_NO_OPTION_FRAME; - hfmpi2c->XferISR = FMPI2C_Master_ISR_DMA; + hfmpi2c->XferISR = FMPI2C_Mem_ISR_DMA; + hfmpi2c->Devaddress = DevAddress; if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) { hfmpi2c->XferSize = MAX_NBYTE_SIZE; - xfermode = FMPI2C_RELOAD_MODE; } else { hfmpi2c->XferSize = hfmpi2c->XferCount; - xfermode = FMPI2C_AUTOEND_MODE; } - /* Send Slave Address and Memory Address */ - if (FMPI2C_RequestMemoryWrite(hfmpi2c, DevAddress, MemAddress, MemAddSize, FMPI2C_TIMEOUT_FLAG, tickstart) - != HAL_OK) + /* If Memory address size is 8Bit */ + if (MemAddSize == FMPI2C_MEMADD_SIZE_8BIT) { - /* Process Unlocked */ - __HAL_UNLOCK(hfmpi2c); - return HAL_ERROR; + /* Prefetch Memory Address */ + hfmpi2c->Instance->TXDR = FMPI2C_MEM_ADD_LSB(MemAddress); + + /* Reset Memaddress content */ + hfmpi2c->Memaddress = 0xFFFFFFFFU; } + /* If Memory address size is 16Bit */ + else + { + /* Prefetch Memory Address (MSB part, LSB will be manage through interrupt) */ + hfmpi2c->Instance->TXDR = FMPI2C_MEM_ADD_MSB(MemAddress); + /* Prepare Memaddress buffer for LSB part */ + hfmpi2c->Memaddress = FMPI2C_MEM_ADD_LSB(MemAddress); + } if (hfmpi2c->hdmatx != NULL) { @@ -2913,12 +3054,8 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Write_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16 if (dmaxferstatus == HAL_OK) { - /* Send Slave Address */ - /* Set NBYTES to write and reload if hfmpi2c->XferCount > MAX_NBYTE_SIZE and generate RESTART */ - FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, xfermode, FMPI2C_NO_STARTSTOP); - - /* Update XferCount value */ - hfmpi2c->XferCount -= hfmpi2c->XferSize; + /* Send Slave Address and Memory Address */ + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)MemAddSize, FMPI2C_RELOAD_MODE, FMPI2C_GENERATE_START_WRITE); /* Process Unlocked */ __HAL_UNLOCK(hfmpi2c); @@ -2926,11 +3063,11 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Write_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16 /* Note : The FMPI2C interrupts must be enabled after unlocking current process to avoid the risk of FMPI2C interrupt handle execution before current process unlock */ - /* Enable ERR and NACK interrupts */ - FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_ERROR_IT); - - /* Enable DMA Request */ - hfmpi2c->Instance->CR1 |= FMPI2C_CR1_TXDMAEN; + /* Enable ERR, TC, STOP, NACK, TXI interrupt */ + /* possible to enable all of these */ + /* FMPI2C_IT_ERRI | FMPI2C_IT_TCI | FMPI2C_IT_STOPI | FMPI2C_IT_NACKI | + FMPI2C_IT_ADDRI | FMPI2C_IT_RXI | FMPI2C_IT_TXI */ + FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_TX_IT); } else { @@ -2968,10 +3105,8 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Write_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16 * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint16_t MemAddress, - uint16_t MemAddSize, uint8_t *pData, uint16_t Size) + uint16_t MemAddSize, uint8_t *pData, uint16_t Size) { - uint32_t tickstart; - uint32_t xfermode; HAL_StatusTypeDef dmaxferstatus; /* Check the parameters */ @@ -2993,9 +3128,6 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_ /* Process Locked */ __HAL_LOCK(hfmpi2c); - /* Init tickstart for timeout management*/ - tickstart = HAL_GetTick(); - hfmpi2c->State = HAL_FMPI2C_STATE_BUSY_RX; hfmpi2c->Mode = HAL_FMPI2C_MODE_MEM; hfmpi2c->ErrorCode = HAL_FMPI2C_ERROR_NONE; @@ -3004,25 +3136,35 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_ hfmpi2c->pBuffPtr = pData; hfmpi2c->XferCount = Size; hfmpi2c->XferOptions = FMPI2C_NO_OPTION_FRAME; - hfmpi2c->XferISR = FMPI2C_Master_ISR_DMA; + hfmpi2c->XferISR = FMPI2C_Mem_ISR_DMA; + hfmpi2c->Devaddress = DevAddress; if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) { hfmpi2c->XferSize = MAX_NBYTE_SIZE; - xfermode = FMPI2C_RELOAD_MODE; } else { hfmpi2c->XferSize = hfmpi2c->XferCount; - xfermode = FMPI2C_AUTOEND_MODE; } - /* Send Slave Address and Memory Address */ - if (FMPI2C_RequestMemoryRead(hfmpi2c, DevAddress, MemAddress, MemAddSize, FMPI2C_TIMEOUT_FLAG, tickstart) != HAL_OK) + /* If Memory address size is 8Bit */ + if (MemAddSize == FMPI2C_MEMADD_SIZE_8BIT) { - /* Process Unlocked */ - __HAL_UNLOCK(hfmpi2c); - return HAL_ERROR; + /* Prefetch Memory Address */ + hfmpi2c->Instance->TXDR = FMPI2C_MEM_ADD_LSB(MemAddress); + + /* Reset Memaddress content */ + hfmpi2c->Memaddress = 0xFFFFFFFFU; + } + /* If Memory address size is 16Bit */ + else + { + /* Prefetch Memory Address (MSB part, LSB will be manage through interrupt) */ + hfmpi2c->Instance->TXDR = FMPI2C_MEM_ADD_MSB(MemAddress); + + /* Prepare Memaddress buffer for LSB part */ + hfmpi2c->Memaddress = FMPI2C_MEM_ADD_LSB(MemAddress); } if (hfmpi2c->hdmarx != NULL) @@ -3058,11 +3200,8 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_ if (dmaxferstatus == HAL_OK) { - /* Set NBYTES to write and reload if hfmpi2c->XferCount > MAX_NBYTE_SIZE and generate RESTART */ - FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, xfermode, FMPI2C_GENERATE_START_READ); - - /* Update XferCount value */ - hfmpi2c->XferCount -= hfmpi2c->XferSize; + /* Send Slave Address and Memory Address */ + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)MemAddSize, FMPI2C_SOFTEND_MODE, FMPI2C_GENERATE_START_WRITE); /* Process Unlocked */ __HAL_UNLOCK(hfmpi2c); @@ -3070,11 +3209,11 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_ /* Note : The FMPI2C interrupts must be enabled after unlocking current process to avoid the risk of FMPI2C interrupt handle execution before current process unlock */ - /* Enable ERR and NACK interrupts */ - FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_ERROR_IT); - - /* Enable DMA Request */ - hfmpi2c->Instance->CR1 |= FMPI2C_CR1_RXDMAEN; + /* Enable ERR, TC, STOP, NACK, TXI interrupt */ + /* possible to enable all of these */ + /* FMPI2C_IT_ERRI | FMPI2C_IT_TCI | FMPI2C_IT_STOPI | FMPI2C_IT_NACKI | + FMPI2C_IT_ADDRI | FMPI2C_IT_RXI | FMPI2C_IT_TXI */ + FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_TX_IT); } else { @@ -3111,7 +3250,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Mem_Read_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_ * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_IsDeviceReady(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint32_t Trials, - uint32_t Timeout) + uint32_t Timeout) { uint32_t tickstart; @@ -3203,22 +3342,6 @@ HAL_StatusTypeDef HAL_FMPI2C_IsDeviceReady(FMPI2C_HandleTypeDef *hfmpi2c, uint16 __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_STOPF); } - /* Check if the maximum allowed number of trials has been reached */ - if (FMPI2C_Trials == Trials) - { - /* Generate Stop */ - hfmpi2c->Instance->CR2 |= FMPI2C_CR2_STOP; - - /* Wait until STOPF flag is reset */ - if (FMPI2C_WaitOnFlagUntilTimeout(hfmpi2c, FMPI2C_FLAG_STOPF, RESET, Timeout, tickstart) != HAL_OK) - { - return HAL_ERROR; - } - - /* Clear STOP Flag */ - __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_STOPF); - } - /* Increment Trials */ FMPI2C_Trials++; } while (FMPI2C_Trials < Trials); @@ -3253,10 +3376,11 @@ HAL_StatusTypeDef HAL_FMPI2C_IsDeviceReady(FMPI2C_HandleTypeDef *hfmpi2c, uint16 * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size, uint32_t XferOptions) + uint16_t Size, uint32_t XferOptions) { uint32_t xfermode; uint32_t xferrequest = FMPI2C_GENERATE_START_WRITE; + uint32_t sizetoxfer = 0U; /* Check the parameters */ assert_param(IS_FMPI2C_TRANSFER_OPTIONS_REQUEST(XferOptions)); @@ -3288,6 +3412,21 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2 xfermode = hfmpi2c->XferOptions; } + if ((hfmpi2c->XferSize > 0U) && ((XferOptions == FMPI2C_FIRST_FRAME) || \ + (XferOptions == FMPI2C_FIRST_AND_LAST_FRAME))) + { + /* Preload TX register */ + /* Write data to TXDR */ + hfmpi2c->Instance->TXDR = *hfmpi2c->pBuffPtr; + + /* Increment Buffer pointer */ + hfmpi2c->pBuffPtr++; + + sizetoxfer = hfmpi2c->XferSize; + hfmpi2c->XferCount--; + hfmpi2c->XferSize--; + } + /* If transfer direction not change and there is no request to start another frame, do not generate Restart Condition */ /* Mean Previous state is same as current state */ @@ -3309,7 +3448,14 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2 } /* Send Slave Address and set NBYTES to write */ - FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, xfermode, xferrequest); + if ((XferOptions == FMPI2C_FIRST_FRAME) || (XferOptions == FMPI2C_FIRST_AND_LAST_FRAME)) + { + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)sizetoxfer, xfermode, xferrequest); + } + else + { + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, xfermode, xferrequest); + } /* Process Unlocked */ __HAL_UNLOCK(hfmpi2c); @@ -3317,6 +3463,10 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2 /* Note : The FMPI2C interrupts must be enabled after unlocking current process to avoid the risk of FMPI2C interrupt handle execution before current process unlock */ + /* Enable ERR, TC, STOP, NACK, TXI interrupt */ + /* possible to enable all of these */ + /* FMPI2C_IT_ERRI | FMPI2C_IT_TCI | FMPI2C_IT_STOPI | FMPI2C_IT_NACKI | + FMPI2C_IT_ADDRI | FMPI2C_IT_RXI | FMPI2C_IT_TXI */ FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_TX_IT); return HAL_OK; @@ -3340,11 +3490,12 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2 * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size, uint32_t XferOptions) + uint16_t Size, uint32_t XferOptions) { uint32_t xfermode; uint32_t xferrequest = FMPI2C_GENERATE_START_WRITE; HAL_StatusTypeDef dmaxferstatus; + uint32_t sizetoxfer = 0U; /* Check the parameters */ assert_param(IS_FMPI2C_TRANSFER_OPTIONS_REQUEST(XferOptions)); @@ -3376,6 +3527,21 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi xfermode = hfmpi2c->XferOptions; } + if ((hfmpi2c->XferSize > 0U) && ((XferOptions == FMPI2C_FIRST_FRAME) || \ + (XferOptions == FMPI2C_FIRST_AND_LAST_FRAME))) + { + /* Preload TX register */ + /* Write data to TXDR */ + hfmpi2c->Instance->TXDR = *hfmpi2c->pBuffPtr; + + /* Increment Buffer pointer */ + hfmpi2c->pBuffPtr++; + + sizetoxfer = hfmpi2c->XferSize; + hfmpi2c->XferCount--; + hfmpi2c->XferSize--; + } + /* If transfer direction not change and there is no request to start another frame, do not generate Restart Condition */ /* Mean Previous state is same as current state */ @@ -3411,8 +3577,8 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi hfmpi2c->hdmatx->XferAbortCallback = NULL; /* Enable the DMA stream */ - dmaxferstatus = HAL_DMA_Start_IT(hfmpi2c->hdmatx, (uint32_t)pData, (uint32_t)&hfmpi2c->Instance->TXDR, - hfmpi2c->XferSize); + dmaxferstatus = HAL_DMA_Start_IT(hfmpi2c->hdmatx, (uint32_t)hfmpi2c->pBuffPtr, + (uint32_t)&hfmpi2c->Instance->TXDR, hfmpi2c->XferSize); } else { @@ -3432,7 +3598,14 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi if (dmaxferstatus == HAL_OK) { /* Send Slave Address and set NBYTES to write */ - FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, xfermode, xferrequest); + if ((XferOptions == FMPI2C_FIRST_FRAME) || (XferOptions == FMPI2C_FIRST_AND_LAST_FRAME)) + { + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)sizetoxfer, xfermode, xferrequest); + } + else + { + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, xfermode, xferrequest); + } /* Update XferCount value */ hfmpi2c->XferCount -= hfmpi2c->XferSize; @@ -3471,8 +3644,14 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi /* Send Slave Address */ /* Set NBYTES to write and generate START condition */ - FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_AUTOEND_MODE, - FMPI2C_GENERATE_START_WRITE); + if ((XferOptions == FMPI2C_FIRST_FRAME) || (XferOptions == FMPI2C_FIRST_AND_LAST_FRAME)) + { + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)sizetoxfer, xfermode, xferrequest); + } + else + { + FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, xfermode, xferrequest); + } /* Process Unlocked */ __HAL_UNLOCK(hfmpi2c); @@ -3508,7 +3687,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Receive_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size, uint32_t XferOptions) + uint16_t Size, uint32_t XferOptions) { uint32_t xfermode; uint32_t xferrequest = FMPI2C_GENERATE_START_READ; @@ -3595,7 +3774,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Receive_IT(FMPI2C_HandleTypeDef *hfmpi2c * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t *pData, - uint16_t Size, uint32_t XferOptions) + uint16_t Size, uint32_t XferOptions) { uint32_t xfermode; uint32_t xferrequest = FMPI2C_GENERATE_START_READ; @@ -3727,7 +3906,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2 /* Send Slave Address */ /* Set NBYTES to read and generate START condition */ FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_AUTOEND_MODE, - FMPI2C_GENERATE_START_READ); + FMPI2C_GENERATE_START_READ); /* Process Unlocked */ __HAL_UNLOCK(hfmpi2c); @@ -3735,11 +3914,11 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2 /* Note : The FMPI2C interrupts must be enabled after unlocking current process to avoid the risk of FMPI2C interrupt handle execution before current process unlock */ - /* Enable ERR, TC, STOP, NACK, TXI interrupt */ + /* Enable ERR, TC, STOP, NACK, RXI interrupt */ /* possible to enable all of these */ /* FMPI2C_IT_ERRI | FMPI2C_IT_TCI | FMPI2C_IT_STOPI | FMPI2C_IT_NACKI | FMPI2C_IT_ADDRI | FMPI2C_IT_RXI | FMPI2C_IT_TXI */ - FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_TX_IT); + FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_RX_IT); } return HAL_OK; @@ -3761,8 +3940,11 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Seq_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2 * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size, - uint32_t XferOptions) + uint32_t XferOptions) { + /* Declaration of tmp to prevent undefined behavior of volatile usage */ + FlagStatus tmp; + /* Check the parameters */ assert_param(IS_FMPI2C_TRANSFER_OPTIONS_REQUEST(XferOptions)); @@ -3822,7 +4004,8 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2c hfmpi2c->XferOptions = XferOptions; hfmpi2c->XferISR = FMPI2C_Slave_ISR_IT; - if (FMPI2C_GET_DIR(hfmpi2c) == FMPI2C_DIRECTION_RECEIVE) + tmp = __HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_ADDR); + if ((FMPI2C_GET_DIR(hfmpi2c) == FMPI2C_DIRECTION_RECEIVE) && (tmp != RESET)) { /* Clear ADDR flag after prepare the transfer parameters */ /* This action will generate an acknowledge to the Master */ @@ -3857,8 +4040,10 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Transmit_IT(FMPI2C_HandleTypeDef *hfmpi2c * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size, - uint32_t XferOptions) + uint32_t XferOptions) { + /* Declaration of tmp to prevent undefined behavior of volatile usage */ + FlagStatus tmp; HAL_StatusTypeDef dmaxferstatus; /* Check the parameters */ @@ -3893,7 +4078,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2 hfmpi2c->Instance->CR1 &= ~FMPI2C_CR1_RXDMAEN; /* Set the FMPI2C DMA Abort callback : - will lead to call HAL_FMPI2C_ErrorCallback() at end of DMA abort procedure */ + will lead to call HAL_FMPI2C_ErrorCallback() at end of DMA abort procedure */ hfmpi2c->hdmarx->XferAbortCallback = FMPI2C_DMAAbort; /* Abort DMA RX */ @@ -3915,7 +4100,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2 if (hfmpi2c->hdmatx != NULL) { /* Set the FMPI2C DMA Abort callback : - will lead to call HAL_FMPI2C_ErrorCallback() at end of DMA abort procedure */ + will lead to call HAL_FMPI2C_ErrorCallback() at end of DMA abort procedure */ hfmpi2c->hdmatx->XferAbortCallback = FMPI2C_DMAAbort; /* Abort DMA TX */ @@ -4000,7 +4185,8 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2 return HAL_ERROR; } - if (FMPI2C_GET_DIR(hfmpi2c) == FMPI2C_DIRECTION_RECEIVE) + tmp = __HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_ADDR); + if ((FMPI2C_GET_DIR(hfmpi2c) == FMPI2C_DIRECTION_RECEIVE) && (tmp != RESET)) { /* Clear ADDR flag after prepare the transfer parameters */ /* This action will generate an acknowledge to the Master */ @@ -4010,15 +4196,15 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2 /* Process Unlocked */ __HAL_UNLOCK(hfmpi2c); + /* Enable DMA Request */ + hfmpi2c->Instance->CR1 |= FMPI2C_CR1_TXDMAEN; + /* Note : The FMPI2C interrupts must be enabled after unlocking current process to avoid the risk of FMPI2C interrupt handle execution before current process unlock */ /* Enable ERR, STOP, NACK, ADDR interrupts */ FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_LISTEN_IT); - /* Enable DMA Request */ - hfmpi2c->Instance->CR1 |= FMPI2C_CR1_TXDMAEN; - return HAL_OK; } else @@ -4038,8 +4224,11 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Transmit_DMA(FMPI2C_HandleTypeDef *hfmpi2 * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Receive_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size, - uint32_t XferOptions) + uint32_t XferOptions) { + /* Declaration of tmp to prevent undefined behavior of volatile usage */ + FlagStatus tmp; + /* Check the parameters */ assert_param(IS_FMPI2C_TRANSFER_OPTIONS_REQUEST(XferOptions)); @@ -4099,7 +4288,8 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Receive_IT(FMPI2C_HandleTypeDef *hfmpi2c, hfmpi2c->XferOptions = XferOptions; hfmpi2c->XferISR = FMPI2C_Slave_ISR_IT; - if (FMPI2C_GET_DIR(hfmpi2c) == FMPI2C_DIRECTION_TRANSMIT) + tmp = __HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_ADDR); + if ((FMPI2C_GET_DIR(hfmpi2c) == FMPI2C_DIRECTION_TRANSMIT) && (tmp != RESET)) { /* Clear ADDR flag after prepare the transfer parameters */ /* This action will generate an acknowledge to the Master */ @@ -4134,8 +4324,10 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Receive_IT(FMPI2C_HandleTypeDef *hfmpi2c, * @retval HAL status */ HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2c, uint8_t *pData, uint16_t Size, - uint32_t XferOptions) + uint32_t XferOptions) { + /* Declaration of tmp to prevent undefined behavior of volatile usage */ + FlagStatus tmp; HAL_StatusTypeDef dmaxferstatus; /* Check the parameters */ @@ -4277,7 +4469,8 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2c return HAL_ERROR; } - if (FMPI2C_GET_DIR(hfmpi2c) == FMPI2C_DIRECTION_TRANSMIT) + tmp = __HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_ADDR); + if ((FMPI2C_GET_DIR(hfmpi2c) == FMPI2C_DIRECTION_TRANSMIT) && (tmp != RESET)) { /* Clear ADDR flag after prepare the transfer parameters */ /* This action will generate an acknowledge to the Master */ @@ -4287,15 +4480,15 @@ HAL_StatusTypeDef HAL_FMPI2C_Slave_Seq_Receive_DMA(FMPI2C_HandleTypeDef *hfmpi2c /* Process Unlocked */ __HAL_UNLOCK(hfmpi2c); + /* Enable DMA Request */ + hfmpi2c->Instance->CR1 |= FMPI2C_CR1_RXDMAEN; + /* Note : The FMPI2C interrupts must be enabled after unlocking current process to avoid the risk of FMPI2C interrupt handle execution before current process unlock */ /* REnable ADDR interrupt */ FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_RX_IT | FMPI2C_XFER_LISTEN_IT); - /* Enable DMA Request */ - hfmpi2c->Instance->CR1 |= FMPI2C_CR1_RXDMAEN; - return HAL_OK; } else @@ -4429,7 +4622,7 @@ HAL_StatusTypeDef HAL_FMPI2C_Master_Abort_IT(FMPI2C_HandleTypeDef *hfmpi2c, uint * the configuration information for the specified FMPI2C. * @retval None */ -void HAL_FMPI2C_EV_IRQHandler(FMPI2C_HandleTypeDef *hfmpi2c) +void HAL_FMPI2C_EV_IRQHandler(FMPI2C_HandleTypeDef *hfmpi2c) /* Derogation MISRAC2012-Rule-8.13 */ { /* Get current IT Flags and IT sources value */ uint32_t itflags = READ_REG(hfmpi2c->Instance->ISR); @@ -4682,7 +4875,7 @@ __weak void HAL_FMPI2C_AbortCpltCallback(FMPI2C_HandleTypeDef *hfmpi2c) * the configuration information for the specified FMPI2C. * @retval HAL state */ -HAL_FMPI2C_StateTypeDef HAL_FMPI2C_GetState(FMPI2C_HandleTypeDef *hfmpi2c) +HAL_FMPI2C_StateTypeDef HAL_FMPI2C_GetState(const FMPI2C_HandleTypeDef *hfmpi2c) { /* Return FMPI2C handle state */ return hfmpi2c->State; @@ -4694,7 +4887,7 @@ HAL_FMPI2C_StateTypeDef HAL_FMPI2C_GetState(FMPI2C_HandleTypeDef *hfmpi2c) * the configuration information for FMPI2C module * @retval HAL mode */ -HAL_FMPI2C_ModeTypeDef HAL_FMPI2C_GetMode(FMPI2C_HandleTypeDef *hfmpi2c) +HAL_FMPI2C_ModeTypeDef HAL_FMPI2C_GetMode(const FMPI2C_HandleTypeDef *hfmpi2c) { return hfmpi2c->Mode; } @@ -4705,7 +4898,7 @@ HAL_FMPI2C_ModeTypeDef HAL_FMPI2C_GetMode(FMPI2C_HandleTypeDef *hfmpi2c) * the configuration information for the specified FMPI2C. * @retval FMPI2C Error Code */ -uint32_t HAL_FMPI2C_GetError(FMPI2C_HandleTypeDef *hfmpi2c) +uint32_t HAL_FMPI2C_GetError(const FMPI2C_HandleTypeDef *hfmpi2c) { return hfmpi2c->ErrorCode; } @@ -4731,7 +4924,7 @@ uint32_t HAL_FMPI2C_GetError(FMPI2C_HandleTypeDef *hfmpi2c) * @retval HAL status */ static HAL_StatusTypeDef FMPI2C_Master_ISR_IT(struct __FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags, - uint32_t ITSources) + uint32_t ITSources) { uint16_t devaddress; uint32_t tmpITFlags = ITFlags; @@ -4768,17 +4961,22 @@ static HAL_StatusTypeDef FMPI2C_Master_ISR_IT(struct __FMPI2C_HandleTypeDef *hfm hfmpi2c->XferSize--; hfmpi2c->XferCount--; } - else if ((FMPI2C_CHECK_FLAG(tmpITFlags, FMPI2C_FLAG_TXIS) != RESET) && \ - (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_TXI) != RESET)) + else if ((FMPI2C_CHECK_FLAG(tmpITFlags, FMPI2C_FLAG_TC) == RESET) && \ + ((FMPI2C_CHECK_FLAG(tmpITFlags, FMPI2C_FLAG_TXIS) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_TXI) != RESET))) { /* Write data to TXDR */ - hfmpi2c->Instance->TXDR = *hfmpi2c->pBuffPtr; + if (hfmpi2c->XferCount != 0U) + { + /* Write data to TXDR */ + hfmpi2c->Instance->TXDR = *hfmpi2c->pBuffPtr; - /* Increment Buffer pointer */ - hfmpi2c->pBuffPtr++; + /* Increment Buffer pointer */ + hfmpi2c->pBuffPtr++; - hfmpi2c->XferSize--; - hfmpi2c->XferCount--; + hfmpi2c->XferSize--; + hfmpi2c->XferCount--; + } } else if ((FMPI2C_CHECK_FLAG(tmpITFlags, FMPI2C_FLAG_TCR) != RESET) && \ (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_TCI) != RESET)) @@ -4789,7 +4987,15 @@ static HAL_StatusTypeDef FMPI2C_Master_ISR_IT(struct __FMPI2C_HandleTypeDef *hfm if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) { - hfmpi2c->XferSize = MAX_NBYTE_SIZE; + /* Errata workaround 170323 */ + if (FMPI2C_GET_DIR(hfmpi2c) == FMPI2C_DIRECTION_RECEIVE) + { + hfmpi2c->XferSize = 1U; + } + else + { + hfmpi2c->XferSize = MAX_NBYTE_SIZE; + } FMPI2C_TransferConfig(hfmpi2c, devaddress, (uint8_t)hfmpi2c->XferSize, FMPI2C_RELOAD_MODE, FMPI2C_NO_STARTSTOP); } else @@ -4798,12 +5004,12 @@ static HAL_StatusTypeDef FMPI2C_Master_ISR_IT(struct __FMPI2C_HandleTypeDef *hfm if (hfmpi2c->XferOptions != FMPI2C_NO_OPTION_FRAME) { FMPI2C_TransferConfig(hfmpi2c, devaddress, (uint8_t)hfmpi2c->XferSize, - hfmpi2c->XferOptions, FMPI2C_NO_STARTSTOP); + hfmpi2c->XferOptions, FMPI2C_NO_STARTSTOP); } else { FMPI2C_TransferConfig(hfmpi2c, devaddress, (uint8_t)hfmpi2c->XferSize, - FMPI2C_AUTOEND_MODE, FMPI2C_NO_STARTSTOP); + FMPI2C_AUTOEND_MODE, FMPI2C_NO_STARTSTOP); } } } @@ -4869,50 +5075,208 @@ static HAL_StatusTypeDef FMPI2C_Master_ISR_IT(struct __FMPI2C_HandleTypeDef *hfm } /** - * @brief Interrupt Sub-Routine which handle the Interrupt Flags Slave Mode with Interrupt. + * @brief Interrupt Sub-Routine which handle the Interrupt Flags Memory Mode with Interrupt. * @param hfmpi2c Pointer to a FMPI2C_HandleTypeDef structure that contains * the configuration information for the specified FMPI2C. * @param ITFlags Interrupt flags to handle. * @param ITSources Interrupt sources enabled. * @retval HAL status */ -static HAL_StatusTypeDef FMPI2C_Slave_ISR_IT(struct __FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags, - uint32_t ITSources) +static HAL_StatusTypeDef FMPI2C_Mem_ISR_IT(struct __FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags, + uint32_t ITSources) { - uint32_t tmpoptions = hfmpi2c->XferOptions; + uint32_t direction = FMPI2C_GENERATE_START_WRITE; uint32_t tmpITFlags = ITFlags; - /* Process locked */ + /* Process Locked */ __HAL_LOCK(hfmpi2c); - /* Check if STOPF is set */ - if ((FMPI2C_CHECK_FLAG(tmpITFlags, FMPI2C_FLAG_STOPF) != RESET) && \ - (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_STOPI) != RESET)) + if ((FMPI2C_CHECK_FLAG(tmpITFlags, FMPI2C_FLAG_AF) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_NACKI) != RESET)) { - /* Call FMPI2C Slave complete process */ - FMPI2C_ITSlaveCplt(hfmpi2c, tmpITFlags); + /* Clear NACK Flag */ + __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_AF); + + /* Set corresponding Error Code */ + /* No need to generate STOP, it is automatically done */ + /* Error callback will be send during stop flag treatment */ + hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_AF; + + /* Flush TX register */ + FMPI2C_Flush_TXDR(hfmpi2c); } + else if ((FMPI2C_CHECK_FLAG(tmpITFlags, FMPI2C_FLAG_RXNE) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_RXI) != RESET)) + { + /* Remove RXNE flag on temporary variable as read done */ + tmpITFlags &= ~FMPI2C_FLAG_RXNE; - if ((FMPI2C_CHECK_FLAG(tmpITFlags, FMPI2C_FLAG_AF) != RESET) && \ - (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_NACKI) != RESET)) + /* Read data from RXDR */ + *hfmpi2c->pBuffPtr = (uint8_t)hfmpi2c->Instance->RXDR; + + /* Increment Buffer pointer */ + hfmpi2c->pBuffPtr++; + + hfmpi2c->XferSize--; + hfmpi2c->XferCount--; + } + else if ((FMPI2C_CHECK_FLAG(tmpITFlags, FMPI2C_FLAG_TXIS) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_TXI) != RESET)) { - /* Check that FMPI2C transfer finished */ - /* if yes, normal use case, a NACK is sent by the MASTER when Transfer is finished */ - /* Mean XferCount == 0*/ - /* So clear Flag NACKF only */ - if (hfmpi2c->XferCount == 0U) + if (hfmpi2c->Memaddress == 0xFFFFFFFFU) { - if ((hfmpi2c->State == HAL_FMPI2C_STATE_LISTEN) && (tmpoptions == FMPI2C_FIRST_AND_LAST_FRAME)) - /* Same action must be done for (tmpoptions == FMPI2C_LAST_FRAME) which removed for - Warning[Pa134]: left and right operands are identical */ - { - /* Call FMPI2C Listen complete process */ - FMPI2C_ITListenCplt(hfmpi2c, tmpITFlags); - } - else if ((hfmpi2c->State == HAL_FMPI2C_STATE_BUSY_TX_LISTEN) && (tmpoptions != FMPI2C_NO_OPTION_FRAME)) - { - /* Clear NACK Flag */ - __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_AF); + /* Write data to TXDR */ + hfmpi2c->Instance->TXDR = *hfmpi2c->pBuffPtr; + + /* Increment Buffer pointer */ + hfmpi2c->pBuffPtr++; + + hfmpi2c->XferSize--; + hfmpi2c->XferCount--; + } + else + { + /* Write LSB part of Memory Address */ + hfmpi2c->Instance->TXDR = hfmpi2c->Memaddress; + + /* Reset Memaddress content */ + hfmpi2c->Memaddress = 0xFFFFFFFFU; + } + } + else if ((FMPI2C_CHECK_FLAG(tmpITFlags, FMPI2C_FLAG_TCR) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_TCI) != RESET)) + { + if ((hfmpi2c->XferCount != 0U) && (hfmpi2c->XferSize == 0U)) + { + if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) + { + /* Errata workaround 170323 */ + if (FMPI2C_GET_DIR(hfmpi2c) == FMPI2C_DIRECTION_RECEIVE) + { + hfmpi2c->XferSize = 1U; + } + else + { + hfmpi2c->XferSize = MAX_NBYTE_SIZE; + } + FMPI2C_TransferConfig(hfmpi2c, (uint16_t)hfmpi2c->Devaddress, (uint8_t)hfmpi2c->XferSize, + FMPI2C_RELOAD_MODE, FMPI2C_NO_STARTSTOP); + } + else + { + hfmpi2c->XferSize = hfmpi2c->XferCount; + FMPI2C_TransferConfig(hfmpi2c, (uint16_t)hfmpi2c->Devaddress, (uint8_t)hfmpi2c->XferSize, + FMPI2C_AUTOEND_MODE, FMPI2C_NO_STARTSTOP); + } + } + else + { + /* Wrong size Status regarding TCR flag event */ + /* Call the corresponding callback to inform upper layer of End of Transfer */ + FMPI2C_ITError(hfmpi2c, HAL_FMPI2C_ERROR_SIZE); + } + } + else if ((FMPI2C_CHECK_FLAG(tmpITFlags, FMPI2C_FLAG_TC) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_TCI) != RESET)) + { + /* Disable Interrupt related to address step */ + FMPI2C_Disable_IRQ(hfmpi2c, FMPI2C_XFER_TX_IT); + + /* Enable ERR, TC, STOP, NACK and RXI interrupts */ + FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_RX_IT); + + if (hfmpi2c->State == HAL_FMPI2C_STATE_BUSY_RX) + { + direction = FMPI2C_GENERATE_START_READ; + } + + if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) + { + /* Errata workaround 170323 */ + if (FMPI2C_GET_DIR(hfmpi2c) == FMPI2C_DIRECTION_RECEIVE) + { + hfmpi2c->XferSize = 1U; + } + else + { + hfmpi2c->XferSize = MAX_NBYTE_SIZE; + } + + /* Set NBYTES to write and reload if hfmpi2c->XferCount > MAX_NBYTE_SIZE and generate RESTART */ + FMPI2C_TransferConfig(hfmpi2c, (uint16_t)hfmpi2c->Devaddress, (uint8_t)hfmpi2c->XferSize, + FMPI2C_RELOAD_MODE, direction); + } + else + { + hfmpi2c->XferSize = hfmpi2c->XferCount; + + /* Set NBYTES to write and generate RESTART */ + FMPI2C_TransferConfig(hfmpi2c, (uint16_t)hfmpi2c->Devaddress, (uint8_t)hfmpi2c->XferSize, + FMPI2C_AUTOEND_MODE, direction); + } + } + else + { + /* Nothing to do */ + } + + if ((FMPI2C_CHECK_FLAG(tmpITFlags, FMPI2C_FLAG_STOPF) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_STOPI) != RESET)) + { + /* Call FMPI2C Master complete process */ + FMPI2C_ITMasterCplt(hfmpi2c, tmpITFlags); + } + + /* Process Unlocked */ + __HAL_UNLOCK(hfmpi2c); + + return HAL_OK; +} + +/** + * @brief Interrupt Sub-Routine which handle the Interrupt Flags Slave Mode with Interrupt. + * @param hfmpi2c Pointer to a FMPI2C_HandleTypeDef structure that contains + * the configuration information for the specified FMPI2C. + * @param ITFlags Interrupt flags to handle. + * @param ITSources Interrupt sources enabled. + * @retval HAL status + */ +static HAL_StatusTypeDef FMPI2C_Slave_ISR_IT(struct __FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags, + uint32_t ITSources) +{ + uint32_t tmpoptions = hfmpi2c->XferOptions; + uint32_t tmpITFlags = ITFlags; + + /* Process locked */ + __HAL_LOCK(hfmpi2c); + + /* Check if STOPF is set */ + if ((FMPI2C_CHECK_FLAG(tmpITFlags, FMPI2C_FLAG_STOPF) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_STOPI) != RESET)) + { + /* Call FMPI2C Slave complete process */ + FMPI2C_ITSlaveCplt(hfmpi2c, tmpITFlags); + } + else if ((FMPI2C_CHECK_FLAG(tmpITFlags, FMPI2C_FLAG_AF) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_NACKI) != RESET)) + { + /* Check that FMPI2C transfer finished */ + /* if yes, normal use case, a NACK is sent by the MASTER when Transfer is finished */ + /* Mean XferCount == 0*/ + /* So clear Flag NACKF only */ + if (hfmpi2c->XferCount == 0U) + { + if ((hfmpi2c->State == HAL_FMPI2C_STATE_LISTEN) && (tmpoptions == FMPI2C_FIRST_AND_LAST_FRAME)) + /* Same action must be done for (tmpoptions == FMPI2C_LAST_FRAME) which removed for + Warning[Pa134]: left and right operands are identical */ + { + /* Call FMPI2C Listen complete process */ + FMPI2C_ITListenCplt(hfmpi2c, tmpITFlags); + } + else if ((hfmpi2c->State == HAL_FMPI2C_STATE_BUSY_TX_LISTEN) && (tmpoptions != FMPI2C_NO_OPTION_FRAME)) + { + /* Clear NACK Flag */ + __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_AF); /* Flush TX register */ FMPI2C_Flush_TXDR(hfmpi2c); @@ -5018,7 +5382,7 @@ static HAL_StatusTypeDef FMPI2C_Slave_ISR_IT(struct __FMPI2C_HandleTypeDef *hfmp * @retval HAL status */ static HAL_StatusTypeDef FMPI2C_Master_ISR_DMA(struct __FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags, - uint32_t ITSources) + uint32_t ITSources) { uint16_t devaddress; uint32_t xfermode; @@ -5057,7 +5421,15 @@ static HAL_StatusTypeDef FMPI2C_Master_ISR_DMA(struct __FMPI2C_HandleTypeDef *hf /* Prepare the new XferSize to transfer */ if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) { - hfmpi2c->XferSize = MAX_NBYTE_SIZE; + /* Errata workaround 170323 */ + if (FMPI2C_GET_DIR(hfmpi2c) == FMPI2C_DIRECTION_RECEIVE) + { + hfmpi2c->XferSize = 1U; + } + else + { + hfmpi2c->XferSize = MAX_NBYTE_SIZE; + } xfermode = FMPI2C_RELOAD_MODE; } else @@ -5149,6 +5521,170 @@ static HAL_StatusTypeDef FMPI2C_Master_ISR_DMA(struct __FMPI2C_HandleTypeDef *hf return HAL_OK; } +/** + * @brief Interrupt Sub-Routine which handle the Interrupt Flags Memory Mode with DMA. + * @param hfmpi2c Pointer to a FMPI2C_HandleTypeDef structure that contains + * the configuration information for the specified FMPI2C. + * @param ITFlags Interrupt flags to handle. + * @param ITSources Interrupt sources enabled. + * @retval HAL status + */ +static HAL_StatusTypeDef FMPI2C_Mem_ISR_DMA(struct __FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags, + uint32_t ITSources) +{ + uint32_t direction = FMPI2C_GENERATE_START_WRITE; + + /* Process Locked */ + __HAL_LOCK(hfmpi2c); + + if ((FMPI2C_CHECK_FLAG(ITFlags, FMPI2C_FLAG_AF) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_NACKI) != RESET)) + { + /* Clear NACK Flag */ + __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_AF); + + /* Set corresponding Error Code */ + hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_AF; + + /* No need to generate STOP, it is automatically done */ + /* But enable STOP interrupt, to treat it */ + /* Error callback will be send during stop flag treatment */ + FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_CPLT_IT); + + /* Flush TX register */ + FMPI2C_Flush_TXDR(hfmpi2c); + } + else if ((FMPI2C_CHECK_FLAG(ITFlags, FMPI2C_FLAG_TXIS) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_TXI) != RESET)) + { + /* Write LSB part of Memory Address */ + hfmpi2c->Instance->TXDR = hfmpi2c->Memaddress; + + /* Reset Memaddress content */ + hfmpi2c->Memaddress = 0xFFFFFFFFU; + } + else if ((FMPI2C_CHECK_FLAG(ITFlags, FMPI2C_FLAG_TCR) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_TCI) != RESET)) + { + /* Disable Interrupt related to address step */ + FMPI2C_Disable_IRQ(hfmpi2c, FMPI2C_XFER_TX_IT); + + /* Enable only Error interrupt */ + FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_ERROR_IT); + + if (hfmpi2c->XferCount != 0U) + { + /* Prepare the new XferSize to transfer */ + if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) + { + /* Errata workaround 170323 */ + if (FMPI2C_GET_DIR(hfmpi2c) == FMPI2C_DIRECTION_RECEIVE) + { + hfmpi2c->XferSize = 1U; + } + else + { + hfmpi2c->XferSize = MAX_NBYTE_SIZE; + } + FMPI2C_TransferConfig(hfmpi2c, (uint16_t)hfmpi2c->Devaddress, (uint8_t)hfmpi2c->XferSize, + FMPI2C_RELOAD_MODE, FMPI2C_NO_STARTSTOP); + } + else + { + hfmpi2c->XferSize = hfmpi2c->XferCount; + FMPI2C_TransferConfig(hfmpi2c, (uint16_t)hfmpi2c->Devaddress, (uint8_t)hfmpi2c->XferSize, + FMPI2C_AUTOEND_MODE, FMPI2C_NO_STARTSTOP); + } + + /* Update XferCount value */ + hfmpi2c->XferCount -= hfmpi2c->XferSize; + + /* Enable DMA Request */ + if (hfmpi2c->State == HAL_FMPI2C_STATE_BUSY_RX) + { + hfmpi2c->Instance->CR1 |= FMPI2C_CR1_RXDMAEN; + } + else + { + hfmpi2c->Instance->CR1 |= FMPI2C_CR1_TXDMAEN; + } + } + else + { + /* Wrong size Status regarding TCR flag event */ + /* Call the corresponding callback to inform upper layer of End of Transfer */ + FMPI2C_ITError(hfmpi2c, HAL_FMPI2C_ERROR_SIZE); + } + } + else if ((FMPI2C_CHECK_FLAG(ITFlags, FMPI2C_FLAG_TC) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_TCI) != RESET)) + { + /* Disable Interrupt related to address step */ + FMPI2C_Disable_IRQ(hfmpi2c, FMPI2C_XFER_TX_IT); + + /* Enable only Error and NACK interrupt for data transfer */ + FMPI2C_Enable_IRQ(hfmpi2c, FMPI2C_XFER_ERROR_IT); + + if (hfmpi2c->State == HAL_FMPI2C_STATE_BUSY_RX) + { + direction = FMPI2C_GENERATE_START_READ; + } + + if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) + { + /* Errata workaround 170323 */ + if (FMPI2C_GET_DIR(hfmpi2c) == FMPI2C_DIRECTION_RECEIVE) + { + hfmpi2c->XferSize = 1U; + } + else + { + hfmpi2c->XferSize = MAX_NBYTE_SIZE; + } + + /* Set NBYTES to write and reload if hfmpi2c->XferCount > MAX_NBYTE_SIZE and generate RESTART */ + FMPI2C_TransferConfig(hfmpi2c, (uint16_t)hfmpi2c->Devaddress, (uint8_t)hfmpi2c->XferSize, + FMPI2C_RELOAD_MODE, direction); + } + else + { + hfmpi2c->XferSize = hfmpi2c->XferCount; + + /* Set NBYTES to write and generate RESTART */ + FMPI2C_TransferConfig(hfmpi2c, (uint16_t)hfmpi2c->Devaddress, (uint8_t)hfmpi2c->XferSize, + FMPI2C_AUTOEND_MODE, direction); + } + + /* Update XferCount value */ + hfmpi2c->XferCount -= hfmpi2c->XferSize; + + /* Enable DMA Request */ + if (hfmpi2c->State == HAL_FMPI2C_STATE_BUSY_RX) + { + hfmpi2c->Instance->CR1 |= FMPI2C_CR1_RXDMAEN; + } + else + { + hfmpi2c->Instance->CR1 |= FMPI2C_CR1_TXDMAEN; + } + } + else if ((FMPI2C_CHECK_FLAG(ITFlags, FMPI2C_FLAG_STOPF) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_STOPI) != RESET)) + { + /* Call FMPI2C Master complete process */ + FMPI2C_ITMasterCplt(hfmpi2c, ITFlags); + } + else + { + /* Nothing to do */ + } + + /* Process Unlocked */ + __HAL_UNLOCK(hfmpi2c); + + return HAL_OK; +} + /** * @brief Interrupt Sub-Routine which handle the Interrupt Flags Slave Mode with DMA. * @param hfmpi2c Pointer to a FMPI2C_HandleTypeDef structure that contains @@ -5158,7 +5694,7 @@ static HAL_StatusTypeDef FMPI2C_Master_ISR_DMA(struct __FMPI2C_HandleTypeDef *hf * @retval HAL status */ static HAL_StatusTypeDef FMPI2C_Slave_ISR_DMA(struct __FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags, - uint32_t ITSources) + uint32_t ITSources) { uint32_t tmpoptions = hfmpi2c->XferOptions; uint32_t treatdmanack = 0U; @@ -5174,9 +5710,8 @@ static HAL_StatusTypeDef FMPI2C_Slave_ISR_DMA(struct __FMPI2C_HandleTypeDef *hfm /* Call FMPI2C Slave complete process */ FMPI2C_ITSlaveCplt(hfmpi2c, ITFlags); } - - if ((FMPI2C_CHECK_FLAG(ITFlags, FMPI2C_FLAG_AF) != RESET) && \ - (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_NACKI) != RESET)) + else if ((FMPI2C_CHECK_FLAG(ITFlags, FMPI2C_FLAG_AF) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_IT_NACKI) != RESET)) { /* Check that FMPI2C transfer finished */ /* if yes, normal use case, a NACK is sent by the MASTER when Transfer is finished */ @@ -5190,7 +5725,7 @@ static HAL_StatusTypeDef FMPI2C_Slave_ISR_DMA(struct __FMPI2C_HandleTypeDef *hfm { if (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_CR1_RXDMAEN) != RESET) { - if (__HAL_DMA_GET_COUNTER(hfmpi2c->hdmarx) == 0U) + if (FMPI2C_GET_DMA_REMAIN_DATA(hfmpi2c->hdmarx) == 0U) { treatdmanack = 1U; } @@ -5202,7 +5737,7 @@ static HAL_StatusTypeDef FMPI2C_Slave_ISR_DMA(struct __FMPI2C_HandleTypeDef *hfm { if (FMPI2C_CHECK_IT_SOURCE(ITSources, FMPI2C_CR1_TXDMAEN) != RESET) { - if (__HAL_DMA_GET_COUNTER(hfmpi2c->hdmatx) == 0U) + if (FMPI2C_GET_DMA_REMAIN_DATA(hfmpi2c->hdmatx) == 0U) { treatdmanack = 1U; } @@ -5303,8 +5838,8 @@ static HAL_StatusTypeDef FMPI2C_Slave_ISR_DMA(struct __FMPI2C_HandleTypeDef *hfm * @retval HAL status */ static HAL_StatusTypeDef FMPI2C_RequestMemoryWrite(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, - uint16_t MemAddress, uint16_t MemAddSize, uint32_t Timeout, - uint32_t Tickstart) + uint16_t MemAddress, uint16_t MemAddSize, uint32_t Timeout, + uint32_t Tickstart) { FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)MemAddSize, FMPI2C_RELOAD_MODE, FMPI2C_GENERATE_START_WRITE); @@ -5358,8 +5893,8 @@ static HAL_StatusTypeDef FMPI2C_RequestMemoryWrite(FMPI2C_HandleTypeDef *hfmpi2c * @retval HAL status */ static HAL_StatusTypeDef FMPI2C_RequestMemoryRead(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, - uint16_t MemAddress, uint16_t MemAddSize, uint32_t Timeout, - uint32_t Tickstart) + uint16_t MemAddress, uint16_t MemAddSize, uint32_t Timeout, + uint32_t Tickstart) { FMPI2C_TransferConfig(hfmpi2c, DevAddress, (uint8_t)MemAddSize, FMPI2C_SOFTEND_MODE, FMPI2C_GENERATE_START_WRITE); @@ -5775,6 +6310,7 @@ static void FMPI2C_ITSlaveCplt(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags) { uint32_t tmpcr1value = READ_REG(hfmpi2c->Instance->CR1); uint32_t tmpITFlags = ITFlags; + uint32_t tmpoptions = hfmpi2c->XferOptions; HAL_FMPI2C_StateTypeDef tmpstate = hfmpi2c->State; /* Clear STOP Flag */ @@ -5791,6 +6327,11 @@ static void FMPI2C_ITSlaveCplt(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags) FMPI2C_Disable_IRQ(hfmpi2c, FMPI2C_XFER_LISTEN_IT | FMPI2C_XFER_RX_IT); hfmpi2c->PreviousState = FMPI2C_STATE_SLAVE_BUSY_RX; } + else if (tmpstate == HAL_FMPI2C_STATE_LISTEN) + { + FMPI2C_Disable_IRQ(hfmpi2c, FMPI2C_XFER_LISTEN_IT | FMPI2C_XFER_TX_IT | FMPI2C_XFER_RX_IT); + hfmpi2c->PreviousState = FMPI2C_STATE_NONE; + } else { /* Do nothing */ @@ -5813,7 +6354,7 @@ static void FMPI2C_ITSlaveCplt(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags) if (hfmpi2c->hdmatx != NULL) { - hfmpi2c->XferCount = (uint16_t)__HAL_DMA_GET_COUNTER(hfmpi2c->hdmatx); + hfmpi2c->XferCount = (uint16_t)FMPI2C_GET_DMA_REMAIN_DATA(hfmpi2c->hdmatx); } } else if (FMPI2C_CHECK_IT_SOURCE(tmpcr1value, FMPI2C_CR1_RXDMAEN) != RESET) @@ -5823,7 +6364,7 @@ static void FMPI2C_ITSlaveCplt(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags) if (hfmpi2c->hdmarx != NULL) { - hfmpi2c->XferCount = (uint16_t)__HAL_DMA_GET_COUNTER(hfmpi2c->hdmarx); + hfmpi2c->XferCount = (uint16_t)FMPI2C_GET_DMA_REMAIN_DATA(hfmpi2c->hdmarx); } } else @@ -5857,6 +6398,57 @@ static void FMPI2C_ITSlaveCplt(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags) hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_AF; } + if ((FMPI2C_CHECK_FLAG(tmpITFlags, FMPI2C_FLAG_AF) != RESET) && \ + (FMPI2C_CHECK_IT_SOURCE(tmpcr1value, FMPI2C_IT_NACKI) != RESET)) + { + /* Check that FMPI2C transfer finished */ + /* if yes, normal use case, a NACK is sent by the MASTER when Transfer is finished */ + /* Mean XferCount == 0*/ + /* So clear Flag NACKF only */ + if (hfmpi2c->XferCount == 0U) + { + if ((hfmpi2c->State == HAL_FMPI2C_STATE_LISTEN) && (tmpoptions == FMPI2C_FIRST_AND_LAST_FRAME)) + /* Same action must be done for (tmpoptions == FMPI2C_LAST_FRAME) which removed for + Warning[Pa134]: left and right operands are identical */ + { + /* Call FMPI2C Listen complete process */ + FMPI2C_ITListenCplt(hfmpi2c, tmpITFlags); + } + else if ((hfmpi2c->State == HAL_FMPI2C_STATE_BUSY_TX_LISTEN) && (tmpoptions != FMPI2C_NO_OPTION_FRAME)) + { + /* Clear NACK Flag */ + __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_AF); + + /* Flush TX register */ + FMPI2C_Flush_TXDR(hfmpi2c); + + /* Last Byte is Transmitted */ + /* Call FMPI2C Slave Sequential complete process */ + FMPI2C_ITSlaveSeqCplt(hfmpi2c); + } + else + { + /* Clear NACK Flag */ + __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_AF); + } + } + else + { + /* if no, error use case, a Non-Acknowledge of last Data is generated by the MASTER*/ + /* Clear NACK Flag */ + __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_AF); + + /* Set ErrorCode corresponding to a Non-Acknowledge */ + hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_AF; + + if ((tmpoptions == FMPI2C_FIRST_FRAME) || (tmpoptions == FMPI2C_NEXT_FRAME)) + { + /* Call the corresponding callback to inform upper layer of End of Transfer */ + FMPI2C_ITError(hfmpi2c, hfmpi2c->ErrorCode); + } + } + } + hfmpi2c->Mode = HAL_FMPI2C_MODE_NONE; hfmpi2c->XferISR = NULL; @@ -5984,6 +6576,7 @@ static void FMPI2C_ITListenCplt(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ITFlags) static void FMPI2C_ITError(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ErrorCode) { HAL_FMPI2C_StateTypeDef tmpstate = hfmpi2c->State; + uint32_t tmppreviousstate; /* Reset handle parameters */ @@ -6011,20 +6604,38 @@ static void FMPI2C_ITError(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ErrorCode) /* Disable all interrupts */ FMPI2C_Disable_IRQ(hfmpi2c, FMPI2C_XFER_LISTEN_IT | FMPI2C_XFER_RX_IT | FMPI2C_XFER_TX_IT); + /* Flush TX register */ + FMPI2C_Flush_TXDR(hfmpi2c); + /* If state is an abort treatment on going, don't change state */ /* This change will be do later */ if (hfmpi2c->State != HAL_FMPI2C_STATE_ABORT) { /* Set HAL_FMPI2C_STATE_READY */ hfmpi2c->State = HAL_FMPI2C_STATE_READY; + + /* Check if a STOPF is detected */ + if (__HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_STOPF) == SET) + { + if (__HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_AF) == SET) + { + __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_AF); + hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_AF; + } + + /* Clear STOP Flag */ + __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_STOPF); + } + } hfmpi2c->XferISR = NULL; } /* Abort DMA TX transfer if any */ tmppreviousstate = hfmpi2c->PreviousState; + if ((hfmpi2c->hdmatx != NULL) && ((tmppreviousstate == FMPI2C_STATE_MASTER_BUSY_TX) || \ - (tmppreviousstate == FMPI2C_STATE_SLAVE_BUSY_TX))) + (tmppreviousstate == FMPI2C_STATE_SLAVE_BUSY_TX))) { if ((hfmpi2c->Instance->CR1 & FMPI2C_CR1_TXDMAEN) == FMPI2C_CR1_TXDMAEN) { @@ -6054,7 +6665,7 @@ static void FMPI2C_ITError(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t ErrorCode) } /* Abort DMA RX transfer if any */ else if ((hfmpi2c->hdmarx != NULL) && ((tmppreviousstate == FMPI2C_STATE_MASTER_BUSY_RX) || \ - (tmppreviousstate == FMPI2C_STATE_SLAVE_BUSY_RX))) + (tmppreviousstate == FMPI2C_STATE_SLAVE_BUSY_RX))) { if ((hfmpi2c->Instance->CR1 & FMPI2C_CR1_RXDMAEN) == FMPI2C_CR1_RXDMAEN) { @@ -6197,6 +6808,7 @@ static void FMPI2C_DMAMasterTransmitCplt(DMA_HandleTypeDef *hdma) } } + /** * @brief DMA FMPI2C slave transmit process complete callback. * @param hdma DMA handle @@ -6225,6 +6837,7 @@ static void FMPI2C_DMASlaveTransmitCplt(DMA_HandleTypeDef *hdma) } } + /** * @brief DMA FMPI2C master receive process complete callback. * @param hdma DMA handle @@ -6253,7 +6866,15 @@ static void FMPI2C_DMAMasterReceiveCplt(DMA_HandleTypeDef *hdma) /* Set the XferSize to transfer */ if (hfmpi2c->XferCount > MAX_NBYTE_SIZE) { - hfmpi2c->XferSize = MAX_NBYTE_SIZE; + /* Errata workaround 170323 */ + if (FMPI2C_GET_DIR(hfmpi2c) == FMPI2C_DIRECTION_RECEIVE) + { + hfmpi2c->XferSize = 1U; + } + else + { + hfmpi2c->XferSize = MAX_NBYTE_SIZE; + } } else { @@ -6275,6 +6896,7 @@ static void FMPI2C_DMAMasterReceiveCplt(DMA_HandleTypeDef *hdma) } } + /** * @brief DMA FMPI2C slave receive process complete callback. * @param hdma DMA handle @@ -6286,7 +6908,7 @@ static void FMPI2C_DMASlaveReceiveCplt(DMA_HandleTypeDef *hdma) FMPI2C_HandleTypeDef *hfmpi2c = (FMPI2C_HandleTypeDef *)(((DMA_HandleTypeDef *)hdma)->Parent); uint32_t tmpoptions = hfmpi2c->XferOptions; - if ((__HAL_DMA_GET_COUNTER(hfmpi2c->hdmarx) == 0U) && \ + if ((FMPI2C_GET_DMA_REMAIN_DATA(hfmpi2c->hdmarx) == 0U) && \ (tmpoptions != FMPI2C_NO_OPTION_FRAME)) { /* Disable DMA Request */ @@ -6303,6 +6925,7 @@ static void FMPI2C_DMASlaveReceiveCplt(DMA_HandleTypeDef *hdma) } } + /** * @brief DMA FMPI2C communication error callback. * @param hdma DMA handle @@ -6316,7 +6939,7 @@ static void FMPI2C_DMAError(DMA_HandleTypeDef *hdma) if (hfmpi2c->hdmatx != NULL) { - if (__HAL_DMA_GET_COUNTER(hfmpi2c->hdmatx) == 0U) + if (FMPI2C_GET_DMA_REMAIN_DATA(hfmpi2c->hdmatx) == 0U) { treatdmaerror = 1U; } @@ -6324,7 +6947,7 @@ static void FMPI2C_DMAError(DMA_HandleTypeDef *hdma) if (hfmpi2c->hdmarx != NULL) { - if (__HAL_DMA_GET_COUNTER(hfmpi2c->hdmarx) == 0U) + if (FMPI2C_GET_DMA_REMAIN_DATA(hfmpi2c->hdmarx) == 0U) { treatdmaerror = 1U; } @@ -6341,6 +6964,7 @@ static void FMPI2C_DMAError(DMA_HandleTypeDef *hdma) } } + /** * @brief DMA FMPI2C communication abort callback * (To be called at end of DMA Abort procedure). @@ -6365,6 +6989,7 @@ static void FMPI2C_DMAAbort(DMA_HandleTypeDef *hdma) FMPI2C_TreatErrorCallback(hfmpi2c); } + /** * @brief This function handles FMPI2C Communication Timeout. It waits * until a flag is no longer in the specified status. @@ -6377,22 +7002,31 @@ static void FMPI2C_DMAAbort(DMA_HandleTypeDef *hdma) * @retval HAL status */ static HAL_StatusTypeDef FMPI2C_WaitOnFlagUntilTimeout(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t Flag, FlagStatus Status, - uint32_t Timeout, uint32_t Tickstart) + uint32_t Timeout, uint32_t Tickstart) { while (__HAL_FMPI2C_GET_FLAG(hfmpi2c, Flag) == Status) { + /* Check if an error is detected */ + if (FMPI2C_IsErrorOccurred(hfmpi2c, Timeout, Tickstart) != HAL_OK) + { + return HAL_ERROR; + } + /* Check for the Timeout */ if (Timeout != HAL_MAX_DELAY) { if (((HAL_GetTick() - Tickstart) > Timeout) || (Timeout == 0U)) { - hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_TIMEOUT; - hfmpi2c->State = HAL_FMPI2C_STATE_READY; - hfmpi2c->Mode = HAL_FMPI2C_MODE_NONE; + if ((__HAL_FMPI2C_GET_FLAG(hfmpi2c, Flag) == Status)) + { + hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_TIMEOUT; + hfmpi2c->State = HAL_FMPI2C_STATE_READY; + hfmpi2c->Mode = HAL_FMPI2C_MODE_NONE; - /* Process Unlocked */ - __HAL_UNLOCK(hfmpi2c); - return HAL_ERROR; + /* Process Unlocked */ + __HAL_UNLOCK(hfmpi2c); + return HAL_ERROR; + } } } } @@ -6408,7 +7042,7 @@ static HAL_StatusTypeDef FMPI2C_WaitOnFlagUntilTimeout(FMPI2C_HandleTypeDef *hfm * @retval HAL status */ static HAL_StatusTypeDef FMPI2C_WaitOnTXISFlagUntilTimeout(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t Timeout, - uint32_t Tickstart) + uint32_t Tickstart) { while (__HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_TXIS) == RESET) { @@ -6423,14 +7057,17 @@ static HAL_StatusTypeDef FMPI2C_WaitOnTXISFlagUntilTimeout(FMPI2C_HandleTypeDef { if (((HAL_GetTick() - Tickstart) > Timeout) || (Timeout == 0U)) { - hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_TIMEOUT; - hfmpi2c->State = HAL_FMPI2C_STATE_READY; - hfmpi2c->Mode = HAL_FMPI2C_MODE_NONE; + if ((__HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_TXIS) == RESET)) + { + hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_TIMEOUT; + hfmpi2c->State = HAL_FMPI2C_STATE_READY; + hfmpi2c->Mode = HAL_FMPI2C_MODE_NONE; - /* Process Unlocked */ - __HAL_UNLOCK(hfmpi2c); + /* Process Unlocked */ + __HAL_UNLOCK(hfmpi2c); - return HAL_ERROR; + return HAL_ERROR; + } } } } @@ -6446,7 +7083,7 @@ static HAL_StatusTypeDef FMPI2C_WaitOnTXISFlagUntilTimeout(FMPI2C_HandleTypeDef * @retval HAL status */ static HAL_StatusTypeDef FMPI2C_WaitOnSTOPFlagUntilTimeout(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t Timeout, - uint32_t Tickstart) + uint32_t Tickstart) { while (__HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_STOPF) == RESET) { @@ -6459,14 +7096,17 @@ static HAL_StatusTypeDef FMPI2C_WaitOnSTOPFlagUntilTimeout(FMPI2C_HandleTypeDef /* Check for the Timeout */ if (((HAL_GetTick() - Tickstart) > Timeout) || (Timeout == 0U)) { - hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_TIMEOUT; - hfmpi2c->State = HAL_FMPI2C_STATE_READY; - hfmpi2c->Mode = HAL_FMPI2C_MODE_NONE; + if ((__HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_STOPF) == RESET)) + { + hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_TIMEOUT; + hfmpi2c->State = HAL_FMPI2C_STATE_READY; + hfmpi2c->Mode = HAL_FMPI2C_MODE_NONE; - /* Process Unlocked */ - __HAL_UNLOCK(hfmpi2c); + /* Process Unlocked */ + __HAL_UNLOCK(hfmpi2c); - return HAL_ERROR; + return HAL_ERROR; + } } } return HAL_OK; @@ -6481,18 +7121,20 @@ static HAL_StatusTypeDef FMPI2C_WaitOnSTOPFlagUntilTimeout(FMPI2C_HandleTypeDef * @retval HAL status */ static HAL_StatusTypeDef FMPI2C_WaitOnRXNEFlagUntilTimeout(FMPI2C_HandleTypeDef *hfmpi2c, uint32_t Timeout, - uint32_t Tickstart) + uint32_t Tickstart) { - while (__HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_RXNE) == RESET) + HAL_StatusTypeDef status = HAL_OK; + + while ((__HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_RXNE) == RESET) && (status == HAL_OK)) { /* Check if an error is detected */ if (FMPI2C_IsErrorOccurred(hfmpi2c, Timeout, Tickstart) != HAL_OK) { - return HAL_ERROR; + status = HAL_ERROR; } /* Check if a STOPF is detected */ - if (__HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_STOPF) == SET) + if ((__HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_STOPF) == SET) && (status == HAL_OK)) { /* Check if an RXNE is pending */ /* Store Last receive data if any */ @@ -6500,40 +7142,51 @@ static HAL_StatusTypeDef FMPI2C_WaitOnRXNEFlagUntilTimeout(FMPI2C_HandleTypeDef { /* Return HAL_OK */ /* The Reading of data from RXDR will be done in caller function */ - return HAL_OK; + status = HAL_OK; } - else + + /* Check a no-acknowledge have been detected */ + if (__HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_AF) == SET) { + __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_AF); + hfmpi2c->ErrorCode = HAL_FMPI2C_ERROR_AF; + /* Clear STOP Flag */ __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_STOPF); /* Clear Configuration Register 2 */ FMPI2C_RESET_CR2(hfmpi2c); - hfmpi2c->ErrorCode = HAL_FMPI2C_ERROR_NONE; hfmpi2c->State = HAL_FMPI2C_STATE_READY; hfmpi2c->Mode = HAL_FMPI2C_MODE_NONE; /* Process Unlocked */ __HAL_UNLOCK(hfmpi2c); - return HAL_ERROR; + status = HAL_ERROR; + } + else + { + hfmpi2c->ErrorCode = HAL_FMPI2C_ERROR_NONE; } } /* Check for the Timeout */ - if (((HAL_GetTick() - Tickstart) > Timeout) || (Timeout == 0U)) + if ((((HAL_GetTick() - Tickstart) > Timeout) || (Timeout == 0U)) && (status == HAL_OK)) { - hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_TIMEOUT; - hfmpi2c->State = HAL_FMPI2C_STATE_READY; + if ((__HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_RXNE) == RESET)) + { + hfmpi2c->ErrorCode |= HAL_FMPI2C_ERROR_TIMEOUT; + hfmpi2c->State = HAL_FMPI2C_STATE_READY; - /* Process Unlocked */ - __HAL_UNLOCK(hfmpi2c); + /* Process Unlocked */ + __HAL_UNLOCK(hfmpi2c); - return HAL_ERROR; + status = HAL_ERROR; + } } } - return HAL_OK; + return status; } /** @@ -6549,15 +7202,14 @@ static HAL_StatusTypeDef FMPI2C_IsErrorOccurred(FMPI2C_HandleTypeDef *hfmpi2c, u HAL_StatusTypeDef status = HAL_OK; uint32_t itflag = hfmpi2c->Instance->ISR; uint32_t error_code = 0; + uint32_t tickstart = Tickstart; + uint32_t tmp1; + HAL_FMPI2C_ModeTypeDef tmp2; if (HAL_IS_BIT_SET(itflag, FMPI2C_FLAG_AF)) { - /* In case of Soft End condition, generate the STOP condition */ - if (FMPI2C_GET_STOP_MODE(hfmpi2c) != FMPI2C_AUTOEND_MODE) - { - /* Generate Stop */ - hfmpi2c->Instance->CR2 |= FMPI2C_CR2_STOP; - } + /* Clear NACKF Flag */ + __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_AF); /* Wait until STOP Flag is set or timeout occurred */ /* AutoEnd should be initiate after AF */ @@ -6566,11 +7218,35 @@ static HAL_StatusTypeDef FMPI2C_IsErrorOccurred(FMPI2C_HandleTypeDef *hfmpi2c, u /* Check for the Timeout */ if (Timeout != HAL_MAX_DELAY) { - if (((HAL_GetTick() - Tickstart) > Timeout) || (Timeout == 0U)) + if (((HAL_GetTick() - tickstart) > Timeout) || (Timeout == 0U)) { - error_code |= HAL_FMPI2C_ERROR_TIMEOUT; + tmp1 = (uint32_t)(hfmpi2c->Instance->CR2 & FMPI2C_CR2_STOP); + tmp2 = hfmpi2c->Mode; - status = HAL_ERROR; + /* In case of FMPI2C still busy, try to regenerate a STOP manually */ + if ((__HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_BUSY) != RESET) && \ + (tmp1 != FMPI2C_CR2_STOP) && \ + (tmp2 != HAL_FMPI2C_MODE_SLAVE)) + { + /* Generate Stop */ + hfmpi2c->Instance->CR2 |= FMPI2C_CR2_STOP; + + /* Update Tick with new reference */ + tickstart = HAL_GetTick(); + } + + while (__HAL_FMPI2C_GET_FLAG(hfmpi2c, FMPI2C_FLAG_STOPF) == RESET) + { + /* Check for the Timeout */ + if ((HAL_GetTick() - tickstart) > FMPI2C_TIMEOUT_STOPF) + { + error_code |= HAL_FMPI2C_ERROR_TIMEOUT; + + status = HAL_ERROR; + + break; + } + } } } } @@ -6582,9 +7258,6 @@ static HAL_StatusTypeDef FMPI2C_IsErrorOccurred(FMPI2C_HandleTypeDef *hfmpi2c, u __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_STOPF); } - /* Clear NACKF Flag */ - __HAL_FMPI2C_CLEAR_FLAG(hfmpi2c, FMPI2C_FLAG_AF); - error_code |= HAL_FMPI2C_ERROR_AF; status = HAL_ERROR; @@ -6666,7 +7339,7 @@ static HAL_StatusTypeDef FMPI2C_IsErrorOccurred(FMPI2C_HandleTypeDef *hfmpi2c, u * @retval None */ static void FMPI2C_TransferConfig(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAddress, uint8_t Size, uint32_t Mode, - uint32_t Request) + uint32_t Request) { /* Check the parameters */ assert_param(IS_FMPI2C_ALL_INSTANCE(hfmpi2c->Instance)); @@ -6675,14 +7348,14 @@ static void FMPI2C_TransferConfig(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t DevAdd /* Declaration of tmp to prevent undefined behavior of volatile usage */ uint32_t tmp = ((uint32_t)(((uint32_t)DevAddress & FMPI2C_CR2_SADD) | \ - (((uint32_t)Size << FMPI2C_CR2_NBYTES_Pos) & FMPI2C_CR2_NBYTES) | \ - (uint32_t)Mode | (uint32_t)Request) & (~0x80000000U)); + (((uint32_t)Size << FMPI2C_CR2_NBYTES_Pos) & FMPI2C_CR2_NBYTES) | \ + (uint32_t)Mode | (uint32_t)Request) & (~0x80000000U)); /* update CR2 register */ MODIFY_REG(hfmpi2c->Instance->CR2, \ ((FMPI2C_CR2_SADD | FMPI2C_CR2_NBYTES | FMPI2C_CR2_RELOAD | FMPI2C_CR2_AUTOEND | \ (FMPI2C_CR2_RD_WRN & (uint32_t)(Request >> (31U - FMPI2C_CR2_RD_WRN_Pos))) | \ - FMPI2C_CR2_START | FMPI2C_CR2_STOP)), tmp); + FMPI2C_CR2_START | FMPI2C_CR2_STOP)), tmp); } /** @@ -6696,8 +7369,9 @@ static void FMPI2C_Enable_IRQ(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t InterruptR { uint32_t tmpisr = 0U; - if ((hfmpi2c->XferISR == FMPI2C_Master_ISR_DMA) || \ - (hfmpi2c->XferISR == FMPI2C_Slave_ISR_DMA)) + if ((hfmpi2c->XferISR != FMPI2C_Master_ISR_DMA) && \ + (hfmpi2c->XferISR != FMPI2C_Slave_ISR_DMA) && \ + (hfmpi2c->XferISR != FMPI2C_Mem_ISR_DMA)) { if ((InterruptRequest & FMPI2C_XFER_LISTEN_IT) == FMPI2C_XFER_LISTEN_IT) { @@ -6705,6 +7379,18 @@ static void FMPI2C_Enable_IRQ(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t InterruptR tmpisr |= FMPI2C_IT_ADDRI | FMPI2C_IT_STOPI | FMPI2C_IT_NACKI | FMPI2C_IT_ERRI; } + if ((InterruptRequest & FMPI2C_XFER_TX_IT) == FMPI2C_XFER_TX_IT) + { + /* Enable ERR, TC, STOP, NACK and TXI interrupts */ + tmpisr |= FMPI2C_IT_ERRI | FMPI2C_IT_TCI | FMPI2C_IT_STOPI | FMPI2C_IT_NACKI | FMPI2C_IT_TXI; + } + + if ((InterruptRequest & FMPI2C_XFER_RX_IT) == FMPI2C_XFER_RX_IT) + { + /* Enable ERR, TC, STOP, NACK and RXI interrupts */ + tmpisr |= FMPI2C_IT_ERRI | FMPI2C_IT_TCI | FMPI2C_IT_STOPI | FMPI2C_IT_NACKI | FMPI2C_IT_RXI; + } + if (InterruptRequest == FMPI2C_XFER_ERROR_IT) { /* Enable ERR and NACK interrupts */ @@ -6714,39 +7400,46 @@ static void FMPI2C_Enable_IRQ(FMPI2C_HandleTypeDef *hfmpi2c, uint16_t InterruptR if (InterruptRequest == FMPI2C_XFER_CPLT_IT) { /* Enable STOP interrupts */ - tmpisr |= (FMPI2C_IT_STOPI | FMPI2C_IT_TCI); - } - - if (InterruptRequest == FMPI2C_XFER_RELOAD_IT) - { - /* Enable TC interrupts */ - tmpisr |= FMPI2C_IT_TCI; + tmpisr |= FMPI2C_IT_STOPI; } } + else { if ((InterruptRequest & FMPI2C_XFER_LISTEN_IT) == FMPI2C_XFER_LISTEN_IT) { - /* Enable ERR, STOP, NACK, and ADDR interrupts */ + /* Enable ERR, STOP, NACK and ADDR interrupts */ tmpisr |= FMPI2C_IT_ADDRI | FMPI2C_IT_STOPI | FMPI2C_IT_NACKI | FMPI2C_IT_ERRI; } if ((InterruptRequest & FMPI2C_XFER_TX_IT) == FMPI2C_XFER_TX_IT) { - /* Enable ERR, TC, STOP, NACK and RXI interrupts */ + /* Enable ERR, TC, STOP, NACK and TXI interrupts */ tmpisr |= FMPI2C_IT_ERRI | FMPI2C_IT_TCI | FMPI2C_IT_STOPI | FMPI2C_IT_NACKI | FMPI2C_IT_TXI; } if ((InterruptRequest & FMPI2C_XFER_RX_IT) == FMPI2C_XFER_RX_IT) { - /* Enable ERR, TC, STOP, NACK and TXI interrupts */ + /* Enable ERR, TC, STOP, NACK and RXI interrupts */ tmpisr |= FMPI2C_IT_ERRI | FMPI2C_IT_TCI | FMPI2C_IT_STOPI | FMPI2C_IT_NACKI | FMPI2C_IT_RXI; } + if (InterruptRequest == FMPI2C_XFER_ERROR_IT) + { + /* Enable ERR and NACK interrupts */ + tmpisr |= FMPI2C_IT_ERRI | FMPI2C_IT_NACKI; + } + if (InterruptRequest == FMPI2C_XFER_CPLT_IT) { /* Enable STOP interrupts */ - tmpisr |= FMPI2C_IT_STOPI; + tmpisr |= (FMPI2C_IT_STOPI | FMPI2C_IT_TCI); + } + + if (InterruptRequest == FMPI2C_XFER_RELOAD_IT) + { + /* Enable TC interrupts */ + tmpisr |= FMPI2C_IT_TCI; } } diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_fmpsmbus.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_fmpsmbus.c index 700f25af..4c913c62 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_fmpsmbus.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_fmpsmbus.c @@ -209,20 +209,28 @@ /** @addtogroup FMPSMBUS_Private_Functions FMPSMBUS Private Functions * @{ */ +/* Private functions to handle flags during polling transfer */ static HAL_StatusTypeDef FMPSMBUS_WaitOnFlagUntilTimeout(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint32_t Flag, - FlagStatus Status, uint32_t Timeout); + FlagStatus Status, uint32_t Timeout); -static void FMPSMBUS_Enable_IRQ(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint32_t InterruptRequest); -static void FMPSMBUS_Disable_IRQ(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint32_t InterruptRequest); +/* Private functions for FMPSMBUS transfer IRQ handler */ static HAL_StatusTypeDef FMPSMBUS_Master_ISR(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint32_t StatusFlags); static HAL_StatusTypeDef FMPSMBUS_Slave_ISR(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint32_t StatusFlags); +static void FMPSMBUS_ITErrorHandler(FMPSMBUS_HandleTypeDef *hfmpsmbus); -static void FMPSMBUS_ConvertOtherXferOptions(FMPSMBUS_HandleTypeDef *hfmpsmbus); +/* Private functions to centralize the enable/disable of Interrupts */ +static void FMPSMBUS_Enable_IRQ(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint32_t InterruptRequest); +static void FMPSMBUS_Disable_IRQ(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint32_t InterruptRequest); -static void FMPSMBUS_ITErrorHandler(FMPSMBUS_HandleTypeDef *hfmpsmbus); +/* Private function to flush TXDR register */ +static void FMPSMBUS_Flush_TXDR(FMPSMBUS_HandleTypeDef *hfmpsmbus); +/* Private function to handle start, restart or stop a transfer */ static void FMPSMBUS_TransferConfig(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint16_t DevAddress, uint8_t Size, - uint32_t Mode, uint32_t Request); + uint32_t Mode, uint32_t Request); + +/* Private function to Convert Specific options */ +static void FMPSMBUS_ConvertOtherXferOptions(FMPSMBUS_HandleTypeDef *hfmpsmbus); /** * @} */ @@ -371,13 +379,13 @@ HAL_StatusTypeDef HAL_FMPSMBUS_Init(FMPSMBUS_HandleTypeDef *hfmpsmbus) /*---------------------------- FMPSMBUSx OAR2 Configuration -----------------------*/ /* Configure FMPSMBUSx: Dual mode and Own Address2 */ hfmpsmbus->Instance->OAR2 = (hfmpsmbus->Init.DualAddressMode | hfmpsmbus->Init.OwnAddress2 | \ - (hfmpsmbus->Init.OwnAddress2Masks << 8U)); + (hfmpsmbus->Init.OwnAddress2Masks << 8U)); /*---------------------------- FMPSMBUSx CR1 Configuration ------------------------*/ /* Configure FMPSMBUSx: Generalcall and NoStretch mode */ hfmpsmbus->Instance->CR1 = (hfmpsmbus->Init.GeneralCallMode | hfmpsmbus->Init.NoStretchMode | \ - hfmpsmbus->Init.PacketErrorCheckMode | hfmpsmbus->Init.PeripheralMode | \ - hfmpsmbus->Init.AnalogFilter); + hfmpsmbus->Init.PacketErrorCheckMode | hfmpsmbus->Init.PeripheralMode | \ + hfmpsmbus->Init.AnalogFilter); /* Enable Slave Byte Control only in case of Packet Error Check is enabled and FMPSMBUS Peripheral is set in Slave mode */ @@ -577,6 +585,9 @@ HAL_StatusTypeDef HAL_FMPSMBUS_ConfigDigitalFilter(FMPSMBUS_HandleTypeDef *hfmps /** * @brief Register a User FMPSMBUS Callback * To be used instead of the weak predefined callback + * @note The HAL_FMPSMBUS_RegisterCallback() may be called before HAL_FMPSMBUS_Init() in + * HAL_FMPSMBUS_STATE_RESET to register callbacks for HAL_FMPSMBUS_MSPINIT_CB_ID and + * HAL_FMPSMBUS_MSPDEINIT_CB_ID. * @param hfmpsmbus Pointer to a FMPSMBUS_HandleTypeDef structure that contains * the configuration information for the specified FMPSMBUS. * @param CallbackID ID of the callback to be registered @@ -593,8 +604,8 @@ HAL_StatusTypeDef HAL_FMPSMBUS_ConfigDigitalFilter(FMPSMBUS_HandleTypeDef *hfmps * @retval HAL status */ HAL_StatusTypeDef HAL_FMPSMBUS_RegisterCallback(FMPSMBUS_HandleTypeDef *hfmpsmbus, - HAL_FMPSMBUS_CallbackIDTypeDef CallbackID, - pFMPSMBUS_CallbackTypeDef pCallback) + HAL_FMPSMBUS_CallbackIDTypeDef CallbackID, + pFMPSMBUS_CallbackTypeDef pCallback) { HAL_StatusTypeDef status = HAL_OK; @@ -606,9 +617,6 @@ HAL_StatusTypeDef HAL_FMPSMBUS_RegisterCallback(FMPSMBUS_HandleTypeDef *hfmpsmbu return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(hfmpsmbus); - if (HAL_FMPSMBUS_STATE_READY == hfmpsmbus->State) { switch (CallbackID) @@ -684,14 +692,15 @@ HAL_StatusTypeDef HAL_FMPSMBUS_RegisterCallback(FMPSMBUS_HandleTypeDef *hfmpsmbu status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hfmpsmbus); return status; } /** * @brief Unregister an FMPSMBUS Callback * FMPSMBUS callback is redirected to the weak predefined callback + * @note The HAL_FMPSMBUS_UnRegisterCallback() may be called before HAL_FMPSMBUS_Init() in + * HAL_FMPSMBUS_STATE_RESET to un-register callbacks for HAL_FMPSMBUS_MSPINIT_CB_ID and + * HAL_FMPSMBUS_MSPDEINIT_CB_ID * @param hfmpsmbus Pointer to a FMPSMBUS_HandleTypeDef structure that contains * the configuration information for the specified FMPSMBUS. * @param CallbackID ID of the callback to be unregistered @@ -708,13 +717,10 @@ HAL_StatusTypeDef HAL_FMPSMBUS_RegisterCallback(FMPSMBUS_HandleTypeDef *hfmpsmbu * @retval HAL status */ HAL_StatusTypeDef HAL_FMPSMBUS_UnRegisterCallback(FMPSMBUS_HandleTypeDef *hfmpsmbus, - HAL_FMPSMBUS_CallbackIDTypeDef CallbackID) + HAL_FMPSMBUS_CallbackIDTypeDef CallbackID) { HAL_StatusTypeDef status = HAL_OK; - /* Process locked */ - __HAL_LOCK(hfmpsmbus); - if (HAL_FMPSMBUS_STATE_READY == hfmpsmbus->State) { switch (CallbackID) @@ -790,8 +796,6 @@ HAL_StatusTypeDef HAL_FMPSMBUS_UnRegisterCallback(FMPSMBUS_HandleTypeDef *hfmpsm status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hfmpsmbus); return status; } @@ -804,7 +808,7 @@ HAL_StatusTypeDef HAL_FMPSMBUS_UnRegisterCallback(FMPSMBUS_HandleTypeDef *hfmpsm * @retval HAL status */ HAL_StatusTypeDef HAL_FMPSMBUS_RegisterAddrCallback(FMPSMBUS_HandleTypeDef *hfmpsmbus, - pFMPSMBUS_AddrCallbackTypeDef pCallback) + pFMPSMBUS_AddrCallbackTypeDef pCallback) { HAL_StatusTypeDef status = HAL_OK; @@ -815,8 +819,6 @@ HAL_StatusTypeDef HAL_FMPSMBUS_RegisterAddrCallback(FMPSMBUS_HandleTypeDef *hfmp return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(hfmpsmbus); if (HAL_FMPSMBUS_STATE_READY == hfmpsmbus->State) { @@ -831,8 +833,6 @@ HAL_StatusTypeDef HAL_FMPSMBUS_RegisterAddrCallback(FMPSMBUS_HandleTypeDef *hfmp status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hfmpsmbus); return status; } @@ -847,9 +847,6 @@ HAL_StatusTypeDef HAL_FMPSMBUS_UnRegisterAddrCallback(FMPSMBUS_HandleTypeDef *hf { HAL_StatusTypeDef status = HAL_OK; - /* Process locked */ - __HAL_LOCK(hfmpsmbus); - if (HAL_FMPSMBUS_STATE_READY == hfmpsmbus->State) { hfmpsmbus->AddrCallback = HAL_FMPSMBUS_AddrCallback; /* Legacy weak AddrCallback */ @@ -863,8 +860,6 @@ HAL_StatusTypeDef HAL_FMPSMBUS_UnRegisterAddrCallback(FMPSMBUS_HandleTypeDef *hf status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hfmpsmbus); return status; } @@ -929,9 +924,10 @@ HAL_StatusTypeDef HAL_FMPSMBUS_UnRegisterAddrCallback(FMPSMBUS_HandleTypeDef *hf * @retval HAL status */ HAL_StatusTypeDef HAL_FMPSMBUS_Master_Transmit_IT(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint16_t DevAddress, - uint8_t *pData, uint16_t Size, uint32_t XferOptions) + uint8_t *pData, uint16_t Size, uint32_t XferOptions) { uint32_t tmp; + uint32_t sizetoxfer; /* Check the parameters */ assert_param(IS_FMPSMBUS_TRANSFER_OPTIONS_REQUEST(XferOptions)); @@ -964,13 +960,37 @@ HAL_StatusTypeDef HAL_FMPSMBUS_Master_Transmit_IT(FMPSMBUS_HandleTypeDef *hfmpsm hfmpsmbus->XferSize = Size; } + sizetoxfer = hfmpsmbus->XferSize; + if ((sizetoxfer > 0U) && ((XferOptions == FMPSMBUS_FIRST_FRAME) || + (XferOptions == FMPSMBUS_FIRST_AND_LAST_FRAME_NO_PEC) || + (XferOptions == FMPSMBUS_FIRST_FRAME_WITH_PEC) || + (XferOptions == FMPSMBUS_FIRST_AND_LAST_FRAME_WITH_PEC))) + { + if (hfmpsmbus->pBuffPtr != NULL) + { + /* Preload TX register */ + /* Write data to TXDR */ + hfmpsmbus->Instance->TXDR = *hfmpsmbus->pBuffPtr; + + /* Increment Buffer pointer */ + hfmpsmbus->pBuffPtr++; + + hfmpsmbus->XferCount--; + hfmpsmbus->XferSize--; + } + else + { + return HAL_ERROR; + } + } + /* Send Slave Address */ /* Set NBYTES to write and reload if size > MAX_NBYTE_SIZE and generate RESTART */ - if ((hfmpsmbus->XferSize < hfmpsmbus->XferCount) && (hfmpsmbus->XferSize == MAX_NBYTE_SIZE)) + if ((sizetoxfer < hfmpsmbus->XferCount) && (sizetoxfer == MAX_NBYTE_SIZE)) { - FMPSMBUS_TransferConfig(hfmpsmbus, DevAddress, (uint8_t)hfmpsmbus->XferSize, - FMPSMBUS_RELOAD_MODE | (hfmpsmbus->XferOptions & FMPSMBUS_SENDPEC_MODE), - FMPSMBUS_GENERATE_START_WRITE); + FMPSMBUS_TransferConfig(hfmpsmbus, DevAddress, (uint8_t)sizetoxfer, + FMPSMBUS_RELOAD_MODE | (hfmpsmbus->XferOptions & FMPSMBUS_SENDPEC_MODE), + FMPSMBUS_GENERATE_START_WRITE); } else { @@ -983,8 +1003,8 @@ HAL_StatusTypeDef HAL_FMPSMBUS_Master_Transmit_IT(FMPSMBUS_HandleTypeDef *hfmpsm if ((hfmpsmbus->PreviousState == HAL_FMPSMBUS_STATE_MASTER_BUSY_TX) && \ (IS_FMPSMBUS_TRANSFER_OTHER_OPTIONS_REQUEST(tmp) == 0)) { - FMPSMBUS_TransferConfig(hfmpsmbus, DevAddress, (uint8_t)hfmpsmbus->XferSize, hfmpsmbus->XferOptions, - FMPSMBUS_NO_STARTSTOP); + FMPSMBUS_TransferConfig(hfmpsmbus, DevAddress, (uint8_t)sizetoxfer, hfmpsmbus->XferOptions, + FMPSMBUS_NO_STARTSTOP); } /* Else transfer direction change, so generate Restart with new transfer direction */ else @@ -993,17 +1013,24 @@ HAL_StatusTypeDef HAL_FMPSMBUS_Master_Transmit_IT(FMPSMBUS_HandleTypeDef *hfmpsm FMPSMBUS_ConvertOtherXferOptions(hfmpsmbus); /* Handle Transfer */ - FMPSMBUS_TransferConfig(hfmpsmbus, DevAddress, (uint8_t)hfmpsmbus->XferSize, - hfmpsmbus->XferOptions, - FMPSMBUS_GENERATE_START_WRITE); + FMPSMBUS_TransferConfig(hfmpsmbus, DevAddress, (uint8_t)sizetoxfer, + hfmpsmbus->XferOptions, + FMPSMBUS_GENERATE_START_WRITE); } /* If PEC mode is enable, size to transmit manage by SW part should be Size-1 byte, corresponding to PEC byte */ /* PEC byte is automatically sent by HW block, no need to manage it in Transmit process */ if (FMPSMBUS_GET_PEC_MODE(hfmpsmbus) != 0UL) { - hfmpsmbus->XferSize--; - hfmpsmbus->XferCount--; + if (hfmpsmbus->XferSize > 0U) + { + hfmpsmbus->XferSize--; + hfmpsmbus->XferCount--; + } + else + { + return HAL_ERROR; + } } } @@ -1035,7 +1062,7 @@ HAL_StatusTypeDef HAL_FMPSMBUS_Master_Transmit_IT(FMPSMBUS_HandleTypeDef *hfmpsm * @retval HAL status */ HAL_StatusTypeDef HAL_FMPSMBUS_Master_Receive_IT(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint16_t DevAddress, uint8_t *pData, - uint16_t Size, uint32_t XferOptions) + uint16_t Size, uint32_t XferOptions) { uint32_t tmp; @@ -1076,8 +1103,8 @@ HAL_StatusTypeDef HAL_FMPSMBUS_Master_Receive_IT(FMPSMBUS_HandleTypeDef *hfmpsmb if ((hfmpsmbus->XferSize < hfmpsmbus->XferCount) && (hfmpsmbus->XferSize == MAX_NBYTE_SIZE)) { FMPSMBUS_TransferConfig(hfmpsmbus, DevAddress, (uint8_t)hfmpsmbus->XferSize, - FMPSMBUS_RELOAD_MODE | (hfmpsmbus->XferOptions & FMPSMBUS_SENDPEC_MODE), - FMPSMBUS_GENERATE_START_READ); + FMPSMBUS_RELOAD_MODE | (hfmpsmbus->XferOptions & FMPSMBUS_SENDPEC_MODE), + FMPSMBUS_GENERATE_START_READ); } else { @@ -1091,7 +1118,7 @@ HAL_StatusTypeDef HAL_FMPSMBUS_Master_Receive_IT(FMPSMBUS_HandleTypeDef *hfmpsmb (IS_FMPSMBUS_TRANSFER_OTHER_OPTIONS_REQUEST(tmp) == 0)) { FMPSMBUS_TransferConfig(hfmpsmbus, DevAddress, (uint8_t)hfmpsmbus->XferSize, hfmpsmbus->XferOptions, - FMPSMBUS_NO_STARTSTOP); + FMPSMBUS_NO_STARTSTOP); } /* Else transfer direction change, so generate Restart with new transfer direction */ else @@ -1101,8 +1128,8 @@ HAL_StatusTypeDef HAL_FMPSMBUS_Master_Receive_IT(FMPSMBUS_HandleTypeDef *hfmpsmb /* Handle Transfer */ FMPSMBUS_TransferConfig(hfmpsmbus, DevAddress, (uint8_t)hfmpsmbus->XferSize, - hfmpsmbus->XferOptions, - FMPSMBUS_GENERATE_START_READ); + hfmpsmbus->XferOptions, + FMPSMBUS_GENERATE_START_READ); } } @@ -1197,7 +1224,7 @@ HAL_StatusTypeDef HAL_FMPSMBUS_Master_Abort_IT(FMPSMBUS_HandleTypeDef *hfmpsmbus * @retval HAL status */ HAL_StatusTypeDef HAL_FMPSMBUS_Slave_Transmit_IT(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint8_t *pData, uint16_t Size, - uint32_t XferOptions) + uint32_t XferOptions) { /* Check the parameters */ assert_param(IS_FMPSMBUS_TRANSFER_OPTIONS_REQUEST(XferOptions)); @@ -1246,14 +1273,14 @@ HAL_StatusTypeDef HAL_FMPSMBUS_Slave_Transmit_IT(FMPSMBUS_HandleTypeDef *hfmpsmb if ((hfmpsmbus->XferSize < hfmpsmbus->XferCount) && (hfmpsmbus->XferSize == MAX_NBYTE_SIZE)) { FMPSMBUS_TransferConfig(hfmpsmbus, 0, (uint8_t)hfmpsmbus->XferSize, - FMPSMBUS_RELOAD_MODE | (hfmpsmbus->XferOptions & FMPSMBUS_SENDPEC_MODE), - FMPSMBUS_NO_STARTSTOP); + FMPSMBUS_RELOAD_MODE | (hfmpsmbus->XferOptions & FMPSMBUS_SENDPEC_MODE), + FMPSMBUS_NO_STARTSTOP); } else { /* Set NBYTE to transmit */ FMPSMBUS_TransferConfig(hfmpsmbus, 0, (uint8_t)hfmpsmbus->XferSize, hfmpsmbus->XferOptions, - FMPSMBUS_NO_STARTSTOP); + FMPSMBUS_NO_STARTSTOP); /* If PEC mode is enable, size to transmit should be Size-1 byte, corresponding to PEC byte */ /* PEC byte is automatically sent by HW block, no need to manage it in Transmit process */ @@ -1295,7 +1322,7 @@ HAL_StatusTypeDef HAL_FMPSMBUS_Slave_Transmit_IT(FMPSMBUS_HandleTypeDef *hfmpsmb * @retval HAL status */ HAL_StatusTypeDef HAL_FMPSMBUS_Slave_Receive_IT(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint8_t *pData, uint16_t Size, - uint32_t XferOptions) + uint32_t XferOptions) { /* Check the parameters */ assert_param(IS_FMPSMBUS_TRANSFER_OPTIONS_REQUEST(XferOptions)); @@ -1340,7 +1367,7 @@ HAL_StatusTypeDef HAL_FMPSMBUS_Slave_Receive_IT(FMPSMBUS_HandleTypeDef *hfmpsmbu if (((FMPSMBUS_GET_PEC_MODE(hfmpsmbus) != 0UL) && (hfmpsmbus->XferSize == 2U)) || (hfmpsmbus->XferSize == 1U)) { FMPSMBUS_TransferConfig(hfmpsmbus, 0, (uint8_t)hfmpsmbus->XferSize, hfmpsmbus->XferOptions, - FMPSMBUS_NO_STARTSTOP); + FMPSMBUS_NO_STARTSTOP); } else { @@ -1455,7 +1482,7 @@ HAL_StatusTypeDef HAL_FMPSMBUS_DisableAlert_IT(FMPSMBUS_HandleTypeDef *hfmpsmbus * @retval HAL status */ HAL_StatusTypeDef HAL_FMPSMBUS_IsDeviceReady(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint16_t DevAddress, uint32_t Trials, - uint32_t Timeout) + uint32_t Timeout) { uint32_t tickstart; @@ -1604,7 +1631,7 @@ void HAL_FMPSMBUS_EV_IRQHandler(FMPSMBUS_HandleTypeDef *hfmpsmbus) /* FMPSMBUS in mode Transmitter ---------------------------------------------------*/ if ((FMPSMBUS_CHECK_IT_SOURCE(tmpcr1value, (FMPSMBUS_IT_TCI | FMPSMBUS_IT_STOPI | - FMPSMBUS_IT_NACKI | FMPSMBUS_IT_TXI)) != RESET) && + FMPSMBUS_IT_NACKI | FMPSMBUS_IT_TXI)) != RESET) && ((FMPSMBUS_CHECK_FLAG(tmpisrvalue, FMPSMBUS_FLAG_TXIS) != RESET) || (FMPSMBUS_CHECK_FLAG(tmpisrvalue, FMPSMBUS_FLAG_TCR) != RESET) || (FMPSMBUS_CHECK_FLAG(tmpisrvalue, FMPSMBUS_FLAG_TC) != RESET) || @@ -1629,7 +1656,7 @@ void HAL_FMPSMBUS_EV_IRQHandler(FMPSMBUS_HandleTypeDef *hfmpsmbus) /* FMPSMBUS in mode Receiver ----------------------------------------------------*/ if ((FMPSMBUS_CHECK_IT_SOURCE(tmpcr1value, (FMPSMBUS_IT_TCI | FMPSMBUS_IT_STOPI | - FMPSMBUS_IT_NACKI | FMPSMBUS_IT_RXI)) != RESET) && + FMPSMBUS_IT_NACKI | FMPSMBUS_IT_RXI)) != RESET) && ((FMPSMBUS_CHECK_FLAG(tmpisrvalue, FMPSMBUS_FLAG_RXNE) != RESET) || (FMPSMBUS_CHECK_FLAG(tmpisrvalue, FMPSMBUS_FLAG_TCR) != RESET) || (FMPSMBUS_CHECK_FLAG(tmpisrvalue, FMPSMBUS_FLAG_TC) != RESET) || @@ -1750,7 +1777,7 @@ __weak void HAL_FMPSMBUS_SlaveRxCpltCallback(FMPSMBUS_HandleTypeDef *hfmpsmbus) * @retval None */ __weak void HAL_FMPSMBUS_AddrCallback(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint8_t TransferDirection, - uint16_t AddrMatchCode) + uint16_t AddrMatchCode) { /* Prevent unused argument(s) compilation warning */ UNUSED(hfmpsmbus); @@ -1819,7 +1846,7 @@ __weak void HAL_FMPSMBUS_ErrorCallback(FMPSMBUS_HandleTypeDef *hfmpsmbus) * the configuration information for the specified FMPSMBUS. * @retval HAL state */ -uint32_t HAL_FMPSMBUS_GetState(FMPSMBUS_HandleTypeDef *hfmpsmbus) +uint32_t HAL_FMPSMBUS_GetState(const FMPSMBUS_HandleTypeDef *hfmpsmbus) { /* Return FMPSMBUS handle state */ return hfmpsmbus->State; @@ -1831,7 +1858,7 @@ uint32_t HAL_FMPSMBUS_GetState(FMPSMBUS_HandleTypeDef *hfmpsmbus) * the configuration information for the specified FMPSMBUS. * @retval FMPSMBUS Error Code */ -uint32_t HAL_FMPSMBUS_GetError(FMPSMBUS_HandleTypeDef *hfmpsmbus) +uint32_t HAL_FMPSMBUS_GetError(const FMPSMBUS_HandleTypeDef *hfmpsmbus) { return hfmpsmbus->ErrorCode; } @@ -1872,6 +1899,9 @@ static HAL_StatusTypeDef FMPSMBUS_Master_ISR(FMPSMBUS_HandleTypeDef *hfmpsmbus, /* No need to generate STOP, it is automatically done */ hfmpsmbus->ErrorCode |= HAL_FMPSMBUS_ERROR_ACKF; + /* Flush TX register */ + FMPSMBUS_Flush_TXDR(hfmpsmbus); + /* Process Unlocked */ __HAL_UNLOCK(hfmpsmbus); @@ -1997,15 +2027,15 @@ static HAL_StatusTypeDef FMPSMBUS_Master_ISR(FMPSMBUS_HandleTypeDef *hfmpsmbus, if (hfmpsmbus->XferCount > MAX_NBYTE_SIZE) { FMPSMBUS_TransferConfig(hfmpsmbus, DevAddress, MAX_NBYTE_SIZE, - (FMPSMBUS_RELOAD_MODE | (hfmpsmbus->XferOptions & FMPSMBUS_SENDPEC_MODE)), - FMPSMBUS_NO_STARTSTOP); + (FMPSMBUS_RELOAD_MODE | (hfmpsmbus->XferOptions & FMPSMBUS_SENDPEC_MODE)), + FMPSMBUS_NO_STARTSTOP); hfmpsmbus->XferSize = MAX_NBYTE_SIZE; } else { hfmpsmbus->XferSize = hfmpsmbus->XferCount; FMPSMBUS_TransferConfig(hfmpsmbus, DevAddress, (uint8_t)hfmpsmbus->XferSize, hfmpsmbus->XferOptions, - FMPSMBUS_NO_STARTSTOP); + FMPSMBUS_NO_STARTSTOP); /* If PEC mode is enable, size to transmit should be Size-1 byte, corresponding to PEC byte */ /* PEC byte is automatically sent by HW block, no need to manage it in Transmit process */ if (FMPSMBUS_GET_PEC_MODE(hfmpsmbus) != 0UL) @@ -2162,6 +2192,9 @@ static HAL_StatusTypeDef FMPSMBUS_Slave_ISR(FMPSMBUS_HandleTypeDef *hfmpsmbus, u /* Clear NACK Flag */ __HAL_FMPSMBUS_CLEAR_FLAG(hfmpsmbus, FMPSMBUS_FLAG_AF); + /* Flush TX register */ + FMPSMBUS_Flush_TXDR(hfmpsmbus); + /* Process Unlocked */ __HAL_UNLOCK(hfmpsmbus); } @@ -2183,6 +2216,9 @@ static HAL_StatusTypeDef FMPSMBUS_Slave_ISR(FMPSMBUS_HandleTypeDef *hfmpsmbus, u /* Set ErrorCode corresponding to a Non-Acknowledge */ hfmpsmbus->ErrorCode |= HAL_FMPSMBUS_ERROR_ACKF; + /* Flush TX register */ + FMPSMBUS_Flush_TXDR(hfmpsmbus); + /* Process Unlocked */ __HAL_UNLOCK(hfmpsmbus); @@ -2258,8 +2294,8 @@ static HAL_StatusTypeDef FMPSMBUS_Slave_ISR(FMPSMBUS_HandleTypeDef *hfmpsmbus, u { /* Set Reload for next Bytes */ FMPSMBUS_TransferConfig(hfmpsmbus, 0, 1, - FMPSMBUS_RELOAD_MODE | (hfmpsmbus->XferOptions & FMPSMBUS_SENDPEC_MODE), - FMPSMBUS_NO_STARTSTOP); + FMPSMBUS_RELOAD_MODE | (hfmpsmbus->XferOptions & FMPSMBUS_SENDPEC_MODE), + FMPSMBUS_NO_STARTSTOP); /* Ack last Byte Read */ hfmpsmbus->Instance->CR2 &= ~FMPI2C_CR2_NACK; @@ -2272,15 +2308,15 @@ static HAL_StatusTypeDef FMPSMBUS_Slave_ISR(FMPSMBUS_HandleTypeDef *hfmpsmbus, u if (hfmpsmbus->XferCount > MAX_NBYTE_SIZE) { FMPSMBUS_TransferConfig(hfmpsmbus, 0, MAX_NBYTE_SIZE, - (FMPSMBUS_RELOAD_MODE | (hfmpsmbus->XferOptions & FMPSMBUS_SENDPEC_MODE)), - FMPSMBUS_NO_STARTSTOP); + (FMPSMBUS_RELOAD_MODE | (hfmpsmbus->XferOptions & FMPSMBUS_SENDPEC_MODE)), + FMPSMBUS_NO_STARTSTOP); hfmpsmbus->XferSize = MAX_NBYTE_SIZE; } else { hfmpsmbus->XferSize = hfmpsmbus->XferCount; FMPSMBUS_TransferConfig(hfmpsmbus, 0, (uint8_t)hfmpsmbus->XferSize, hfmpsmbus->XferOptions, - FMPSMBUS_NO_STARTSTOP); + FMPSMBUS_NO_STARTSTOP); /* If PEC mode is enable, size to transmit should be Size-1 byte, corresponding to PEC byte */ /* PEC byte is automatically sent by HW block, no need to manage it in Transmit process */ if (FMPSMBUS_GET_PEC_MODE(hfmpsmbus) != 0UL) @@ -2584,7 +2620,13 @@ static void FMPSMBUS_ITErrorHandler(FMPSMBUS_HandleTypeDef *hfmpsmbus) __HAL_FMPSMBUS_CLEAR_FLAG(hfmpsmbus, FMPSMBUS_FLAG_PECERR); } - /* Store current volatile hfmpsmbus->State, misra rule */ + if (hfmpsmbus->ErrorCode != HAL_FMPSMBUS_ERROR_NONE) + { + /* Flush TX register */ + FMPSMBUS_Flush_TXDR(hfmpsmbus); + } + + /* Store current volatile hfmpsmbus->ErrorCode, misra rule */ tmperror = hfmpsmbus->ErrorCode; /* Call the Error Callback in case of Error detected */ @@ -2625,7 +2667,7 @@ static void FMPSMBUS_ITErrorHandler(FMPSMBUS_HandleTypeDef *hfmpsmbus) * @retval HAL status */ static HAL_StatusTypeDef FMPSMBUS_WaitOnFlagUntilTimeout(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint32_t Flag, - FlagStatus Status, uint32_t Timeout) + FlagStatus Status, uint32_t Timeout) { uint32_t tickstart = HAL_GetTick(); @@ -2654,6 +2696,27 @@ static HAL_StatusTypeDef FMPSMBUS_WaitOnFlagUntilTimeout(FMPSMBUS_HandleTypeDef return HAL_OK; } +/** + * @brief FMPSMBUS Tx data register flush process. + * @param hfmpsmbus FMPSMBUS handle. + * @retval None + */ +static void FMPSMBUS_Flush_TXDR(FMPSMBUS_HandleTypeDef *hfmpsmbus) +{ + /* If a pending TXIS flag is set */ + /* Write a dummy data in TXDR to clear it */ + if (__HAL_FMPSMBUS_GET_FLAG(hfmpsmbus, FMPSMBUS_FLAG_TXIS) != RESET) + { + hfmpsmbus->Instance->TXDR = 0x00U; + } + + /* Flush TX register if not empty */ + if (__HAL_FMPSMBUS_GET_FLAG(hfmpsmbus, FMPSMBUS_FLAG_TXE) == RESET) + { + __HAL_FMPSMBUS_CLEAR_FLAG(hfmpsmbus, FMPSMBUS_FLAG_TXE); + } +} + /** * @brief Handle FMPSMBUSx communication when starting transfer or during transfer (TC or TCR flag are set). * @param hfmpsmbus FMPSMBUS handle. @@ -2675,7 +2738,7 @@ static HAL_StatusTypeDef FMPSMBUS_WaitOnFlagUntilTimeout(FMPSMBUS_HandleTypeDef * @retval None */ static void FMPSMBUS_TransferConfig(FMPSMBUS_HandleTypeDef *hfmpsmbus, uint16_t DevAddress, uint8_t Size, - uint32_t Mode, uint32_t Request) + uint32_t Mode, uint32_t Request) { /* Check the parameters */ assert_param(IS_FMPSMBUS_ALL_INSTANCE(hfmpsmbus->Instance)); diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_hash.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_hash.c index f3aafe0a..5fc5cc9d 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_hash.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_hash.c @@ -1657,7 +1657,7 @@ static void HASH_DMAXferCplt(DMA_HandleTypeDef *hdma) HASH_HandleTypeDef *hhash = (HASH_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent; uint32_t inputaddr; uint32_t buffersize; - HAL_StatusTypeDef status = HAL_OK; + HAL_StatusTypeDef status; if (hhash->State != HAL_HASH_STATE_SUSPENDED) { diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_hcd.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_hcd.c index c7c5b70b..7ab12227 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_hcd.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_hcd.c @@ -109,7 +109,9 @@ static void HCD_Port_IRQHandler(HCD_HandleTypeDef *hhcd); */ HAL_StatusTypeDef HAL_HCD_Init(HCD_HandleTypeDef *hhcd) { - USB_OTG_GlobalTypeDef *USBx; +#if defined (USB_OTG_FS) + const USB_OTG_GlobalTypeDef *USBx; +#endif /* defined (USB_OTG_FS) */ /* Check the HCD handle allocation */ if (hhcd == NULL) @@ -120,7 +122,9 @@ HAL_StatusTypeDef HAL_HCD_Init(HCD_HandleTypeDef *hhcd) /* Check the parameters */ assert_param(IS_HCD_ALL_INSTANCE(hhcd->Instance)); +#if defined (USB_OTG_FS) USBx = hhcd->Instance; +#endif /* defined (USB_OTG_FS) */ if (hhcd->State == HAL_HCD_STATE_RESET) { @@ -150,23 +154,37 @@ HAL_StatusTypeDef HAL_HCD_Init(HCD_HandleTypeDef *hhcd) hhcd->State = HAL_HCD_STATE_BUSY; +#if defined (USB_OTG_FS) /* Disable DMA mode for FS instance */ - if ((USBx->CID & (0x1U << 8)) == 0U) + if (USBx == USB_OTG_FS) { hhcd->Init.dma_enable = 0U; } +#endif /* defined (USB_OTG_FS) */ /* Disable the Interrupts */ __HAL_HCD_DISABLE(hhcd); /* Init the Core (common init.) */ - (void)USB_CoreInit(hhcd->Instance, hhcd->Init); + if (USB_CoreInit(hhcd->Instance, hhcd->Init) != HAL_OK) + { + hhcd->State = HAL_HCD_STATE_ERROR; + return HAL_ERROR; + } - /* Force Host Mode*/ - (void)USB_SetCurrentMode(hhcd->Instance, USB_HOST_MODE); + /* Force Host Mode */ + if (USB_SetCurrentMode(hhcd->Instance, USB_HOST_MODE) != HAL_OK) + { + hhcd->State = HAL_HCD_STATE_ERROR; + return HAL_ERROR; + } /* Init Host */ - (void)USB_HostInit(hhcd->Instance, hhcd->Init); + if (USB_HostInit(hhcd->Instance, hhcd->Init) != HAL_OK) + { + hhcd->State = HAL_HCD_STATE_ERROR; + return HAL_ERROR; + } hhcd->State = HAL_HCD_STATE_READY; @@ -197,24 +215,22 @@ HAL_StatusTypeDef HAL_HCD_Init(HCD_HandleTypeDef *hhcd) * This parameter can be a value from 0 to32K * @retval HAL status */ -HAL_StatusTypeDef HAL_HCD_HC_Init(HCD_HandleTypeDef *hhcd, - uint8_t ch_num, - uint8_t epnum, - uint8_t dev_address, - uint8_t speed, - uint8_t ep_type, - uint16_t mps) +HAL_StatusTypeDef HAL_HCD_HC_Init(HCD_HandleTypeDef *hhcd, uint8_t ch_num, uint8_t epnum, + uint8_t dev_address, uint8_t speed, uint8_t ep_type, uint16_t mps) { HAL_StatusTypeDef status; + uint32_t HostCoreSpeed; + uint32_t HCcharMps = mps; __HAL_LOCK(hhcd); hhcd->hc[ch_num].do_ping = 0U; hhcd->hc[ch_num].dev_addr = dev_address; - hhcd->hc[ch_num].max_packet = mps; hhcd->hc[ch_num].ch_num = ch_num; hhcd->hc[ch_num].ep_type = ep_type; hhcd->hc[ch_num].ep_num = epnum & 0x7FU; + (void)HAL_HCD_HC_ClearHubInfo(hhcd, ch_num); + if ((epnum & 0x80U) == 0x80U) { hhcd->hc[ch_num].ep_is_in = 1U; @@ -224,15 +240,27 @@ HAL_StatusTypeDef HAL_HCD_HC_Init(HCD_HandleTypeDef *hhcd, hhcd->hc[ch_num].ep_is_in = 0U; } + HostCoreSpeed = USB_GetHostSpeed(hhcd->Instance); + + if (ep_type == EP_TYPE_ISOC) + { + /* FS device plugged to HS HUB */ + if ((speed == HCD_DEVICE_SPEED_FULL) && (HostCoreSpeed == HPRT0_PRTSPD_HIGH_SPEED)) + { + if (HCcharMps > ISO_SPLT_MPS) + { + /* ISO Max Packet Size for Split mode */ + HCcharMps = ISO_SPLT_MPS; + } + } + } + hhcd->hc[ch_num].speed = speed; + hhcd->hc[ch_num].max_packet = (uint16_t)HCcharMps; + + status = USB_HC_Init(hhcd->Instance, ch_num, epnum, + dev_address, speed, ep_type, (uint16_t)HCcharMps); - status = USB_HC_Init(hhcd->Instance, - ch_num, - epnum, - dev_address, - speed, - ep_type, - mps); __HAL_UNLOCK(hhcd); return status; @@ -250,7 +278,7 @@ HAL_StatusTypeDef HAL_HCD_HC_Halt(HCD_HandleTypeDef *hhcd, uint8_t ch_num) HAL_StatusTypeDef status = HAL_OK; __HAL_LOCK(hhcd); - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); + (void)USB_HC_Halt(hhcd->Instance, ch_num); __HAL_UNLOCK(hhcd); return status; @@ -389,24 +417,41 @@ HAL_StatusTypeDef HAL_HCD_HC_SubmitRequest(HCD_HandleTypeDef *hhcd, switch (ep_type) { case EP_TYPE_CTRL: - if ((token == 1U) && (direction == 0U)) /*send data */ + if (token == 1U) /* send data */ { - if (length == 0U) + if (direction == 0U) { - /* For Status OUT stage, Length==0, Status Out PID = 1 */ - hhcd->hc[ch_num].toggle_out = 1U; - } + if (length == 0U) + { + /* For Status OUT stage, Length == 0U, Status Out PID = 1 */ + hhcd->hc[ch_num].toggle_out = 1U; + } - /* Set the Data Toggle bit as per the Flag */ - if (hhcd->hc[ch_num].toggle_out == 0U) - { - /* Put the PID 0 */ - hhcd->hc[ch_num].data_pid = HC_PID_DATA0; + /* Set the Data Toggle bit as per the Flag */ + if (hhcd->hc[ch_num].toggle_out == 0U) + { + /* Put the PID 0 */ + hhcd->hc[ch_num].data_pid = HC_PID_DATA0; + } + else + { + /* Put the PID 1 */ + hhcd->hc[ch_num].data_pid = HC_PID_DATA1; + } } else { - /* Put the PID 1 */ - hhcd->hc[ch_num].data_pid = HC_PID_DATA1; + if (hhcd->hc[ch_num].do_ssplit == 1U) + { + if (hhcd->hc[ch_num].toggle_in == 0U) + { + hhcd->hc[ch_num].data_pid = HC_PID_DATA0; + } + else + { + hhcd->hc[ch_num].data_pid = HC_PID_DATA1; + } + } } } break; @@ -541,8 +586,11 @@ void HAL_HCD_IRQHandler(HCD_HandleTypeDef *hhcd) (void)USB_FlushTxFifo(USBx, 0x10U); (void)USB_FlushRxFifo(USBx); - /* Restore FS Clock */ - (void)USB_InitFSLSPClkSel(hhcd->Instance, HCFG_48_MHZ); + if (hhcd->Init.phy_itface == USB_OTG_EMBEDDED_PHY) + { + /* Restore FS Clock */ + (void)USB_InitFSLSPClkSel(hhcd->Instance, HCFG_48_MHZ); + } /* Handle Host Port Disconnect Interrupt */ #if (USE_HAL_HCD_REGISTER_CALLBACKS == 1U) @@ -571,16 +619,6 @@ void HAL_HCD_IRQHandler(HCD_HandleTypeDef *hhcd) __HAL_HCD_CLEAR_FLAG(hhcd, USB_OTG_GINTSTS_SOF); } - /* Handle Rx Queue Level Interrupts */ - if ((__HAL_HCD_GET_FLAG(hhcd, USB_OTG_GINTSTS_RXFLVL)) != 0U) - { - USB_MASK_INTERRUPT(hhcd->Instance, USB_OTG_GINTSTS_RXFLVL); - - HCD_RXQLVL_IRQHandler(hhcd); - - USB_UNMASK_INTERRUPT(hhcd->Instance, USB_OTG_GINTSTS_RXFLVL); - } - /* Handle Host channel Interrupt */ if (__HAL_HCD_GET_FLAG(hhcd, USB_OTG_GINTSTS_HCINT)) { @@ -601,6 +639,16 @@ void HAL_HCD_IRQHandler(HCD_HandleTypeDef *hhcd) } __HAL_HCD_CLEAR_FLAG(hhcd, USB_OTG_GINTSTS_HCINT); } + + /* Handle Rx Queue Level Interrupts */ + if ((__HAL_HCD_GET_FLAG(hhcd, USB_OTG_GINTSTS_RXFLVL)) != 0U) + { + USB_MASK_INTERRUPT(hhcd->Instance, USB_OTG_GINTSTS_RXFLVL); + + HCD_RXQLVL_IRQHandler(hhcd); + + USB_UNMASK_INTERRUPT(hhcd->Instance, USB_OTG_GINTSTS_RXFLVL); + } } } @@ -1084,7 +1132,7 @@ HAL_StatusTypeDef HAL_HCD_ResetPort(HCD_HandleTypeDef *hhcd) * @param hhcd HCD handle * @retval HAL state */ -HCD_StateTypeDef HAL_HCD_GetState(HCD_HandleTypeDef *hhcd) +HCD_StateTypeDef HAL_HCD_GetState(HCD_HandleTypeDef const *hhcd) { return hhcd->State; } @@ -1103,7 +1151,7 @@ HCD_StateTypeDef HAL_HCD_GetState(HCD_HandleTypeDef *hhcd) * URB_ERROR/ * URB_STALL */ -HCD_URBStateTypeDef HAL_HCD_HC_GetURBState(HCD_HandleTypeDef *hhcd, uint8_t chnum) +HCD_URBStateTypeDef HAL_HCD_HC_GetURBState(HCD_HandleTypeDef const *hhcd, uint8_t chnum) { return hhcd->hc[chnum].urb_state; } @@ -1116,7 +1164,7 @@ HCD_URBStateTypeDef HAL_HCD_HC_GetURBState(HCD_HandleTypeDef *hhcd, uint8_t chnu * This parameter can be a value from 1 to 15 * @retval last transfer size in byte */ -uint32_t HAL_HCD_HC_GetXferCount(HCD_HandleTypeDef *hhcd, uint8_t chnum) +uint32_t HAL_HCD_HC_GetXferCount(HCD_HandleTypeDef const *hhcd, uint8_t chnum) { return hhcd->hc[chnum].xfer_count; } @@ -1138,7 +1186,7 @@ uint32_t HAL_HCD_HC_GetXferCount(HCD_HandleTypeDef *hhcd, uint8_t chnum) * HC_BBLERR/ * HC_DATATGLERR */ -HCD_HCStateTypeDef HAL_HCD_HC_GetState(HCD_HandleTypeDef *hhcd, uint8_t chnum) +HCD_HCStateTypeDef HAL_HCD_HC_GetState(HCD_HandleTypeDef const *hhcd, uint8_t chnum) { return hhcd->hc[chnum].state; } @@ -1163,6 +1211,54 @@ uint32_t HAL_HCD_GetCurrentSpeed(HCD_HandleTypeDef *hhcd) return (USB_GetHostSpeed(hhcd->Instance)); } +/** + * @brief Set host channel Hub information. + * @param hhcd HCD handle + * @param ch_num Channel number. + * This parameter can be a value from 1 to 15 + * @param addr Hub address + * @param PortNbr Hub port number + * @retval HAL status + */ +HAL_StatusTypeDef HAL_HCD_HC_SetHubInfo(HCD_HandleTypeDef *hhcd, uint8_t ch_num, + uint8_t addr, uint8_t PortNbr) +{ + uint32_t HostCoreSpeed = USB_GetHostSpeed(hhcd->Instance); + + /* LS/FS device plugged to HS HUB */ + if ((hhcd->hc[ch_num].speed != HCD_DEVICE_SPEED_HIGH) && (HostCoreSpeed == HPRT0_PRTSPD_HIGH_SPEED)) + { + hhcd->hc[ch_num].do_ssplit = 1U; + + if ((hhcd->hc[ch_num].ep_type == EP_TYPE_CTRL) && (hhcd->hc[ch_num].ep_is_in != 0U)) + { + hhcd->hc[ch_num].toggle_in = 1U; + } + } + + hhcd->hc[ch_num].hub_addr = addr; + hhcd->hc[ch_num].hub_port_nbr = PortNbr; + + return HAL_OK; +} + + +/** + * @brief Clear host channel hub information. + * @param hhcd HCD handle + * @param ch_num Channel number. + * This parameter can be a value from 1 to 15 + * @retval HAL status + */ +HAL_StatusTypeDef HAL_HCD_HC_ClearHubInfo(HCD_HandleTypeDef *hhcd, uint8_t ch_num) +{ + hhcd->hc[ch_num].do_ssplit = 0U; + hhcd->hc[ch_num].do_csplit = 0U; + hhcd->hc[ch_num].hub_addr = 0U; + hhcd->hc[ch_num].hub_port_nbr = 0U; + + return HAL_OK; +} /** * @} */ @@ -1183,84 +1279,86 @@ uint32_t HAL_HCD_GetCurrentSpeed(HCD_HandleTypeDef *hhcd) */ static void HCD_HC_IN_IRQHandler(HCD_HandleTypeDef *hhcd, uint8_t chnum) { - USB_OTG_GlobalTypeDef *USBx = hhcd->Instance; + const USB_OTG_GlobalTypeDef *USBx = hhcd->Instance; uint32_t USBx_BASE = (uint32_t)USBx; - uint32_t ch_num = (uint32_t)chnum; - uint32_t tmpreg; - if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_AHBERR) == USB_OTG_HCINT_AHBERR) + if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_AHBERR)) { - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_AHBERR); - hhcd->hc[ch_num].state = HC_XACTERR; - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_AHBERR); + hhcd->hc[chnum].state = HC_XACTERR; + (void)USB_HC_Halt(hhcd->Instance, chnum); } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_BBERR) == USB_OTG_HCINT_BBERR) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_BBERR)) { - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_BBERR); - hhcd->hc[ch_num].state = HC_BBLERR; - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_BBERR); + hhcd->hc[chnum].state = HC_BBLERR; + (void)USB_HC_Halt(hhcd->Instance, chnum); } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_ACK) == USB_OTG_HCINT_ACK) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_STALL)) { - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_ACK); + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_STALL); + hhcd->hc[chnum].state = HC_STALL; + (void)USB_HC_Halt(hhcd->Instance, chnum); } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_STALL) == USB_OTG_HCINT_STALL) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_DTERR)) { - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_STALL); - hhcd->hc[ch_num].state = HC_STALL; - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_DTERR); + hhcd->hc[chnum].state = HC_DATATGLERR; + (void)USB_HC_Halt(hhcd->Instance, chnum); } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_DTERR) == USB_OTG_HCINT_DTERR) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_TXERR)) { - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_DTERR); - hhcd->hc[ch_num].state = HC_DATATGLERR; - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); - } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_TXERR) == USB_OTG_HCINT_TXERR) - { - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_TXERR); - hhcd->hc[ch_num].state = HC_XACTERR; - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_TXERR); + hhcd->hc[chnum].state = HC_XACTERR; + (void)USB_HC_Halt(hhcd->Instance, chnum); } else { /* ... */ } - if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_FRMOR) == USB_OTG_HCINT_FRMOR) + if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_FRMOR)) { - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_FRMOR); + (void)USB_HC_Halt(hhcd->Instance, chnum); + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_FRMOR); } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_XFRC) == USB_OTG_HCINT_XFRC) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_XFRC)) { + /* Clear any pending ACK IT */ + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_ACK); + + if (hhcd->hc[chnum].do_csplit == 1U) + { + hhcd->hc[chnum].do_csplit = 0U; + __HAL_HCD_CLEAR_HC_CSPLT(chnum); + } + if (hhcd->Init.dma_enable != 0U) { - hhcd->hc[ch_num].xfer_count = hhcd->hc[ch_num].XferSize - \ - (USBx_HC(ch_num)->HCTSIZ & USB_OTG_HCTSIZ_XFRSIZ); + hhcd->hc[chnum].xfer_count = hhcd->hc[chnum].XferSize - (USBx_HC(chnum)->HCTSIZ & USB_OTG_HCTSIZ_XFRSIZ); } - hhcd->hc[ch_num].state = HC_XFRC; - hhcd->hc[ch_num].ErrCnt = 0U; - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_XFRC); + hhcd->hc[chnum].state = HC_XFRC; + hhcd->hc[chnum].ErrCnt = 0U; + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_XFRC); - if ((hhcd->hc[ch_num].ep_type == EP_TYPE_CTRL) || - (hhcd->hc[ch_num].ep_type == EP_TYPE_BULK)) + if ((hhcd->hc[chnum].ep_type == EP_TYPE_CTRL) || + (hhcd->hc[chnum].ep_type == EP_TYPE_BULK)) { - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_NAK); + (void)USB_HC_Halt(hhcd->Instance, chnum); + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_NAK); } - else if ((hhcd->hc[ch_num].ep_type == EP_TYPE_INTR) || - (hhcd->hc[ch_num].ep_type == EP_TYPE_ISOC)) + else if ((hhcd->hc[chnum].ep_type == EP_TYPE_INTR) || + (hhcd->hc[chnum].ep_type == EP_TYPE_ISOC)) { - USBx_HC(ch_num)->HCCHAR |= USB_OTG_HCCHAR_ODDFRM; - hhcd->hc[ch_num].urb_state = URB_DONE; + USBx_HC(chnum)->HCCHAR |= USB_OTG_HCCHAR_ODDFRM; + hhcd->hc[chnum].urb_state = URB_DONE; #if (USE_HAL_HCD_REGISTER_CALLBACKS == 1U) - hhcd->HC_NotifyURBChangeCallback(hhcd, (uint8_t)ch_num, hhcd->hc[ch_num].urb_state); + hhcd->HC_NotifyURBChangeCallback(hhcd, chnum, hhcd->hc[chnum].urb_state); #else - HAL_HCD_HC_NotifyURBChange_Callback(hhcd, (uint8_t)ch_num, hhcd->hc[ch_num].urb_state); + HAL_HCD_HC_NotifyURBChange_Callback(hhcd, chnum, hhcd->hc[chnum].urb_state); #endif /* USE_HAL_HCD_REGISTER_CALLBACKS */ } else @@ -1270,96 +1368,220 @@ static void HCD_HC_IN_IRQHandler(HCD_HandleTypeDef *hhcd, uint8_t chnum) if (hhcd->Init.dma_enable == 1U) { - if (((hhcd->hc[ch_num].XferSize / hhcd->hc[ch_num].max_packet) & 1U) != 0U) + if ((((hhcd->hc[chnum].xfer_count + hhcd->hc[chnum].max_packet - 1U) / hhcd->hc[chnum].max_packet) & 1U) != 0U) { - hhcd->hc[ch_num].toggle_in ^= 1U; + hhcd->hc[chnum].toggle_in ^= 1U; } } else { - hhcd->hc[ch_num].toggle_in ^= 1U; + hhcd->hc[chnum].toggle_in ^= 1U; + } + } + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_ACK)) + { + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_ACK); + + if (hhcd->hc[chnum].do_ssplit == 1U) + { + hhcd->hc[chnum].do_csplit = 1U; + hhcd->hc[chnum].state = HC_ACK; + + (void)USB_HC_Halt(hhcd->Instance, chnum); } } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_CHH) == USB_OTG_HCINT_CHH) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_CHH)) { - if (hhcd->hc[ch_num].state == HC_XFRC) + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_CHH); + + if (hhcd->hc[chnum].state == HC_XFRC) { - hhcd->hc[ch_num].urb_state = URB_DONE; + hhcd->hc[chnum].state = HC_HALTED; + hhcd->hc[chnum].urb_state = URB_DONE; } - else if (hhcd->hc[ch_num].state == HC_STALL) + else if (hhcd->hc[chnum].state == HC_STALL) { - hhcd->hc[ch_num].urb_state = URB_STALL; + hhcd->hc[chnum].state = HC_HALTED; + hhcd->hc[chnum].urb_state = URB_STALL; } - else if ((hhcd->hc[ch_num].state == HC_XACTERR) || - (hhcd->hc[ch_num].state == HC_DATATGLERR)) + else if ((hhcd->hc[chnum].state == HC_XACTERR) || + (hhcd->hc[chnum].state == HC_DATATGLERR)) { - hhcd->hc[ch_num].ErrCnt++; - if (hhcd->hc[ch_num].ErrCnt > 2U) + hhcd->hc[chnum].state = HC_HALTED; + hhcd->hc[chnum].ErrCnt++; + if (hhcd->hc[chnum].ErrCnt > 2U) { - hhcd->hc[ch_num].ErrCnt = 0U; - hhcd->hc[ch_num].urb_state = URB_ERROR; + hhcd->hc[chnum].ErrCnt = 0U; + + if (hhcd->hc[chnum].do_ssplit == 1U) + { + hhcd->hc[chnum].do_csplit = 0U; + hhcd->hc[chnum].ep_ss_schedule = 0U; + __HAL_HCD_CLEAR_HC_CSPLT(chnum); + } + + hhcd->hc[chnum].urb_state = URB_ERROR; } else { - hhcd->hc[ch_num].urb_state = URB_NOTREADY; + hhcd->hc[chnum].urb_state = URB_NOTREADY; - /* re-activate the channel */ - tmpreg = USBx_HC(ch_num)->HCCHAR; - tmpreg &= ~USB_OTG_HCCHAR_CHDIS; - tmpreg |= USB_OTG_HCCHAR_CHENA; - USBx_HC(ch_num)->HCCHAR = tmpreg; + if ((hhcd->hc[chnum].ep_type == EP_TYPE_CTRL) || + (hhcd->hc[chnum].ep_type == EP_TYPE_BULK)) + { + /* re-activate the channel */ + tmpreg = USBx_HC(chnum)->HCCHAR; + tmpreg &= ~USB_OTG_HCCHAR_CHDIS; + tmpreg |= USB_OTG_HCCHAR_CHENA; + USBx_HC(chnum)->HCCHAR = tmpreg; + } } } - else if (hhcd->hc[ch_num].state == HC_NAK) + else if (hhcd->hc[chnum].state == HC_NYET) { - hhcd->hc[ch_num].urb_state = URB_NOTREADY; + hhcd->hc[chnum].state = HC_HALTED; - /* re-activate the channel */ - tmpreg = USBx_HC(ch_num)->HCCHAR; - tmpreg &= ~USB_OTG_HCCHAR_CHDIS; - tmpreg |= USB_OTG_HCCHAR_CHENA; - USBx_HC(ch_num)->HCCHAR = tmpreg; + if (hhcd->hc[chnum].do_csplit == 1U) + { + if (hhcd->hc[chnum].ep_type == EP_TYPE_INTR) + { + hhcd->hc[chnum].NyetErrCnt++; + if (hhcd->hc[chnum].NyetErrCnt > 2U) + { + hhcd->hc[chnum].NyetErrCnt = 0U; + hhcd->hc[chnum].do_csplit = 0U; + + if (hhcd->hc[chnum].ErrCnt < 3U) + { + hhcd->hc[chnum].ep_ss_schedule = 1U; + } + __HAL_HCD_CLEAR_HC_CSPLT(chnum); + hhcd->hc[chnum].urb_state = URB_ERROR; + } + else + { + hhcd->hc[chnum].urb_state = URB_NOTREADY; + } + } + else + { + hhcd->hc[chnum].urb_state = URB_NOTREADY; + } + + if ((hhcd->hc[chnum].ep_type == EP_TYPE_CTRL) || + (hhcd->hc[chnum].ep_type == EP_TYPE_BULK)) + { + /* re-activate the channel */ + tmpreg = USBx_HC(chnum)->HCCHAR; + tmpreg &= ~USB_OTG_HCCHAR_CHDIS; + tmpreg |= USB_OTG_HCCHAR_CHENA; + USBx_HC(chnum)->HCCHAR = tmpreg; + } + } + } + else if (hhcd->hc[chnum].state == HC_ACK) + { + hhcd->hc[chnum].state = HC_HALTED; + + if (hhcd->hc[chnum].do_csplit == 1U) + { + hhcd->hc[chnum].urb_state = URB_NOTREADY; + + /* Set Complete split and re-activate the channel */ + USBx_HC(chnum)->HCSPLT |= USB_OTG_HCSPLT_COMPLSPLT; + USBx_HC(chnum)->HCINTMSK |= USB_OTG_HCINTMSK_NYET; + USBx_HC(chnum)->HCINTMSK &= ~USB_OTG_HCINT_ACK; + + if ((hhcd->hc[chnum].ep_type == EP_TYPE_CTRL) || + (hhcd->hc[chnum].ep_type == EP_TYPE_BULK)) + { + /* re-activate the channel */ + tmpreg = USBx_HC(chnum)->HCCHAR; + tmpreg &= ~USB_OTG_HCCHAR_CHDIS; + tmpreg |= USB_OTG_HCCHAR_CHENA; + USBx_HC(chnum)->HCCHAR = tmpreg; + } + } + } + else if (hhcd->hc[chnum].state == HC_NAK) + { + hhcd->hc[chnum].state = HC_HALTED; + hhcd->hc[chnum].urb_state = URB_NOTREADY; + + if ((hhcd->hc[chnum].ep_type == EP_TYPE_CTRL) || + (hhcd->hc[chnum].ep_type == EP_TYPE_BULK)) + { + /* re-activate the channel */ + tmpreg = USBx_HC(chnum)->HCCHAR; + tmpreg &= ~USB_OTG_HCCHAR_CHDIS; + tmpreg |= USB_OTG_HCCHAR_CHENA; + USBx_HC(chnum)->HCCHAR = tmpreg; + } } - else if (hhcd->hc[ch_num].state == HC_BBLERR) + else if (hhcd->hc[chnum].state == HC_BBLERR) { - hhcd->hc[ch_num].ErrCnt++; - hhcd->hc[ch_num].urb_state = URB_ERROR; + hhcd->hc[chnum].state = HC_HALTED; + hhcd->hc[chnum].ErrCnt++; + hhcd->hc[chnum].urb_state = URB_ERROR; } else { - /* ... */ + if (hhcd->hc[chnum].state == HC_HALTED) + { + return; + } } - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_CHH); #if (USE_HAL_HCD_REGISTER_CALLBACKS == 1U) - hhcd->HC_NotifyURBChangeCallback(hhcd, (uint8_t)ch_num, hhcd->hc[ch_num].urb_state); + hhcd->HC_NotifyURBChangeCallback(hhcd, chnum, hhcd->hc[chnum].urb_state); #else - HAL_HCD_HC_NotifyURBChange_Callback(hhcd, (uint8_t)ch_num, hhcd->hc[ch_num].urb_state); + HAL_HCD_HC_NotifyURBChange_Callback(hhcd, chnum, hhcd->hc[chnum].urb_state); #endif /* USE_HAL_HCD_REGISTER_CALLBACKS */ } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_NAK) == USB_OTG_HCINT_NAK) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_NYET)) + { + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_NYET); + hhcd->hc[chnum].state = HC_NYET; + + if (hhcd->hc[chnum].do_ssplit == 0U) + { + hhcd->hc[chnum].ErrCnt = 0U; + } + + (void)USB_HC_Halt(hhcd->Instance, chnum); + } + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_NAK)) { - if (hhcd->hc[ch_num].ep_type == EP_TYPE_INTR) + if (hhcd->hc[chnum].ep_type == EP_TYPE_INTR) { - hhcd->hc[ch_num].ErrCnt = 0U; - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); + hhcd->hc[chnum].ErrCnt = 0U; + hhcd->hc[chnum].state = HC_NAK; + (void)USB_HC_Halt(hhcd->Instance, chnum); } - else if ((hhcd->hc[ch_num].ep_type == EP_TYPE_CTRL) || - (hhcd->hc[ch_num].ep_type == EP_TYPE_BULK)) + else if ((hhcd->hc[chnum].ep_type == EP_TYPE_CTRL) || + (hhcd->hc[chnum].ep_type == EP_TYPE_BULK)) { - hhcd->hc[ch_num].ErrCnt = 0U; + hhcd->hc[chnum].ErrCnt = 0U; - if (hhcd->Init.dma_enable == 0U) + if ((hhcd->Init.dma_enable == 0U) || (hhcd->hc[chnum].do_csplit == 1U)) { - hhcd->hc[ch_num].state = HC_NAK; - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); + hhcd->hc[chnum].state = HC_NAK; + (void)USB_HC_Halt(hhcd->Instance, chnum); } } else { /* ... */ } - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_NAK); + + if (hhcd->hc[chnum].do_csplit == 1U) + { + hhcd->hc[chnum].do_csplit = 0U; + __HAL_HCD_CLEAR_HC_CSPLT(chnum); + __HAL_HCD_UNMASK_ACK_HC_INT(chnum); + } + + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_NAK); } else { @@ -1376,184 +1598,231 @@ static void HCD_HC_IN_IRQHandler(HCD_HandleTypeDef *hhcd, uint8_t chnum) */ static void HCD_HC_OUT_IRQHandler(HCD_HandleTypeDef *hhcd, uint8_t chnum) { - USB_OTG_GlobalTypeDef *USBx = hhcd->Instance; + const USB_OTG_GlobalTypeDef *USBx = hhcd->Instance; uint32_t USBx_BASE = (uint32_t)USBx; - uint32_t ch_num = (uint32_t)chnum; uint32_t tmpreg; uint32_t num_packets; - if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_AHBERR) == USB_OTG_HCINT_AHBERR) + if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_AHBERR)) { - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_AHBERR); - hhcd->hc[ch_num].state = HC_XACTERR; - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_AHBERR); + hhcd->hc[chnum].state = HC_XACTERR; + (void)USB_HC_Halt(hhcd->Instance, chnum); } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_ACK) == USB_OTG_HCINT_ACK) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_ACK)) { - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_ACK); + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_ACK); + + if (hhcd->hc[chnum].do_ping == 1U) + { + hhcd->hc[chnum].do_ping = 0U; + hhcd->hc[chnum].urb_state = URB_NOTREADY; + hhcd->hc[chnum].state = HC_ACK; + (void)USB_HC_Halt(hhcd->Instance, chnum); + } - if (hhcd->hc[ch_num].do_ping == 1U) + if ((hhcd->hc[chnum].do_ssplit == 1U) && (hhcd->hc[chnum].do_csplit == 0U)) { - hhcd->hc[ch_num].do_ping = 0U; - hhcd->hc[ch_num].urb_state = URB_NOTREADY; - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); + if (hhcd->hc[chnum].ep_type != EP_TYPE_ISOC) + { + hhcd->hc[chnum].do_csplit = 1U; + } + + hhcd->hc[chnum].state = HC_ACK; + (void)USB_HC_Halt(hhcd->Instance, chnum); + + /* reset error_count */ + hhcd->hc[chnum].ErrCnt = 0U; } } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_FRMOR) == USB_OTG_HCINT_FRMOR) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_FRMOR)) { - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_FRMOR); - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_FRMOR); + (void)USB_HC_Halt(hhcd->Instance, chnum); } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_XFRC) == USB_OTG_HCINT_XFRC) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_XFRC)) { - hhcd->hc[ch_num].ErrCnt = 0U; + hhcd->hc[chnum].ErrCnt = 0U; /* transaction completed with NYET state, update do ping state */ - if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_NYET) == USB_OTG_HCINT_NYET) + if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_NYET)) { - hhcd->hc[ch_num].do_ping = 1U; - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_NYET); + hhcd->hc[chnum].do_ping = 1U; + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_NYET); } - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_XFRC); - hhcd->hc[ch_num].state = HC_XFRC; - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); + + if (hhcd->hc[chnum].do_csplit != 0U) + { + hhcd->hc[chnum].do_csplit = 0U; + __HAL_HCD_CLEAR_HC_CSPLT(chnum); + } + + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_XFRC); + hhcd->hc[chnum].state = HC_XFRC; + (void)USB_HC_Halt(hhcd->Instance, chnum); } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_NYET) == USB_OTG_HCINT_NYET) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_NYET)) { - hhcd->hc[ch_num].state = HC_NYET; - hhcd->hc[ch_num].do_ping = 1U; - hhcd->hc[ch_num].ErrCnt = 0U; - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_NYET); + hhcd->hc[chnum].state = HC_NYET; + + if (hhcd->hc[chnum].do_ssplit == 0U) + { + hhcd->hc[chnum].do_ping = 1U; + } + + hhcd->hc[chnum].ErrCnt = 0U; + (void)USB_HC_Halt(hhcd->Instance, chnum); + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_NYET); } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_STALL) == USB_OTG_HCINT_STALL) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_STALL)) { - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_STALL); - hhcd->hc[ch_num].state = HC_STALL; - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_STALL); + hhcd->hc[chnum].state = HC_STALL; + (void)USB_HC_Halt(hhcd->Instance, chnum); } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_NAK) == USB_OTG_HCINT_NAK) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_NAK)) { - hhcd->hc[ch_num].ErrCnt = 0U; - hhcd->hc[ch_num].state = HC_NAK; + hhcd->hc[chnum].ErrCnt = 0U; + hhcd->hc[chnum].state = HC_NAK; - if (hhcd->hc[ch_num].do_ping == 0U) + if (hhcd->hc[chnum].do_ping == 0U) { - if (hhcd->hc[ch_num].speed == HCD_DEVICE_SPEED_HIGH) + if (hhcd->hc[chnum].speed == HCD_DEVICE_SPEED_HIGH) { - hhcd->hc[ch_num].do_ping = 1U; + hhcd->hc[chnum].do_ping = 1U; } } - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_NAK); + (void)USB_HC_Halt(hhcd->Instance, chnum); + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_NAK); } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_TXERR) == USB_OTG_HCINT_TXERR) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_TXERR)) { if (hhcd->Init.dma_enable == 0U) { - hhcd->hc[ch_num].state = HC_XACTERR; - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); + hhcd->hc[chnum].state = HC_XACTERR; + (void)USB_HC_Halt(hhcd->Instance, chnum); } else { - hhcd->hc[ch_num].ErrCnt++; - if (hhcd->hc[ch_num].ErrCnt > 2U) + hhcd->hc[chnum].ErrCnt++; + if (hhcd->hc[chnum].ErrCnt > 2U) { - hhcd->hc[ch_num].ErrCnt = 0U; - hhcd->hc[ch_num].urb_state = URB_ERROR; + hhcd->hc[chnum].ErrCnt = 0U; + hhcd->hc[chnum].urb_state = URB_ERROR; #if (USE_HAL_HCD_REGISTER_CALLBACKS == 1U) - hhcd->HC_NotifyURBChangeCallback(hhcd, (uint8_t)ch_num, hhcd->hc[ch_num].urb_state); + hhcd->HC_NotifyURBChangeCallback(hhcd, chnum, hhcd->hc[chnum].urb_state); #else - HAL_HCD_HC_NotifyURBChange_Callback(hhcd, (uint8_t)ch_num, hhcd->hc[ch_num].urb_state); + HAL_HCD_HC_NotifyURBChange_Callback(hhcd, chnum, hhcd->hc[chnum].urb_state); #endif /* USE_HAL_HCD_REGISTER_CALLBACKS */ } else { - hhcd->hc[ch_num].urb_state = URB_NOTREADY; + hhcd->hc[chnum].urb_state = URB_NOTREADY; } } - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_TXERR); + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_TXERR); } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_DTERR) == USB_OTG_HCINT_DTERR) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_DTERR)) { - hhcd->hc[ch_num].state = HC_DATATGLERR; - (void)USB_HC_Halt(hhcd->Instance, (uint8_t)ch_num); - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_DTERR); + hhcd->hc[chnum].state = HC_DATATGLERR; + (void)USB_HC_Halt(hhcd->Instance, chnum); + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_DTERR); } - else if ((USBx_HC(ch_num)->HCINT & USB_OTG_HCINT_CHH) == USB_OTG_HCINT_CHH) + else if (__HAL_HCD_GET_CH_FLAG(hhcd, chnum, USB_OTG_HCINT_CHH)) { - if (hhcd->hc[ch_num].state == HC_XFRC) + __HAL_HCD_CLEAR_HC_INT(chnum, USB_OTG_HCINT_CHH); + + if (hhcd->hc[chnum].state == HC_XFRC) { - hhcd->hc[ch_num].urb_state = URB_DONE; - if ((hhcd->hc[ch_num].ep_type == EP_TYPE_BULK) || - (hhcd->hc[ch_num].ep_type == EP_TYPE_INTR)) + hhcd->hc[chnum].state = HC_HALTED; + hhcd->hc[chnum].urb_state = URB_DONE; + + if ((hhcd->hc[chnum].ep_type == EP_TYPE_BULK) || + (hhcd->hc[chnum].ep_type == EP_TYPE_INTR)) { if (hhcd->Init.dma_enable == 0U) { - hhcd->hc[ch_num].toggle_out ^= 1U; + hhcd->hc[chnum].toggle_out ^= 1U; } - if ((hhcd->Init.dma_enable == 1U) && (hhcd->hc[ch_num].xfer_len > 0U)) + if ((hhcd->Init.dma_enable == 1U) && (hhcd->hc[chnum].xfer_len > 0U)) { - num_packets = (hhcd->hc[ch_num].xfer_len + hhcd->hc[ch_num].max_packet - 1U) / hhcd->hc[ch_num].max_packet; + num_packets = (hhcd->hc[chnum].xfer_len + hhcd->hc[chnum].max_packet - 1U) / hhcd->hc[chnum].max_packet; if ((num_packets & 1U) != 0U) { - hhcd->hc[ch_num].toggle_out ^= 1U; + hhcd->hc[chnum].toggle_out ^= 1U; } } } } - else if (hhcd->hc[ch_num].state == HC_NAK) + else if (hhcd->hc[chnum].state == HC_ACK) { - hhcd->hc[ch_num].urb_state = URB_NOTREADY; + hhcd->hc[chnum].state = HC_HALTED; + + if (hhcd->hc[chnum].do_csplit == 1U) + { + hhcd->hc[chnum].urb_state = URB_NOTREADY; + } } - else if (hhcd->hc[ch_num].state == HC_NYET) + else if (hhcd->hc[chnum].state == HC_NAK) { - hhcd->hc[ch_num].urb_state = URB_NOTREADY; + hhcd->hc[chnum].state = HC_HALTED; + hhcd->hc[chnum].urb_state = URB_NOTREADY; + + if (hhcd->hc[chnum].do_csplit == 1U) + { + hhcd->hc[chnum].do_csplit = 0U; + __HAL_HCD_CLEAR_HC_CSPLT(chnum); + } } - else if (hhcd->hc[ch_num].state == HC_STALL) + else if (hhcd->hc[chnum].state == HC_NYET) { - hhcd->hc[ch_num].urb_state = URB_STALL; + hhcd->hc[chnum].state = HC_HALTED; + hhcd->hc[chnum].urb_state = URB_NOTREADY; } - else if ((hhcd->hc[ch_num].state == HC_XACTERR) || - (hhcd->hc[ch_num].state == HC_DATATGLERR)) + else if (hhcd->hc[chnum].state == HC_STALL) { - hhcd->hc[ch_num].ErrCnt++; - if (hhcd->hc[ch_num].ErrCnt > 2U) + hhcd->hc[chnum].state = HC_HALTED; + hhcd->hc[chnum].urb_state = URB_STALL; + } + else if ((hhcd->hc[chnum].state == HC_XACTERR) || + (hhcd->hc[chnum].state == HC_DATATGLERR)) + { + hhcd->hc[chnum].state = HC_HALTED; + hhcd->hc[chnum].ErrCnt++; + if (hhcd->hc[chnum].ErrCnt > 2U) { - hhcd->hc[ch_num].ErrCnt = 0U; - hhcd->hc[ch_num].urb_state = URB_ERROR; + hhcd->hc[chnum].ErrCnt = 0U; + hhcd->hc[chnum].urb_state = URB_ERROR; } else { - hhcd->hc[ch_num].urb_state = URB_NOTREADY; + hhcd->hc[chnum].urb_state = URB_NOTREADY; /* re-activate the channel */ - tmpreg = USBx_HC(ch_num)->HCCHAR; + tmpreg = USBx_HC(chnum)->HCCHAR; tmpreg &= ~USB_OTG_HCCHAR_CHDIS; tmpreg |= USB_OTG_HCCHAR_CHENA; - USBx_HC(ch_num)->HCCHAR = tmpreg; + USBx_HC(chnum)->HCCHAR = tmpreg; } } else { - /* ... */ + return; } - __HAL_HCD_CLEAR_HC_INT(ch_num, USB_OTG_HCINT_CHH); - #if (USE_HAL_HCD_REGISTER_CALLBACKS == 1U) - hhcd->HC_NotifyURBChangeCallback(hhcd, (uint8_t)ch_num, hhcd->hc[ch_num].urb_state); + hhcd->HC_NotifyURBChangeCallback(hhcd, chnum, hhcd->hc[chnum].urb_state); #else - HAL_HCD_HC_NotifyURBChange_Callback(hhcd, (uint8_t)ch_num, hhcd->hc[ch_num].urb_state); + HAL_HCD_HC_NotifyURBChange_Callback(hhcd, chnum, hhcd->hc[chnum].urb_state); #endif /* USE_HAL_HCD_REGISTER_CALLBACKS */ } else { - /* ... */ + return; } } @@ -1564,17 +1833,17 @@ static void HCD_HC_OUT_IRQHandler(HCD_HandleTypeDef *hhcd, uint8_t chnum) */ static void HCD_RXQLVL_IRQHandler(HCD_HandleTypeDef *hhcd) { - USB_OTG_GlobalTypeDef *USBx = hhcd->Instance; + const USB_OTG_GlobalTypeDef *USBx = hhcd->Instance; uint32_t USBx_BASE = (uint32_t)USBx; uint32_t pktsts; uint32_t pktcnt; uint32_t GrxstspReg; uint32_t xferSizePktCnt; uint32_t tmpreg; - uint32_t ch_num; + uint32_t chnum; GrxstspReg = hhcd->Instance->GRXSTSP; - ch_num = GrxstspReg & USB_OTG_GRXSTSP_EPNUM; + chnum = GrxstspReg & USB_OTG_GRXSTSP_EPNUM; pktsts = (GrxstspReg & USB_OTG_GRXSTSP_PKTSTS) >> 17; pktcnt = (GrxstspReg & USB_OTG_GRXSTSP_BCNT) >> 4; @@ -1582,33 +1851,33 @@ static void HCD_RXQLVL_IRQHandler(HCD_HandleTypeDef *hhcd) { case GRXSTS_PKTSTS_IN: /* Read the data into the host buffer. */ - if ((pktcnt > 0U) && (hhcd->hc[ch_num].xfer_buff != (void *)0)) + if ((pktcnt > 0U) && (hhcd->hc[chnum].xfer_buff != (void *)0)) { - if ((hhcd->hc[ch_num].xfer_count + pktcnt) <= hhcd->hc[ch_num].xfer_len) + if ((hhcd->hc[chnum].xfer_count + pktcnt) <= hhcd->hc[chnum].xfer_len) { (void)USB_ReadPacket(hhcd->Instance, - hhcd->hc[ch_num].xfer_buff, (uint16_t)pktcnt); + hhcd->hc[chnum].xfer_buff, (uint16_t)pktcnt); /* manage multiple Xfer */ - hhcd->hc[ch_num].xfer_buff += pktcnt; - hhcd->hc[ch_num].xfer_count += pktcnt; + hhcd->hc[chnum].xfer_buff += pktcnt; + hhcd->hc[chnum].xfer_count += pktcnt; /* get transfer size packet count */ - xferSizePktCnt = (USBx_HC(ch_num)->HCTSIZ & USB_OTG_HCTSIZ_PKTCNT) >> 19; + xferSizePktCnt = (USBx_HC(chnum)->HCTSIZ & USB_OTG_HCTSIZ_PKTCNT) >> 19; - if ((hhcd->hc[ch_num].max_packet == pktcnt) && (xferSizePktCnt > 0U)) + if ((hhcd->hc[chnum].max_packet == pktcnt) && (xferSizePktCnt > 0U)) { /* re-activate the channel when more packets are expected */ - tmpreg = USBx_HC(ch_num)->HCCHAR; + tmpreg = USBx_HC(chnum)->HCCHAR; tmpreg &= ~USB_OTG_HCCHAR_CHDIS; tmpreg |= USB_OTG_HCCHAR_CHENA; - USBx_HC(ch_num)->HCCHAR = tmpreg; - hhcd->hc[ch_num].toggle_in ^= 1U; + USBx_HC(chnum)->HCCHAR = tmpreg; + hhcd->hc[chnum].toggle_in ^= 1U; } } else { - hhcd->hc[ch_num].urb_state = URB_ERROR; + hhcd->hc[chnum].urb_state = URB_ERROR; } } break; @@ -1630,7 +1899,7 @@ static void HCD_RXQLVL_IRQHandler(HCD_HandleTypeDef *hhcd) */ static void HCD_Port_IRQHandler(HCD_HandleTypeDef *hhcd) { - USB_OTG_GlobalTypeDef *USBx = hhcd->Instance; + const USB_OTG_GlobalTypeDef *USBx = hhcd->Instance; uint32_t USBx_BASE = (uint32_t)USBx; __IO uint32_t hprt0; __IO uint32_t hprt0_dup; @@ -1663,7 +1932,7 @@ static void HCD_Port_IRQHandler(HCD_HandleTypeDef *hhcd) if ((hprt0 & USB_OTG_HPRT_PENA) == USB_OTG_HPRT_PENA) { - if (hhcd->Init.phy_itface == USB_OTG_EMBEDDED_PHY) + if (hhcd->Init.phy_itface == USB_OTG_EMBEDDED_PHY) { if ((hprt0 & USB_OTG_HPRT_PSPD) == (HPRT0_PRTSPD_LOW_SPEED << 17)) { @@ -1678,7 +1947,7 @@ static void HCD_Port_IRQHandler(HCD_HandleTypeDef *hhcd) { if (hhcd->Init.speed == HCD_SPEED_FULL) { - USBx_HOST->HFIR = 60000U; + USBx_HOST->HFIR = HFIR_60_MHZ; } } #if (USE_HAL_HCD_REGISTER_CALLBACKS == 1U) diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_i2c.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_i2c.c index c2a8eb75..f2884b69 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_i2c.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_i2c.c @@ -89,7 +89,7 @@ add his own code by customization of function pointer HAL_I2C_SlaveRxCpltCallback() (+) In case of transfer Error, HAL_I2C_ErrorCallback() function is executed and user can add his own code by customization of function pointer HAL_I2C_ErrorCallback() - (+) Abort a master I2C process communication with Interrupt using HAL_I2C_Master_Abort_IT() + (+) Abort a master or memory I2C process communication with Interrupt using HAL_I2C_Master_Abort_IT() (+) End of abort process, HAL_I2C_AbortCpltCallback() is executed and user can add his own code by customization of function pointer HAL_I2C_AbortCpltCallback() @@ -139,7 +139,7 @@ or using HAL_I2C_Master_Seq_Receive_DMA() (+++) At reception end of current frame transfer, HAL_I2C_MasterRxCpltCallback() is executed and user can add his own code by customization of function pointer HAL_I2C_MasterRxCpltCallback() - (++) Abort a master IT or DMA I2C process communication with Interrupt using HAL_I2C_Master_Abort_IT() + (++) Abort a master or memory IT or DMA I2C process communication with Interrupt using HAL_I2C_Master_Abort_IT() (+++) End of abort process, HAL_I2C_AbortCpltCallback() is executed and user can add his own code by customization of function pointer HAL_I2C_AbortCpltCallback() (++) Enable/disable the Address listen mode in slave I2C mode using HAL_I2C_EnableListen_IT() HAL_I2C_DisableListen_IT() @@ -193,7 +193,7 @@ add his own code by customization of function pointer HAL_I2C_SlaveRxCpltCallback() (+) In case of transfer Error, HAL_I2C_ErrorCallback() function is executed and user can add his own code by customization of function pointer HAL_I2C_ErrorCallback() - (+) Abort a master I2C process communication with Interrupt using HAL_I2C_Master_Abort_IT() + (+) Abort a master or memory I2C process communication with Interrupt using HAL_I2C_Master_Abort_IT() (+) End of abort process, HAL_I2C_AbortCpltCallback() is executed and user can add his own code by customization of function pointer HAL_I2C_AbortCpltCallback() @@ -313,7 +313,7 @@ /* Private typedef -----------------------------------------------------------*/ /* Private define ------------------------------------------------------------*/ -/** @addtogroup I2C_Private_Define +/** @defgroup I2C_Private_Define I2C Private Define * @{ */ #define I2C_TIMEOUT_FLAG 35U /*!< Timeout 35 ms */ @@ -334,6 +334,14 @@ */ /* Private macro -------------------------------------------------------------*/ +/** @addtogroup I2C_Private_Macros + * @{ + */ +/* Macro to get remaining data to transfer on DMA side */ +#define I2C_GET_DMA_REMAIN_DATA(__HANDLE__) __HAL_DMA_GET_COUNTER(__HANDLE__) +/** + * @} + */ /* Private variables ---------------------------------------------------------*/ /* Private function prototypes -----------------------------------------------*/ @@ -383,6 +391,9 @@ static void I2C_MemoryTransmit_TXE_BTF(I2C_HandleTypeDef *hi2c); /* Private function to Convert Specific options */ static void I2C_ConvertOtherXferOptions(I2C_HandleTypeDef *hi2c); + +/* Private function to flush DR register */ +static void I2C_Flush_DR(I2C_HandleTypeDef *hi2c); /** * @} */ @@ -940,6 +951,20 @@ HAL_StatusTypeDef HAL_I2C_UnRegisterAddrCallback(I2C_HandleTypeDef *hi2c) #endif /* USE_HAL_I2C_REGISTER_CALLBACKS */ +/** + * @brief I2C data register flush process. + * @param hi2c I2C handle. + * @retval None + */ +static void I2C_Flush_DR(I2C_HandleTypeDef *hi2c) +{ + /* Write a dummy data in DR to clear TXE flag */ + if (__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_TXE) != RESET) + { + hi2c->Instance->DR = 0x00U; + } +} + /** * @} */ @@ -1357,6 +1382,13 @@ HAL_StatusTypeDef HAL_I2C_Master_Receive(I2C_HandleTypeDef *hi2c, uint16_t DevAd if (__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BTF) == SET) { + + if (hi2c->XferSize == 3U) + { + /* Disable Acknowledge */ + CLEAR_BIT(hi2c->Instance->CR1, I2C_CR1_ACK); + } + /* Read data from DR */ *hi2c->pBuffPtr = (uint8_t)hi2c->Instance->DR; @@ -1662,10 +1694,7 @@ HAL_StatusTypeDef HAL_I2C_Master_Transmit_IT(I2C_HandleTypeDef *hi2c, uint16_t D hi2c->Mode = HAL_I2C_MODE_NONE; hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); - - return HAL_ERROR; + return HAL_BUSY; } } while (__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BUSY) != RESET); @@ -1742,10 +1771,7 @@ HAL_StatusTypeDef HAL_I2C_Master_Receive_IT(I2C_HandleTypeDef *hi2c, uint16_t De hi2c->Mode = HAL_I2C_MODE_NONE; hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); - - return HAL_ERROR; + return HAL_BUSY; } } while (__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BUSY) != RESET); @@ -1952,10 +1978,7 @@ HAL_StatusTypeDef HAL_I2C_Master_Transmit_DMA(I2C_HandleTypeDef *hi2c, uint16_t hi2c->Mode = HAL_I2C_MODE_NONE; hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); - - return HAL_ERROR; + return HAL_BUSY; } } while (__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BUSY) != RESET); @@ -2110,10 +2133,7 @@ HAL_StatusTypeDef HAL_I2C_Master_Receive_DMA(I2C_HandleTypeDef *hi2c, uint16_t D hi2c->Mode = HAL_I2C_MODE_NONE; hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); - - return HAL_ERROR; + return HAL_BUSY; } } while (__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BUSY) != RESET); @@ -2811,6 +2831,11 @@ HAL_StatusTypeDef HAL_I2C_Mem_Read(I2C_HandleTypeDef *hi2c, uint16_t DevAddress, if (__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BTF) == SET) { + if (hi2c->XferSize == 3U) + { + /* Disable Acknowledge */ + CLEAR_BIT(hi2c->Instance->CR1, I2C_CR1_ACK); + } /* Read data from DR */ *hi2c->pBuffPtr = (uint8_t)hi2c->Instance->DR; @@ -2871,10 +2896,7 @@ HAL_StatusTypeDef HAL_I2C_Mem_Write_IT(I2C_HandleTypeDef *hi2c, uint16_t DevAddr hi2c->Mode = HAL_I2C_MODE_NONE; hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); - - return HAL_ERROR; + return HAL_BUSY; } } while (__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BUSY) != RESET); @@ -2959,10 +2981,7 @@ HAL_StatusTypeDef HAL_I2C_Mem_Read_IT(I2C_HandleTypeDef *hi2c, uint16_t DevAddre hi2c->Mode = HAL_I2C_MODE_NONE; hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); - - return HAL_ERROR; + return HAL_BUSY; } } while (__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BUSY) != RESET); @@ -3057,10 +3076,7 @@ HAL_StatusTypeDef HAL_I2C_Mem_Write_DMA(I2C_HandleTypeDef *hi2c, uint16_t DevAdd hi2c->Mode = HAL_I2C_MODE_NONE; hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); - - return HAL_ERROR; + return HAL_BUSY; } } while (__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BUSY) != RESET); @@ -3241,10 +3257,7 @@ HAL_StatusTypeDef HAL_I2C_Mem_Read_DMA(I2C_HandleTypeDef *hi2c, uint16_t DevAddr hi2c->Mode = HAL_I2C_MODE_NONE; hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); - - return HAL_ERROR; + return HAL_BUSY; } } while (__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BUSY) != RESET); @@ -3577,10 +3590,7 @@ HAL_StatusTypeDef HAL_I2C_Master_Seq_Transmit_IT(I2C_HandleTypeDef *hi2c, uint16 hi2c->Mode = HAL_I2C_MODE_NONE; hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); - - return HAL_ERROR; + return HAL_BUSY; } } while (__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BUSY) != RESET); @@ -3676,10 +3686,7 @@ HAL_StatusTypeDef HAL_I2C_Master_Seq_Transmit_DMA(I2C_HandleTypeDef *hi2c, uint1 hi2c->Mode = HAL_I2C_MODE_NONE; hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); - - return HAL_ERROR; + return HAL_BUSY; } } while (__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BUSY) != RESET); @@ -3859,10 +3866,7 @@ HAL_StatusTypeDef HAL_I2C_Master_Seq_Receive_IT(I2C_HandleTypeDef *hi2c, uint16_ hi2c->Mode = HAL_I2C_MODE_NONE; hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); - - return HAL_ERROR; + return HAL_BUSY; } } while (__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BUSY) != RESET); @@ -3984,10 +3988,7 @@ HAL_StatusTypeDef HAL_I2C_Master_Seq_Receive_DMA(I2C_HandleTypeDef *hi2c, uint16 hi2c->Mode = HAL_I2C_MODE_NONE; hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); - - return HAL_ERROR; + return HAL_BUSY; } } while (__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BUSY) != RESET); @@ -4712,7 +4713,7 @@ HAL_StatusTypeDef HAL_I2C_DisableListen_IT(I2C_HandleTypeDef *hi2c) } /** - * @brief Abort a master I2C IT or DMA process communication with Interrupt. + * @brief Abort a master or memory I2C IT or DMA process communication with Interrupt. * @param hi2c Pointer to a I2C_HandleTypeDef structure that contains * the configuration information for the specified I2C. * @param DevAddress Target device address: The device 7 bits address value @@ -4728,7 +4729,8 @@ HAL_StatusTypeDef HAL_I2C_Master_Abort_IT(I2C_HandleTypeDef *hi2c, uint16_t DevA UNUSED(DevAddress); /* Abort Master transfer during Receive or Transmit process */ - if ((__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BUSY) != RESET) && (CurrentMode == HAL_I2C_MODE_MASTER)) + if ((__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BUSY) != RESET) && ((CurrentMode == HAL_I2C_MODE_MASTER) || + (CurrentMode == HAL_I2C_MODE_MEM))) { /* Process Locked */ __HAL_LOCK(hi2c); @@ -5504,7 +5506,8 @@ static void I2C_MemoryTransmit_TXE_BTF(I2C_HandleTypeDef *hi2c) } else { - /* Do nothing */ + /* Clear TXE and BTF flags */ + I2C_Flush_DR(hi2c); } } @@ -5519,7 +5522,9 @@ static void I2C_MasterReceive_RXNE(I2C_HandleTypeDef *hi2c) if (hi2c->State == HAL_I2C_STATE_BUSY_RX) { uint32_t tmp; + uint32_t CurrentXferOptions; + CurrentXferOptions = hi2c->XferOptions; tmp = hi2c->XferCount; if (tmp > 3U) { @@ -5575,7 +5580,14 @@ static void I2C_MasterReceive_RXNE(I2C_HandleTypeDef *hi2c) else { hi2c->Mode = HAL_I2C_MODE_NONE; - hi2c->PreviousState = I2C_STATE_MASTER_BUSY_RX; + if ((CurrentXferOptions == I2C_FIRST_AND_LAST_FRAME) || (CurrentXferOptions == I2C_LAST_FRAME)) + { + hi2c->PreviousState = I2C_STATE_NONE; + } + else + { + hi2c->PreviousState = I2C_STATE_MASTER_BUSY_RX; + } #if (USE_HAL_I2C_REGISTER_CALLBACKS == 1) hi2c->MasterRxCpltCallback(hi2c); @@ -5723,7 +5735,14 @@ static void I2C_MasterReceive_BTF(I2C_HandleTypeDef *hi2c) else { hi2c->Mode = HAL_I2C_MODE_NONE; - hi2c->PreviousState = I2C_STATE_MASTER_BUSY_RX; + if ((CurrentXferOptions == I2C_FIRST_AND_LAST_FRAME) || (CurrentXferOptions == I2C_LAST_FRAME)) + { + hi2c->PreviousState = I2C_STATE_NONE; + } + else + { + hi2c->PreviousState = I2C_STATE_MASTER_BUSY_RX; + } #if (USE_HAL_I2C_REGISTER_CALLBACKS == 1) hi2c->MasterRxCpltCallback(hi2c); #else @@ -6170,7 +6189,7 @@ static void I2C_Slave_STOPF(I2C_HandleTypeDef *hi2c) { if ((CurrentState == HAL_I2C_STATE_BUSY_RX) || (CurrentState == HAL_I2C_STATE_BUSY_RX_LISTEN)) { - hi2c->XferCount = (uint16_t)(__HAL_DMA_GET_COUNTER(hi2c->hdmarx)); + hi2c->XferCount = (uint16_t)(I2C_GET_DMA_REMAIN_DATA(hi2c->hdmarx)); if (hi2c->XferCount != 0U) { @@ -6198,7 +6217,7 @@ static void I2C_Slave_STOPF(I2C_HandleTypeDef *hi2c) } else { - hi2c->XferCount = (uint16_t)(__HAL_DMA_GET_COUNTER(hi2c->hdmatx)); + hi2c->XferCount = (uint16_t)(I2C_GET_DMA_REMAIN_DATA(hi2c->hdmatx)); if (hi2c->XferCount != 0U) { @@ -6367,6 +6386,9 @@ static void I2C_Slave_AF(I2C_HandleTypeDef *hi2c) /* Disable Acknowledge */ CLEAR_BIT(hi2c->Instance->CR1, I2C_CR1_ACK); + /* Clear TXE flag */ + I2C_Flush_DR(hi2c); + #if (USE_HAL_I2C_REGISTER_CALLBACKS == 1) hi2c->SlaveTxCpltCallback(hi2c); #else @@ -7028,7 +7050,14 @@ static void I2C_DMAXferCplt(DMA_HandleTypeDef *hdma) else { hi2c->Mode = HAL_I2C_MODE_NONE; - hi2c->PreviousState = I2C_STATE_MASTER_BUSY_RX; + if ((CurrentXferOptions == I2C_FIRST_AND_LAST_FRAME) || (CurrentXferOptions == I2C_LAST_FRAME)) + { + hi2c->PreviousState = I2C_STATE_NONE; + } + else + { + hi2c->PreviousState = I2C_STATE_MASTER_BUSY_RX; + } #if (USE_HAL_I2C_REGISTER_CALLBACKS == 1) hi2c->MasterRxCpltCallback(hi2c); @@ -7203,15 +7232,18 @@ static HAL_StatusTypeDef I2C_WaitOnFlagUntilTimeout(I2C_HandleTypeDef *hi2c, uin { if (((HAL_GetTick() - Tickstart) > Timeout) || (Timeout == 0U)) { - hi2c->PreviousState = I2C_STATE_NONE; - hi2c->State = HAL_I2C_STATE_READY; - hi2c->Mode = HAL_I2C_MODE_NONE; - hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; + if ((__HAL_I2C_GET_FLAG(hi2c, Flag) == Status)) + { + hi2c->PreviousState = I2C_STATE_NONE; + hi2c->State = HAL_I2C_STATE_READY; + hi2c->Mode = HAL_I2C_MODE_NONE; + hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); + /* Process Unlocked */ + __HAL_UNLOCK(hi2c); - return HAL_ERROR; + return HAL_ERROR; + } } } } @@ -7255,15 +7287,18 @@ static HAL_StatusTypeDef I2C_WaitOnMasterAddressFlagUntilTimeout(I2C_HandleTypeD { if (((HAL_GetTick() - Tickstart) > Timeout) || (Timeout == 0U)) { - hi2c->PreviousState = I2C_STATE_NONE; - hi2c->State = HAL_I2C_STATE_READY; - hi2c->Mode = HAL_I2C_MODE_NONE; - hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; + if ((__HAL_I2C_GET_FLAG(hi2c, Flag) == RESET)) + { + hi2c->PreviousState = I2C_STATE_NONE; + hi2c->State = HAL_I2C_STATE_READY; + hi2c->Mode = HAL_I2C_MODE_NONE; + hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); + /* Process Unlocked */ + __HAL_UNLOCK(hi2c); - return HAL_ERROR; + return HAL_ERROR; + } } } } @@ -7293,15 +7328,18 @@ static HAL_StatusTypeDef I2C_WaitOnTXEFlagUntilTimeout(I2C_HandleTypeDef *hi2c, { if (((HAL_GetTick() - Tickstart) > Timeout) || (Timeout == 0U)) { - hi2c->PreviousState = I2C_STATE_NONE; - hi2c->State = HAL_I2C_STATE_READY; - hi2c->Mode = HAL_I2C_MODE_NONE; - hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; + if ((__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_TXE) == RESET)) + { + hi2c->PreviousState = I2C_STATE_NONE; + hi2c->State = HAL_I2C_STATE_READY; + hi2c->Mode = HAL_I2C_MODE_NONE; + hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); + /* Process Unlocked */ + __HAL_UNLOCK(hi2c); - return HAL_ERROR; + return HAL_ERROR; + } } } } @@ -7331,15 +7369,18 @@ static HAL_StatusTypeDef I2C_WaitOnBTFFlagUntilTimeout(I2C_HandleTypeDef *hi2c, { if (((HAL_GetTick() - Tickstart) > Timeout) || (Timeout == 0U)) { - hi2c->PreviousState = I2C_STATE_NONE; - hi2c->State = HAL_I2C_STATE_READY; - hi2c->Mode = HAL_I2C_MODE_NONE; - hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; + if ((__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_BTF) == RESET)) + { + hi2c->PreviousState = I2C_STATE_NONE; + hi2c->State = HAL_I2C_STATE_READY; + hi2c->Mode = HAL_I2C_MODE_NONE; + hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); + /* Process Unlocked */ + __HAL_UNLOCK(hi2c); - return HAL_ERROR; + return HAL_ERROR; + } } } } @@ -7367,15 +7408,18 @@ static HAL_StatusTypeDef I2C_WaitOnSTOPFlagUntilTimeout(I2C_HandleTypeDef *hi2c, /* Check for the Timeout */ if (((HAL_GetTick() - Tickstart) > Timeout) || (Timeout == 0U)) { - hi2c->PreviousState = I2C_STATE_NONE; - hi2c->State = HAL_I2C_STATE_READY; - hi2c->Mode = HAL_I2C_MODE_NONE; - hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; + if ((__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_STOPF) == RESET)) + { + hi2c->PreviousState = I2C_STATE_NONE; + hi2c->State = HAL_I2C_STATE_READY; + hi2c->Mode = HAL_I2C_MODE_NONE; + hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); + /* Process Unlocked */ + __HAL_UNLOCK(hi2c); - return HAL_ERROR; + return HAL_ERROR; + } } } return HAL_OK; @@ -7441,15 +7485,18 @@ static HAL_StatusTypeDef I2C_WaitOnRXNEFlagUntilTimeout(I2C_HandleTypeDef *hi2c, /* Check for the Timeout */ if (((HAL_GetTick() - Tickstart) > Timeout) || (Timeout == 0U)) { - hi2c->PreviousState = I2C_STATE_NONE; - hi2c->State = HAL_I2C_STATE_READY; - hi2c->Mode = HAL_I2C_MODE_NONE; - hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; + if ((__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_RXNE) == RESET)) + { + hi2c->PreviousState = I2C_STATE_NONE; + hi2c->State = HAL_I2C_STATE_READY; + hi2c->Mode = HAL_I2C_MODE_NONE; + hi2c->ErrorCode |= HAL_I2C_ERROR_TIMEOUT; - /* Process Unlocked */ - __HAL_UNLOCK(hi2c); + /* Process Unlocked */ + __HAL_UNLOCK(hi2c); - return HAL_ERROR; + return HAL_ERROR; + } } } return HAL_OK; diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_irda.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_irda.c index 47b44ffd..0072dc19 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_irda.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_irda.c @@ -439,6 +439,8 @@ __weak void HAL_IRDA_MspDeInit(IRDA_HandleTypeDef *hirda) /** * @brief Register a User IRDA Callback * To be used instead of the weak predefined callback + * @note The HAL_IRDA_RegisterCallback() may be called before HAL_IRDA_Init() in HAL_IRDA_STATE_RESET + * to register callbacks for HAL_IRDA_MSPINIT_CB_ID and HAL_IRDA_MSPDEINIT_CB_ID * @param hirda irda handle * @param CallbackID ID of the callback to be registered * This parameter can be one of the following values: @@ -466,8 +468,6 @@ HAL_StatusTypeDef HAL_IRDA_RegisterCallback(IRDA_HandleTypeDef *hirda, HAL_IRDA_ return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(hirda); if (hirda->gState == HAL_IRDA_STATE_READY) { @@ -552,15 +552,14 @@ HAL_StatusTypeDef HAL_IRDA_RegisterCallback(IRDA_HandleTypeDef *hirda, HAL_IRDA_ status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hirda); - return status; } /** * @brief Unregister an IRDA callback * IRDA callback is redirected to the weak predefined callback + * @note The HAL_IRDA_UnRegisterCallback() may be called before HAL_IRDA_Init() in HAL_IRDA_STATE_RESET + * to un-register callbacks for HAL_IRDA_MSPINIT_CB_ID and HAL_IRDA_MSPDEINIT_CB_ID * @param hirda irda handle * @param CallbackID ID of the callback to be unregistered * This parameter can be one of the following values: @@ -580,9 +579,6 @@ HAL_StatusTypeDef HAL_IRDA_UnRegisterCallback(IRDA_HandleTypeDef *hirda, HAL_IRD { HAL_StatusTypeDef status = HAL_OK; - /* Process locked */ - __HAL_LOCK(hirda); - if (HAL_IRDA_STATE_READY == hirda->gState) { switch (CallbackID) @@ -666,9 +662,6 @@ HAL_StatusTypeDef HAL_IRDA_UnRegisterCallback(IRDA_HandleTypeDef *hirda, HAL_IRD status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hirda); - return status; } #endif /* USE_HAL_IRDA_REGISTER_CALLBACKS */ @@ -2030,7 +2023,7 @@ __weak void HAL_IRDA_AbortReceiveCpltCallback(IRDA_HandleTypeDef *hirda) * the configuration information for the specified IRDA. * @retval HAL state */ -HAL_IRDA_StateTypeDef HAL_IRDA_GetState(IRDA_HandleTypeDef *hirda) +HAL_IRDA_StateTypeDef HAL_IRDA_GetState(const IRDA_HandleTypeDef *hirda) { uint32_t temp1 = 0x00U, temp2 = 0x00U; temp1 = hirda->gState; @@ -2045,7 +2038,7 @@ HAL_IRDA_StateTypeDef HAL_IRDA_GetState(IRDA_HandleTypeDef *hirda) * the configuration information for the specified IRDA. * @retval IRDA Error Code */ -uint32_t HAL_IRDA_GetError(IRDA_HandleTypeDef *hirda) +uint32_t HAL_IRDA_GetError(const IRDA_HandleTypeDef *hirda) { return hirda->ErrorCode; } diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_lptim.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_lptim.c index 7b569513..05fa66da 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_lptim.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_lptim.c @@ -185,7 +185,7 @@ #if (USE_HAL_LPTIM_REGISTER_CALLBACKS == 1) static void LPTIM_ResetCallback(LPTIM_HandleTypeDef *lptim); #endif /* USE_HAL_LPTIM_REGISTER_CALLBACKS */ -static HAL_StatusTypeDef LPTIM_WaitForFlag(LPTIM_HandleTypeDef *hlptim, uint32_t flag); +static HAL_StatusTypeDef LPTIM_WaitForFlag(const LPTIM_HandleTypeDef *hlptim, uint32_t flag); /* Exported functions --------------------------------------------------------*/ @@ -1753,7 +1753,7 @@ HAL_StatusTypeDef HAL_LPTIM_Counter_Stop_IT(LPTIM_HandleTypeDef *hlptim) * @param hlptim LPTIM handle * @retval Counter value. */ -uint32_t HAL_LPTIM_ReadCounter(LPTIM_HandleTypeDef *hlptim) +uint32_t HAL_LPTIM_ReadCounter(const LPTIM_HandleTypeDef *hlptim) { /* Check the parameters */ assert_param(IS_LPTIM_INSTANCE(hlptim->Instance)); @@ -1766,7 +1766,7 @@ uint32_t HAL_LPTIM_ReadCounter(LPTIM_HandleTypeDef *hlptim) * @param hlptim LPTIM handle * @retval Autoreload value. */ -uint32_t HAL_LPTIM_ReadAutoReload(LPTIM_HandleTypeDef *hlptim) +uint32_t HAL_LPTIM_ReadAutoReload(const LPTIM_HandleTypeDef *hlptim) { /* Check the parameters */ assert_param(IS_LPTIM_INSTANCE(hlptim->Instance)); @@ -1779,7 +1779,7 @@ uint32_t HAL_LPTIM_ReadAutoReload(LPTIM_HandleTypeDef *hlptim) * @param hlptim LPTIM handle * @retval Compare value. */ -uint32_t HAL_LPTIM_ReadCompare(LPTIM_HandleTypeDef *hlptim) +uint32_t HAL_LPTIM_ReadCompare(const LPTIM_HandleTypeDef *hlptim) { /* Check the parameters */ assert_param(IS_LPTIM_INSTANCE(hlptim->Instance)); @@ -2077,9 +2077,6 @@ HAL_StatusTypeDef HAL_LPTIM_RegisterCallback(LPTIM_HandleTypeDef *hlptim, return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(hlptim); - if (hlptim->State == HAL_LPTIM_STATE_READY) { switch (CallbackID) @@ -2150,9 +2147,6 @@ HAL_StatusTypeDef HAL_LPTIM_RegisterCallback(LPTIM_HandleTypeDef *hlptim, status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hlptim); - return status; } @@ -2178,9 +2172,6 @@ HAL_StatusTypeDef HAL_LPTIM_UnRegisterCallback(LPTIM_HandleTypeDef *hlpti { HAL_StatusTypeDef status = HAL_OK; - /* Process locked */ - __HAL_LOCK(hlptim); - if (hlptim->State == HAL_LPTIM_STATE_READY) { switch (CallbackID) @@ -2262,9 +2253,6 @@ HAL_StatusTypeDef HAL_LPTIM_UnRegisterCallback(LPTIM_HandleTypeDef *hlpti status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hlptim); - return status; } #endif /* USE_HAL_LPTIM_REGISTER_CALLBACKS */ @@ -2292,7 +2280,7 @@ HAL_StatusTypeDef HAL_LPTIM_UnRegisterCallback(LPTIM_HandleTypeDef *hlpti * @param hlptim LPTIM handle * @retval HAL state */ -HAL_LPTIM_StateTypeDef HAL_LPTIM_GetState(LPTIM_HandleTypeDef *hlptim) +HAL_LPTIM_StateTypeDef HAL_LPTIM_GetState(const LPTIM_HandleTypeDef *hlptim) { /* Return LPTIM handle state */ return hlptim->State; @@ -2339,7 +2327,7 @@ static void LPTIM_ResetCallback(LPTIM_HandleTypeDef *lptim) * @param flag The lptim flag * @retval HAL status */ -static HAL_StatusTypeDef LPTIM_WaitForFlag(LPTIM_HandleTypeDef *hlptim, uint32_t flag) +static HAL_StatusTypeDef LPTIM_WaitForFlag(const LPTIM_HandleTypeDef *hlptim, uint32_t flag) { HAL_StatusTypeDef result = HAL_OK; uint32_t count = TIMEOUT * (SystemCoreClock / 20UL / 1000UL); diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_ltdc.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_ltdc.c index fcc5fa16..f5f81482 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_ltdc.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_ltdc.c @@ -178,7 +178,13 @@ /* Private typedef -----------------------------------------------------------*/ /* Private define ------------------------------------------------------------*/ +/** @defgroup LTDC_Private_Define LTDC Private Define + * @{ + */ #define LTDC_TIMEOUT_VALUE ((uint32_t)100U) /* 100ms */ +/** + * @} + */ /* Private macro -------------------------------------------------------------*/ /* Private variables ---------------------------------------------------------*/ /* Private function prototypes -----------------------------------------------*/ @@ -549,7 +555,7 @@ HAL_StatusTypeDef HAL_LTDC_UnRegisterCallback(LTDC_HandleTypeDef *hltdc, HAL_LTD break; case HAL_LTDC_MSPINIT_CB_ID : - hltdc->MspInitCallback = HAL_LTDC_MspInit; /* Legcay weak MspInit Callback */ + hltdc->MspInitCallback = HAL_LTDC_MspInit; /* Legcay weak MspInit Callback */ break; case HAL_LTDC_MSPDEINIT_CB_ID : diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_ltdc_ex.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_ltdc_ex.c index 2ee7795b..ab2ca72e 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_ltdc_ex.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_ltdc_ex.c @@ -74,16 +74,18 @@ HAL_StatusTypeDef HAL_LTDCEx_StructInitFromVideoConfig(LTDC_HandleTypeDef *hltdc /* The following polarity is inverted: LTDC_DEPOLARITY_AL <-> LTDC_DEPOLARITY_AH */ +#if !defined(POLARITIES_INVERSION_UPDATED) /* Note 1 : Code in line w/ Current LTDC specification */ hltdc->Init.DEPolarity = (VidCfg->DEPolarity == \ DSI_DATA_ENABLE_ACTIVE_HIGH) ? LTDC_DEPOLARITY_AL : LTDC_DEPOLARITY_AH; hltdc->Init.VSPolarity = (VidCfg->VSPolarity == DSI_VSYNC_ACTIVE_HIGH) ? LTDC_VSPOLARITY_AH : LTDC_VSPOLARITY_AL; hltdc->Init.HSPolarity = (VidCfg->HSPolarity == DSI_HSYNC_ACTIVE_HIGH) ? LTDC_HSPOLARITY_AH : LTDC_HSPOLARITY_AL; - +#else /* Note 2: Code to be used in case LTDC polarities inversion updated in the specification */ - /* hltdc->Init.DEPolarity = VidCfg->DEPolarity << 29; - hltdc->Init.VSPolarity = VidCfg->VSPolarity << 29; - hltdc->Init.HSPolarity = VidCfg->HSPolarity << 29; */ + hltdc->Init.DEPolarity = VidCfg->DEPolarity << 29; + hltdc->Init.VSPolarity = VidCfg->VSPolarity << 29; + hltdc->Init.HSPolarity = VidCfg->HSPolarity << 29; +#endif /* POLARITIES_INVERSION_UPDATED */ /* Retrieve vertical timing parameters from DSI */ hltdc->Init.VerticalSync = VidCfg->VerticalSyncActive - 1U; @@ -115,17 +117,18 @@ HAL_StatusTypeDef HAL_LTDCEx_StructInitFromAdaptedCommandConfig(LTDC_HandleTypeD LTDC_VSPOLARITY_AL <-> LTDC_VSPOLARITY_AH LTDC_HSPOLARITY_AL <-> LTDC_HSPOLARITY_AH)*/ +#if !defined(POLARITIES_INVERSION_UPDATED) /* Note 1 : Code in line w/ Current LTDC specification */ hltdc->Init.DEPolarity = (CmdCfg->DEPolarity == \ DSI_DATA_ENABLE_ACTIVE_HIGH) ? LTDC_DEPOLARITY_AL : LTDC_DEPOLARITY_AH; hltdc->Init.VSPolarity = (CmdCfg->VSPolarity == DSI_VSYNC_ACTIVE_HIGH) ? LTDC_VSPOLARITY_AL : LTDC_VSPOLARITY_AH; hltdc->Init.HSPolarity = (CmdCfg->HSPolarity == DSI_HSYNC_ACTIVE_HIGH) ? LTDC_HSPOLARITY_AL : LTDC_HSPOLARITY_AH; - +#else /* Note 2: Code to be used in case LTDC polarities inversion updated in the specification */ - /* hltdc->Init.DEPolarity = CmdCfg->DEPolarity << 29; - hltdc->Init.VSPolarity = CmdCfg->VSPolarity << 29; - hltdc->Init.HSPolarity = CmdCfg->HSPolarity << 29; */ - + hltdc->Init.DEPolarity = CmdCfg->DEPolarity << 29; + hltdc->Init.VSPolarity = CmdCfg->VSPolarity << 29; + hltdc->Init.HSPolarity = CmdCfg->HSPolarity << 29; +#endif /* POLARITIES_INVERSION_UPDATED */ return HAL_OK; } diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_nand.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_nand.c index a0b89e62..5cd4ab2d 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_nand.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_nand.c @@ -77,15 +77,15 @@ and a pointer to the user callback function. Use function HAL_NAND_UnRegisterCallback() to reset a callback to the default - weak (surcharged) function. It allows to reset following callbacks: + weak (overridden) function. It allows to reset following callbacks: (+) MspInitCallback : NAND MspInit. (+) MspDeInitCallback : NAND MspDeInit. This function) takes as parameters the HAL peripheral handle and the Callback ID. By default, after the HAL_NAND_Init and if the state is HAL_NAND_STATE_RESET - all callbacks are reset to the corresponding legacy weak (surcharged) functions. + all callbacks are reset to the corresponding legacy weak (overridden) functions. Exception done for MspInit and MspDeInit callbacks that are respectively - reset to the legacy weak (surcharged) functions in the HAL_NAND_Init + reset to the legacy weak (overridden) functions in the HAL_NAND_Init and HAL_NAND_DeInit only when these callbacks are null (not registered beforehand). If not, MspInit or MspDeInit are not null, the HAL_NAND_Init and HAL_NAND_DeInit keep and use the user MspInit/MspDeInit callbacks (registered beforehand) @@ -100,7 +100,7 @@ When The compilation define USE_HAL_NAND_REGISTER_CALLBACKS is set to 0 or not defined, the callback registering feature is not available - and weak (surcharged) callbacks are used. + and weak (overridden) callbacks are used. @endverbatim ****************************************************************************** @@ -199,7 +199,7 @@ HAL_StatusTypeDef HAL_NAND_Init(NAND_HandleTypeDef *hnand, FMC_NAND_PCC_TimingT __FMC_NAND_ENABLE(hnand->Instance, hnand->Init.NandBank); #else __FMC_NAND_ENABLE(hnand->Instance); -#endif +#endif /* (FMC_Bank2_3) || (FSMC_Bank2_3) */ /* Update the NAND controller state */ hnand->State = HAL_NAND_STATE_READY; @@ -428,7 +428,7 @@ HAL_StatusTypeDef HAL_NAND_Read_ID(NAND_HandleTypeDef *hnand, NAND_IDTypeDef *pN } #else deviceaddress = NAND_DEVICE; -#endif +#endif /* FMC_Bank2_3 */ /* Send Read ID command sequence */ *(__IO uint8_t *)((uint32_t)(deviceaddress | CMD_AREA)) = NAND_CMD_READID; @@ -441,7 +441,7 @@ HAL_StatusTypeDef HAL_NAND_Read_ID(NAND_HandleTypeDef *hnand, NAND_IDTypeDef *pN if (hnand->Init.MemoryDataWidth == FSMC_NAND_PCC_MEM_BUS_WIDTH_8) #else /* FMC_PCR2_PWID is defined */ if (hnand->Init.MemoryDataWidth == FMC_NAND_PCC_MEM_BUS_WIDTH_8) -#endif +#endif /* FSMC_PCR2_PWID */ { data = *(__IO uint32_t *)deviceaddress; @@ -512,7 +512,7 @@ HAL_StatusTypeDef HAL_NAND_Reset(NAND_HandleTypeDef *hnand) } #else deviceaddress = NAND_DEVICE; -#endif +#endif /* FMC_Bank2_3 */ /* Send NAND reset command */ *(__IO uint8_t *)((uint32_t)(deviceaddress | CMD_AREA)) = 0xFF; @@ -561,8 +561,8 @@ HAL_StatusTypeDef HAL_NAND_ConfigDevice(NAND_HandleTypeDef *hnand, NAND_DeviceC * @param NumPageToRead number of pages to read from block * @retval HAL status */ -HAL_StatusTypeDef HAL_NAND_Read_Page_8b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, uint8_t *pBuffer, - uint32_t NumPageToRead) +HAL_StatusTypeDef HAL_NAND_Read_Page_8b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, + uint8_t *pBuffer, uint32_t NumPageToRead) { uint32_t index; uint32_t tickstart; @@ -597,7 +597,7 @@ HAL_StatusTypeDef HAL_NAND_Read_Page_8b(NAND_HandleTypeDef *hnand, NAND_AddressT } #else deviceaddress = NAND_DEVICE; -#endif +#endif /* FMC_Bank2_3 */ /* NAND raw address calculation */ nandaddress = ARRAY_ADDRESS(pAddress, hnand); @@ -730,8 +730,8 @@ HAL_StatusTypeDef HAL_NAND_Read_Page_8b(NAND_HandleTypeDef *hnand, NAND_AddressT * @param NumPageToRead number of pages to read from block * @retval HAL status */ -HAL_StatusTypeDef HAL_NAND_Read_Page_16b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, uint16_t *pBuffer, - uint32_t NumPageToRead) +HAL_StatusTypeDef HAL_NAND_Read_Page_16b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, + uint16_t *pBuffer, uint32_t NumPageToRead) { uint32_t index; uint32_t tickstart; @@ -766,7 +766,7 @@ HAL_StatusTypeDef HAL_NAND_Read_Page_16b(NAND_HandleTypeDef *hnand, NAND_Address } #else deviceaddress = NAND_DEVICE; -#endif +#endif /* FMC_Bank2_3 */ /* NAND raw address calculation */ nandaddress = ARRAY_ADDRESS(pAddress, hnand); @@ -860,9 +860,9 @@ HAL_StatusTypeDef HAL_NAND_Read_Page_16b(NAND_HandleTypeDef *hnand, NAND_Address /* Calculate PageSize */ #if defined(FSMC_PCR2_PWID) - if (hnand->Init.MemoryDataWidth == FSMC_NAND_PCC_MEM_BUS_WIDTH_8) + if (hnand->Init.MemoryDataWidth == FSMC_NAND_PCC_MEM_BUS_WIDTH_8) #else - if (hnand->Init.MemoryDataWidth == FMC_NAND_PCC_MEM_BUS_WIDTH_8) + if (hnand->Init.MemoryDataWidth == FMC_NAND_PCC_MEM_BUS_WIDTH_8) #endif /* FSMC_PCR2_PWID */ { hnand->Config.PageSize = hnand->Config.PageSize / 2U; @@ -913,8 +913,8 @@ HAL_StatusTypeDef HAL_NAND_Read_Page_16b(NAND_HandleTypeDef *hnand, NAND_Address * @param NumPageToWrite number of pages to write to block * @retval HAL status */ -HAL_StatusTypeDef HAL_NAND_Write_Page_8b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, uint8_t *pBuffer, - uint32_t NumPageToWrite) +HAL_StatusTypeDef HAL_NAND_Write_Page_8b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, + const uint8_t *pBuffer, uint32_t NumPageToWrite) { uint32_t index; uint32_t tickstart; @@ -922,7 +922,7 @@ HAL_StatusTypeDef HAL_NAND_Write_Page_8b(NAND_HandleTypeDef *hnand, NAND_Address uint32_t numpageswritten = 0U; uint32_t nandaddress; uint32_t nbpages = NumPageToWrite; - uint8_t *buff = pBuffer; + const uint8_t *buff = pBuffer; /* Check the NAND controller state */ if (hnand->State == HAL_NAND_STATE_BUSY) @@ -949,7 +949,7 @@ HAL_StatusTypeDef HAL_NAND_Write_Page_8b(NAND_HandleTypeDef *hnand, NAND_Address } #else deviceaddress = NAND_DEVICE; -#endif +#endif /* FMC_Bank2_3 */ /* NAND raw address calculation */ nandaddress = ARRAY_ADDRESS(pAddress, hnand); @@ -1077,8 +1077,8 @@ HAL_StatusTypeDef HAL_NAND_Write_Page_8b(NAND_HandleTypeDef *hnand, NAND_Address * @param NumPageToWrite number of pages to write to block * @retval HAL status */ -HAL_StatusTypeDef HAL_NAND_Write_Page_16b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, uint16_t *pBuffer, - uint32_t NumPageToWrite) +HAL_StatusTypeDef HAL_NAND_Write_Page_16b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, + const uint16_t *pBuffer, uint32_t NumPageToWrite) { uint32_t index; uint32_t tickstart; @@ -1086,7 +1086,7 @@ HAL_StatusTypeDef HAL_NAND_Write_Page_16b(NAND_HandleTypeDef *hnand, NAND_Addres uint32_t numpageswritten = 0U; uint32_t nandaddress; uint32_t nbpages = NumPageToWrite; - uint16_t *buff = pBuffer; + const uint16_t *buff = pBuffer; /* Check the NAND controller state */ if (hnand->State == HAL_NAND_STATE_BUSY) @@ -1113,7 +1113,7 @@ HAL_StatusTypeDef HAL_NAND_Write_Page_16b(NAND_HandleTypeDef *hnand, NAND_Addres } #else deviceaddress = NAND_DEVICE; -#endif +#endif /* FMC_Bank2_3 */ /* NAND raw address calculation */ nandaddress = ARRAY_ADDRESS(pAddress, hnand); @@ -1181,9 +1181,9 @@ HAL_StatusTypeDef HAL_NAND_Write_Page_16b(NAND_HandleTypeDef *hnand, NAND_Addres /* Calculate PageSize */ #if defined(FSMC_PCR2_PWID) - if (hnand->Init.MemoryDataWidth == FSMC_NAND_PCC_MEM_BUS_WIDTH_8) + if (hnand->Init.MemoryDataWidth == FSMC_NAND_PCC_MEM_BUS_WIDTH_8) #else - if (hnand->Init.MemoryDataWidth == FMC_NAND_PCC_MEM_BUS_WIDTH_8) + if (hnand->Init.MemoryDataWidth == FMC_NAND_PCC_MEM_BUS_WIDTH_8) #endif /* FSMC_PCR2_PWID */ { hnand->Config.PageSize = hnand->Config.PageSize / 2U; @@ -1256,8 +1256,8 @@ HAL_StatusTypeDef HAL_NAND_Write_Page_16b(NAND_HandleTypeDef *hnand, NAND_Addres * @param NumSpareAreaToRead Number of spare area to read * @retval HAL status */ -HAL_StatusTypeDef HAL_NAND_Read_SpareArea_8b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, uint8_t *pBuffer, - uint32_t NumSpareAreaToRead) +HAL_StatusTypeDef HAL_NAND_Read_SpareArea_8b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, + uint8_t *pBuffer, uint32_t NumSpareAreaToRead) { uint32_t index; uint32_t tickstart; @@ -1293,7 +1293,7 @@ HAL_StatusTypeDef HAL_NAND_Read_SpareArea_8b(NAND_HandleTypeDef *hnand, NAND_Add } #else deviceaddress = NAND_DEVICE; -#endif +#endif /* FMC_Bank2_3 */ /* NAND raw address calculation */ nandaddress = ARRAY_ADDRESS(pAddress, hnand); @@ -1432,7 +1432,7 @@ HAL_StatusTypeDef HAL_NAND_Read_SpareArea_8b(NAND_HandleTypeDef *hnand, NAND_Add * @param NumSpareAreaToRead Number of spare area to read * @retval HAL status */ -HAL_StatusTypeDef HAL_NAND_Read_SpareArea_16b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, +HAL_StatusTypeDef HAL_NAND_Read_SpareArea_16b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, uint16_t *pBuffer, uint32_t NumSpareAreaToRead) { uint32_t index; @@ -1469,7 +1469,7 @@ HAL_StatusTypeDef HAL_NAND_Read_SpareArea_16b(NAND_HandleTypeDef *hnand, NAND_Ad } #else deviceaddress = NAND_DEVICE; -#endif +#endif /* FMC_Bank2_3 */ /* NAND raw address calculation */ nandaddress = ARRAY_ADDRESS(pAddress, hnand); @@ -1608,8 +1608,8 @@ HAL_StatusTypeDef HAL_NAND_Read_SpareArea_16b(NAND_HandleTypeDef *hnand, NAND_Ad * @param NumSpareAreaTowrite number of spare areas to write to block * @retval HAL status */ -HAL_StatusTypeDef HAL_NAND_Write_SpareArea_8b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, - uint8_t *pBuffer, uint32_t NumSpareAreaTowrite) +HAL_StatusTypeDef HAL_NAND_Write_SpareArea_8b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, + const uint8_t *pBuffer, uint32_t NumSpareAreaTowrite) { uint32_t index; uint32_t tickstart; @@ -1618,7 +1618,7 @@ HAL_StatusTypeDef HAL_NAND_Write_SpareArea_8b(NAND_HandleTypeDef *hnand, NAND_Ad uint32_t nandaddress; uint32_t columnaddress; uint32_t nbspare = NumSpareAreaTowrite; - uint8_t *buff = pBuffer; + const uint8_t *buff = pBuffer; /* Check the NAND controller state */ if (hnand->State == HAL_NAND_STATE_BUSY) @@ -1645,7 +1645,7 @@ HAL_StatusTypeDef HAL_NAND_Write_SpareArea_8b(NAND_HandleTypeDef *hnand, NAND_Ad } #else deviceaddress = NAND_DEVICE; -#endif +#endif /* FMC_Bank2_3 */ /* Page address calculation */ nandaddress = ARRAY_ADDRESS(pAddress, hnand); @@ -1782,8 +1782,8 @@ HAL_StatusTypeDef HAL_NAND_Write_SpareArea_8b(NAND_HandleTypeDef *hnand, NAND_Ad * @param NumSpareAreaTowrite number of spare areas to write to block * @retval HAL status */ -HAL_StatusTypeDef HAL_NAND_Write_SpareArea_16b(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress, - uint16_t *pBuffer, uint32_t NumSpareAreaTowrite) +HAL_StatusTypeDef HAL_NAND_Write_SpareArea_16b(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress, + const uint16_t *pBuffer, uint32_t NumSpareAreaTowrite) { uint32_t index; uint32_t tickstart; @@ -1792,7 +1792,7 @@ HAL_StatusTypeDef HAL_NAND_Write_SpareArea_16b(NAND_HandleTypeDef *hnand, NAND_A uint32_t nandaddress; uint32_t columnaddress; uint32_t nbspare = NumSpareAreaTowrite; - uint16_t *buff = pBuffer; + const uint16_t *buff = pBuffer; /* Check the NAND controller state */ if (hnand->State == HAL_NAND_STATE_BUSY) @@ -1819,7 +1819,7 @@ HAL_StatusTypeDef HAL_NAND_Write_SpareArea_16b(NAND_HandleTypeDef *hnand, NAND_A } #else deviceaddress = NAND_DEVICE; -#endif +#endif /* FMC_Bank2_3 */ /* NAND raw address calculation */ nandaddress = ARRAY_ADDRESS(pAddress, hnand); @@ -1954,7 +1954,7 @@ HAL_StatusTypeDef HAL_NAND_Write_SpareArea_16b(NAND_HandleTypeDef *hnand, NAND_A * @param pAddress pointer to NAND address structure * @retval HAL status */ -HAL_StatusTypeDef HAL_NAND_Erase_Block(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress) +HAL_StatusTypeDef HAL_NAND_Erase_Block(NAND_HandleTypeDef *hnand, const NAND_AddressTypeDef *pAddress) { uint32_t deviceaddress; @@ -1983,7 +1983,7 @@ HAL_StatusTypeDef HAL_NAND_Erase_Block(NAND_HandleTypeDef *hnand, NAND_AddressTy } #else deviceaddress = NAND_DEVICE; -#endif +#endif /* FMC_Bank2_3 */ /* Send Erase block command sequence */ *(__IO uint8_t *)((uint32_t)(deviceaddress | CMD_AREA)) = NAND_CMD_ERASE0; @@ -2021,7 +2021,7 @@ HAL_StatusTypeDef HAL_NAND_Erase_Block(NAND_HandleTypeDef *hnand, NAND_AddressTy * - NAND_VALID_ADDRESS: When the new address is valid address * - NAND_INVALID_ADDRESS: When the new address is invalid address */ -uint32_t HAL_NAND_Address_Inc(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress) +uint32_t HAL_NAND_Address_Inc(const NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pAddress) { uint32_t status = NAND_VALID_ADDRESS; @@ -2052,7 +2052,7 @@ uint32_t HAL_NAND_Address_Inc(NAND_HandleTypeDef *hnand, NAND_AddressTypeDef *pA #if (USE_HAL_NAND_REGISTER_CALLBACKS == 1) /** * @brief Register a User NAND Callback - * To be used instead of the weak (surcharged) predefined callback + * To be used to override the weak predefined callback * @param hnand : NAND handle * @param CallbackId : ID of the callback to be registered * This parameter can be one of the following values: @@ -2072,9 +2072,6 @@ HAL_StatusTypeDef HAL_NAND_RegisterCallback(NAND_HandleTypeDef *hnand, HAL_NAND_ return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(hnand); - if (hnand->State == HAL_NAND_STATE_READY) { switch (CallbackId) @@ -2116,14 +2113,12 @@ HAL_StatusTypeDef HAL_NAND_RegisterCallback(NAND_HandleTypeDef *hnand, HAL_NAND_ status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hnand); return status; } /** * @brief Unregister a User NAND Callback - * NAND Callback is redirected to the weak (surcharged) predefined callback + * NAND Callback is redirected to the weak predefined callback * @param hnand : NAND handle * @param CallbackId : ID of the callback to be unregistered * This parameter can be one of the following values: @@ -2136,9 +2131,6 @@ HAL_StatusTypeDef HAL_NAND_UnRegisterCallback(NAND_HandleTypeDef *hnand, HAL_NAN { HAL_StatusTypeDef status = HAL_OK; - /* Process locked */ - __HAL_LOCK(hnand); - if (hnand->State == HAL_NAND_STATE_READY) { switch (CallbackId) @@ -2180,8 +2172,6 @@ HAL_StatusTypeDef HAL_NAND_UnRegisterCallback(NAND_HandleTypeDef *hnand, HAL_NAN status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hnand); return status; } #endif /* USE_HAL_NAND_REGISTER_CALLBACKS */ @@ -2332,7 +2322,7 @@ HAL_StatusTypeDef HAL_NAND_GetECC(NAND_HandleTypeDef *hnand, uint32_t *ECCval, * the configuration information for NAND module. * @retval HAL state */ -HAL_NAND_StateTypeDef HAL_NAND_GetState(NAND_HandleTypeDef *hnand) +HAL_NAND_StateTypeDef HAL_NAND_GetState(const NAND_HandleTypeDef *hnand) { return hnand->State; } @@ -2343,7 +2333,7 @@ HAL_NAND_StateTypeDef HAL_NAND_GetState(NAND_HandleTypeDef *hnand) * the configuration information for NAND module. * @retval NAND status */ -uint32_t HAL_NAND_Read_Status(NAND_HandleTypeDef *hnand) +uint32_t HAL_NAND_Read_Status(const NAND_HandleTypeDef *hnand) { uint32_t data; uint32_t deviceaddress; @@ -2361,7 +2351,7 @@ uint32_t HAL_NAND_Read_Status(NAND_HandleTypeDef *hnand) } #else deviceaddress = NAND_DEVICE; -#endif +#endif /* FMC_Bank2_3 */ /* Send Read status operation command */ *(__IO uint8_t *)((uint32_t)(deviceaddress | CMD_AREA)) = NAND_CMD_STATUS; diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_nor.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_nor.c index 0a820448..22366b40 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_nor.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_nor.c @@ -74,15 +74,15 @@ and a pointer to the user callback function. Use function HAL_NOR_UnRegisterCallback() to reset a callback to the default - weak (surcharged) function. It allows to reset following callbacks: + weak (overridden) function. It allows to reset following callbacks: (+) MspInitCallback : NOR MspInit. (+) MspDeInitCallback : NOR MspDeInit. This function) takes as parameters the HAL peripheral handle and the Callback ID. By default, after the HAL_NOR_Init and if the state is HAL_NOR_STATE_RESET - all callbacks are reset to the corresponding legacy weak (surcharged) functions. + all callbacks are reset to the corresponding legacy weak (overridden) functions. Exception done for MspInit and MspDeInit callbacks that are respectively - reset to the legacy weak (surcharged) functions in the HAL_NOR_Init + reset to the legacy weak (overridden) functions in the HAL_NOR_Init and HAL_NOR_DeInit only when these callbacks are null (not registered beforehand). If not, MspInit or MspDeInit are not null, the HAL_NOR_Init and HAL_NOR_DeInit keep and use the user MspInit/MspDeInit callbacks (registered beforehand) @@ -97,7 +97,7 @@ When The compilation define USE_HAL_NOR_REGISTER_CALLBACKS is set to 0 or not defined, the callback registering feature is not available - and weak (surcharged) callbacks are used. + and weak (overridden) callbacks are used. @endverbatim ****************************************************************************** @@ -106,7 +106,7 @@ /* Includes ------------------------------------------------------------------*/ #include "stm32f4xx_hal.h" -#if defined(FMC_Bank1) || defined(FSMC_Bank1) +#if defined(FMC_Bank1) || defined(FSMC_Bank1) /** @addtogroup STM32F4xx_HAL_Driver * @{ @@ -127,6 +127,11 @@ */ /* Constants to define address to set to write a command */ +#define NOR_CMD_ADDRESS_FIRST_BYTE (uint16_t)0x0AAA +#define NOR_CMD_ADDRESS_FIRST_CFI_BYTE (uint16_t)0x00AA +#define NOR_CMD_ADDRESS_SECOND_BYTE (uint16_t)0x0555 +#define NOR_CMD_ADDRESS_THIRD_BYTE (uint16_t)0x0AAA + #define NOR_CMD_ADDRESS_FIRST (uint16_t)0x0555 #define NOR_CMD_ADDRESS_FIRST_CFI (uint16_t)0x0055 #define NOR_CMD_ADDRESS_SECOND (uint16_t)0x02AA @@ -264,7 +269,8 @@ HAL_StatusTypeDef HAL_NOR_Init(NOR_HandleTypeDef *hnor, FMC_NORSRAM_TimingTypeDe (void)FMC_NORSRAM_Timing_Init(hnor->Instance, Timing, hnor->Init.NSBank); /* Initialize NOR extended mode timing Interface */ - (void)FMC_NORSRAM_Extended_Timing_Init(hnor->Extended, ExtTiming, hnor->Init.NSBank, hnor->Init.ExtendedMode); + (void)FMC_NORSRAM_Extended_Timing_Init(hnor->Extended, ExtTiming, + hnor->Init.NSBank, hnor->Init.ExtendedMode); /* Enable the NORSRAM device */ __FMC_NORSRAM_ENABLE(hnor->Instance, hnor->Init.NSBank); @@ -310,7 +316,16 @@ HAL_StatusTypeDef HAL_NOR_Init(NOR_HandleTypeDef *hnor, FMC_NORSRAM_TimingTypeDe else { /* Get the value of the command set */ - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST_CFI), NOR_CMD_DATA_CFI); + if (uwNORMemoryDataWidth == NOR_MEMORY_8B) + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST_CFI_BYTE), + NOR_CMD_DATA_CFI); + } + else + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST_CFI), NOR_CMD_DATA_CFI); + } + hnor->CommandSet = *(__IO uint16_t *) NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_ADDRESS_COMMAND_SET); status = HAL_NOR_ReturnToReadMode(hnor); @@ -472,9 +487,22 @@ HAL_StatusTypeDef HAL_NOR_Read_ID(NOR_HandleTypeDef *hnor, NOR_IDTypeDef *pNOR_I /* Send read ID command */ if (hnor->CommandSet == NOR_AMD_FUJITSU_COMMAND_SET) { - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST), NOR_CMD_DATA_FIRST); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND), NOR_CMD_DATA_SECOND); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD), NOR_CMD_DATA_AUTO_SELECT); + if (uwNORMemoryDataWidth == NOR_MEMORY_8B) + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST_BYTE), + NOR_CMD_DATA_FIRST); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND_BYTE), + NOR_CMD_DATA_SECOND); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD_BYTE), + NOR_CMD_DATA_AUTO_SELECT); + } + else + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST), NOR_CMD_DATA_FIRST); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND), NOR_CMD_DATA_SECOND); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD), + NOR_CMD_DATA_AUTO_SELECT); + } } else if (hnor->CommandSet == NOR_INTEL_SHARP_EXT_COMMAND_SET) { @@ -641,9 +669,22 @@ HAL_StatusTypeDef HAL_NOR_Read(NOR_HandleTypeDef *hnor, uint32_t *pAddress, uint /* Send read data command */ if (hnor->CommandSet == NOR_AMD_FUJITSU_COMMAND_SET) { - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST), NOR_CMD_DATA_FIRST); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND), NOR_CMD_DATA_SECOND); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD), NOR_CMD_DATA_READ_RESET); + if (uwNORMemoryDataWidth == NOR_MEMORY_8B) + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST_BYTE), + NOR_CMD_DATA_FIRST); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND_BYTE), + NOR_CMD_DATA_SECOND); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD_BYTE), + NOR_CMD_DATA_READ_RESET); + } + else + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST), NOR_CMD_DATA_FIRST); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND), NOR_CMD_DATA_SECOND); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD), + NOR_CMD_DATA_READ_RESET); + } } else if (hnor->CommandSet == NOR_INTEL_SHARP_EXT_COMMAND_SET) { @@ -722,9 +763,21 @@ HAL_StatusTypeDef HAL_NOR_Program(NOR_HandleTypeDef *hnor, uint32_t *pAddress, u /* Send program data command */ if (hnor->CommandSet == NOR_AMD_FUJITSU_COMMAND_SET) { - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST), NOR_CMD_DATA_FIRST); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND), NOR_CMD_DATA_SECOND); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD), NOR_CMD_DATA_PROGRAM); + if (uwNORMemoryDataWidth == NOR_MEMORY_8B) + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST_BYTE), + NOR_CMD_DATA_FIRST); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND_BYTE), + NOR_CMD_DATA_SECOND); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD_BYTE), + NOR_CMD_DATA_PROGRAM); + } + else + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST), NOR_CMD_DATA_FIRST); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND), NOR_CMD_DATA_SECOND); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD), NOR_CMD_DATA_PROGRAM); + } } else if (hnor->CommandSet == NOR_INTEL_SHARP_EXT_COMMAND_SET) { @@ -814,9 +867,22 @@ HAL_StatusTypeDef HAL_NOR_ReadBuffer(NOR_HandleTypeDef *hnor, uint32_t uwAddress /* Send read data command */ if (hnor->CommandSet == NOR_AMD_FUJITSU_COMMAND_SET) { - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST), NOR_CMD_DATA_FIRST); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND), NOR_CMD_DATA_SECOND); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD), NOR_CMD_DATA_READ_RESET); + if (uwNORMemoryDataWidth == NOR_MEMORY_8B) + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST_BYTE), + NOR_CMD_DATA_FIRST); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND_BYTE), + NOR_CMD_DATA_SECOND); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD_BYTE), + NOR_CMD_DATA_READ_RESET); + } + else + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST), NOR_CMD_DATA_FIRST); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND), NOR_CMD_DATA_SECOND); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD), + NOR_CMD_DATA_READ_RESET); + } } else if (hnor->CommandSet == NOR_INTEL_SHARP_EXT_COMMAND_SET) { @@ -909,10 +975,20 @@ HAL_StatusTypeDef HAL_NOR_ProgramBuffer(NOR_HandleTypeDef *hnor, uint32_t uwAddr if (hnor->CommandSet == NOR_AMD_FUJITSU_COMMAND_SET) { - /* Issue unlock command sequence */ - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST), NOR_CMD_DATA_FIRST); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND), NOR_CMD_DATA_SECOND); - + if (uwNORMemoryDataWidth == NOR_MEMORY_8B) + { + /* Issue unlock command sequence */ + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST_BYTE), + NOR_CMD_DATA_FIRST); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND_BYTE), + NOR_CMD_DATA_SECOND); + } + else + { + /* Issue unlock command sequence */ + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST), NOR_CMD_DATA_FIRST); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND), NOR_CMD_DATA_SECOND); + } /* Write Buffer Load Command */ NOR_WRITE((deviceaddress + uwAddress), NOR_CMD_DATA_BUFFER_AND_PROG); NOR_WRITE((deviceaddress + uwAddress), (uint16_t)(uwBufferSize - 1U)); @@ -1012,14 +1088,26 @@ HAL_StatusTypeDef HAL_NOR_Erase_Block(NOR_HandleTypeDef *hnor, uint32_t BlockAdd /* Send block erase command sequence */ if (hnor->CommandSet == NOR_AMD_FUJITSU_COMMAND_SET) { - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST), NOR_CMD_DATA_FIRST); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND), NOR_CMD_DATA_SECOND); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD), - NOR_CMD_DATA_CHIP_BLOCK_ERASE_THIRD); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FOURTH), - NOR_CMD_DATA_CHIP_BLOCK_ERASE_FOURTH); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIFTH), - NOR_CMD_DATA_CHIP_BLOCK_ERASE_FIFTH); + if (uwNORMemoryDataWidth == NOR_MEMORY_8B) + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST_BYTE), + NOR_CMD_DATA_FIRST); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND_BYTE), + NOR_CMD_DATA_SECOND); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD_BYTE), + NOR_CMD_DATA_CHIP_BLOCK_ERASE_THIRD); + } + else + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST), NOR_CMD_DATA_FIRST); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND), NOR_CMD_DATA_SECOND); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD), + NOR_CMD_DATA_CHIP_BLOCK_ERASE_THIRD); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FOURTH), + NOR_CMD_DATA_CHIP_BLOCK_ERASE_FOURTH); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIFTH), + NOR_CMD_DATA_CHIP_BLOCK_ERASE_FIFTH); + } NOR_WRITE((uint32_t)(BlockAddress + Address), NOR_CMD_DATA_BLOCK_ERASE); } else if (hnor->CommandSet == NOR_INTEL_SHARP_EXT_COMMAND_SET) @@ -1097,15 +1185,28 @@ HAL_StatusTypeDef HAL_NOR_Erase_Chip(NOR_HandleTypeDef *hnor, uint32_t Address) /* Send NOR chip erase command sequence */ if (hnor->CommandSet == NOR_AMD_FUJITSU_COMMAND_SET) { - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST), NOR_CMD_DATA_FIRST); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND), NOR_CMD_DATA_SECOND); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD), - NOR_CMD_DATA_CHIP_BLOCK_ERASE_THIRD); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FOURTH), - NOR_CMD_DATA_CHIP_BLOCK_ERASE_FOURTH); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIFTH), - NOR_CMD_DATA_CHIP_BLOCK_ERASE_FIFTH); - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SIXTH), NOR_CMD_DATA_CHIP_ERASE); + if (uwNORMemoryDataWidth == NOR_MEMORY_8B) + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST_BYTE), + NOR_CMD_DATA_FIRST); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND_BYTE), + NOR_CMD_DATA_SECOND); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD_BYTE), + NOR_CMD_DATA_CHIP_BLOCK_ERASE_THIRD); + } + else + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST), NOR_CMD_DATA_FIRST); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SECOND), NOR_CMD_DATA_SECOND); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_THIRD), + NOR_CMD_DATA_CHIP_BLOCK_ERASE_THIRD); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FOURTH), + NOR_CMD_DATA_CHIP_BLOCK_ERASE_FOURTH); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIFTH), + NOR_CMD_DATA_CHIP_BLOCK_ERASE_FIFTH); + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_SIXTH), + NOR_CMD_DATA_CHIP_ERASE); + } } else { @@ -1176,8 +1277,15 @@ HAL_StatusTypeDef HAL_NOR_Read_CFI(NOR_HandleTypeDef *hnor, NOR_CFITypeDef *pNOR } /* Send read CFI query command */ - NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST_CFI), NOR_CMD_DATA_CFI); - + if (uwNORMemoryDataWidth == NOR_MEMORY_8B) + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST_CFI_BYTE), + NOR_CMD_DATA_CFI); + } + else + { + NOR_WRITE(NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, NOR_CMD_ADDRESS_FIRST_CFI), NOR_CMD_DATA_CFI); + } /* read the NOR CFI information */ pNOR_CFI->CFI_1 = *(__IO uint16_t *) NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, CFI1_ADDRESS); pNOR_CFI->CFI_2 = *(__IO uint16_t *) NOR_ADDR_SHIFT(deviceaddress, uwNORMemoryDataWidth, CFI2_ADDRESS); @@ -1201,7 +1309,7 @@ HAL_StatusTypeDef HAL_NOR_Read_CFI(NOR_HandleTypeDef *hnor, NOR_CFITypeDef *pNOR #if (USE_HAL_NOR_REGISTER_CALLBACKS == 1) /** * @brief Register a User NOR Callback - * To be used instead of the weak (surcharged) predefined callback + * To be used to override the weak predefined callback * @param hnor : NOR handle * @param CallbackId : ID of the callback to be registered * This parameter can be one of the following values: @@ -1221,9 +1329,6 @@ HAL_StatusTypeDef HAL_NOR_RegisterCallback(NOR_HandleTypeDef *hnor, HAL_NOR_Call return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(hnor); - state = hnor->State; if ((state == HAL_NOR_STATE_READY) || (state == HAL_NOR_STATE_RESET) || (state == HAL_NOR_STATE_PROTECTED)) { @@ -1247,14 +1352,12 @@ HAL_StatusTypeDef HAL_NOR_RegisterCallback(NOR_HandleTypeDef *hnor, HAL_NOR_Call status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hnor); return status; } /** * @brief Unregister a User NOR Callback - * NOR Callback is redirected to the weak (surcharged) predefined callback + * NOR Callback is redirected to the weak predefined callback * @param hnor : NOR handle * @param CallbackId : ID of the callback to be unregistered * This parameter can be one of the following values: @@ -1267,9 +1370,6 @@ HAL_StatusTypeDef HAL_NOR_UnRegisterCallback(NOR_HandleTypeDef *hnor, HAL_NOR_Ca HAL_StatusTypeDef status = HAL_OK; HAL_NOR_StateTypeDef state; - /* Process locked */ - __HAL_LOCK(hnor); - state = hnor->State; if ((state == HAL_NOR_STATE_READY) || (state == HAL_NOR_STATE_RESET) || (state == HAL_NOR_STATE_PROTECTED)) { @@ -1293,8 +1393,6 @@ HAL_StatusTypeDef HAL_NOR_UnRegisterCallback(NOR_HandleTypeDef *hnor, HAL_NOR_Ca status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hnor); return status; } #endif /* (USE_HAL_NOR_REGISTER_CALLBACKS) */ @@ -1411,7 +1509,7 @@ HAL_StatusTypeDef HAL_NOR_WriteOperation_Disable(NOR_HandleTypeDef *hnor) * the configuration information for NOR module. * @retval NOR controller state */ -HAL_NOR_StateTypeDef HAL_NOR_GetState(NOR_HandleTypeDef *hnor) +HAL_NOR_StateTypeDef HAL_NOR_GetState(const NOR_HandleTypeDef *hnor) { return hnor->State; } diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_pcd.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_pcd.c index 7e46592b..fa50ea18 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_pcd.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_pcd.c @@ -122,7 +122,9 @@ static HAL_StatusTypeDef PCD_EP_OutSetupPacket_int(PCD_HandleTypeDef *hpcd, uint */ HAL_StatusTypeDef HAL_PCD_Init(PCD_HandleTypeDef *hpcd) { - USB_OTG_GlobalTypeDef *USBx; +#if defined (USB_OTG_FS) + const USB_OTG_GlobalTypeDef *USBx; +#endif /* defined (USB_OTG_FS) */ uint8_t i; /* Check the PCD handle allocation */ @@ -134,7 +136,9 @@ HAL_StatusTypeDef HAL_PCD_Init(PCD_HandleTypeDef *hpcd) /* Check the parameters */ assert_param(IS_PCD_ALL_INSTANCE(hpcd->Instance)); +#if defined (USB_OTG_FS) USBx = hpcd->Instance; +#endif /* defined (USB_OTG_FS) */ if (hpcd->State == HAL_PCD_STATE_RESET) { @@ -171,11 +175,13 @@ HAL_StatusTypeDef HAL_PCD_Init(PCD_HandleTypeDef *hpcd) hpcd->State = HAL_PCD_STATE_BUSY; +#if defined (USB_OTG_FS) /* Disable DMA mode for FS instance */ - if ((USBx->CID & (0x1U << 8)) == 0U) + if (USBx == USB_OTG_FS) { hpcd->Init.dma_enable = 0U; } +#endif /* defined (USB_OTG_FS) */ /* Disable the Interrupts */ __HAL_PCD_DISABLE(hpcd); @@ -187,8 +193,12 @@ HAL_StatusTypeDef HAL_PCD_Init(PCD_HandleTypeDef *hpcd) return HAL_ERROR; } - /* Force Device Mode*/ - (void)USB_SetCurrentMode(hpcd->Instance, USB_DEVICE_MODE); + /* Force Device Mode */ + if (USB_SetCurrentMode(hpcd->Instance, USB_DEVICE_MODE) != HAL_OK) + { + hpcd->State = HAL_PCD_STATE_ERROR; + return HAL_ERROR; + } /* Init endpoints structures */ for (i = 0U; i < hpcd->Init.dev_endpoints; i++) @@ -224,13 +234,17 @@ HAL_StatusTypeDef HAL_PCD_Init(PCD_HandleTypeDef *hpcd) hpcd->USB_Address = 0U; hpcd->State = HAL_PCD_STATE_READY; -#if defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) +#if defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) \ + || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) \ + || defined(STM32F423xx) /* Activate LPM */ if (hpcd->Init.lpm_enable == 1U) { (void)HAL_PCDEx_ActivateLPM(hpcd); } -#endif /* defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) */ +#endif /* defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || + defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || + defined(STM32F423xx) */ (void)USB_DevDisconnect(hpcd->Instance); return HAL_OK; @@ -318,7 +332,7 @@ __weak void HAL_PCD_MspDeInit(PCD_HandleTypeDef *hpcd) * @arg @ref HAL_PCD_SUSPEND_CB_ID USB PCD Suspend callback ID * @arg @ref HAL_PCD_RESUME_CB_ID USB PCD Resume callback ID * @arg @ref HAL_PCD_CONNECT_CB_ID USB PCD Connect callback ID - * @arg @ref HAL_PCD_DISCONNECT_CB_ID OTG PCD Disconnect callback ID + * @arg @ref HAL_PCD_DISCONNECT_CB_ID USB PCD Disconnect callback ID * @arg @ref HAL_PCD_MSPINIT_CB_ID MspDeInit callback ID * @arg @ref HAL_PCD_MSPDEINIT_CB_ID MspDeInit callback ID * @param pCallback pointer to the Callback function @@ -432,7 +446,7 @@ HAL_StatusTypeDef HAL_PCD_RegisterCallback(PCD_HandleTypeDef *hpcd, * @arg @ref HAL_PCD_SUSPEND_CB_ID USB PCD Suspend callback ID * @arg @ref HAL_PCD_RESUME_CB_ID USB PCD Resume callback ID * @arg @ref HAL_PCD_CONNECT_CB_ID USB PCD Connect callback ID - * @arg @ref HAL_PCD_DISCONNECT_CB_ID OTG PCD Disconnect callback ID + * @arg @ref HAL_PCD_DISCONNECT_CB_ID USB PCD Disconnect callback ID * @arg @ref HAL_PCD_MSPINIT_CB_ID MspDeInit callback ID * @arg @ref HAL_PCD_MSPDEINIT_CB_ID MspDeInit callback ID * @retval HAL status @@ -1004,8 +1018,8 @@ HAL_StatusTypeDef HAL_PCD_Start(PCD_HandleTypeDef *hpcd) __HAL_LOCK(hpcd); - if ((hpcd->Init.battery_charging_enable == 1U) && - (hpcd->Init.phy_itface != USB_OTG_ULPI_PHY)) + if (((USBx->GUSBCFG & USB_OTG_GUSBCFG_PHYSEL) != 0U) && + (hpcd->Init.battery_charging_enable == 1U)) { /* Enable USB Transceiver */ USBx->GCCFG |= USB_OTG_GCCFG_PWRDWN; @@ -1033,8 +1047,8 @@ HAL_StatusTypeDef HAL_PCD_Stop(PCD_HandleTypeDef *hpcd) (void)USB_FlushTxFifo(hpcd->Instance, 0x10U); - if ((hpcd->Init.battery_charging_enable == 1U) && - (hpcd->Init.phy_itface != USB_OTG_ULPI_PHY)) + if (((USBx->GUSBCFG & USB_OTG_GUSBCFG_PHYSEL) != 0U) && + (hpcd->Init.battery_charging_enable == 1U)) { /* Disable USB Transceiver */ USBx->GCCFG &= ~(USB_OTG_GCCFG_PWRDWN); @@ -1306,7 +1320,9 @@ void HAL_PCD_IRQHandler(PCD_HandleTypeDef *hpcd) } __HAL_PCD_CLEAR_FLAG(hpcd, USB_OTG_GINTSTS_USBSUSP); } -#if defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) +#if defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) \ + || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) \ + || defined(STM32F423xx) /* Handle LPM Interrupt */ if (__HAL_PCD_GET_FLAG(hpcd, USB_OTG_GINTSTS_LPMINT)) { @@ -1332,7 +1348,9 @@ void HAL_PCD_IRQHandler(PCD_HandleTypeDef *hpcd) #endif /* USE_HAL_PCD_REGISTER_CALLBACKS */ } } -#endif /* defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) */ +#endif /* defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || + defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || + defined(STM32F423xx) */ /* Handle Reset Interrupt */ if (__HAL_PCD_GET_FLAG(hpcd, USB_OTG_GINTSTS_USBRST)) { @@ -1513,16 +1531,17 @@ void HAL_PCD_IRQHandler(PCD_HandleTypeDef *hpcd) */ void HAL_PCD_WKUP_IRQHandler(PCD_HandleTypeDef *hpcd) { +#if defined (USB_OTG_FS) USB_OTG_GlobalTypeDef *USBx; - USBx = hpcd->Instance; - if ((USBx->CID & (0x1U << 8)) == 0U) + if (USBx == USB_OTG_FS) { /* Clear EXTI pending Bit */ __HAL_USB_OTG_FS_WAKEUP_EXTI_CLEAR_FLAG(); } else +#endif /* defined (USB_OTG_FS) */ { /* Clear EXTI pending Bit */ __HAL_USB_OTG_HS_WAKEUP_EXTI_CLEAR_FLAG(); @@ -1733,8 +1752,8 @@ HAL_StatusTypeDef HAL_PCD_DevConnect(PCD_HandleTypeDef *hpcd) __HAL_LOCK(hpcd); - if ((hpcd->Init.battery_charging_enable == 1U) && - (hpcd->Init.phy_itface != USB_OTG_ULPI_PHY)) + if (((USBx->GUSBCFG & USB_OTG_GUSBCFG_PHYSEL) != 0U) && + (hpcd->Init.battery_charging_enable == 1U)) { /* Enable USB Transceiver */ USBx->GCCFG |= USB_OTG_GCCFG_PWRDWN; @@ -1757,8 +1776,8 @@ HAL_StatusTypeDef HAL_PCD_DevDisconnect(PCD_HandleTypeDef *hpcd) __HAL_LOCK(hpcd); (void)USB_DevDisconnect(hpcd->Instance); - if ((hpcd->Init.battery_charging_enable == 1U) && - (hpcd->Init.phy_itface != USB_OTG_ULPI_PHY)) + if (((USBx->GUSBCFG & USB_OTG_GUSBCFG_PHYSEL) != 0U) && + (hpcd->Init.battery_charging_enable == 1U)) { /* Disable USB Transceiver */ USBx->GCCFG &= ~(USB_OTG_GCCFG_PWRDWN); @@ -1818,6 +1837,7 @@ HAL_StatusTypeDef HAL_PCD_EP_Open(PCD_HandleTypeDef *hpcd, uint8_t ep_addr, /* Assign a Tx FIFO */ ep->tx_fifo_num = ep->num; } + /* Set initial data PID. */ if (ep_type == EP_TYPE_BULK) { @@ -1851,7 +1871,7 @@ HAL_StatusTypeDef HAL_PCD_EP_Close(PCD_HandleTypeDef *hpcd, uint8_t ep_addr) ep = &hpcd->OUT_ep[ep_addr & EP_ADDR_MSK]; ep->is_in = 0U; } - ep->num = ep_addr & EP_ADDR_MSK; + ep->num = ep_addr & EP_ADDR_MSK; __HAL_LOCK(hpcd); (void)USB_DeactivateEndpoint(hpcd->Instance, ep); @@ -1886,14 +1906,7 @@ HAL_StatusTypeDef HAL_PCD_EP_Receive(PCD_HandleTypeDef *hpcd, uint8_t ep_addr, u ep->dma_addr = (uint32_t)pBuf; } - if ((ep_addr & EP_ADDR_MSK) == 0U) - { - (void)USB_EP0StartXfer(hpcd->Instance, ep, (uint8_t)hpcd->Init.dma_enable); - } - else - { - (void)USB_EPStartXfer(hpcd->Instance, ep, (uint8_t)hpcd->Init.dma_enable); - } + (void)USB_EPStartXfer(hpcd->Instance, ep, (uint8_t)hpcd->Init.dma_enable); return HAL_OK; } @@ -1904,7 +1917,7 @@ HAL_StatusTypeDef HAL_PCD_EP_Receive(PCD_HandleTypeDef *hpcd, uint8_t ep_addr, u * @param ep_addr endpoint address * @retval Data Size */ -uint32_t HAL_PCD_EP_GetRxCount(PCD_HandleTypeDef *hpcd, uint8_t ep_addr) +uint32_t HAL_PCD_EP_GetRxCount(PCD_HandleTypeDef const *hpcd, uint8_t ep_addr) { return hpcd->OUT_ep[ep_addr & EP_ADDR_MSK].xfer_count; } @@ -1934,14 +1947,7 @@ HAL_StatusTypeDef HAL_PCD_EP_Transmit(PCD_HandleTypeDef *hpcd, uint8_t ep_addr, ep->dma_addr = (uint32_t)pBuf; } - if ((ep_addr & EP_ADDR_MSK) == 0U) - { - (void)USB_EP0StartXfer(hpcd->Instance, ep, (uint8_t)hpcd->Init.dma_enable); - } - else - { - (void)USB_EPStartXfer(hpcd->Instance, ep, (uint8_t)hpcd->Init.dma_enable); - } + (void)USB_EPStartXfer(hpcd->Instance, ep, (uint8_t)hpcd->Init.dma_enable); return HAL_OK; } @@ -2119,20 +2125,21 @@ HAL_StatusTypeDef HAL_PCD_DeActivateRemoteWakeup(PCD_HandleTypeDef *hpcd) * @param hpcd PCD handle * @retval HAL state */ -PCD_StateTypeDef HAL_PCD_GetState(PCD_HandleTypeDef *hpcd) +PCD_StateTypeDef HAL_PCD_GetState(PCD_HandleTypeDef const *hpcd) { return hpcd->State; } +#if defined (USB_OTG_FS) || defined (USB_OTG_HS) /** * @brief Set the USB Device high speed test mode. * @param hpcd PCD handle * @param testmode USB Device high speed test mode * @retval HAL status */ -HAL_StatusTypeDef HAL_PCD_SetTestMode(PCD_HandleTypeDef *hpcd, uint8_t testmode) +HAL_StatusTypeDef HAL_PCD_SetTestMode(const PCD_HandleTypeDef *hpcd, uint8_t testmode) { - USB_OTG_GlobalTypeDef *USBx = hpcd->Instance; + const USB_OTG_GlobalTypeDef *USBx = hpcd->Instance; uint32_t USBx_BASE = (uint32_t)USBx; switch (testmode) @@ -2151,6 +2158,7 @@ HAL_StatusTypeDef HAL_PCD_SetTestMode(PCD_HandleTypeDef *hpcd, uint8_t testmode) return HAL_OK; } +#endif /* defined (USB_OTG_FS) || defined (USB_OTG_HS) */ /** * @} */ @@ -2233,9 +2241,9 @@ static HAL_StatusTypeDef PCD_WriteEmptyTxFifo(PCD_HandleTypeDef *hpcd, uint32_t static HAL_StatusTypeDef PCD_EP_OutXfrComplete_int(PCD_HandleTypeDef *hpcd, uint32_t epnum) { USB_OTG_EPTypeDef *ep; - USB_OTG_GlobalTypeDef *USBx = hpcd->Instance; + const USB_OTG_GlobalTypeDef *USBx = hpcd->Instance; uint32_t USBx_BASE = (uint32_t)USBx; - uint32_t gSNPSiD = *(__IO uint32_t *)(&USBx->CID + 0x1U); + uint32_t gSNPSiD = *(__IO const uint32_t *)(&USBx->CID + 0x1U); uint32_t DoepintReg = USBx_OUTEP(epnum)->DOEPINT; if (hpcd->Init.dma_enable == 1U) @@ -2344,9 +2352,9 @@ static HAL_StatusTypeDef PCD_EP_OutXfrComplete_int(PCD_HandleTypeDef *hpcd, uint */ static HAL_StatusTypeDef PCD_EP_OutSetupPacket_int(PCD_HandleTypeDef *hpcd, uint32_t epnum) { - USB_OTG_GlobalTypeDef *USBx = hpcd->Instance; + const USB_OTG_GlobalTypeDef *USBx = hpcd->Instance; uint32_t USBx_BASE = (uint32_t)USBx; - uint32_t gSNPSiD = *(__IO uint32_t *)(&USBx->CID + 0x1U); + uint32_t gSNPSiD = *(__IO const uint32_t *)(&USBx->CID + 0x1U); uint32_t DoepintReg = USBx_OUTEP(epnum)->DOEPINT; if ((gSNPSiD > USB_OTG_CORE_ID_300A) && diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_pcd_ex.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_pcd_ex.c index 292faf13..b66be6ac 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_pcd_ex.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_pcd_ex.c @@ -115,7 +115,9 @@ HAL_StatusTypeDef HAL_PCDEx_SetRxFiFo(PCD_HandleTypeDef *hpcd, uint16_t size) return HAL_OK; } -#if defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) +#if defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) \ + || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) \ + || defined(STM32F423xx) /** * @brief Activate LPM feature. * @param hpcd PCD handle @@ -148,8 +150,11 @@ HAL_StatusTypeDef HAL_PCDEx_DeActivateLPM(PCD_HandleTypeDef *hpcd) return HAL_OK; } -#endif /* defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) */ -#if defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) +#endif /* defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || + defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || + defined(STM32F423xx) */ +#if defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) \ + || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) /** * @brief Handle BatteryCharging Process. * @param hpcd PCD handle @@ -178,9 +183,9 @@ void HAL_PCDEx_BCD_VBUSDetect(PCD_HandleTypeDef *hpcd) /* Primary detection: checks if connected to Standard Downstream Port (without charging capability) */ - USBx->GCCFG &= ~ USB_OTG_GCCFG_DCDEN; + USBx->GCCFG &= ~USB_OTG_GCCFG_DCDEN; HAL_Delay(50U); - USBx->GCCFG |= USB_OTG_GCCFG_PDEN; + USBx->GCCFG |= USB_OTG_GCCFG_PDEN; HAL_Delay(50U); if ((USBx->GCCFG & USB_OTG_GCCFG_PDET) == 0U) @@ -196,9 +201,9 @@ void HAL_PCDEx_BCD_VBUSDetect(PCD_HandleTypeDef *hpcd) { /* start secondary detection to check connection to Charging Downstream Port or Dedicated Charging Port */ - USBx->GCCFG &= ~ USB_OTG_GCCFG_PDEN; + USBx->GCCFG &= ~(USB_OTG_GCCFG_PDEN); HAL_Delay(50U); - USBx->GCCFG |= USB_OTG_GCCFG_SDEN; + USBx->GCCFG |= USB_OTG_GCCFG_SDEN; HAL_Delay(50U); if ((USBx->GCCFG & USB_OTG_GCCFG_SDET) == USB_OTG_GCCFG_SDET) @@ -285,7 +290,8 @@ HAL_StatusTypeDef HAL_PCDEx_DeActivateBCD(PCD_HandleTypeDef *hpcd) return HAL_OK; } -#endif /* defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) */ +#endif /* defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || + defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) */ #endif /* defined (USB_OTG_FS) || defined (USB_OTG_HS) */ /** diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_pwr.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_pwr.c index b9f7cc20..6b298881 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_pwr.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_pwr.c @@ -179,10 +179,12 @@ void HAL_PWR_DisableBkUpAccess(void) ================== [..] (+) Entry: - The Sleep mode is entered by using the HAL_PWR_EnterSLEEPMode(PWR_MAINREGULATOR_ON, PWR_SLEEPENTRY_WFI) + The Sleep mode is entered by using the HAL_PWR_EnterSLEEPMode(Regulator, SLEEPEntry) functions with (++) PWR_SLEEPENTRY_WFI: enter SLEEP mode with WFI instruction (++) PWR_SLEEPENTRY_WFE: enter SLEEP mode with WFE instruction + (++) PWR_SLEEPENTRY_WFE_NO_EVT_CLEAR: Enter SLEEP mode with WFE instruction and + no clear of pending event before. -@@- The Regulator parameter is not used for the STM32F4 family and is kept as parameter just to maintain compatibility with the @@ -204,10 +206,17 @@ void HAL_PWR_DisableBkUpAccess(void) the HAL_PWREx_DisableFlashPowerDown() function. (+) Entry: - The Stop mode is entered using the HAL_PWR_EnterSTOPMode(PWR_MAINREGULATOR_ON) + The Stop mode is entered using the HAL_PWR_EnterSTOPMode(Regulator, STOPEntry) function with: - (++) Main regulator ON. - (++) Low Power regulator ON. + (++) Regulator: + (+++) Main regulator ON. + (+++) Low Power regulator ON. + (++) STOPEntry: + (+++) PWR_STOPENTRY_WFI : Enter STOP mode with WFI instruction. + (+++) PWR_STOPENTRY_WFE : Enter STOP mode with WFE instruction and + clear of pending events before. + (+++) PWR_STOPENTRY_WFE_NO_EVT_CLEAR : Enter STOP mode with WFE instruction and + no clear of pending event before. (+) Exit: Any EXTI Line (Internal or External) configured in Interrupt/Event mode. @@ -372,12 +381,18 @@ void HAL_PWR_DisableWakeUpPin(uint32_t WakeUpPinx) * just to maintain compatibility with the lower power families. * @param SLEEPEntry Specifies if SLEEP mode in entered with WFI or WFE instruction. * This parameter can be one of the following values: - * @arg PWR_SLEEPENTRY_WFI: enter SLEEP mode with WFI instruction - * @arg PWR_SLEEPENTRY_WFE: enter SLEEP mode with WFE instruction + * @arg PWR_SLEEPENTRY_WFI : Enter SLEEP mode with WFI instruction + * @arg PWR_SLEEPENTRY_WFE : Enter SLEEP mode with WFE instruction and + * clear of pending events before. + * @arg PWR_SLEEPENTRY_WFE_NO_EVT_CLEAR : Enter SLEEP mode with WFE instruction and + * no clear of pending event before. * @retval None */ void HAL_PWR_EnterSLEEPMode(uint32_t Regulator, uint8_t SLEEPEntry) { + /* Prevent unused argument(s) compilation warning */ + UNUSED(Regulator); + /* Check the parameters */ assert_param(IS_PWR_REGULATOR(Regulator)); assert_param(IS_PWR_SLEEP_ENTRY(SLEEPEntry)); @@ -393,9 +408,14 @@ void HAL_PWR_EnterSLEEPMode(uint32_t Regulator, uint8_t SLEEPEntry) } else { + if(SLEEPEntry != PWR_SLEEPENTRY_WFE_NO_EVT_CLEAR) + { + /* Clear all pending event */ + __SEV(); + __WFE(); + } + /* Request Wait For Event */ - __SEV(); - __WFE(); __WFE(); } } @@ -415,8 +435,11 @@ void HAL_PWR_EnterSLEEPMode(uint32_t Regulator, uint8_t SLEEPEntry) * @arg PWR_LOWPOWERREGULATOR_ON: Stop mode with low power regulator ON * @param STOPEntry Specifies if Stop mode in entered with WFI or WFE instruction. * This parameter can be one of the following values: - * @arg PWR_STOPENTRY_WFI: Enter Stop mode with WFI instruction - * @arg PWR_STOPENTRY_WFE: Enter Stop mode with WFE instruction + * @arg PWR_STOPENTRY_WFI : Enter Stop mode with WFI instruction + * @arg PWR_STOPENTRY_WFE : Enter Stop mode with WFE instruction and + * clear of pending events before. + * @arg PWR_STOPENTRY_WFE_NO_EVT_CLEAR : Enter STOP mode with WFE instruction and + * no clear of pending event before. * @retval None */ void HAL_PWR_EnterSTOPMode(uint32_t Regulator, uint8_t STOPEntry) @@ -439,9 +462,13 @@ void HAL_PWR_EnterSTOPMode(uint32_t Regulator, uint8_t STOPEntry) } else { + if(STOPEntry != PWR_STOPENTRY_WFE_NO_EVT_CLEAR) + { + /* Clear all pending event */ + __SEV(); + __WFE(); + } /* Request Wait For Event */ - __SEV(); - __WFE(); __WFE(); } /* Reset SLEEPDEEP bit of Cortex System Control Register */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_qspi.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_qspi.c index 74ffe164..508f0b6d 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_qspi.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_qspi.c @@ -162,7 +162,7 @@ and a pointer to the user callback function. Use function HAL_QSPI_UnRegisterCallback() to reset a callback to the default - weak (surcharged) function. It allows to reset following callbacks: + weak (overridden) function. It allows to reset following callbacks: (+) ErrorCallback : callback when error occurs. (+) AbortCpltCallback : callback when abort is completed. (+) FifoThresholdCallback : callback when the fifo threshold is reached. @@ -178,9 +178,9 @@ This function) takes as parameters the HAL peripheral handle and the Callback ID. By default, after the HAL_QSPI_Init and if the state is HAL_QSPI_STATE_RESET - all callbacks are reset to the corresponding legacy weak (surcharged) functions. + all callbacks are reset to the corresponding legacy weak (overridden) functions. Exception done for MspInit and MspDeInit callbacks that are respectively - reset to the legacy weak (surcharged) functions in the HAL_QSPI_Init + reset to the legacy weak (overridden) functions in the HAL_QSPI_Init and HAL_QSPI_DeInit only when these callbacks are null (not registered beforehand). If not, MspInit or MspDeInit are not null, the HAL_QSPI_Init and HAL_QSPI_DeInit keep and use the user MspInit/MspDeInit callbacks (registered beforehand) @@ -195,7 +195,7 @@ When The compilation define USE_HAL_QSPI_REGISTER_CALLBACKS is set to 0 or not defined, the callback registering feature is not available - and weak (surcharged) callbacks are used. + and weak (overridden) callbacks are used. *** Workarounds linked to Silicon Limitation *** ==================================================== @@ -2075,7 +2075,7 @@ __weak void HAL_QSPI_TimeOutCallback(QSPI_HandleTypeDef *hqspi) #if (USE_HAL_QSPI_REGISTER_CALLBACKS == 1) /** * @brief Register a User QSPI Callback - * To be used instead of the weak (surcharged) predefined callback + * To be used to override the weak predefined callback * @param hqspi QSPI handle * @param CallbackId ID of the callback to be registered * This parameter can be one of the following values: @@ -2189,7 +2189,7 @@ HAL_StatusTypeDef HAL_QSPI_RegisterCallback (QSPI_HandleTypeDef *hqspi, HAL_QSPI /** * @brief Unregister a User QSPI Callback - * QSPI Callback is redirected to the weak (surcharged) predefined callback + * QSPI Callback is redirected to the weak predefined callback * @param hqspi QSPI handle * @param CallbackId ID of the callback to be unregistered * This parameter can be one of the following values: diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rcc.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rcc.c index f1873487..8e494e63 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rcc.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rcc.c @@ -479,7 +479,7 @@ __weak HAL_StatusTypeDef HAL_RCC_OscConfig(RCC_OscInitTypeDef *RCC_OscInitStruc /* Get Start Tick */ tickstart = HAL_GetTick(); - /* Wait till PLL is ready */ + /* Wait till PLL is disabled */ while(__HAL_RCC_GET_FLAG(RCC_FLAG_PLLRDY) != RESET) { if((HAL_GetTick() - tickstart ) > PLL_TIMEOUT_VALUE) @@ -517,7 +517,7 @@ __weak HAL_StatusTypeDef HAL_RCC_OscConfig(RCC_OscInitTypeDef *RCC_OscInitStruc /* Get Start Tick */ tickstart = HAL_GetTick(); - /* Wait till PLL is ready */ + /* Wait till PLL is disabled */ while(__HAL_RCC_GET_FLAG(RCC_FLAG_PLLRDY) != RESET) { if((HAL_GetTick() - tickstart ) > PLL_TIMEOUT_VALUE) diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rcc_ex.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rcc_ex.c index 114e09e0..0250aa72 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rcc_ex.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rcc_ex.c @@ -843,6 +843,10 @@ uint32_t HAL_RCCEx_GetPeriphCLKFreq(uint32_t PeriphClk) } break; } + default: + { + break; + } } return frequency; } @@ -1253,6 +1257,10 @@ uint32_t HAL_RCCEx_GetPeriphCLKFreq(uint32_t PeriphClk) } break; } + default: + { + break; + } } return frequency; } @@ -1910,6 +1918,10 @@ uint32_t HAL_RCCEx_GetPeriphCLKFreq(uint32_t PeriphClk) } break; } + default: + { + break; + } } return frequency; } @@ -2140,6 +2152,10 @@ uint32_t HAL_RCCEx_GetPeriphCLKFreq(uint32_t PeriphClk) } break; } + default: + { + break; + } } return frequency; } @@ -2491,6 +2507,10 @@ uint32_t HAL_RCCEx_GetPeriphCLKFreq(uint32_t PeriphClk) } break; } + default: + { + break; + } } return frequency; } @@ -2745,6 +2765,10 @@ uint32_t HAL_RCCEx_GetPeriphCLKFreq(uint32_t PeriphClk) } break; } + default: + { + break; + } } return frequency; } diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rng.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rng.c index bd50438d..885ce69a 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rng.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rng.c @@ -52,7 +52,7 @@ [..] Use function HAL_RNG_UnRegisterCallback() to reset a callback to the default - weak (surcharged) function. + weak (overridden) function. HAL_RNG_UnRegisterCallback() takes as parameters the HAL peripheral handle, and the Callback ID. This function allows to reset following callbacks: @@ -66,10 +66,10 @@ [..] By default, after the HAL_RNG_Init() and when the state is HAL_RNG_STATE_RESET - all callbacks are set to the corresponding weak (surcharged) functions: + all callbacks are set to the corresponding weak (overridden) functions: example HAL_RNG_ErrorCallback(). Exception done for MspInit and MspDeInit functions that are respectively - reset to the legacy weak (surcharged) functions in the HAL_RNG_Init() + reset to the legacy weak (overridden) functions in the HAL_RNG_Init() and HAL_RNG_DeInit() only when these callbacks are null (not registered beforehand). If not, MspInit or MspDeInit are not null, the HAL_RNG_Init() and HAL_RNG_DeInit() keep and use the user MspInit/MspDeInit callbacks (registered beforehand). @@ -86,7 +86,7 @@ [..] When The compilation define USE_HAL_RNG_REGISTER_CALLBACKS is set to 0 or not defined, the callback registration feature is not available - and weak (surcharged) callbacks are used. + and weak (overridden) callbacks are used. @endverbatim ****************************************************************************** @@ -307,8 +307,6 @@ HAL_StatusTypeDef HAL_RNG_RegisterCallback(RNG_HandleTypeDef *hrng, HAL_RNG_Call hrng->ErrorCode = HAL_RNG_ERROR_INVALID_CALLBACK; return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(hrng); if (HAL_RNG_STATE_READY == hrng->State) { @@ -362,8 +360,6 @@ HAL_StatusTypeDef HAL_RNG_RegisterCallback(RNG_HandleTypeDef *hrng, HAL_RNG_Call status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hrng); return status; } @@ -382,8 +378,6 @@ HAL_StatusTypeDef HAL_RNG_UnRegisterCallback(RNG_HandleTypeDef *hrng, HAL_RNG_Ca { HAL_StatusTypeDef status = HAL_OK; - /* Process locked */ - __HAL_LOCK(hrng); if (HAL_RNG_STATE_READY == hrng->State) { @@ -437,8 +431,6 @@ HAL_StatusTypeDef HAL_RNG_UnRegisterCallback(RNG_HandleTypeDef *hrng, HAL_RNG_Ca status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hrng); return status; } @@ -697,15 +689,16 @@ uint32_t HAL_RNG_GetRandomNumber_IT(RNG_HandleTypeDef *hrng) void HAL_RNG_IRQHandler(RNG_HandleTypeDef *hrng) { uint32_t rngclockerror = 0U; + uint32_t itflag = hrng->Instance->SR; /* RNG clock error interrupt occurred */ - if (__HAL_RNG_GET_IT(hrng, RNG_IT_CEI) != RESET) + if ((itflag & RNG_IT_CEI) == RNG_IT_CEI) { /* Update the error code */ hrng->ErrorCode = HAL_RNG_ERROR_CLOCK; rngclockerror = 1U; } - else if (__HAL_RNG_GET_IT(hrng, RNG_IT_SEI) != RESET) + else if ((itflag & RNG_IT_SEI) == RNG_IT_SEI) { /* Update the error code */ hrng->ErrorCode = HAL_RNG_ERROR_SEED; @@ -736,7 +729,7 @@ void HAL_RNG_IRQHandler(RNG_HandleTypeDef *hrng) } /* Check RNG data ready interrupt occurred */ - if (__HAL_RNG_GET_IT(hrng, RNG_IT_DRDY) != RESET) + if ((itflag & RNG_IT_DRDY) == RNG_IT_DRDY) { /* Generate random number once, so disable the IT */ __HAL_RNG_DISABLE_IT(hrng); @@ -768,7 +761,7 @@ void HAL_RNG_IRQHandler(RNG_HandleTypeDef *hrng) * the configuration information for RNG. * @retval random value */ -uint32_t HAL_RNG_ReadLastRandomNumber(RNG_HandleTypeDef *hrng) +uint32_t HAL_RNG_ReadLastRandomNumber(const RNG_HandleTypeDef *hrng) { return (hrng->RandomNumber); } @@ -830,7 +823,7 @@ __weak void HAL_RNG_ErrorCallback(RNG_HandleTypeDef *hrng) * the configuration information for RNG. * @retval HAL state */ -HAL_RNG_StateTypeDef HAL_RNG_GetState(RNG_HandleTypeDef *hrng) +HAL_RNG_StateTypeDef HAL_RNG_GetState(const RNG_HandleTypeDef *hrng) { return hrng->State; } @@ -840,7 +833,7 @@ HAL_RNG_StateTypeDef HAL_RNG_GetState(RNG_HandleTypeDef *hrng) * @param hrng: pointer to a RNG_HandleTypeDef structure. * @retval RNG Error Code */ -uint32_t HAL_RNG_GetError(RNG_HandleTypeDef *hrng) +uint32_t HAL_RNG_GetError(const RNG_HandleTypeDef *hrng) { /* Return RNG Error Code */ return hrng->ErrorCode; diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rtc.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rtc.c index 2d2be66d..0365accf 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rtc.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rtc.c @@ -14,7 +14,7 @@ ****************************************************************************** * @attention * - * Copyright (c) 2017 STMicroelectronics. + * Copyright (c) 2016 STMicroelectronics. * All rights reserved. * * This software is licensed under terms that can be found in the LICENSE file @@ -306,38 +306,50 @@ HAL_StatusTypeDef HAL_RTC_Init(RTC_HandleTypeDef *hrtc) /* Set RTC state */ hrtc->State = HAL_RTC_STATE_BUSY; - /* Disable the write protection for RTC registers */ - __HAL_RTC_WRITEPROTECTION_DISABLE(hrtc); + /* Check whether the calendar needs to be initialized */ + if (__HAL_RTC_IS_CALENDAR_INITIALIZED(hrtc) == 0U) + { + /* Disable the write protection for RTC registers */ + __HAL_RTC_WRITEPROTECTION_DISABLE(hrtc); - /* Enter Initialization mode */ - status = RTC_EnterInitMode(hrtc); + /* Enter Initialization mode */ + status = RTC_EnterInitMode(hrtc); - if (status == HAL_OK) - { - /* Clear RTC_CR FMT, OSEL and POL Bits */ - hrtc->Instance->CR &= ((uint32_t)~(RTC_CR_FMT | RTC_CR_OSEL | RTC_CR_POL)); - /* Set RTC_CR register */ - hrtc->Instance->CR |= (uint32_t)(hrtc->Init.HourFormat | hrtc->Init.OutPut | hrtc->Init.OutPutPolarity); + if (status == HAL_OK) + { + /* Clear RTC_CR FMT, OSEL and POL Bits */ + hrtc->Instance->CR &= ((uint32_t)~(RTC_CR_FMT | RTC_CR_OSEL | RTC_CR_POL)); + /* Set RTC_CR register */ + hrtc->Instance->CR |= (uint32_t)(hrtc->Init.HourFormat | hrtc->Init.OutPut | hrtc->Init.OutPutPolarity); - /* Configure the RTC PRER */ - hrtc->Instance->PRER = (uint32_t)(hrtc->Init.SynchPrediv); - hrtc->Instance->PRER |= (uint32_t)(hrtc->Init.AsynchPrediv << RTC_PRER_PREDIV_A_Pos); + /* Configure the RTC PRER */ + hrtc->Instance->PRER = (uint32_t)(hrtc->Init.SynchPrediv); + hrtc->Instance->PRER |= (uint32_t)(hrtc->Init.AsynchPrediv << RTC_PRER_PREDIV_A_Pos); - /* Exit Initialization mode */ - status = RTC_ExitInitMode(hrtc); + /* Exit Initialization mode */ + status = RTC_ExitInitMode(hrtc); + } + + if (status == HAL_OK) + { + hrtc->Instance->TAFCR &= (uint32_t)~RTC_OUTPUT_TYPE_PUSHPULL; + hrtc->Instance->TAFCR |= (uint32_t)(hrtc->Init.OutPutType); + } + + /* Enable the write protection for RTC registers */ + __HAL_RTC_WRITEPROTECTION_ENABLE(hrtc); + } + else + { + /* The calendar is already initialized */ + status = HAL_OK; } if (status == HAL_OK) { - hrtc->Instance->TAFCR &= (uint32_t)~RTC_OUTPUT_TYPE_PUSHPULL; - hrtc->Instance->TAFCR |= (uint32_t)(hrtc->Init.OutPutType); - hrtc->State = HAL_RTC_STATE_READY; } - /* Enable the write protection for RTC registers */ - __HAL_RTC_WRITEPROTECTION_ENABLE(hrtc); - return status; } @@ -522,7 +534,7 @@ HAL_StatusTypeDef HAL_RTC_RegisterCallback(RTC_HandleTypeDef *hrtc, HAL_RTC_Call /** * @brief Unregisters an RTC Callback - * RTC callabck is redirected to the weak predefined callback + * RTC callback is redirected to the weak predefined callback * @param hrtc pointer to a RTC_HandleTypeDef structure that contains * the configuration information for RTC. * @param CallbackID ID of the callback to be unregistered @@ -1293,7 +1305,8 @@ HAL_StatusTypeDef HAL_RTC_SetAlarm_IT(RTC_HandleTypeDef *hrtc, RTC_AlarmTypeDef /* Wait till RTC ALRAWF flag is set and if timeout is reached exit */ do { - if (count-- == 0U) + count = count - 1U; + if (count == 0U) { /* Enable the write protection for RTC registers */ __HAL_RTC_WRITEPROTECTION_ENABLE(hrtc); @@ -1329,7 +1342,8 @@ HAL_StatusTypeDef HAL_RTC_SetAlarm_IT(RTC_HandleTypeDef *hrtc, RTC_AlarmTypeDef /* Wait till RTC ALRBWF flag is set and if timeout is reached exit */ do { - if (count-- == 0U) + count = count - 1U; + if (count == 0U) { /* Enable the write protection for RTC registers */ __HAL_RTC_WRITEPROTECTION_ENABLE(hrtc); @@ -1529,21 +1543,24 @@ HAL_StatusTypeDef HAL_RTC_GetAlarm(RTC_HandleTypeDef *hrtc, RTC_AlarmTypeDef *sA */ void HAL_RTC_AlarmIRQHandler(RTC_HandleTypeDef *hrtc) { + /* Clear the EXTI's line Flag for RTC Alarm */ + __HAL_RTC_ALARM_EXTI_CLEAR_FLAG(); + /* Get the Alarm A interrupt source enable status */ if (__HAL_RTC_ALARM_GET_IT_SOURCE(hrtc, RTC_IT_ALRA) != 0U) { /* Get the pending status of the Alarm A Interrupt */ if (__HAL_RTC_ALARM_GET_FLAG(hrtc, RTC_FLAG_ALRAF) != 0U) { + /* Clear the Alarm A interrupt pending bit */ + __HAL_RTC_ALARM_CLEAR_FLAG(hrtc, RTC_FLAG_ALRAF); + /* Alarm A callback */ #if (USE_HAL_RTC_REGISTER_CALLBACKS == 1) hrtc->AlarmAEventCallback(hrtc); #else HAL_RTC_AlarmAEventCallback(hrtc); #endif /* USE_HAL_RTC_REGISTER_CALLBACKS */ - - /* Clear the Alarm A interrupt pending bit */ - __HAL_RTC_ALARM_CLEAR_FLAG(hrtc, RTC_FLAG_ALRAF); } } @@ -1553,21 +1570,18 @@ void HAL_RTC_AlarmIRQHandler(RTC_HandleTypeDef *hrtc) /* Get the pending status of the Alarm B Interrupt */ if (__HAL_RTC_ALARM_GET_FLAG(hrtc, RTC_FLAG_ALRBF) != 0U) { + /* Clear the Alarm B interrupt pending bit */ + __HAL_RTC_ALARM_CLEAR_FLAG(hrtc, RTC_FLAG_ALRBF); + /* Alarm B callback */ #if (USE_HAL_RTC_REGISTER_CALLBACKS == 1) hrtc->AlarmBEventCallback(hrtc); #else HAL_RTCEx_AlarmBEventCallback(hrtc); #endif /* USE_HAL_RTC_REGISTER_CALLBACKS */ - - /* Clear the Alarm B interrupt pending bit */ - __HAL_RTC_ALARM_CLEAR_FLAG(hrtc, RTC_FLAG_ALRBF); } } - /* Clear the EXTI's line Flag for RTC Alarm */ - __HAL_RTC_ALARM_EXTI_CLEAR_FLAG(); - /* Change RTC state */ hrtc->State = HAL_RTC_STATE_READY; } @@ -1663,8 +1677,8 @@ HAL_StatusTypeDef HAL_RTC_WaitForSynchro(RTC_HandleTypeDef *hrtc) { uint32_t tickstart = 0U; - /* Clear RSF flag */ - hrtc->Instance->ISR &= (uint32_t)RTC_RSF_MASK; + /* Clear RSF flag, keep reserved bits at reset values (setting other flags has no effect) */ + hrtc->Instance->ISR = ((uint32_t)(RTC_RSF_MASK & RTC_ISR_RESERVED_MASK)); /* Get tick */ tickstart = HAL_GetTick(); @@ -1859,7 +1873,7 @@ HAL_StatusTypeDef RTC_ExitInitMode(RTC_HandleTypeDef *hrtc) */ uint8_t RTC_ByteToBcd2(uint8_t number) { - uint8_t bcdhigh = 0U; + uint32_t bcdhigh = 0U; while (number >= 10U) { @@ -1877,9 +1891,9 @@ uint8_t RTC_ByteToBcd2(uint8_t number) */ uint8_t RTC_Bcd2ToByte(uint8_t number) { - uint8_t tmp = 0U; - tmp = ((uint8_t)(number & (uint8_t)0xF0) >> (uint8_t)0x4) * 10; - return (tmp + (number & (uint8_t)0x0F)); + uint32_t tens = 0U; + tens = (((uint32_t)number & 0xF0U) >> 4U) * 10U; + return (uint8_t)(tens + ((uint32_t)number & 0x0FU)); } /** diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rtc_ex.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rtc_ex.c index eb5708fa..f6aec5ea 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rtc_ex.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_rtc_ex.c @@ -14,7 +14,7 @@ ****************************************************************************** * @attention * - * Copyright (c) 2017 STMicroelectronics. + * Copyright (c) 2016 STMicroelectronics. * All rights reserved. * * This software is licensed under terms that can be found in the LICENSE file @@ -605,6 +605,9 @@ HAL_StatusTypeDef HAL_RTCEx_DeactivateTamper(RTC_HandleTypeDef *hrtc, uint32_t T */ void HAL_RTCEx_TamperTimeStampIRQHandler(RTC_HandleTypeDef *hrtc) { + /* Clear the EXTI's Flag for RTC Timestamp and Tamper */ + __HAL_RTC_TAMPER_TIMESTAMP_EXTI_CLEAR_FLAG(); + /* Get the Timestamp interrupt source enable status */ if (__HAL_RTC_TIMESTAMP_GET_IT_SOURCE(hrtc, RTC_IT_TS) != 0U) { @@ -618,7 +621,8 @@ void HAL_RTCEx_TamperTimeStampIRQHandler(RTC_HandleTypeDef *hrtc) HAL_RTCEx_TimeStampEventCallback(hrtc); #endif /* USE_HAL_RTC_REGISTER_CALLBACKS */ - /* Clear the Timestamp interrupt pending bit */ + /* Clear the Timestamp interrupt pending bit after returning from callback + as RTC_TSTR and RTC_TSDR registers are cleared when TSF bit is reset */ __HAL_RTC_TIMESTAMP_CLEAR_FLAG(hrtc, RTC_FLAG_TSF); } } @@ -629,15 +633,15 @@ void HAL_RTCEx_TamperTimeStampIRQHandler(RTC_HandleTypeDef *hrtc) /* Get the pending status of the Tamper 1 Interrupt */ if (__HAL_RTC_TAMPER_GET_FLAG(hrtc, RTC_FLAG_TAMP1F) != 0U) { + /* Clear the Tamper interrupt pending bit */ + __HAL_RTC_TAMPER_CLEAR_FLAG(hrtc, RTC_FLAG_TAMP1F); + /* Tamper callback */ #if (USE_HAL_RTC_REGISTER_CALLBACKS == 1) hrtc->Tamper1EventCallback(hrtc); #else HAL_RTCEx_Tamper1EventCallback(hrtc); #endif /* USE_HAL_RTC_REGISTER_CALLBACKS */ - - /* Clear the Tamper interrupt pending bit */ - __HAL_RTC_TAMPER_CLEAR_FLAG(hrtc, RTC_FLAG_TAMP1F); } } @@ -648,22 +652,19 @@ void HAL_RTCEx_TamperTimeStampIRQHandler(RTC_HandleTypeDef *hrtc) /* Get the pending status of the Tamper 2 Interrupt */ if (__HAL_RTC_TAMPER_GET_FLAG(hrtc, RTC_FLAG_TAMP2F) != 0U) { + /* Clear the Tamper interrupt pending bit */ + __HAL_RTC_TAMPER_CLEAR_FLAG(hrtc, RTC_FLAG_TAMP2F); + /* Tamper callback */ #if (USE_HAL_RTC_REGISTER_CALLBACKS == 1) hrtc->Tamper2EventCallback(hrtc); #else HAL_RTCEx_Tamper2EventCallback(hrtc); #endif /* USE_HAL_RTC_REGISTER_CALLBACKS */ - - /* Clear the Tamper interrupt pending bit */ - __HAL_RTC_TAMPER_CLEAR_FLAG(hrtc, RTC_FLAG_TAMP2F); } } #endif /* RTC_TAMPER2_SUPPORT */ - /* Clear the EXTI's Flag for RTC Timestamp and Tamper */ - __HAL_RTC_TAMPER_TIMESTAMP_EXTI_CLEAR_FLAG(); - /* Change RTC state */ hrtc->State = HAL_RTC_STATE_READY; } @@ -979,7 +980,8 @@ HAL_StatusTypeDef HAL_RTCEx_SetWakeUpTimer_IT(RTC_HandleTypeDef *hrtc, uint32_t /* Wait till RTC WUTWF flag is reset and if timeout is reached exit */ do { - if (count-- == 0U) + count = count - 1U; + if (count == 0U) { /* Enable the write protection for RTC registers */ __HAL_RTC_WRITEPROTECTION_ENABLE(hrtc); @@ -1006,7 +1008,8 @@ HAL_StatusTypeDef HAL_RTCEx_SetWakeUpTimer_IT(RTC_HandleTypeDef *hrtc, uint32_t /* Wait till RTC WUTWF flag is set and if timeout is reached exit */ do { - if (count-- == 0U) + count = count - 1U; + if (count == 0U) { /* Enable the write protection for RTC registers */ __HAL_RTC_WRITEPROTECTION_ENABLE(hrtc); @@ -1130,23 +1133,23 @@ uint32_t HAL_RTCEx_GetWakeUpTimer(RTC_HandleTypeDef *hrtc) */ void HAL_RTCEx_WakeUpTimerIRQHandler(RTC_HandleTypeDef *hrtc) { + /* Clear the EXTI's line Flag for RTC WakeUpTimer */ + __HAL_RTC_WAKEUPTIMER_EXTI_CLEAR_FLAG(); + /* Get the pending status of the Wakeup timer Interrupt */ if (__HAL_RTC_WAKEUPTIMER_GET_FLAG(hrtc, RTC_FLAG_WUTF) != 0U) { + /* Clear the Wakeup timer interrupt pending bit */ + __HAL_RTC_WAKEUPTIMER_CLEAR_FLAG(hrtc, RTC_FLAG_WUTF); + /* Wakeup timer callback */ #if (USE_HAL_RTC_REGISTER_CALLBACKS == 1) hrtc->WakeUpTimerEventCallback(hrtc); #else HAL_RTCEx_WakeUpTimerEventCallback(hrtc); #endif /* USE_HAL_RTC_REGISTER_CALLBACKS */ - - /* Clear the Wakeup timer interrupt pending bit */ - __HAL_RTC_WAKEUPTIMER_CLEAR_FLAG(hrtc, RTC_FLAG_WUTF); } - /* Clear the EXTI's line Flag for RTC WakeUpTimer */ - __HAL_RTC_WAKEUPTIMER_EXTI_CLEAR_FLAG(); - /* Change RTC state */ hrtc->State = HAL_RTC_STATE_READY; } diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sai.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sai.c index 19e37482..e881d252 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sai.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sai.c @@ -172,7 +172,7 @@ [..] Use function HAL_SAI_UnRegisterCallback() to reset a callback to the default - weak (surcharged) function. + weak function. HAL_SAI_UnRegisterCallback() takes as parameters the HAL peripheral handle, and the callback ID. [..] @@ -187,10 +187,10 @@ [..] By default, after the HAL_SAI_Init and if the state is HAL_SAI_STATE_RESET - all callbacks are reset to the corresponding legacy weak (surcharged) functions: + all callbacks are reset to the corresponding legacy weak functions: examples HAL_SAI_RxCpltCallback(), HAL_SAI_ErrorCallback(). Exception done for MspInit and MspDeInit callbacks that are respectively - reset to the legacy weak (surcharged) functions in the HAL_SAI_Init + reset to the legacy weak functions in the HAL_SAI_Init and HAL_SAI_DeInit only when these callbacks are null (not registered beforehand). If not, MspInit or MspDeInit are not null, the HAL_SAI_Init and HAL_SAI_DeInit keep and use the user MspInit/MspDeInit callbacks (registered beforehand). @@ -207,7 +207,7 @@ [..] When the compilation define USE_HAL_SAI_REGISTER_CALLBACKS is set to 0 or not defined, the callback registering feature is not available - and weak (surcharged) callbacks are used. + and weak callbacks are used. @endverbatim */ @@ -260,7 +260,7 @@ typedef enum * @{ */ static void SAI_FillFifo(SAI_HandleTypeDef *hsai); -static uint32_t SAI_InterruptFlag(SAI_HandleTypeDef *hsai, uint32_t mode); +static uint32_t SAI_InterruptFlag(const SAI_HandleTypeDef *hsai, uint32_t mode); static HAL_StatusTypeDef SAI_InitI2S(SAI_HandleTypeDef *hsai, uint32_t protocol, uint32_t datasize, uint32_t nbslot); static HAL_StatusTypeDef SAI_InitPCM(SAI_HandleTypeDef *hsai, uint32_t protocol, uint32_t datasize, uint32_t nbslot); @@ -1243,6 +1243,9 @@ HAL_StatusTypeDef HAL_SAI_DMAStop(SAI_HandleTypeDef *hsai) /* Process Locked */ __HAL_LOCK(hsai); + /* Disable SAI peripheral */ + SAI_Disable(hsai); + /* Disable the SAI DMA request */ hsai->Instance->CR1 &= ~SAI_xCR1_DMAEN; @@ -1274,9 +1277,6 @@ HAL_StatusTypeDef HAL_SAI_DMAStop(SAI_HandleTypeDef *hsai) } } - /* Disable SAI peripheral */ - SAI_Disable(hsai); - /* Flush the fifo */ SET_BIT(hsai->Instance->CR2, SAI_xCR2_FFLUSH); @@ -1302,6 +1302,9 @@ HAL_StatusTypeDef HAL_SAI_Abort(SAI_HandleTypeDef *hsai) /* Process Locked */ __HAL_LOCK(hsai); + /* Disable SAI peripheral */ + SAI_Disable(hsai); + /* Check SAI DMA is enabled or not */ if ((hsai->Instance->CR1 & SAI_xCR1_DMAEN) == SAI_xCR1_DMAEN) { @@ -1341,9 +1344,6 @@ HAL_StatusTypeDef HAL_SAI_Abort(SAI_HandleTypeDef *hsai) hsai->Instance->IMR = 0U; hsai->Instance->CLRFR = 0xFFFFFFFFU; - /* Disable SAI peripheral */ - SAI_Disable(hsai); - /* Flush the fifo */ SET_BIT(hsai->Instance->CR2, SAI_xCR2_FFLUSH); @@ -1909,7 +1909,7 @@ __weak void HAL_SAI_ErrorCallback(SAI_HandleTypeDef *hsai) * the configuration information for SAI module. * @retval HAL state */ -HAL_SAI_StateTypeDef HAL_SAI_GetState(SAI_HandleTypeDef *hsai) +HAL_SAI_StateTypeDef HAL_SAI_GetState(const SAI_HandleTypeDef *hsai) { return hsai->State; } @@ -1920,7 +1920,7 @@ HAL_SAI_StateTypeDef HAL_SAI_GetState(SAI_HandleTypeDef *hsai) * the configuration information for the specified SAI Block. * @retval SAI Error Code */ -uint32_t HAL_SAI_GetError(SAI_HandleTypeDef *hsai) +uint32_t HAL_SAI_GetError(const SAI_HandleTypeDef *hsai) { return hsai->ErrorCode; } @@ -2138,7 +2138,7 @@ static void SAI_FillFifo(SAI_HandleTypeDef *hsai) * @param mode SAI_MODE_DMA or SAI_MODE_IT * @retval the list of the IT flag to enable */ -static uint32_t SAI_InterruptFlag(SAI_HandleTypeDef *hsai, uint32_t mode) +static uint32_t SAI_InterruptFlag(const SAI_HandleTypeDef *hsai, uint32_t mode) { uint32_t tmpIT = SAI_IT_OVRUDR; diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sai_ex.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sai_ex.c index 78a73f80..2d5e8c4d 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sai_ex.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sai_ex.c @@ -95,7 +95,7 @@ * the configuration information for SAI module. * @retval SAI Clock Input */ -void SAI_BlockSynchroConfig(SAI_HandleTypeDef *hsai) +void SAI_BlockSynchroConfig(const SAI_HandleTypeDef *hsai) { uint32_t tmpregisterGCR; @@ -158,7 +158,7 @@ void SAI_BlockSynchroConfig(SAI_HandleTypeDef *hsai) * the configuration information for SAI module. * @retval SAI Clock Input */ -uint32_t SAI_GetInputClock(SAI_HandleTypeDef *hsai) +uint32_t SAI_GetInputClock(const SAI_HandleTypeDef *hsai) { /* This variable used to store the SAI_CK_x (value in Hz) */ uint32_t saiclocksource = 0U; diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sd.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sd.c index 32e54ea1..ebafec70 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sd.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sd.c @@ -696,7 +696,11 @@ HAL_StatusTypeDef HAL_SD_ReadBlocks(SD_HandleTypeDef *hsd, uint8_t *pData, uint3 } /* Get error state */ +#if defined(SDIO_STA_STBITERR) + if(__HAL_SD_GET_FLAG(hsd, SDIO_FLAG_DTIMEOUT) || (__HAL_SD_GET_FLAG(hsd, SDIO_FLAG_STBITERR))) +#else /* SDIO_STA_STBITERR not defined */ if(__HAL_SD_GET_FLAG(hsd, SDIO_FLAG_DTIMEOUT)) +#endif /* SDIO_STA_STBITERR */ { /* Clear all the static flags */ __HAL_SD_CLEAR_FLAG(hsd, SDIO_STATIC_FLAGS); @@ -911,7 +915,11 @@ HAL_StatusTypeDef HAL_SD_WriteBlocks(SD_HandleTypeDef *hsd, uint8_t *pData, uint } /* Get error state */ +#if defined(SDIO_STA_STBITERR) + if(__HAL_SD_GET_FLAG(hsd, SDIO_FLAG_DTIMEOUT) || (__HAL_SD_GET_FLAG(hsd, SDIO_FLAG_STBITERR))) +#else /* SDIO_STA_STBITERR not defined */ if(__HAL_SD_GET_FLAG(hsd, SDIO_FLAG_DTIMEOUT)) +#endif /* SDIO_STA_STBITERR */ { /* Clear all the static flags */ __HAL_SD_CLEAR_FLAG(hsd, SDIO_STATIC_FLAGS); @@ -2921,13 +2929,17 @@ static uint32_t SD_SendSDStatus(SD_HandleTypeDef *hsd, uint32_t *pSDstatus) } } - if((HAL_GetTick() - tickstart) >= SDMMC_DATATIMEOUT) + if((HAL_GetTick() - tickstart) >= SDMMC_SWDATATIMEOUT) { return HAL_SD_ERROR_TIMEOUT; } } +#if defined(SDIO_STA_STBITERR) + if(__HAL_SD_GET_FLAG(hsd, SDIO_FLAG_DTIMEOUT) || (__HAL_SD_GET_FLAG(hsd, SDIO_FLAG_STBITERR))) +#else /* SDIO_STA_STBITERR not defined */ if(__HAL_SD_GET_FLAG(hsd, SDIO_FLAG_DTIMEOUT)) +#endif /* SDIO_STA_STBITERR */ { return HAL_SD_ERROR_DATA_TIMEOUT; } @@ -2949,7 +2961,7 @@ static uint32_t SD_SendSDStatus(SD_HandleTypeDef *hsd, uint32_t *pSDstatus) *pData = SDIO_ReadFIFO(hsd->Instance); pData++; - if((HAL_GetTick() - tickstart) >= SDMMC_DATATIMEOUT) + if((HAL_GetTick() - tickstart) >= SDMMC_SWDATATIMEOUT) { return HAL_SD_ERROR_TIMEOUT; } @@ -3141,13 +3153,17 @@ static uint32_t SD_FindSCR(SD_HandleTypeDef *hsd, uint32_t *pSCR) break; } - if((HAL_GetTick() - tickstart) >= SDMMC_DATATIMEOUT) + if((HAL_GetTick() - tickstart) >= SDMMC_SWDATATIMEOUT) { return HAL_SD_ERROR_TIMEOUT; } } +#if defined(SDIO_STA_STBITERR) + if(__HAL_SD_GET_FLAG(hsd, SDIO_FLAG_DTIMEOUT) || (__HAL_SD_GET_FLAG(hsd, SDIO_FLAG_STBITERR))) +#else /* SDIO_STA_STBITERR not defined */ if(__HAL_SD_GET_FLAG(hsd, SDIO_FLAG_DTIMEOUT)) +#endif /* SDIO_STA_STBITERR */ { __HAL_SD_CLEAR_FLAG(hsd, SDIO_FLAG_DTIMEOUT); diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sdram.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sdram.c index 31633a2b..eb31bdee 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sdram.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sdram.c @@ -82,15 +82,15 @@ and a pointer to the user callback function. Use function HAL_SDRAM_UnRegisterCallback() to reset a callback to the default - weak (surcharged) function. It allows to reset following callbacks: + weak (overridden) function. It allows to reset following callbacks: (+) MspInitCallback : SDRAM MspInit. (+) MspDeInitCallback : SDRAM MspDeInit. This function) takes as parameters the HAL peripheral handle and the Callback ID. By default, after the HAL_SDRAM_Init and if the state is HAL_SDRAM_STATE_RESET - all callbacks are reset to the corresponding legacy weak (surcharged) functions. + all callbacks are reset to the corresponding legacy weak (overridden) functions. Exception done for MspInit and MspDeInit callbacks that are respectively - reset to the legacy weak (surcharged) functions in the HAL_SDRAM_Init + reset to the legacy weak (overridden) functions in the HAL_SDRAM_Init and HAL_SDRAM_DeInit only when these callbacks are null (not registered beforehand). If not, MspInit or MspDeInit are not null, the HAL_SDRAM_Init and HAL_SDRAM_DeInit keep and use the user MspInit/MspDeInit callbacks (registered beforehand) @@ -105,7 +105,7 @@ When The compilation define USE_HAL_SDRAM_REGISTER_CALLBACKS is set to 0 or not defined, the callback registering feature is not available - and weak (surcharged) callbacks are used. + and weak (overridden) callbacks are used. @endverbatim ****************************************************************************** @@ -132,9 +132,15 @@ /* Private macro -------------------------------------------------------------*/ /* Private variables ---------------------------------------------------------*/ /* Private function prototypes -----------------------------------------------*/ +/** @addtogroup SDRAM_Private_Functions SDRAM Private Functions + * @{ + */ static void SDRAM_DMACplt(DMA_HandleTypeDef *hdma); static void SDRAM_DMACpltProt(DMA_HandleTypeDef *hdma); static void SDRAM_DMAError(DMA_HandleTypeDef *hdma); +/** + * @} + */ /* Exported functions --------------------------------------------------------*/ /** @defgroup SDRAM_Exported_Functions SDRAM Exported Functions @@ -785,7 +791,7 @@ HAL_StatusTypeDef HAL_SDRAM_Write_DMA(SDRAM_HandleTypeDef *hsdram, uint32_t *pAd #if (USE_HAL_SDRAM_REGISTER_CALLBACKS == 1) /** * @brief Register a User SDRAM Callback - * To be used instead of the weak (surcharged) predefined callback + * To be used to override the weak predefined callback * @param hsdram : SDRAM handle * @param CallbackId : ID of the callback to be registered * This parameter can be one of the following values: @@ -806,9 +812,6 @@ HAL_StatusTypeDef HAL_SDRAM_RegisterCallback(SDRAM_HandleTypeDef *hsdram, HAL_SD return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(hsdram); - state = hsdram->State; if ((state == HAL_SDRAM_STATE_READY) || (state == HAL_SDRAM_STATE_WRITE_PROTECTED)) { @@ -851,14 +854,12 @@ HAL_StatusTypeDef HAL_SDRAM_RegisterCallback(SDRAM_HandleTypeDef *hsdram, HAL_SD status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hsdram); return status; } /** * @brief Unregister a User SDRAM Callback - * SDRAM Callback is redirected to the weak (surcharged) predefined callback + * SDRAM Callback is redirected to the weak predefined callback * @param hsdram : SDRAM handle * @param CallbackId : ID of the callback to be unregistered * This parameter can be one of the following values: @@ -874,9 +875,6 @@ HAL_StatusTypeDef HAL_SDRAM_UnRegisterCallback(SDRAM_HandleTypeDef *hsdram, HAL_ HAL_StatusTypeDef status = HAL_OK; HAL_SDRAM_StateTypeDef state; - /* Process locked */ - __HAL_LOCK(hsdram); - state = hsdram->State; if ((state == HAL_SDRAM_STATE_READY) || (state == HAL_SDRAM_STATE_WRITE_PROTECTED)) { @@ -925,14 +923,12 @@ HAL_StatusTypeDef HAL_SDRAM_UnRegisterCallback(SDRAM_HandleTypeDef *hsdram, HAL_ status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hsdram); return status; } /** * @brief Register a User SDRAM Callback for DMA transfers - * To be used instead of the weak (surcharged) predefined callback + * To be used to override the weak predefined callback * @param hsdram : SDRAM handle * @param CallbackId : ID of the callback to be registered * This parameter can be one of the following values: @@ -1229,6 +1225,9 @@ HAL_SDRAM_StateTypeDef HAL_SDRAM_GetState(SDRAM_HandleTypeDef *hsdram) * @} */ +/** @addtogroup SDRAM_Private_Functions SDRAM Private Functions + * @{ + */ /** * @brief DMA SDRAM process complete callback. * @param hdma : DMA handle @@ -1295,6 +1294,9 @@ static void SDRAM_DMAError(DMA_HandleTypeDef *hdma) #endif /* USE_HAL_SDRAM_REGISTER_CALLBACKS */ } +/** + * @} + */ /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_smartcard.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_smartcard.c index e721d80c..2295e7ec 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_smartcard.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_smartcard.c @@ -449,6 +449,9 @@ __weak void HAL_SMARTCARD_MspDeInit(SMARTCARD_HandleTypeDef *hsc) /** * @brief Register a User SMARTCARD Callback * To be used instead of the weak predefined callback + * @note The HAL_SMARTCARD_RegisterCallback() may be called before HAL_SMARTCARD_Init() + * in HAL_SMARTCARD_STATE_RESET to register callbacks for HAL_SMARTCARD_MSPINIT_CB_ID + * and HAL_SMARTCARD_MSPDEINIT_CB_ID * @param hsc smartcard handle * @param CallbackID ID of the callback to be registered * This parameter can be one of the following values: @@ -474,8 +477,6 @@ HAL_StatusTypeDef HAL_SMARTCARD_RegisterCallback(SMARTCARD_HandleTypeDef *hsc, H return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(hsc); if (hsc->gState == HAL_SMARTCARD_STATE_READY) { @@ -554,15 +555,15 @@ HAL_StatusTypeDef HAL_SMARTCARD_RegisterCallback(SMARTCARD_HandleTypeDef *hsc, H status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hsc); - return status; } /** * @brief Unregister an SMARTCARD callback * SMARTCARD callback is redirected to the weak predefined callback + * @note The HAL_SMARTCARD_UnRegisterCallback() may be called before HAL_SMARTCARD_Init() + * in HAL_SMARTCARD_STATE_RESET to un-register callbacks for HAL_SMARTCARD_MSPINIT_CB_ID + * and HAL_SMARTCARD_MSPDEINIT_CB_ID * @param hsc smartcard handle * @param CallbackID ID of the callback to be unregistered * This parameter can be one of the following values: @@ -580,9 +581,6 @@ HAL_StatusTypeDef HAL_SMARTCARD_UnRegisterCallback(SMARTCARD_HandleTypeDef *hsc, { HAL_StatusTypeDef status = HAL_OK; - /* Process locked */ - __HAL_LOCK(hsc); - if (HAL_SMARTCARD_STATE_READY == hsc->gState) { switch (CallbackID) @@ -659,9 +657,6 @@ HAL_StatusTypeDef HAL_SMARTCARD_UnRegisterCallback(SMARTCARD_HandleTypeDef *hsc, status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hsc); - return status; } #endif /* USE_HAL_SMARTCARD_REGISTER_CALLBACKS */ @@ -1780,7 +1775,7 @@ __weak void HAL_SMARTCARD_AbortReceiveCpltCallback (SMARTCARD_HandleTypeDef *hsc * the configuration information for SMARTCARD module. * @retval HAL state */ -HAL_SMARTCARD_StateTypeDef HAL_SMARTCARD_GetState(SMARTCARD_HandleTypeDef *hsc) +HAL_SMARTCARD_StateTypeDef HAL_SMARTCARD_GetState(const SMARTCARD_HandleTypeDef *hsc) { uint32_t temp1= 0x00U, temp2 = 0x00U; temp1 = hsc->gState; @@ -1795,7 +1790,7 @@ HAL_SMARTCARD_StateTypeDef HAL_SMARTCARD_GetState(SMARTCARD_HandleTypeDef *hsc) * the configuration information for the specified SMARTCARD. * @retval SMARTCARD Error Code */ -uint32_t HAL_SMARTCARD_GetError(SMARTCARD_HandleTypeDef *hsc) +uint32_t HAL_SMARTCARD_GetError(const SMARTCARD_HandleTypeDef *hsc) { return hsc->ErrorCode; } diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_smbus.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_smbus.c index 25c72fda..f8dbf31f 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_smbus.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_smbus.c @@ -181,7 +181,7 @@ /* Private typedef -----------------------------------------------------------*/ /* Private define ------------------------------------------------------------*/ -/** @addtogroup SMBUS_Private_Define +/** @defgroup SMBUS_Private_Define SMBUS Private Define * @{ */ #define SMBUS_TIMEOUT_FLAG 35U /*!< Timeout 35 ms */ @@ -213,6 +213,7 @@ static HAL_StatusTypeDef SMBUS_WaitOnFlagUntilTimeout(SMBUS_HandleTypeDef *hsmbus, uint32_t Flag, FlagStatus Status, uint32_t Timeout, uint32_t Tickstart); static void SMBUS_ITError(SMBUS_HandleTypeDef *hsmbus); +static void SMBUS_Flush_DR(SMBUS_HandleTypeDef *hsmbus); /* Private functions for SMBUS transfer IRQ handler */ static HAL_StatusTypeDef SMBUS_MasterTransmit_TXE(SMBUS_HandleTypeDef *hsmbus); @@ -845,6 +846,17 @@ HAL_StatusTypeDef HAL_SMBUS_UnRegisterAddrCallback(SMBUS_HandleTypeDef *hsmbus) #endif /* USE_HAL_SMBUS_REGISTER_CALLBACKS */ +/** + * @brief SMBUS data register flush process. + * @param hsmbus SMBUS handle. + * @retval None + */ +static void SMBUS_Flush_DR(SMBUS_HandleTypeDef *hsmbus) +{ + /* Write a dummy data in DR to clear it */ + hsmbus->Instance->DR = 0x00U; +} + /** * @} */ @@ -1680,6 +1692,12 @@ void HAL_SMBUS_ER_IRQHandler(SMBUS_HandleTypeDef *hsmbus) /* Clear AF flag */ __HAL_SMBUS_CLEAR_FLAG(hsmbus, SMBUS_FLAG_AF); + + /* Disable EVT, BUF and ERR interrupts */ + __HAL_SMBUS_DISABLE_IT(hsmbus, SMBUS_IT_EVT | SMBUS_IT_BUF | SMBUS_IT_ERR); + + /* Flush data register */ + SMBUS_Flush_DR(hsmbus); } } @@ -2036,7 +2054,7 @@ static HAL_StatusTypeDef SMBUS_MasterTransmit_BTF(SMBUS_HandleTypeDef *hsmbus) /* Generate Stop */ SET_BIT(hsmbus->Instance->CR1, I2C_CR1_STOP); - hsmbus->PreviousState = HAL_SMBUS_STATE_READY; + hsmbus->PreviousState = SMBUS_STATE_NONE; hsmbus->State = HAL_SMBUS_STATE_READY; hsmbus->Mode = HAL_SMBUS_MODE_NONE; #if (USE_HAL_SMBUS_REGISTER_CALLBACKS == 1) diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_spdifrx.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_spdifrx.c index da7c3021..3b0ab1fa 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_spdifrx.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_spdifrx.c @@ -33,10 +33,12 @@ (##) SPDIFRX pins configuration: (+++) Enable the clock for the SPDIFRX GPIOs. (+++) Configure these SPDIFRX pins as alternate function pull-up. - (##) NVIC configuration if you need to use interrupt process (HAL_SPDIFRX_ReceiveControlFlow_IT() and HAL_SPDIFRX_ReceiveDataFlow_IT() API's). + (##) NVIC configuration if you need to use interrupt process (HAL_SPDIFRX_ReceiveControlFlow_IT() and + HAL_SPDIFRX_ReceiveDataFlow_IT() API's). (+++) Configure the SPDIFRX interrupt priority. (+++) Enable the NVIC SPDIFRX IRQ handle. - (##) DMA Configuration if you need to use DMA process (HAL_SPDIFRX_ReceiveDataFlow_DMA() and HAL_SPDIFRX_ReceiveControlFlow_DMA() API's). + (##) DMA Configuration if you need to use DMA process (HAL_SPDIFRX_ReceiveDataFlow_DMA() and + HAL_SPDIFRX_ReceiveControlFlow_DMA() API's). (+++) Declare a DMA handle structure for the reception of the Data Flow channel. (+++) Declare a DMA handle structure for the reception of the Control Flow channel. (+++) Enable the DMAx interface clock. @@ -46,8 +48,8 @@ (+++) Configure the priority and enable the NVIC for the transfer complete interrupt on the DMA CtrlRx/DataRx channel. - (#) Program the input selection, re-tries number, wait for activity, channel status selection, data format, stereo mode and masking of user bits - using HAL_SPDIFRX_Init() function. + (#) Program the input selection, re-tries number, wait for activity, channel status selection, data format, + stereo mode and masking of user bits using HAL_SPDIFRX_Init() function. -@- The specific SPDIFRX interrupts (RXNE/CSRNE and Error Interrupts) will be managed using the macros __SPDIFRX_ENABLE_IT() and __SPDIFRX_DISABLE_IT() inside the receive process. @@ -90,7 +92,7 @@ ============================================= [..] Below the list of most used macros in SPDIFRX HAL driver. - (+) __HAL_SPDIFRX_IDLE: Disable the specified SPDIFRX peripheral (IDEL State) + (+) __HAL_SPDIFRX_IDLE: Disable the specified SPDIFRX peripheral (IDLE State) (+) __HAL_SPDIFRX_SYNC: Enable the synchronization state of the specified SPDIFRX peripheral (SYNC State) (+) __HAL_SPDIFRX_RCV: Enable the receive state of the specified SPDIFRX peripheral (RCV State) (+) __HAL_SPDIFRX_ENABLE_IT : Enable the specified SPDIFRX interrupts @@ -173,8 +175,13 @@ #if defined(STM32F446xx) /* Private typedef -----------------------------------------------------------*/ /* Private define ------------------------------------------------------------*/ -#define SPDIFRX_TIMEOUT_VALUE 0xFFFFU - +/** @defgroup SPDIFRX_Private_Defines SPDIFRX Private Defines + * @{ + */ +#define SPDIFRX_TIMEOUT_VALUE 10U +/** + * @} + */ /* Private macro -------------------------------------------------------------*/ /* Private variables ---------------------------------------------------------*/ /* Private function prototypes -----------------------------------------------*/ @@ -888,7 +895,8 @@ HAL_StatusTypeDef HAL_SPDIFRX_ReceiveDataFlow_IT(SPDIFRX_HandleTypeDef *hspdif, { if (count == 0U) { - /* Disable TXE, RXNE, PE and ERR (Frame error, noise error, overrun error) interrupts for the interrupt process */ + /* Disable TXE, RXNE, PE and ERR (Frame error, noise error, overrun error) interrupts for the interrupt + process */ __HAL_SPDIFRX_DISABLE_IT(hspdif, SPDIFRX_IT_RXNE); __HAL_SPDIFRX_DISABLE_IT(hspdif, SPDIFRX_IT_CSRNE); __HAL_SPDIFRX_DISABLE_IT(hspdif, SPDIFRX_IT_PERRIE); @@ -973,7 +981,8 @@ HAL_StatusTypeDef HAL_SPDIFRX_ReceiveControlFlow_IT(SPDIFRX_HandleTypeDef *hspdi { if (count == 0U) { - /* Disable TXE, RXNE, PE and ERR (Frame error, noise error, overrun error) interrupts for the interrupt process */ + /* Disable TXE, RXNE, PE and ERR (Frame error, noise error, overrun error) interrupts for the interrupt + process */ __HAL_SPDIFRX_DISABLE_IT(hspdif, SPDIFRX_IT_RXNE); __HAL_SPDIFRX_DISABLE_IT(hspdif, SPDIFRX_IT_CSRNE); __HAL_SPDIFRX_DISABLE_IT(hspdif, SPDIFRX_IT_PERRIE); @@ -1047,7 +1056,8 @@ HAL_StatusTypeDef HAL_SPDIFRX_ReceiveDataFlow_DMA(SPDIFRX_HandleTypeDef *hspdif, hspdif->hdmaDrRx->XferErrorCallback = SPDIFRX_DMAError; /* Enable the DMA request */ - if (HAL_DMA_Start_IT(hspdif->hdmaDrRx, (uint32_t)&hspdif->Instance->DR, (uint32_t)hspdif->pRxBuffPtr, Size) != HAL_OK) + if (HAL_DMA_Start_IT(hspdif->hdmaDrRx, (uint32_t)&hspdif->Instance->DR, (uint32_t)hspdif->pRxBuffPtr, Size) != + HAL_OK) { /* Set SPDIFRX error */ hspdif->ErrorCode = HAL_SPDIFRX_ERROR_DMA; @@ -1074,7 +1084,8 @@ HAL_StatusTypeDef HAL_SPDIFRX_ReceiveDataFlow_DMA(SPDIFRX_HandleTypeDef *hspdif, { if (count == 0U) { - /* Disable TXE, RXNE, PE and ERR (Frame error, noise error, overrun error) interrupts for the interrupt process */ + /* Disable TXE, RXNE, PE and ERR (Frame error, noise error, overrun error) interrupts for the interrupt + process */ __HAL_SPDIFRX_DISABLE_IT(hspdif, SPDIFRX_IT_RXNE); __HAL_SPDIFRX_DISABLE_IT(hspdif, SPDIFRX_IT_CSRNE); __HAL_SPDIFRX_DISABLE_IT(hspdif, SPDIFRX_IT_PERRIE); @@ -1148,7 +1159,8 @@ HAL_StatusTypeDef HAL_SPDIFRX_ReceiveControlFlow_DMA(SPDIFRX_HandleTypeDef *hspd hspdif->hdmaCsRx->XferErrorCallback = SPDIFRX_DMAError; /* Enable the DMA request */ - if (HAL_DMA_Start_IT(hspdif->hdmaCsRx, (uint32_t)&hspdif->Instance->CSR, (uint32_t)hspdif->pCsBuffPtr, Size) != HAL_OK) + if (HAL_DMA_Start_IT(hspdif->hdmaCsRx, (uint32_t)&hspdif->Instance->CSR, (uint32_t)hspdif->pCsBuffPtr, Size) != + HAL_OK) { /* Set SPDIFRX error */ hspdif->ErrorCode = HAL_SPDIFRX_ERROR_DMA; @@ -1175,7 +1187,8 @@ HAL_StatusTypeDef HAL_SPDIFRX_ReceiveControlFlow_DMA(SPDIFRX_HandleTypeDef *hspd { if (count == 0U) { - /* Disable TXE, RXNE, PE and ERR (Frame error, noise error, overrun error) interrupts for the interrupt process */ + /* Disable TXE, RXNE, PE and ERR (Frame error, noise error, overrun error) interrupts for the interrupt + process */ __HAL_SPDIFRX_DISABLE_IT(hspdif, SPDIFRX_IT_RXNE); __HAL_SPDIFRX_DISABLE_IT(hspdif, SPDIFRX_IT_CSRNE); __HAL_SPDIFRX_DISABLE_IT(hspdif, SPDIFRX_IT_PERRIE); @@ -1224,8 +1237,14 @@ HAL_StatusTypeDef HAL_SPDIFRX_DMAStop(SPDIFRX_HandleTypeDef *hspdif) hspdif->Instance->CR &= (uint16_t)(~SPDIFRX_CR_CBDMAEN); /* Disable the SPDIFRX DMA channel */ - __HAL_DMA_DISABLE(hspdif->hdmaDrRx); - __HAL_DMA_DISABLE(hspdif->hdmaCsRx); + if (hspdif->hdmaDrRx != NULL) + { + __HAL_DMA_DISABLE(hspdif->hdmaDrRx); + } + if (hspdif->hdmaCsRx != NULL) + { + __HAL_DMA_DISABLE(hspdif->hdmaCsRx); + } /* Disable SPDIFRX peripheral */ __HAL_SPDIFRX_IDLE(hspdif); @@ -1578,8 +1597,8 @@ static void SPDIFRX_ReceiveControlFlow_IT(SPDIFRX_HandleTypeDef *hspdif) * @param tickstart Tick start value * @retval HAL status */ -static HAL_StatusTypeDef SPDIFRX_WaitOnFlagUntilTimeout(SPDIFRX_HandleTypeDef *hspdif, uint32_t Flag, FlagStatus Status, - uint32_t Timeout, uint32_t tickstart) +static HAL_StatusTypeDef SPDIFRX_WaitOnFlagUntilTimeout(SPDIFRX_HandleTypeDef *hspdif, uint32_t Flag, + FlagStatus Status, uint32_t Timeout, uint32_t tickstart) { /* Wait until flag is set */ while (__HAL_SPDIFRX_GET_FLAG(hspdif, Flag) == Status) @@ -1589,7 +1608,8 @@ static HAL_StatusTypeDef SPDIFRX_WaitOnFlagUntilTimeout(SPDIFRX_HandleTypeDef *h { if (((HAL_GetTick() - tickstart) > Timeout) || (Timeout == 0U)) { - /* Disable TXE, RXNE, PE and ERR (Frame error, noise error, overrun error) interrupts for the interrupt process */ + /* Disable TXE, RXNE, PE and ERR (Frame error, noise error, overrun error) interrupts for the interrupt + process */ __HAL_SPDIFRX_DISABLE_IT(hspdif, SPDIFRX_IT_RXNE); __HAL_SPDIFRX_DISABLE_IT(hspdif, SPDIFRX_IT_CSRNE); __HAL_SPDIFRX_DISABLE_IT(hspdif, SPDIFRX_IT_PERRIE); diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_spi.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_spi.c index 62d5d658..341b7ab5 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_spi.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_spi.c @@ -856,6 +856,7 @@ HAL_StatusTypeDef HAL_SPI_Transmit(SPI_HandleTypeDef *hspi, uint8_t *pData, uint if ((((HAL_GetTick() - tickstart) >= Timeout) && (Timeout != HAL_MAX_DELAY)) || (Timeout == 0U)) { errorcode = HAL_TIMEOUT; + hspi->State = HAL_SPI_STATE_READY; goto error; } } @@ -885,6 +886,7 @@ HAL_StatusTypeDef HAL_SPI_Transmit(SPI_HandleTypeDef *hspi, uint8_t *pData, uint if ((((HAL_GetTick() - tickstart) >= Timeout) && (Timeout != HAL_MAX_DELAY)) || (Timeout == 0U)) { errorcode = HAL_TIMEOUT; + hspi->State = HAL_SPI_STATE_READY; goto error; } } @@ -914,9 +916,12 @@ HAL_StatusTypeDef HAL_SPI_Transmit(SPI_HandleTypeDef *hspi, uint8_t *pData, uint { errorcode = HAL_ERROR; } + else + { + hspi->State = HAL_SPI_STATE_READY; + } error: - hspi->State = HAL_SPI_STATE_READY; /* Process Unlocked */ __HAL_UNLOCK(hspi); return errorcode; @@ -939,6 +944,12 @@ HAL_StatusTypeDef HAL_SPI_Receive(SPI_HandleTypeDef *hspi, uint8_t *pData, uint1 uint32_t tickstart; HAL_StatusTypeDef errorcode = HAL_OK; + if (hspi->State != HAL_SPI_STATE_READY) + { + errorcode = HAL_BUSY; + goto error; + } + if ((hspi->Init.Mode == SPI_MODE_MASTER) && (hspi->Init.Direction == SPI_DIRECTION_2LINES)) { hspi->State = HAL_SPI_STATE_BUSY_RX; @@ -952,12 +963,6 @@ HAL_StatusTypeDef HAL_SPI_Receive(SPI_HandleTypeDef *hspi, uint8_t *pData, uint1 /* Init tickstart for timeout management*/ tickstart = HAL_GetTick(); - if (hspi->State != HAL_SPI_STATE_READY) - { - errorcode = HAL_BUSY; - goto error; - } - if ((pData == NULL) || (Size == 0U)) { errorcode = HAL_ERROR; @@ -1023,6 +1028,7 @@ HAL_StatusTypeDef HAL_SPI_Receive(SPI_HandleTypeDef *hspi, uint8_t *pData, uint1 if ((((HAL_GetTick() - tickstart) >= Timeout) && (Timeout != HAL_MAX_DELAY)) || (Timeout == 0U)) { errorcode = HAL_TIMEOUT; + hspi->State = HAL_SPI_STATE_READY; goto error; } } @@ -1046,6 +1052,7 @@ HAL_StatusTypeDef HAL_SPI_Receive(SPI_HandleTypeDef *hspi, uint8_t *pData, uint1 if ((((HAL_GetTick() - tickstart) >= Timeout) && (Timeout != HAL_MAX_DELAY)) || (Timeout == 0U)) { errorcode = HAL_TIMEOUT; + hspi->State = HAL_SPI_STATE_READY; goto error; } } @@ -1112,9 +1119,12 @@ HAL_StatusTypeDef HAL_SPI_Receive(SPI_HandleTypeDef *hspi, uint8_t *pData, uint1 { errorcode = HAL_ERROR; } + else + { + hspi->State = HAL_SPI_STATE_READY; + } error : - hspi->State = HAL_SPI_STATE_READY; __HAL_UNLOCK(hspi); return errorcode; } @@ -1213,6 +1223,15 @@ HAL_StatusTypeDef HAL_SPI_TransmitReceive(SPI_HandleTypeDef *hspi, uint8_t *pTxD hspi->Instance->DR = *((uint16_t *)hspi->pTxBuffPtr); hspi->pTxBuffPtr += sizeof(uint16_t); hspi->TxXferCount--; + +#if (USE_SPI_CRC != 0U) + /* Enable CRC Transmission */ + if ((hspi->TxXferCount == 0U) && (hspi->Init.CRCCalculation == SPI_CRCCALCULATION_ENABLE)) + { + SET_BIT(hspi->Instance->CR1, SPI_CR1_CRCNEXT); + } +#endif /* USE_SPI_CRC */ + } while ((hspi->TxXferCount > 0U) || (hspi->RxXferCount > 0U)) { @@ -1246,6 +1265,7 @@ HAL_StatusTypeDef HAL_SPI_TransmitReceive(SPI_HandleTypeDef *hspi, uint8_t *pTxD if (((HAL_GetTick() - tickstart) >= Timeout) && (Timeout != HAL_MAX_DELAY)) { errorcode = HAL_TIMEOUT; + hspi->State = HAL_SPI_STATE_READY; goto error; } } @@ -1258,6 +1278,14 @@ HAL_StatusTypeDef HAL_SPI_TransmitReceive(SPI_HandleTypeDef *hspi, uint8_t *pTxD *((__IO uint8_t *)&hspi->Instance->DR) = (*hspi->pTxBuffPtr); hspi->pTxBuffPtr += sizeof(uint8_t); hspi->TxXferCount--; + +#if (USE_SPI_CRC != 0U) + /* Enable CRC Transmission */ + if ((hspi->TxXferCount == 0U) && (hspi->Init.CRCCalculation == SPI_CRCCALCULATION_ENABLE)) + { + SET_BIT(hspi->Instance->CR1, SPI_CR1_CRCNEXT); + } +#endif /* USE_SPI_CRC */ } while ((hspi->TxXferCount > 0U) || (hspi->RxXferCount > 0U)) { @@ -1291,6 +1319,7 @@ HAL_StatusTypeDef HAL_SPI_TransmitReceive(SPI_HandleTypeDef *hspi, uint8_t *pTxD if ((((HAL_GetTick() - tickstart) >= Timeout) && ((Timeout != HAL_MAX_DELAY))) || (Timeout == 0U)) { errorcode = HAL_TIMEOUT; + hspi->State = HAL_SPI_STATE_READY; goto error; } } @@ -1339,8 +1368,16 @@ HAL_StatusTypeDef HAL_SPI_TransmitReceive(SPI_HandleTypeDef *hspi, uint8_t *pTxD __HAL_SPI_CLEAR_OVRFLAG(hspi); } + if (hspi->ErrorCode != HAL_SPI_ERROR_NONE) + { + errorcode = HAL_ERROR; + } + else + { + hspi->State = HAL_SPI_STATE_READY; + } + error : - hspi->State = HAL_SPI_STATE_READY; __HAL_UNLOCK(hspi); return errorcode; } @@ -1360,8 +1397,6 @@ HAL_StatusTypeDef HAL_SPI_Transmit_IT(SPI_HandleTypeDef *hspi, uint8_t *pData, u /* Check Direction parameter */ assert_param(IS_SPI_DIRECTION_2LINES_OR_1LINE(hspi->Init.Direction)); - /* Process Locked */ - __HAL_LOCK(hspi); if ((pData == NULL) || (Size == 0U)) { @@ -1375,6 +1410,9 @@ HAL_StatusTypeDef HAL_SPI_Transmit_IT(SPI_HandleTypeDef *hspi, uint8_t *pData, u goto error; } + /* Process Locked */ + __HAL_LOCK(hspi); + /* Set the transaction information */ hspi->State = HAL_SPI_STATE_BUSY_TX; hspi->ErrorCode = HAL_SPI_ERROR_NONE; @@ -1414,10 +1452,6 @@ HAL_StatusTypeDef HAL_SPI_Transmit_IT(SPI_HandleTypeDef *hspi, uint8_t *pData, u } #endif /* USE_SPI_CRC */ - /* Enable TXE and ERR interrupt */ - __HAL_SPI_ENABLE_IT(hspi, (SPI_IT_TXE | SPI_IT_ERR)); - - /* Check if the SPI is already enabled */ if ((hspi->Instance->CR1 & SPI_CR1_SPE) != SPI_CR1_SPE) { @@ -1425,8 +1459,12 @@ HAL_StatusTypeDef HAL_SPI_Transmit_IT(SPI_HandleTypeDef *hspi, uint8_t *pData, u __HAL_SPI_ENABLE(hspi); } -error : + /* Process Unlocked */ __HAL_UNLOCK(hspi); + /* Enable TXE and ERR interrupt */ + __HAL_SPI_ENABLE_IT(hspi, (SPI_IT_TXE | SPI_IT_ERR)); + +error : return errorcode; } @@ -1442,6 +1480,13 @@ HAL_StatusTypeDef HAL_SPI_Receive_IT(SPI_HandleTypeDef *hspi, uint8_t *pData, ui { HAL_StatusTypeDef errorcode = HAL_OK; + + if (hspi->State != HAL_SPI_STATE_READY) + { + errorcode = HAL_BUSY; + goto error; + } + if ((hspi->Init.Direction == SPI_DIRECTION_2LINES) && (hspi->Init.Mode == SPI_MODE_MASTER)) { hspi->State = HAL_SPI_STATE_BUSY_RX; @@ -1449,14 +1494,6 @@ HAL_StatusTypeDef HAL_SPI_Receive_IT(SPI_HandleTypeDef *hspi, uint8_t *pData, ui return HAL_SPI_TransmitReceive_IT(hspi, pData, pData, Size); } - /* Process Locked */ - __HAL_LOCK(hspi); - - if (hspi->State != HAL_SPI_STATE_READY) - { - errorcode = HAL_BUSY; - goto error; - } if ((pData == NULL) || (Size == 0U)) { @@ -1464,6 +1501,9 @@ HAL_StatusTypeDef HAL_SPI_Receive_IT(SPI_HandleTypeDef *hspi, uint8_t *pData, ui goto error; } + /* Process Locked */ + __HAL_LOCK(hspi); + /* Set the transaction information */ hspi->State = HAL_SPI_STATE_BUSY_RX; hspi->ErrorCode = HAL_SPI_ERROR_NONE; @@ -1503,9 +1543,6 @@ HAL_StatusTypeDef HAL_SPI_Receive_IT(SPI_HandleTypeDef *hspi, uint8_t *pData, ui } #endif /* USE_SPI_CRC */ - /* Enable TXE and ERR interrupt */ - __HAL_SPI_ENABLE_IT(hspi, (SPI_IT_RXNE | SPI_IT_ERR)); - /* Note : The SPI must be enabled after unlocking current process to avoid the risk of SPI interrupt handle execution before current process unlock */ @@ -1517,9 +1554,12 @@ HAL_StatusTypeDef HAL_SPI_Receive_IT(SPI_HandleTypeDef *hspi, uint8_t *pData, ui __HAL_SPI_ENABLE(hspi); } -error : /* Process Unlocked */ __HAL_UNLOCK(hspi); + /* Enable RXNE and ERR interrupt */ + __HAL_SPI_ENABLE_IT(hspi, (SPI_IT_RXNE | SPI_IT_ERR)); + +error : return errorcode; } @@ -1541,9 +1581,6 @@ HAL_StatusTypeDef HAL_SPI_TransmitReceive_IT(SPI_HandleTypeDef *hspi, uint8_t *p /* Check Direction parameter */ assert_param(IS_SPI_DIRECTION_2LINES(hspi->Init.Direction)); - /* Process locked */ - __HAL_LOCK(hspi); - /* Init temporary variables */ tmp_state = hspi->State; tmp_mode = hspi->Init.Mode; @@ -1561,6 +1598,9 @@ HAL_StatusTypeDef HAL_SPI_TransmitReceive_IT(SPI_HandleTypeDef *hspi, uint8_t *p goto error; } + /* Process locked */ + __HAL_LOCK(hspi); + /* Don't overwrite in case of HAL_SPI_STATE_BUSY_RX */ if (hspi->State != HAL_SPI_STATE_BUSY_RX) { @@ -1596,8 +1636,6 @@ HAL_StatusTypeDef HAL_SPI_TransmitReceive_IT(SPI_HandleTypeDef *hspi, uint8_t *p } #endif /* USE_SPI_CRC */ - /* Enable TXE, RXNE and ERR interrupt */ - __HAL_SPI_ENABLE_IT(hspi, (SPI_IT_TXE | SPI_IT_RXNE | SPI_IT_ERR)); /* Check if the SPI is already enabled */ if ((hspi->Instance->CR1 & SPI_CR1_SPE) != SPI_CR1_SPE) @@ -1606,9 +1644,12 @@ HAL_StatusTypeDef HAL_SPI_TransmitReceive_IT(SPI_HandleTypeDef *hspi, uint8_t *p __HAL_SPI_ENABLE(hspi); } -error : /* Process Unlocked */ __HAL_UNLOCK(hspi); + /* Enable TXE, RXNE and ERR interrupt */ + __HAL_SPI_ENABLE_IT(hspi, (SPI_IT_TXE | SPI_IT_RXNE | SPI_IT_ERR)); + +error : return errorcode; } @@ -1695,7 +1736,6 @@ HAL_StatusTypeDef HAL_SPI_Transmit_DMA(SPI_HandleTypeDef *hspi, uint8_t *pData, SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_DMA); errorcode = HAL_ERROR; - hspi->State = HAL_SPI_STATE_READY; goto error; } @@ -1735,6 +1775,12 @@ HAL_StatusTypeDef HAL_SPI_Receive_DMA(SPI_HandleTypeDef *hspi, uint8_t *pData, u /* Check rx dma handle */ assert_param(IS_SPI_DMA_HANDLE(hspi->hdmarx)); + if (hspi->State != HAL_SPI_STATE_READY) + { + errorcode = HAL_BUSY; + goto error; + } + if ((hspi->Init.Direction == SPI_DIRECTION_2LINES) && (hspi->Init.Mode == SPI_MODE_MASTER)) { hspi->State = HAL_SPI_STATE_BUSY_RX; @@ -1749,12 +1795,6 @@ HAL_StatusTypeDef HAL_SPI_Receive_DMA(SPI_HandleTypeDef *hspi, uint8_t *pData, u /* Process Locked */ __HAL_LOCK(hspi); - if (hspi->State != HAL_SPI_STATE_READY) - { - errorcode = HAL_BUSY; - goto error; - } - if ((pData == NULL) || (Size == 0U)) { errorcode = HAL_ERROR; @@ -1810,7 +1850,6 @@ HAL_StatusTypeDef HAL_SPI_Receive_DMA(SPI_HandleTypeDef *hspi, uint8_t *pData, u SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_DMA); errorcode = HAL_ERROR; - hspi->State = HAL_SPI_STATE_READY; goto error; } @@ -1932,7 +1971,6 @@ HAL_StatusTypeDef HAL_SPI_TransmitReceive_DMA(SPI_HandleTypeDef *hspi, uint8_t * SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_DMA); errorcode = HAL_ERROR; - hspi->State = HAL_SPI_STATE_READY; goto error; } @@ -1954,7 +1992,6 @@ HAL_StatusTypeDef HAL_SPI_TransmitReceive_DMA(SPI_HandleTypeDef *hspi, uint8_t * SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_DMA); errorcode = HAL_ERROR; - hspi->State = HAL_SPI_STATE_READY; goto error; } @@ -3595,6 +3632,13 @@ static HAL_StatusTypeDef SPI_EndRxTransaction(SPI_HandleTypeDef *hspi, uint32_t */ static HAL_StatusTypeDef SPI_EndRxTxTransaction(SPI_HandleTypeDef *hspi, uint32_t Timeout, uint32_t Tickstart) { + /* Wait until TXE flag */ + if(SPI_WaitFlagStateUntilTimeout(hspi, SPI_FLAG_TXE, SET, Timeout, Tickstart) != HAL_OK) + { + SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_FLAG); + return HAL_TIMEOUT; + } + /* Timeout in µs */ __IO uint32_t count = SPI_BSY_FLAG_WORKAROUND_TIMEOUT * (SystemCoreClock / 24U / 1000000U); /* Erratasheet: BSY bit may stay high at the end of a data transfer in Slave mode */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sram.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sram.c index ef2eaf09..17884373 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sram.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_sram.c @@ -83,15 +83,15 @@ and a pointer to the user callback function. Use function HAL_SRAM_UnRegisterCallback() to reset a callback to the default - weak (surcharged) function. It allows to reset following callbacks: + weak (overridden) function. It allows to reset following callbacks: (+) MspInitCallback : SRAM MspInit. (+) MspDeInitCallback : SRAM MspDeInit. This function) takes as parameters the HAL peripheral handle and the Callback ID. By default, after the HAL_SRAM_Init and if the state is HAL_SRAM_STATE_RESET - all callbacks are reset to the corresponding legacy weak (surcharged) functions. + all callbacks are reset to the corresponding legacy weak (overridden) functions. Exception done for MspInit and MspDeInit callbacks that are respectively - reset to the legacy weak (surcharged) functions in the HAL_SRAM_Init + reset to the legacy weak (overridden) functions in the HAL_SRAM_Init and HAL_SRAM_DeInit only when these callbacks are null (not registered beforehand). If not, MspInit or MspDeInit are not null, the HAL_SRAM_Init and HAL_SRAM_DeInit keep and use the user MspInit/MspDeInit callbacks (registered beforehand) @@ -106,7 +106,7 @@ When The compilation define USE_HAL_SRAM_REGISTER_CALLBACKS is set to 0 or not defined, the callback registering feature is not available - and weak (surcharged) callbacks are used. + and weak (overridden) callbacks are used. @endverbatim ****************************************************************************** @@ -133,9 +133,15 @@ /* Private macro -------------------------------------------------------------*/ /* Private variables ---------------------------------------------------------*/ /* Private function prototypes -----------------------------------------------*/ +/** @addtogroup SRAM_Private_Functions SRAM Private Functions + * @{ + */ static void SRAM_DMACplt(DMA_HandleTypeDef *hdma); static void SRAM_DMACpltProt(DMA_HandleTypeDef *hdma); static void SRAM_DMAError(DMA_HandleTypeDef *hdma); +/** + * @} + */ /* Exported functions --------------------------------------------------------*/ @@ -731,7 +737,7 @@ HAL_StatusTypeDef HAL_SRAM_Write_DMA(SRAM_HandleTypeDef *hsram, uint32_t *pAddre #if (USE_HAL_SRAM_REGISTER_CALLBACKS == 1) /** * @brief Register a User SRAM Callback - * To be used instead of the weak (surcharged) predefined callback + * To be used to override the weak predefined callback * @param hsram : SRAM handle * @param CallbackId : ID of the callback to be registered * This parameter can be one of the following values: @@ -751,9 +757,6 @@ HAL_StatusTypeDef HAL_SRAM_RegisterCallback(SRAM_HandleTypeDef *hsram, HAL_SRAM_ return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(hsram); - state = hsram->State; if ((state == HAL_SRAM_STATE_READY) || (state == HAL_SRAM_STATE_RESET) || (state == HAL_SRAM_STATE_PROTECTED)) { @@ -777,14 +780,12 @@ HAL_StatusTypeDef HAL_SRAM_RegisterCallback(SRAM_HandleTypeDef *hsram, HAL_SRAM_ status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hsram); return status; } /** * @brief Unregister a User SRAM Callback - * SRAM Callback is redirected to the weak (surcharged) predefined callback + * SRAM Callback is redirected to the weak predefined callback * @param hsram : SRAM handle * @param CallbackId : ID of the callback to be unregistered * This parameter can be one of the following values: @@ -799,9 +800,6 @@ HAL_StatusTypeDef HAL_SRAM_UnRegisterCallback(SRAM_HandleTypeDef *hsram, HAL_SRA HAL_StatusTypeDef status = HAL_OK; HAL_SRAM_StateTypeDef state; - /* Process locked */ - __HAL_LOCK(hsram); - state = hsram->State; if ((state == HAL_SRAM_STATE_READY) || (state == HAL_SRAM_STATE_PROTECTED)) { @@ -847,14 +845,12 @@ HAL_StatusTypeDef HAL_SRAM_UnRegisterCallback(SRAM_HandleTypeDef *hsram, HAL_SRA status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(hsram); return status; } /** * @brief Register a User SRAM Callback for DMA transfers - * To be used instead of the weak (surcharged) predefined callback + * To be used to override the weak predefined callback * @param hsram : SRAM handle * @param CallbackId : ID of the callback to be registered * This parameter can be one of the following values: @@ -1018,7 +1014,7 @@ HAL_StatusTypeDef HAL_SRAM_WriteOperation_Disable(SRAM_HandleTypeDef *hsram) * the configuration information for SRAM module. * @retval HAL state */ -HAL_SRAM_StateTypeDef HAL_SRAM_GetState(SRAM_HandleTypeDef *hsram) +HAL_SRAM_StateTypeDef HAL_SRAM_GetState(const SRAM_HandleTypeDef *hsram) { return hsram->State; } @@ -1031,6 +1027,10 @@ HAL_SRAM_StateTypeDef HAL_SRAM_GetState(SRAM_HandleTypeDef *hsram) * @} */ +/** @addtogroup SRAM_Private_Functions SRAM Private Functions + * @{ + */ + /** * @brief DMA SRAM process complete callback. * @param hdma : DMA handle @@ -1097,6 +1097,10 @@ static void SRAM_DMAError(DMA_HandleTypeDef *hdma) #endif /* USE_HAL_SRAM_REGISTER_CALLBACKS */ } +/** + * @} + */ + /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_tim.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_tim.c index 1ca1781e..d5978cca 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_tim.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_tim.c @@ -204,9 +204,9 @@ all interrupt callbacks are set to the corresponding weak functions: /** @addtogroup TIM_Private_Functions * @{ */ -static void TIM_OC1_SetConfig(TIM_TypeDef *TIMx, TIM_OC_InitTypeDef *OC_Config); -static void TIM_OC3_SetConfig(TIM_TypeDef *TIMx, TIM_OC_InitTypeDef *OC_Config); -static void TIM_OC4_SetConfig(TIM_TypeDef *TIMx, TIM_OC_InitTypeDef *OC_Config); +static void TIM_OC1_SetConfig(TIM_TypeDef *TIMx, const TIM_OC_InitTypeDef *OC_Config); +static void TIM_OC3_SetConfig(TIM_TypeDef *TIMx, const TIM_OC_InitTypeDef *OC_Config); +static void TIM_OC4_SetConfig(TIM_TypeDef *TIMx, const TIM_OC_InitTypeDef *OC_Config); static void TIM_TI1_ConfigInputStage(TIM_TypeDef *TIMx, uint32_t TIM_ICPolarity, uint32_t TIM_ICFilter); static void TIM_TI2_SetConfig(TIM_TypeDef *TIMx, uint32_t TIM_ICPolarity, uint32_t TIM_ICSelection, uint32_t TIM_ICFilter); @@ -222,7 +222,7 @@ static void TIM_DMADelayPulseCplt(DMA_HandleTypeDef *hdma); static void TIM_DMATriggerCplt(DMA_HandleTypeDef *hdma); static void TIM_DMATriggerHalfCplt(DMA_HandleTypeDef *hdma); static HAL_StatusTypeDef TIM_SlaveTimer_SetConfig(TIM_HandleTypeDef *htim, - TIM_SlaveConfigTypeDef *sSlaveConfig); + const TIM_SlaveConfigTypeDef *sSlaveConfig); /** * @} */ @@ -275,6 +275,7 @@ HAL_StatusTypeDef HAL_TIM_Base_Init(TIM_HandleTypeDef *htim) assert_param(IS_TIM_INSTANCE(htim->Instance)); assert_param(IS_TIM_COUNTER_MODE(htim->Init.CounterMode)); assert_param(IS_TIM_CLOCKDIVISION_DIV(htim->Init.ClockDivision)); + assert_param(IS_TIM_PERIOD(htim, htim->Init.Period)); assert_param(IS_TIM_AUTORELOAD_PRELOAD(htim->Init.AutoReloadPreload)); if (htim->State == HAL_TIM_STATE_RESET) @@ -522,7 +523,7 @@ HAL_StatusTypeDef HAL_TIM_Base_Stop_IT(TIM_HandleTypeDef *htim) * @param Length The length of data to be transferred from memory to peripheral. * @retval HAL status */ -HAL_StatusTypeDef HAL_TIM_Base_Start_DMA(TIM_HandleTypeDef *htim, uint32_t *pData, uint16_t Length) +HAL_StatusTypeDef HAL_TIM_Base_Start_DMA(TIM_HandleTypeDef *htim, const uint32_t *pData, uint16_t Length) { uint32_t tmpsmcr; @@ -536,7 +537,7 @@ HAL_StatusTypeDef HAL_TIM_Base_Start_DMA(TIM_HandleTypeDef *htim, uint32_t *pDat } else if (htim->State == HAL_TIM_STATE_READY) { - if ((pData == NULL) && (Length > 0U)) + if ((pData == NULL) || (Length == 0U)) { return HAL_ERROR; } @@ -658,6 +659,7 @@ HAL_StatusTypeDef HAL_TIM_OC_Init(TIM_HandleTypeDef *htim) assert_param(IS_TIM_INSTANCE(htim->Instance)); assert_param(IS_TIM_COUNTER_MODE(htim->Init.CounterMode)); assert_param(IS_TIM_CLOCKDIVISION_DIV(htim->Init.ClockDivision)); + assert_param(IS_TIM_PERIOD(htim, htim->Init.Period)); assert_param(IS_TIM_AUTORELOAD_PRELOAD(htim->Init.AutoReloadPreload)); if (htim->State == HAL_TIM_STATE_RESET) @@ -1043,7 +1045,8 @@ HAL_StatusTypeDef HAL_TIM_OC_Stop_IT(TIM_HandleTypeDef *htim, uint32_t Channel) * @param Length The length of data to be transferred from memory to TIM peripheral * @retval HAL status */ -HAL_StatusTypeDef HAL_TIM_OC_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, uint32_t *pData, uint16_t Length) +HAL_StatusTypeDef HAL_TIM_OC_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, const uint32_t *pData, + uint16_t Length) { HAL_StatusTypeDef status = HAL_OK; uint32_t tmpsmcr; @@ -1058,7 +1061,7 @@ HAL_StatusTypeDef HAL_TIM_OC_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel } else if (TIM_CHANNEL_STATE_GET(htim, Channel) == HAL_TIM_CHANNEL_STATE_READY) { - if ((pData == NULL) && (Length > 0U)) + if ((pData == NULL) || (Length == 0U)) { return HAL_ERROR; } @@ -1321,6 +1324,7 @@ HAL_StatusTypeDef HAL_TIM_PWM_Init(TIM_HandleTypeDef *htim) assert_param(IS_TIM_INSTANCE(htim->Instance)); assert_param(IS_TIM_COUNTER_MODE(htim->Init.CounterMode)); assert_param(IS_TIM_CLOCKDIVISION_DIV(htim->Init.ClockDivision)); + assert_param(IS_TIM_PERIOD(htim, htim->Init.Period)); assert_param(IS_TIM_AUTORELOAD_PRELOAD(htim->Init.AutoReloadPreload)); if (htim->State == HAL_TIM_STATE_RESET) @@ -1706,7 +1710,8 @@ HAL_StatusTypeDef HAL_TIM_PWM_Stop_IT(TIM_HandleTypeDef *htim, uint32_t Channel) * @param Length The length of data to be transferred from memory to TIM peripheral * @retval HAL status */ -HAL_StatusTypeDef HAL_TIM_PWM_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, uint32_t *pData, uint16_t Length) +HAL_StatusTypeDef HAL_TIM_PWM_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, const uint32_t *pData, + uint16_t Length) { HAL_StatusTypeDef status = HAL_OK; uint32_t tmpsmcr; @@ -1721,7 +1726,7 @@ HAL_StatusTypeDef HAL_TIM_PWM_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channe } else if (TIM_CHANNEL_STATE_GET(htim, Channel) == HAL_TIM_CHANNEL_STATE_READY) { - if ((pData == NULL) && (Length > 0U)) + if ((pData == NULL) || (Length == 0U)) { return HAL_ERROR; } @@ -1983,6 +1988,7 @@ HAL_StatusTypeDef HAL_TIM_IC_Init(TIM_HandleTypeDef *htim) assert_param(IS_TIM_INSTANCE(htim->Instance)); assert_param(IS_TIM_COUNTER_MODE(htim->Init.CounterMode)); assert_param(IS_TIM_CLOCKDIVISION_DIV(htim->Init.ClockDivision)); + assert_param(IS_TIM_PERIOD(htim, htim->Init.Period)); assert_param(IS_TIM_AUTORELOAD_PRELOAD(htim->Init.AutoReloadPreload)); if (htim->State == HAL_TIM_STATE_RESET) @@ -2376,7 +2382,7 @@ HAL_StatusTypeDef HAL_TIM_IC_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel else if ((channel_state == HAL_TIM_CHANNEL_STATE_READY) && (complementary_channel_state == HAL_TIM_CHANNEL_STATE_READY)) { - if ((pData == NULL) && (Length > 0U)) + if ((pData == NULL) || (Length == 0U)) { return HAL_ERROR; } @@ -2632,6 +2638,7 @@ HAL_StatusTypeDef HAL_TIM_OnePulse_Init(TIM_HandleTypeDef *htim, uint32_t OnePul assert_param(IS_TIM_COUNTER_MODE(htim->Init.CounterMode)); assert_param(IS_TIM_CLOCKDIVISION_DIV(htim->Init.ClockDivision)); assert_param(IS_TIM_OPM_MODE(OnePulseMode)); + assert_param(IS_TIM_PERIOD(htim, htim->Init.Period)); assert_param(IS_TIM_AUTORELOAD_PRELOAD(htim->Init.AutoReloadPreload)); if (htim->State == HAL_TIM_STATE_RESET) @@ -3009,7 +3016,7 @@ HAL_StatusTypeDef HAL_TIM_OnePulse_Stop_IT(TIM_HandleTypeDef *htim, uint32_t Out * @param sConfig TIM Encoder Interface configuration structure * @retval HAL status */ -HAL_StatusTypeDef HAL_TIM_Encoder_Init(TIM_HandleTypeDef *htim, TIM_Encoder_InitTypeDef *sConfig) +HAL_StatusTypeDef HAL_TIM_Encoder_Init(TIM_HandleTypeDef *htim, const TIM_Encoder_InitTypeDef *sConfig) { uint32_t tmpsmcr; uint32_t tmpccmr1; @@ -3035,6 +3042,7 @@ HAL_StatusTypeDef HAL_TIM_Encoder_Init(TIM_HandleTypeDef *htim, TIM_Encoder_Ini assert_param(IS_TIM_IC_PRESCALER(sConfig->IC2Prescaler)); assert_param(IS_TIM_IC_FILTER(sConfig->IC1Filter)); assert_param(IS_TIM_IC_FILTER(sConfig->IC2Filter)); + assert_param(IS_TIM_PERIOD(htim, htim->Init.Period)); if (htim->State == HAL_TIM_STATE_RESET) { @@ -3544,7 +3552,7 @@ HAL_StatusTypeDef HAL_TIM_Encoder_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Ch else if ((channel_1_state == HAL_TIM_CHANNEL_STATE_READY) && (complementary_channel_1_state == HAL_TIM_CHANNEL_STATE_READY)) { - if ((pData1 == NULL) && (Length > 0U)) + if ((pData1 == NULL) || (Length == 0U)) { return HAL_ERROR; } @@ -3569,7 +3577,7 @@ HAL_StatusTypeDef HAL_TIM_Encoder_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Ch else if ((channel_2_state == HAL_TIM_CHANNEL_STATE_READY) && (complementary_channel_2_state == HAL_TIM_CHANNEL_STATE_READY)) { - if ((pData2 == NULL) && (Length > 0U)) + if ((pData2 == NULL) || (Length == 0U)) { return HAL_ERROR; } @@ -3598,7 +3606,7 @@ HAL_StatusTypeDef HAL_TIM_Encoder_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Ch && (complementary_channel_1_state == HAL_TIM_CHANNEL_STATE_READY) && (complementary_channel_2_state == HAL_TIM_CHANNEL_STATE_READY)) { - if ((((pData1 == NULL) || (pData2 == NULL))) && (Length > 0U)) + if ((((pData1 == NULL) || (pData2 == NULL))) || (Length == 0U)) { return HAL_ERROR; } @@ -3814,13 +3822,16 @@ HAL_StatusTypeDef HAL_TIM_Encoder_Stop_DMA(TIM_HandleTypeDef *htim, uint32_t Cha */ void HAL_TIM_IRQHandler(TIM_HandleTypeDef *htim) { + uint32_t itsource = htim->Instance->DIER; + uint32_t itflag = htim->Instance->SR; + /* Capture compare 1 event */ - if (__HAL_TIM_GET_FLAG(htim, TIM_FLAG_CC1) != RESET) + if ((itflag & (TIM_FLAG_CC1)) == (TIM_FLAG_CC1)) { - if (__HAL_TIM_GET_IT_SOURCE(htim, TIM_IT_CC1) != RESET) + if ((itsource & (TIM_IT_CC1)) == (TIM_IT_CC1)) { { - __HAL_TIM_CLEAR_IT(htim, TIM_IT_CC1); + __HAL_TIM_CLEAR_FLAG(htim, TIM_FLAG_CC1); htim->Channel = HAL_TIM_ACTIVE_CHANNEL_1; /* Input capture event */ @@ -3848,11 +3859,11 @@ void HAL_TIM_IRQHandler(TIM_HandleTypeDef *htim) } } /* Capture compare 2 event */ - if (__HAL_TIM_GET_FLAG(htim, TIM_FLAG_CC2) != RESET) + if ((itflag & (TIM_FLAG_CC2)) == (TIM_FLAG_CC2)) { - if (__HAL_TIM_GET_IT_SOURCE(htim, TIM_IT_CC2) != RESET) + if ((itsource & (TIM_IT_CC2)) == (TIM_IT_CC2)) { - __HAL_TIM_CLEAR_IT(htim, TIM_IT_CC2); + __HAL_TIM_CLEAR_FLAG(htim, TIM_FLAG_CC2); htim->Channel = HAL_TIM_ACTIVE_CHANNEL_2; /* Input capture event */ if ((htim->Instance->CCMR1 & TIM_CCMR1_CC2S) != 0x00U) @@ -3878,11 +3889,11 @@ void HAL_TIM_IRQHandler(TIM_HandleTypeDef *htim) } } /* Capture compare 3 event */ - if (__HAL_TIM_GET_FLAG(htim, TIM_FLAG_CC3) != RESET) + if ((itflag & (TIM_FLAG_CC3)) == (TIM_FLAG_CC3)) { - if (__HAL_TIM_GET_IT_SOURCE(htim, TIM_IT_CC3) != RESET) + if ((itsource & (TIM_IT_CC3)) == (TIM_IT_CC3)) { - __HAL_TIM_CLEAR_IT(htim, TIM_IT_CC3); + __HAL_TIM_CLEAR_FLAG(htim, TIM_FLAG_CC3); htim->Channel = HAL_TIM_ACTIVE_CHANNEL_3; /* Input capture event */ if ((htim->Instance->CCMR2 & TIM_CCMR2_CC3S) != 0x00U) @@ -3908,11 +3919,11 @@ void HAL_TIM_IRQHandler(TIM_HandleTypeDef *htim) } } /* Capture compare 4 event */ - if (__HAL_TIM_GET_FLAG(htim, TIM_FLAG_CC4) != RESET) + if ((itflag & (TIM_FLAG_CC4)) == (TIM_FLAG_CC4)) { - if (__HAL_TIM_GET_IT_SOURCE(htim, TIM_IT_CC4) != RESET) + if ((itsource & (TIM_IT_CC4)) == (TIM_IT_CC4)) { - __HAL_TIM_CLEAR_IT(htim, TIM_IT_CC4); + __HAL_TIM_CLEAR_FLAG(htim, TIM_FLAG_CC4); htim->Channel = HAL_TIM_ACTIVE_CHANNEL_4; /* Input capture event */ if ((htim->Instance->CCMR2 & TIM_CCMR2_CC4S) != 0x00U) @@ -3938,11 +3949,11 @@ void HAL_TIM_IRQHandler(TIM_HandleTypeDef *htim) } } /* TIM Update event */ - if (__HAL_TIM_GET_FLAG(htim, TIM_FLAG_UPDATE) != RESET) + if ((itflag & (TIM_FLAG_UPDATE)) == (TIM_FLAG_UPDATE)) { - if (__HAL_TIM_GET_IT_SOURCE(htim, TIM_IT_UPDATE) != RESET) + if ((itsource & (TIM_IT_UPDATE)) == (TIM_IT_UPDATE)) { - __HAL_TIM_CLEAR_IT(htim, TIM_IT_UPDATE); + __HAL_TIM_CLEAR_FLAG(htim, TIM_FLAG_UPDATE); #if (USE_HAL_TIM_REGISTER_CALLBACKS == 1) htim->PeriodElapsedCallback(htim); #else @@ -3951,11 +3962,11 @@ void HAL_TIM_IRQHandler(TIM_HandleTypeDef *htim) } } /* TIM Break input event */ - if (__HAL_TIM_GET_FLAG(htim, TIM_FLAG_BREAK) != RESET) + if ((itflag & (TIM_FLAG_BREAK)) == (TIM_FLAG_BREAK)) { - if (__HAL_TIM_GET_IT_SOURCE(htim, TIM_IT_BREAK) != RESET) + if ((itsource & (TIM_IT_BREAK)) == (TIM_IT_BREAK)) { - __HAL_TIM_CLEAR_IT(htim, TIM_IT_BREAK); + __HAL_TIM_CLEAR_FLAG(htim, TIM_FLAG_BREAK); #if (USE_HAL_TIM_REGISTER_CALLBACKS == 1) htim->BreakCallback(htim); #else @@ -3964,11 +3975,11 @@ void HAL_TIM_IRQHandler(TIM_HandleTypeDef *htim) } } /* TIM Trigger detection event */ - if (__HAL_TIM_GET_FLAG(htim, TIM_FLAG_TRIGGER) != RESET) + if ((itflag & (TIM_FLAG_TRIGGER)) == (TIM_FLAG_TRIGGER)) { - if (__HAL_TIM_GET_IT_SOURCE(htim, TIM_IT_TRIGGER) != RESET) + if ((itsource & (TIM_IT_TRIGGER)) == (TIM_IT_TRIGGER)) { - __HAL_TIM_CLEAR_IT(htim, TIM_IT_TRIGGER); + __HAL_TIM_CLEAR_FLAG(htim, TIM_FLAG_TRIGGER); #if (USE_HAL_TIM_REGISTER_CALLBACKS == 1) htim->TriggerCallback(htim); #else @@ -3977,11 +3988,11 @@ void HAL_TIM_IRQHandler(TIM_HandleTypeDef *htim) } } /* TIM commutation event */ - if (__HAL_TIM_GET_FLAG(htim, TIM_FLAG_COM) != RESET) + if ((itflag & (TIM_FLAG_COM)) == (TIM_FLAG_COM)) { - if (__HAL_TIM_GET_IT_SOURCE(htim, TIM_IT_COM) != RESET) + if ((itsource & (TIM_IT_COM)) == (TIM_IT_COM)) { - __HAL_TIM_CLEAR_IT(htim, TIM_FLAG_COM); + __HAL_TIM_CLEAR_FLAG(htim, TIM_FLAG_COM); #if (USE_HAL_TIM_REGISTER_CALLBACKS == 1) htim->CommutationCallback(htim); #else @@ -4028,7 +4039,7 @@ void HAL_TIM_IRQHandler(TIM_HandleTypeDef *htim) * @retval HAL status */ HAL_StatusTypeDef HAL_TIM_OC_ConfigChannel(TIM_HandleTypeDef *htim, - TIM_OC_InitTypeDef *sConfig, + const TIM_OC_InitTypeDef *sConfig, uint32_t Channel) { HAL_StatusTypeDef status = HAL_OK; @@ -4106,7 +4117,7 @@ HAL_StatusTypeDef HAL_TIM_OC_ConfigChannel(TIM_HandleTypeDef *htim, * @arg TIM_CHANNEL_4: TIM Channel 4 selected * @retval HAL status */ -HAL_StatusTypeDef HAL_TIM_IC_ConfigChannel(TIM_HandleTypeDef *htim, TIM_IC_InitTypeDef *sConfig, uint32_t Channel) +HAL_StatusTypeDef HAL_TIM_IC_ConfigChannel(TIM_HandleTypeDef *htim, const TIM_IC_InitTypeDef *sConfig, uint32_t Channel) { HAL_StatusTypeDef status = HAL_OK; @@ -4206,7 +4217,7 @@ HAL_StatusTypeDef HAL_TIM_IC_ConfigChannel(TIM_HandleTypeDef *htim, TIM_IC_InitT * @retval HAL status */ HAL_StatusTypeDef HAL_TIM_PWM_ConfigChannel(TIM_HandleTypeDef *htim, - TIM_OC_InitTypeDef *sConfig, + const TIM_OC_InitTypeDef *sConfig, uint32_t Channel) { HAL_StatusTypeDef status = HAL_OK; @@ -4468,7 +4479,8 @@ HAL_StatusTypeDef HAL_TIM_OnePulse_ConfigChannel(TIM_HandleTypeDef *htim, TIM_O * @retval HAL status */ HAL_StatusTypeDef HAL_TIM_DMABurst_WriteStart(TIM_HandleTypeDef *htim, uint32_t BurstBaseAddress, - uint32_t BurstRequestSrc, uint32_t *BurstBuffer, uint32_t BurstLength) + uint32_t BurstRequestSrc, const uint32_t *BurstBuffer, + uint32_t BurstLength) { HAL_StatusTypeDef status; @@ -4520,7 +4532,7 @@ HAL_StatusTypeDef HAL_TIM_DMABurst_WriteStart(TIM_HandleTypeDef *htim, uint32_t * @retval HAL status */ HAL_StatusTypeDef HAL_TIM_DMABurst_MultiWriteStart(TIM_HandleTypeDef *htim, uint32_t BurstBaseAddress, - uint32_t BurstRequestSrc, uint32_t *BurstBuffer, + uint32_t BurstRequestSrc, const uint32_t *BurstBuffer, uint32_t BurstLength, uint32_t DataLength) { HAL_StatusTypeDef status = HAL_OK; @@ -5160,7 +5172,7 @@ HAL_StatusTypeDef HAL_TIM_GenerateEvent(TIM_HandleTypeDef *htim, uint32_t EventS * @retval HAL status */ HAL_StatusTypeDef HAL_TIM_ConfigOCrefClear(TIM_HandleTypeDef *htim, - TIM_ClearInputConfigTypeDef *sClearInputConfig, + const TIM_ClearInputConfigTypeDef *sClearInputConfig, uint32_t Channel) { HAL_StatusTypeDef status = HAL_OK; @@ -5289,7 +5301,7 @@ HAL_StatusTypeDef HAL_TIM_ConfigOCrefClear(TIM_HandleTypeDef *htim, * contains the clock source information for the TIM peripheral. * @retval HAL status */ -HAL_StatusTypeDef HAL_TIM_ConfigClockSource(TIM_HandleTypeDef *htim, TIM_ClockConfigTypeDef *sClockSourceConfig) +HAL_StatusTypeDef HAL_TIM_ConfigClockSource(TIM_HandleTypeDef *htim, const TIM_ClockConfigTypeDef *sClockSourceConfig) { HAL_StatusTypeDef status = HAL_OK; uint32_t tmpsmcr; @@ -5475,7 +5487,7 @@ HAL_StatusTypeDef HAL_TIM_ConfigTI1Input(TIM_HandleTypeDef *htim, uint32_t TI1_S * (Disable, Reset, Gated, Trigger, External clock mode 1). * @retval HAL status */ -HAL_StatusTypeDef HAL_TIM_SlaveConfigSynchro(TIM_HandleTypeDef *htim, TIM_SlaveConfigTypeDef *sSlaveConfig) +HAL_StatusTypeDef HAL_TIM_SlaveConfigSynchro(TIM_HandleTypeDef *htim, const TIM_SlaveConfigTypeDef *sSlaveConfig) { /* Check the parameters */ assert_param(IS_TIM_SLAVE_INSTANCE(htim->Instance)); @@ -5516,7 +5528,7 @@ HAL_StatusTypeDef HAL_TIM_SlaveConfigSynchro(TIM_HandleTypeDef *htim, TIM_SlaveC * @retval HAL status */ HAL_StatusTypeDef HAL_TIM_SlaveConfigSynchro_IT(TIM_HandleTypeDef *htim, - TIM_SlaveConfigTypeDef *sSlaveConfig) + const TIM_SlaveConfigTypeDef *sSlaveConfig) { /* Check the parameters */ assert_param(IS_TIM_SLAVE_INSTANCE(htim->Instance)); @@ -5558,7 +5570,7 @@ HAL_StatusTypeDef HAL_TIM_SlaveConfigSynchro_IT(TIM_HandleTypeDef *htim, * @arg TIM_CHANNEL_4: TIM Channel 4 selected * @retval Captured value */ -uint32_t HAL_TIM_ReadCapturedValue(TIM_HandleTypeDef *htim, uint32_t Channel) +uint32_t HAL_TIM_ReadCapturedValue(const TIM_HandleTypeDef *htim, uint32_t Channel) { uint32_t tmpreg = 0U; @@ -5832,8 +5844,6 @@ HAL_StatusTypeDef HAL_TIM_RegisterCallback(TIM_HandleTypeDef *htim, HAL_TIM_Call { return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(htim); if (htim->State == HAL_TIM_STATE_READY) { @@ -6025,9 +6035,6 @@ HAL_StatusTypeDef HAL_TIM_RegisterCallback(TIM_HandleTypeDef *htim, HAL_TIM_Call status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(htim); - return status; } @@ -6070,9 +6077,6 @@ HAL_StatusTypeDef HAL_TIM_UnRegisterCallback(TIM_HandleTypeDef *htim, HAL_TIM_Ca { HAL_StatusTypeDef status = HAL_OK; - /* Process locked */ - __HAL_LOCK(htim); - if (htim->State == HAL_TIM_STATE_READY) { switch (CallbackID) @@ -6304,9 +6308,6 @@ HAL_StatusTypeDef HAL_TIM_UnRegisterCallback(TIM_HandleTypeDef *htim, HAL_TIM_Ca status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(htim); - return status; } #endif /* USE_HAL_TIM_REGISTER_CALLBACKS */ @@ -6335,7 +6336,7 @@ HAL_StatusTypeDef HAL_TIM_UnRegisterCallback(TIM_HandleTypeDef *htim, HAL_TIM_Ca * @param htim TIM Base handle * @retval HAL state */ -HAL_TIM_StateTypeDef HAL_TIM_Base_GetState(TIM_HandleTypeDef *htim) +HAL_TIM_StateTypeDef HAL_TIM_Base_GetState(const TIM_HandleTypeDef *htim) { return htim->State; } @@ -6345,7 +6346,7 @@ HAL_TIM_StateTypeDef HAL_TIM_Base_GetState(TIM_HandleTypeDef *htim) * @param htim TIM Output Compare handle * @retval HAL state */ -HAL_TIM_StateTypeDef HAL_TIM_OC_GetState(TIM_HandleTypeDef *htim) +HAL_TIM_StateTypeDef HAL_TIM_OC_GetState(const TIM_HandleTypeDef *htim) { return htim->State; } @@ -6355,7 +6356,7 @@ HAL_TIM_StateTypeDef HAL_TIM_OC_GetState(TIM_HandleTypeDef *htim) * @param htim TIM handle * @retval HAL state */ -HAL_TIM_StateTypeDef HAL_TIM_PWM_GetState(TIM_HandleTypeDef *htim) +HAL_TIM_StateTypeDef HAL_TIM_PWM_GetState(const TIM_HandleTypeDef *htim) { return htim->State; } @@ -6365,7 +6366,7 @@ HAL_TIM_StateTypeDef HAL_TIM_PWM_GetState(TIM_HandleTypeDef *htim) * @param htim TIM IC handle * @retval HAL state */ -HAL_TIM_StateTypeDef HAL_TIM_IC_GetState(TIM_HandleTypeDef *htim) +HAL_TIM_StateTypeDef HAL_TIM_IC_GetState(const TIM_HandleTypeDef *htim) { return htim->State; } @@ -6375,7 +6376,7 @@ HAL_TIM_StateTypeDef HAL_TIM_IC_GetState(TIM_HandleTypeDef *htim) * @param htim TIM OPM handle * @retval HAL state */ -HAL_TIM_StateTypeDef HAL_TIM_OnePulse_GetState(TIM_HandleTypeDef *htim) +HAL_TIM_StateTypeDef HAL_TIM_OnePulse_GetState(const TIM_HandleTypeDef *htim) { return htim->State; } @@ -6385,7 +6386,7 @@ HAL_TIM_StateTypeDef HAL_TIM_OnePulse_GetState(TIM_HandleTypeDef *htim) * @param htim TIM Encoder Interface handle * @retval HAL state */ -HAL_TIM_StateTypeDef HAL_TIM_Encoder_GetState(TIM_HandleTypeDef *htim) +HAL_TIM_StateTypeDef HAL_TIM_Encoder_GetState(const TIM_HandleTypeDef *htim) { return htim->State; } @@ -6395,7 +6396,7 @@ HAL_TIM_StateTypeDef HAL_TIM_Encoder_GetState(TIM_HandleTypeDef *htim) * @param htim TIM handle * @retval Active channel */ -HAL_TIM_ActiveChannel HAL_TIM_GetActiveChannel(TIM_HandleTypeDef *htim) +HAL_TIM_ActiveChannel HAL_TIM_GetActiveChannel(const TIM_HandleTypeDef *htim) { return htim->Channel; } @@ -6413,7 +6414,7 @@ HAL_TIM_ActiveChannel HAL_TIM_GetActiveChannel(TIM_HandleTypeDef *htim) * @arg TIM_CHANNEL_6: TIM Channel 6 * @retval TIM Channel state */ -HAL_TIM_ChannelStateTypeDef HAL_TIM_GetChannelState(TIM_HandleTypeDef *htim, uint32_t Channel) +HAL_TIM_ChannelStateTypeDef HAL_TIM_GetChannelState(const TIM_HandleTypeDef *htim, uint32_t Channel) { HAL_TIM_ChannelStateTypeDef channel_state; @@ -6430,7 +6431,7 @@ HAL_TIM_ChannelStateTypeDef HAL_TIM_GetChannelState(TIM_HandleTypeDef *htim, ui * @param htim TIM handle * @retval DMA burst state */ -HAL_TIM_DMABurstStateTypeDef HAL_TIM_DMABurstState(TIM_HandleTypeDef *htim) +HAL_TIM_DMABurstStateTypeDef HAL_TIM_DMABurstState(const TIM_HandleTypeDef *htim) { /* Check the parameters */ assert_param(IS_TIM_DMABURST_INSTANCE(htim->Instance)); @@ -6773,7 +6774,7 @@ static void TIM_DMATriggerHalfCplt(DMA_HandleTypeDef *hdma) * @param Structure TIM Base configuration structure * @retval None */ -void TIM_Base_SetConfig(TIM_TypeDef *TIMx, TIM_Base_InitTypeDef *Structure) +void TIM_Base_SetConfig(TIM_TypeDef *TIMx, const TIM_Base_InitTypeDef *Structure) { uint32_t tmpcr1; tmpcr1 = TIMx->CR1; @@ -6813,6 +6814,13 @@ void TIM_Base_SetConfig(TIM_TypeDef *TIMx, TIM_Base_InitTypeDef *Structure) /* Generate an update event to reload the Prescaler and the repetition counter (only for advanced timer) value immediately */ TIMx->EGR = TIM_EGR_UG; + + /* Check if the update flag is set after the Update Generation, if so clear the UIF flag */ + if (HAL_IS_BIT_SET(TIMx->SR, TIM_FLAG_UPDATE)) + { + /* Clear the update flag */ + CLEAR_BIT(TIMx->SR, TIM_FLAG_UPDATE); + } } /** @@ -6821,17 +6829,18 @@ void TIM_Base_SetConfig(TIM_TypeDef *TIMx, TIM_Base_InitTypeDef *Structure) * @param OC_Config The output configuration structure * @retval None */ -static void TIM_OC1_SetConfig(TIM_TypeDef *TIMx, TIM_OC_InitTypeDef *OC_Config) +static void TIM_OC1_SetConfig(TIM_TypeDef *TIMx, const TIM_OC_InitTypeDef *OC_Config) { uint32_t tmpccmrx; uint32_t tmpccer; uint32_t tmpcr2; + /* Get the TIMx CCER register value */ + tmpccer = TIMx->CCER; + /* Disable the Channel 1: Reset the CC1E Bit */ TIMx->CCER &= ~TIM_CCER_CC1E; - /* Get the TIMx CCER register value */ - tmpccer = TIMx->CCER; /* Get the TIMx CR2 register value */ tmpcr2 = TIMx->CR2; @@ -6896,17 +6905,18 @@ static void TIM_OC1_SetConfig(TIM_TypeDef *TIMx, TIM_OC_InitTypeDef *OC_Config) * @param OC_Config The output configuration structure * @retval None */ -void TIM_OC2_SetConfig(TIM_TypeDef *TIMx, TIM_OC_InitTypeDef *OC_Config) +void TIM_OC2_SetConfig(TIM_TypeDef *TIMx, const TIM_OC_InitTypeDef *OC_Config) { uint32_t tmpccmrx; uint32_t tmpccer; uint32_t tmpcr2; + /* Get the TIMx CCER register value */ + tmpccer = TIMx->CCER; + /* Disable the Channel 2: Reset the CC2E Bit */ TIMx->CCER &= ~TIM_CCER_CC2E; - /* Get the TIMx CCER register value */ - tmpccer = TIMx->CCER; /* Get the TIMx CR2 register value */ tmpcr2 = TIMx->CR2; @@ -6935,7 +6945,6 @@ void TIM_OC2_SetConfig(TIM_TypeDef *TIMx, TIM_OC_InitTypeDef *OC_Config) tmpccer |= (OC_Config->OCNPolarity << 4U); /* Reset the Output N State */ tmpccer &= ~TIM_CCER_CC2NE; - } if (IS_TIM_BREAK_INSTANCE(TIMx)) @@ -6972,17 +6981,18 @@ void TIM_OC2_SetConfig(TIM_TypeDef *TIMx, TIM_OC_InitTypeDef *OC_Config) * @param OC_Config The output configuration structure * @retval None */ -static void TIM_OC3_SetConfig(TIM_TypeDef *TIMx, TIM_OC_InitTypeDef *OC_Config) +static void TIM_OC3_SetConfig(TIM_TypeDef *TIMx, const TIM_OC_InitTypeDef *OC_Config) { uint32_t tmpccmrx; uint32_t tmpccer; uint32_t tmpcr2; + /* Get the TIMx CCER register value */ + tmpccer = TIMx->CCER; + /* Disable the Channel 3: Reset the CC2E Bit */ TIMx->CCER &= ~TIM_CCER_CC3E; - /* Get the TIMx CCER register value */ - tmpccer = TIMx->CCER; /* Get the TIMx CR2 register value */ tmpcr2 = TIMx->CR2; @@ -7046,17 +7056,18 @@ static void TIM_OC3_SetConfig(TIM_TypeDef *TIMx, TIM_OC_InitTypeDef *OC_Config) * @param OC_Config The output configuration structure * @retval None */ -static void TIM_OC4_SetConfig(TIM_TypeDef *TIMx, TIM_OC_InitTypeDef *OC_Config) +static void TIM_OC4_SetConfig(TIM_TypeDef *TIMx, const TIM_OC_InitTypeDef *OC_Config) { uint32_t tmpccmrx; uint32_t tmpccer; uint32_t tmpcr2; + /* Get the TIMx CCER register value */ + tmpccer = TIMx->CCER; + /* Disable the Channel 4: Reset the CC4E Bit */ TIMx->CCER &= ~TIM_CCER_CC4E; - /* Get the TIMx CCER register value */ - tmpccer = TIMx->CCER; /* Get the TIMx CR2 register value */ tmpcr2 = TIMx->CR2; @@ -7107,7 +7118,7 @@ static void TIM_OC4_SetConfig(TIM_TypeDef *TIMx, TIM_OC_InitTypeDef *OC_Config) * @retval None */ static HAL_StatusTypeDef TIM_SlaveTimer_SetConfig(TIM_HandleTypeDef *htim, - TIM_SlaveConfigTypeDef *sSlaveConfig) + const TIM_SlaveConfigTypeDef *sSlaveConfig) { HAL_StatusTypeDef status = HAL_OK; uint32_t tmpsmcr; @@ -7247,9 +7258,9 @@ void TIM_TI1_SetConfig(TIM_TypeDef *TIMx, uint32_t TIM_ICPolarity, uint32_t TIM_ uint32_t tmpccer; /* Disable the Channel 1: Reset the CC1E Bit */ + tmpccer = TIMx->CCER; TIMx->CCER &= ~TIM_CCER_CC1E; tmpccmr1 = TIMx->CCMR1; - tmpccer = TIMx->CCER; /* Select the Input */ if (IS_TIM_CC2_INSTANCE(TIMx) != RESET) @@ -7337,9 +7348,9 @@ static void TIM_TI2_SetConfig(TIM_TypeDef *TIMx, uint32_t TIM_ICPolarity, uint32 uint32_t tmpccer; /* Disable the Channel 2: Reset the CC2E Bit */ + tmpccer = TIMx->CCER; TIMx->CCER &= ~TIM_CCER_CC2E; tmpccmr1 = TIMx->CCMR1; - tmpccer = TIMx->CCER; /* Select the Input */ tmpccmr1 &= ~TIM_CCMR1_CC2S; @@ -7376,9 +7387,9 @@ static void TIM_TI2_ConfigInputStage(TIM_TypeDef *TIMx, uint32_t TIM_ICPolarity, uint32_t tmpccer; /* Disable the Channel 2: Reset the CC2E Bit */ + tmpccer = TIMx->CCER; TIMx->CCER &= ~TIM_CCER_CC2E; tmpccmr1 = TIMx->CCMR1; - tmpccer = TIMx->CCER; /* Set the filter */ tmpccmr1 &= ~TIM_CCMR1_IC2F; @@ -7420,9 +7431,9 @@ static void TIM_TI3_SetConfig(TIM_TypeDef *TIMx, uint32_t TIM_ICPolarity, uint32 uint32_t tmpccer; /* Disable the Channel 3: Reset the CC3E Bit */ + tmpccer = TIMx->CCER; TIMx->CCER &= ~TIM_CCER_CC3E; tmpccmr2 = TIMx->CCMR2; - tmpccer = TIMx->CCER; /* Select the Input */ tmpccmr2 &= ~TIM_CCMR2_CC3S; @@ -7468,9 +7479,9 @@ static void TIM_TI4_SetConfig(TIM_TypeDef *TIMx, uint32_t TIM_ICPolarity, uint32 uint32_t tmpccer; /* Disable the Channel 4: Reset the CC4E Bit */ + tmpccer = TIMx->CCER; TIMx->CCER &= ~TIM_CCER_CC4E; tmpccmr2 = TIMx->CCMR2; - tmpccer = TIMx->CCER; /* Select the Input */ tmpccmr2 &= ~TIM_CCMR2_CC4S; diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_tim_ex.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_tim_ex.c index 092175f5..889f8fb9 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_tim_ex.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_tim_ex.c @@ -135,7 +135,7 @@ static void TIM_CCxNChannelCmd(TIM_TypeDef *TIMx, uint32_t Channel, uint32_t Cha * @param sConfig TIM Hall Sensor configuration structure * @retval HAL status */ -HAL_StatusTypeDef HAL_TIMEx_HallSensor_Init(TIM_HandleTypeDef *htim, TIM_HallSensor_InitTypeDef *sConfig) +HAL_StatusTypeDef HAL_TIMEx_HallSensor_Init(TIM_HandleTypeDef *htim, const TIM_HallSensor_InitTypeDef *sConfig) { TIM_OC_InitTypeDef OC_Config; @@ -151,6 +151,7 @@ HAL_StatusTypeDef HAL_TIMEx_HallSensor_Init(TIM_HandleTypeDef *htim, TIM_HallSen assert_param(IS_TIM_CLOCKDIVISION_DIV(htim->Init.ClockDivision)); assert_param(IS_TIM_AUTORELOAD_PRELOAD(htim->Init.AutoReloadPreload)); assert_param(IS_TIM_IC_POLARITY(sConfig->IC1Polarity)); + assert_param(IS_TIM_PERIOD(htim, htim->Init.Period)); assert_param(IS_TIM_IC_PRESCALER(sConfig->IC1Prescaler)); assert_param(IS_TIM_IC_FILTER(sConfig->IC1Filter)); @@ -501,7 +502,7 @@ HAL_StatusTypeDef HAL_TIMEx_HallSensor_Start_DMA(TIM_HandleTypeDef *htim, uint32 else if ((channel_1_state == HAL_TIM_CHANNEL_STATE_READY) && (complementary_channel_1_state == HAL_TIM_CHANNEL_STATE_READY)) { - if ((pData == NULL) && (Length > 0U)) + if ((pData == NULL) || (Length == 0U)) { return HAL_ERROR; } @@ -834,7 +835,7 @@ HAL_StatusTypeDef HAL_TIMEx_OCN_Stop_IT(TIM_HandleTypeDef *htim, uint32_t Channe /* Disable the TIM Break interrupt (only if no more channel is active) */ tmpccer = htim->Instance->CCER; - if ((tmpccer & (TIM_CCER_CC1NE | TIM_CCER_CC2NE | TIM_CCER_CC3NE)) == (uint32_t)RESET) + if ((tmpccer & TIM_CCER_CCxNE_MASK) == (uint32_t)RESET) { __HAL_TIM_DISABLE_IT(htim, TIM_IT_BREAK); } @@ -866,7 +867,8 @@ HAL_StatusTypeDef HAL_TIMEx_OCN_Stop_IT(TIM_HandleTypeDef *htim, uint32_t Channe * @param Length The length of data to be transferred from memory to TIM peripheral * @retval HAL status */ -HAL_StatusTypeDef HAL_TIMEx_OCN_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, uint32_t *pData, uint16_t Length) +HAL_StatusTypeDef HAL_TIMEx_OCN_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, const uint32_t *pData, + uint16_t Length) { HAL_StatusTypeDef status = HAL_OK; uint32_t tmpsmcr; @@ -881,7 +883,7 @@ HAL_StatusTypeDef HAL_TIMEx_OCN_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Chan } else if (TIM_CHANNEL_N_STATE_GET(htim, Channel) == HAL_TIM_CHANNEL_STATE_READY) { - if ((pData == NULL) && (Length > 0U)) + if ((pData == NULL) || (Length == 0U)) { return HAL_ERROR; } @@ -1079,17 +1081,6 @@ HAL_StatusTypeDef HAL_TIMEx_OCN_Stop_DMA(TIM_HandleTypeDef *htim, uint32_t Chann (+) Stop the Complementary PWM and disable interrupts. (+) Start the Complementary PWM and enable DMA transfers. (+) Stop the Complementary PWM and disable DMA transfers. - (+) Start the Complementary Input Capture measurement. - (+) Stop the Complementary Input Capture. - (+) Start the Complementary Input Capture and enable interrupts. - (+) Stop the Complementary Input Capture and disable interrupts. - (+) Start the Complementary Input Capture and enable DMA transfers. - (+) Stop the Complementary Input Capture and disable DMA transfers. - (+) Start the Complementary One Pulse generation. - (+) Stop the Complementary One Pulse. - (+) Start the Complementary One Pulse and enable interrupts. - (+) Stop the Complementary One Pulse and disable interrupts. - @endverbatim * @{ */ @@ -1315,7 +1306,7 @@ HAL_StatusTypeDef HAL_TIMEx_PWMN_Stop_IT(TIM_HandleTypeDef *htim, uint32_t Chann /* Disable the TIM Break interrupt (only if no more channel is active) */ tmpccer = htim->Instance->CCER; - if ((tmpccer & (TIM_CCER_CC1NE | TIM_CCER_CC2NE | TIM_CCER_CC3NE)) == (uint32_t)RESET) + if ((tmpccer & TIM_CCER_CCxNE_MASK) == (uint32_t)RESET) { __HAL_TIM_DISABLE_IT(htim, TIM_IT_BREAK); } @@ -1347,7 +1338,8 @@ HAL_StatusTypeDef HAL_TIMEx_PWMN_Stop_IT(TIM_HandleTypeDef *htim, uint32_t Chann * @param Length The length of data to be transferred from memory to TIM peripheral * @retval HAL status */ -HAL_StatusTypeDef HAL_TIMEx_PWMN_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, uint32_t *pData, uint16_t Length) +HAL_StatusTypeDef HAL_TIMEx_PWMN_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Channel, const uint32_t *pData, + uint16_t Length) { HAL_StatusTypeDef status = HAL_OK; uint32_t tmpsmcr; @@ -1362,7 +1354,7 @@ HAL_StatusTypeDef HAL_TIMEx_PWMN_Start_DMA(TIM_HandleTypeDef *htim, uint32_t Cha } else if (TIM_CHANNEL_N_STATE_GET(htim, Channel) == HAL_TIM_CHANNEL_STATE_READY) { - if ((pData == NULL) && (Length > 0U)) + if ((pData == NULL) || (Length == 0U)) { return HAL_ERROR; } @@ -1960,7 +1952,7 @@ HAL_StatusTypeDef HAL_TIMEx_ConfigCommutEvent_DMA(TIM_HandleTypeDef *htim, uint3 * @retval HAL status */ HAL_StatusTypeDef HAL_TIMEx_MasterConfigSynchronization(TIM_HandleTypeDef *htim, - TIM_MasterConfigTypeDef *sMasterConfig) + const TIM_MasterConfigTypeDef *sMasterConfig) { uint32_t tmpcr2; uint32_t tmpsmcr; @@ -2021,7 +2013,7 @@ HAL_StatusTypeDef HAL_TIMEx_MasterConfigSynchronization(TIM_HandleTypeDef *htim, * @retval HAL status */ HAL_StatusTypeDef HAL_TIMEx_ConfigBreakDeadTime(TIM_HandleTypeDef *htim, - TIM_BreakDeadTimeConfigTypeDef *sBreakDeadTimeConfig) + const TIM_BreakDeadTimeConfigTypeDef *sBreakDeadTimeConfig) { /* Keep this variable initialized to 0 as it is used to configure BDTR register */ uint32_t tmpbdtr = 0U; @@ -2098,7 +2090,6 @@ HAL_StatusTypeDef HAL_TIMEx_ConfigBreakDeadTime(TIM_HandleTypeDef *htim, */ HAL_StatusTypeDef HAL_TIMEx_RemapConfig(TIM_HandleTypeDef *htim, uint32_t Remap) { - /* Check parameters */ assert_param(IS_TIM_REMAP(htim->Instance, Remap)); @@ -2149,7 +2140,7 @@ HAL_StatusTypeDef HAL_TIMEx_RemapConfig(TIM_HandleTypeDef *htim, uint32_t Remap) */ /** - * @brief Hall commutation changed callback in non-blocking mode + * @brief Commutation callback in non-blocking mode * @param htim TIM handle * @retval None */ @@ -2163,7 +2154,7 @@ __weak void HAL_TIMEx_CommutCallback(TIM_HandleTypeDef *htim) */ } /** - * @brief Hall commutation changed half complete callback in non-blocking mode + * @brief Commutation half complete callback in non-blocking mode * @param htim TIM handle * @retval None */ @@ -2178,7 +2169,7 @@ __weak void HAL_TIMEx_CommutHalfCpltCallback(TIM_HandleTypeDef *htim) } /** - * @brief Hall Break detection callback in non-blocking mode + * @brief Break detection callback in non-blocking mode * @param htim TIM handle * @retval None */ @@ -2215,7 +2206,7 @@ __weak void HAL_TIMEx_BreakCallback(TIM_HandleTypeDef *htim) * @param htim TIM Hall Sensor handle * @retval HAL state */ -HAL_TIM_StateTypeDef HAL_TIMEx_HallSensor_GetState(TIM_HandleTypeDef *htim) +HAL_TIM_StateTypeDef HAL_TIMEx_HallSensor_GetState(const TIM_HandleTypeDef *htim) { return htim->State; } @@ -2230,7 +2221,7 @@ HAL_TIM_StateTypeDef HAL_TIMEx_HallSensor_GetState(TIM_HandleTypeDef *htim) * @arg TIM_CHANNEL_3: TIM Channel 3 * @retval TIM Complementary channel state */ -HAL_TIM_ChannelStateTypeDef HAL_TIMEx_GetChannelNState(TIM_HandleTypeDef *htim, uint32_t ChannelN) +HAL_TIM_ChannelStateTypeDef HAL_TIMEx_GetChannelNState(const TIM_HandleTypeDef *htim, uint32_t ChannelN) { HAL_TIM_ChannelStateTypeDef channel_state; @@ -2329,15 +2320,6 @@ static void TIM_DMADelayPulseNCplt(DMA_HandleTypeDef *hdma) TIM_CHANNEL_N_STATE_SET(htim, TIM_CHANNEL_3, HAL_TIM_CHANNEL_STATE_READY); } } - else if (hdma == htim->hdma[TIM_DMA_ID_CC4]) - { - htim->Channel = HAL_TIM_ACTIVE_CHANNEL_4; - - if (hdma->Init.Mode == DMA_NORMAL) - { - TIM_CHANNEL_N_STATE_SET(htim, TIM_CHANNEL_4, HAL_TIM_CHANNEL_STATE_READY); - } - } else { /* nothing to do */ @@ -2406,13 +2388,13 @@ static void TIM_CCxNChannelCmd(TIM_TypeDef *TIMx, uint32_t Channel, uint32_t Cha { uint32_t tmp; - tmp = TIM_CCER_CC1NE << (Channel & 0x1FU); /* 0x1FU = 31 bits max shift */ + tmp = TIM_CCER_CC1NE << (Channel & 0xFU); /* 0xFU = 15 bits max shift */ /* Reset the CCxNE Bit */ TIMx->CCER &= ~tmp; /* Set or reset the CCxNE Bit */ - TIMx->CCER |= (uint32_t)(ChannelNState << (Channel & 0x1FU)); /* 0x1FU = 31 bits max shift */ + TIMx->CCER |= (uint32_t)(ChannelNState << (Channel & 0xFU)); /* 0xFU = 15 bits max shift */ } /** * @} diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_timebase_rtc_alarm_template.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_timebase_rtc_alarm_template.c index e7f0ace4..e45f04c7 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_timebase_rtc_alarm_template.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_timebase_rtc_alarm_template.c @@ -6,7 +6,7 @@ * * This file override the native HAL time base functions (defined as weak) * to use the RTC ALARM for time base generation: - * + Intializes the RTC peripheral to increment the seconds registers each 1ms + * + Initializes the RTC peripheral to increment the seconds registers each 1ms * + The alarm is configured to assert an interrupt when the RTC reaches 1ms * + HAL_IncTick is called at each Alarm event and the time is reset to 00:00:00 * + HSE (default), LSE or LSI can be selected as RTC clock source @@ -102,19 +102,19 @@ HAL_StatusTypeDef HAL_InitTick(uint32_t TickPriority) HAL_StatusTypeDef status; #ifdef RTC_CLOCK_SOURCE_LSE - /* Configue LSE as RTC clock soucre */ + /* Configure LSE as RTC clock source */ RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_LSE; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE; RCC_OscInitStruct.LSEState = RCC_LSE_ON; PeriphClkInitStruct.RTCClockSelection = RCC_RTCCLKSOURCE_LSE; #elif defined (RTC_CLOCK_SOURCE_LSI) - /* Configue LSI as RTC clock soucre */ + /* Configure LSI as RTC clock source */ RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_LSI; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE; RCC_OscInitStruct.LSIState = RCC_LSI_ON; PeriphClkInitStruct.RTCClockSelection = RCC_RTCCLKSOURCE_LSI; #elif defined (RTC_CLOCK_SOURCE_HSE) - /* Configue HSE as RTC clock soucre */ + /* Configure HSE as RTC clock source */ RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE; RCC_OscInitStruct.HSEState = RCC_HSE_ON; diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_timebase_rtc_wakeup_template.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_timebase_rtc_wakeup_template.c index a9cff87d..897db4e6 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_timebase_rtc_wakeup_template.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_timebase_rtc_wakeup_template.c @@ -6,7 +6,7 @@ * * This file overrides the native HAL time base functions (defined as weak) * to use the RTC WAKEUP for the time base generation: - * + Intializes the RTC peripheral and configures the wakeup timer to be + * + Initializes the RTC peripheral and configures the wakeup timer to be * incremented each 1ms * + The wakeup feature is configured to assert an interrupt each 1ms * + HAL_IncTick is called inside the HAL_RTCEx_WakeUpTimerEventCallback @@ -109,19 +109,19 @@ HAL_StatusTypeDef HAL_InitTick (uint32_t TickPriority) HAL_StatusTypeDef status; #ifdef RTC_CLOCK_SOURCE_LSE - /* Configue LSE as RTC clock soucre */ + /* Configure LSE as RTC clock source */ RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_LSE; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE; RCC_OscInitStruct.LSEState = RCC_LSE_ON; PeriphClkInitStruct.RTCClockSelection = RCC_RTCCLKSOURCE_LSE; #elif defined (RTC_CLOCK_SOURCE_LSI) - /* Configue LSI as RTC clock soucre */ + /* Configure LSI as RTC clock source */ RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_LSI; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE; RCC_OscInitStruct.LSIState = RCC_LSI_ON; PeriphClkInitStruct.RTCClockSelection = RCC_RTCCLKSOURCE_LSI; #elif defined (RTC_CLOCK_SOURCE_HSE) - /* Configue HSE as RTC clock soucre */ + /* Configure HSE as RTC clock source */ RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE; RCC_OscInitStruct.HSEState = RCC_HSE_ON; diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_timebase_tim_template.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_timebase_tim_template.c index 8e18d974..098acf93 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_timebase_tim_template.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_timebase_tim_template.c @@ -6,7 +6,7 @@ * * This file overrides the native HAL time base functions (defined as weak) * the TIM time base: - * + Intializes the TIM peripheral generate a Period elapsed Event each 1ms + * + Initializes the TIM peripheral generate a Period elapsed Event each 1ms * + HAL_IncTick is called inside HAL_TIM_PeriodElapsedCallback ie each 1ms * ****************************************************************************** diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_uart.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_uart.c index 36b7317a..33a5f002 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_uart.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_uart.c @@ -420,6 +420,7 @@ HAL_StatusTypeDef HAL_UART_Init(UART_HandleTypeDef *huart) huart->ErrorCode = HAL_UART_ERROR_NONE; huart->gState = HAL_UART_STATE_READY; huart->RxState = HAL_UART_STATE_READY; + huart->RxEventType = HAL_UART_RXEVENT_TC; return HAL_OK; } @@ -489,6 +490,7 @@ HAL_StatusTypeDef HAL_HalfDuplex_Init(UART_HandleTypeDef *huart) huart->ErrorCode = HAL_UART_ERROR_NONE; huart->gState = HAL_UART_STATE_READY; huart->RxState = HAL_UART_STATE_READY; + huart->RxEventType = HAL_UART_RXEVENT_TC; return HAL_OK; } @@ -569,6 +571,7 @@ HAL_StatusTypeDef HAL_LIN_Init(UART_HandleTypeDef *huart, uint32_t BreakDetectLe huart->ErrorCode = HAL_UART_ERROR_NONE; huart->gState = HAL_UART_STATE_READY; huart->RxState = HAL_UART_STATE_READY; + huart->RxEventType = HAL_UART_RXEVENT_TC; return HAL_OK; } @@ -652,6 +655,7 @@ HAL_StatusTypeDef HAL_MultiProcessor_Init(UART_HandleTypeDef *huart, uint8_t Add huart->ErrorCode = HAL_UART_ERROR_NONE; huart->gState = HAL_UART_STATE_READY; huart->RxState = HAL_UART_STATE_READY; + huart->RxEventType = HAL_UART_RXEVENT_TC; return HAL_OK; } @@ -694,6 +698,7 @@ HAL_StatusTypeDef HAL_UART_DeInit(UART_HandleTypeDef *huart) huart->gState = HAL_UART_STATE_RESET; huart->RxState = HAL_UART_STATE_RESET; huart->ReceptionType = HAL_UART_RECEPTION_STANDARD; + huart->RxEventType = HAL_UART_RXEVENT_TC; /* Process Unlock */ __HAL_UNLOCK(huart); @@ -735,6 +740,8 @@ __weak void HAL_UART_MspDeInit(UART_HandleTypeDef *huart) /** * @brief Register a User UART Callback * To be used instead of the weak predefined callback + * @note The HAL_UART_RegisterCallback() may be called before HAL_UART_Init(), HAL_HalfDuplex_Init(), HAL_LIN_Init(), + * HAL_MultiProcessor_Init() to register callbacks for HAL_UART_MSPINIT_CB_ID and HAL_UART_MSPDEINIT_CB_ID * @param huart uart handle * @param CallbackID ID of the callback to be registered * This parameter can be one of the following values: @@ -763,8 +770,6 @@ HAL_StatusTypeDef HAL_UART_RegisterCallback(UART_HandleTypeDef *huart, HAL_UART_ return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(huart); if (huart->gState == HAL_UART_STATE_READY) { @@ -849,15 +854,15 @@ HAL_StatusTypeDef HAL_UART_RegisterCallback(UART_HandleTypeDef *huart, HAL_UART_ status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(huart); - return status; } /** * @brief Unregister an UART Callback * UART callaback is redirected to the weak predefined callback + * @note The HAL_UART_UnRegisterCallback() may be called before HAL_UART_Init(), HAL_HalfDuplex_Init(), + * HAL_LIN_Init(), HAL_MultiProcessor_Init() to un-register callbacks for HAL_UART_MSPINIT_CB_ID + * and HAL_UART_MSPDEINIT_CB_ID * @param huart uart handle * @param CallbackID ID of the callback to be unregistered * This parameter can be one of the following values: @@ -877,9 +882,6 @@ HAL_StatusTypeDef HAL_UART_UnRegisterCallback(UART_HandleTypeDef *huart, HAL_UAR { HAL_StatusTypeDef status = HAL_OK; - /* Process locked */ - __HAL_LOCK(huart); - if (HAL_UART_STATE_READY == huart->gState) { switch (CallbackID) @@ -963,9 +965,6 @@ HAL_StatusTypeDef HAL_UART_UnRegisterCallback(UART_HandleTypeDef *huart, HAL_UAR status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(huart); - return status; } @@ -1147,9 +1146,6 @@ HAL_StatusTypeDef HAL_UART_Transmit(UART_HandleTypeDef *huart, const uint8_t *pD return HAL_ERROR; } - /* Process Locked */ - __HAL_LOCK(huart); - huart->ErrorCode = HAL_UART_ERROR_NONE; huart->gState = HAL_UART_STATE_BUSY_TX; @@ -1171,13 +1167,12 @@ HAL_StatusTypeDef HAL_UART_Transmit(UART_HandleTypeDef *huart, const uint8_t *pD pdata16bits = NULL; } - /* Process Unlocked */ - __HAL_UNLOCK(huart); - while (huart->TxXferCount > 0U) { if (UART_WaitOnFlagUntilTimeout(huart, UART_FLAG_TXE, RESET, tickstart, Timeout) != HAL_OK) { + huart->gState = HAL_UART_STATE_READY; + return HAL_TIMEOUT; } if (pdata8bits == NULL) @@ -1195,6 +1190,8 @@ HAL_StatusTypeDef HAL_UART_Transmit(UART_HandleTypeDef *huart, const uint8_t *pD if (UART_WaitOnFlagUntilTimeout(huart, UART_FLAG_TC, RESET, tickstart, Timeout) != HAL_OK) { + huart->gState = HAL_UART_STATE_READY; + return HAL_TIMEOUT; } @@ -1235,9 +1232,6 @@ HAL_StatusTypeDef HAL_UART_Receive(UART_HandleTypeDef *huart, uint8_t *pData, ui return HAL_ERROR; } - /* Process Locked */ - __HAL_LOCK(huart); - huart->ErrorCode = HAL_UART_ERROR_NONE; huart->RxState = HAL_UART_STATE_BUSY_RX; huart->ReceptionType = HAL_UART_RECEPTION_STANDARD; @@ -1260,14 +1254,13 @@ HAL_StatusTypeDef HAL_UART_Receive(UART_HandleTypeDef *huart, uint8_t *pData, ui pdata16bits = NULL; } - /* Process Unlocked */ - __HAL_UNLOCK(huart); - /* Check the remain data to be received */ while (huart->RxXferCount > 0U) { if (UART_WaitOnFlagUntilTimeout(huart, UART_FLAG_RXNE, RESET, tickstart, Timeout) != HAL_OK) { + huart->RxState = HAL_UART_STATE_READY; + return HAL_TIMEOUT; } if (pdata8bits == NULL) @@ -1322,9 +1315,6 @@ HAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, const uint8_t return HAL_ERROR; } - /* Process Locked */ - __HAL_LOCK(huart); - huart->pTxBuffPtr = pData; huart->TxXferSize = Size; huart->TxXferCount = Size; @@ -1332,9 +1322,6 @@ HAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, const uint8_t huart->ErrorCode = HAL_UART_ERROR_NONE; huart->gState = HAL_UART_STATE_BUSY_TX; - /* Process Unlocked */ - __HAL_UNLOCK(huart); - /* Enable the UART Transmit data register empty Interrupt */ __HAL_UART_ENABLE_IT(huart, UART_IT_TXE); @@ -1367,9 +1354,6 @@ HAL_StatusTypeDef HAL_UART_Receive_IT(UART_HandleTypeDef *huart, uint8_t *pData, return HAL_ERROR; } - /* Process Locked */ - __HAL_LOCK(huart); - /* Set Reception type to Standard reception */ huart->ReceptionType = HAL_UART_RECEPTION_STANDARD; @@ -1404,9 +1388,6 @@ HAL_StatusTypeDef HAL_UART_Transmit_DMA(UART_HandleTypeDef *huart, const uint8_t return HAL_ERROR; } - /* Process Locked */ - __HAL_LOCK(huart); - huart->pTxBuffPtr = pData; huart->TxXferSize = Size; huart->TxXferCount = Size; @@ -1433,9 +1414,6 @@ HAL_StatusTypeDef HAL_UART_Transmit_DMA(UART_HandleTypeDef *huart, const uint8_t /* Clear the TC flag in the SR register by writing 0 to it */ __HAL_UART_CLEAR_FLAG(huart, UART_FLAG_TC); - /* Process Unlocked */ - __HAL_UNLOCK(huart); - /* Enable the DMA transfer for transmit request by setting the DMAT bit in the UART CR3 register */ ATOMIC_SET_BIT(huart->Instance->CR3, USART_CR3_DMAT); @@ -1470,9 +1448,6 @@ HAL_StatusTypeDef HAL_UART_Receive_DMA(UART_HandleTypeDef *huart, uint8_t *pData return HAL_ERROR; } - /* Process Locked */ - __HAL_LOCK(huart); - /* Set Reception type to Standard reception */ huart->ReceptionType = HAL_UART_RECEPTION_STANDARD; @@ -1494,9 +1469,6 @@ HAL_StatusTypeDef HAL_UART_DMAPause(UART_HandleTypeDef *huart) { uint32_t dmarequest = 0x00U; - /* Process Locked */ - __HAL_LOCK(huart); - dmarequest = HAL_IS_BIT_SET(huart->Instance->CR3, USART_CR3_DMAT); if ((huart->gState == HAL_UART_STATE_BUSY_TX) && dmarequest) { @@ -1515,9 +1487,6 @@ HAL_StatusTypeDef HAL_UART_DMAPause(UART_HandleTypeDef *huart) ATOMIC_CLEAR_BIT(huart->Instance->CR3, USART_CR3_DMAR); } - /* Process Unlocked */ - __HAL_UNLOCK(huart); - return HAL_OK; } @@ -1529,8 +1498,6 @@ HAL_StatusTypeDef HAL_UART_DMAPause(UART_HandleTypeDef *huart) */ HAL_StatusTypeDef HAL_UART_DMAResume(UART_HandleTypeDef *huart) { - /* Process Locked */ - __HAL_LOCK(huart); if (huart->gState == HAL_UART_STATE_BUSY_TX) { @@ -1554,9 +1521,6 @@ HAL_StatusTypeDef HAL_UART_DMAResume(UART_HandleTypeDef *huart) ATOMIC_SET_BIT(huart->Instance->CR3, USART_CR3_DMAR); } - /* Process Unlocked */ - __HAL_UNLOCK(huart); - return HAL_OK; } @@ -1636,11 +1600,10 @@ HAL_StatusTypeDef HAL_UARTEx_ReceiveToIdle(UART_HandleTypeDef *huart, uint8_t *p return HAL_ERROR; } - __HAL_LOCK(huart); - huart->ErrorCode = HAL_UART_ERROR_NONE; huart->RxState = HAL_UART_STATE_BUSY_RX; huart->ReceptionType = HAL_UART_RECEPTION_TOIDLE; + huart->RxEventType = HAL_UART_RXEVENT_TC; /* Init tickstart for timeout management */ tickstart = HAL_GetTick(); @@ -1660,8 +1623,6 @@ HAL_StatusTypeDef HAL_UARTEx_ReceiveToIdle(UART_HandleTypeDef *huart, uint8_t *p pdata16bits = NULL; } - __HAL_UNLOCK(huart); - /* Initialize output number of received elements */ *RxLen = 0U; @@ -1678,6 +1639,7 @@ HAL_StatusTypeDef HAL_UARTEx_ReceiveToIdle(UART_HandleTypeDef *huart, uint8_t *p /* If Set, and data has already been received, this means Idle Event is valid : End reception */ if (*RxLen > 0U) { + huart->RxEventType = HAL_UART_RXEVENT_IDLE; huart->RxState = HAL_UART_STATE_READY; return HAL_OK; @@ -1760,10 +1722,9 @@ HAL_StatusTypeDef HAL_UARTEx_ReceiveToIdle_IT(UART_HandleTypeDef *huart, uint8_t return HAL_ERROR; } - __HAL_LOCK(huart); - /* Set Reception type to reception till IDLE Event*/ huart->ReceptionType = HAL_UART_RECEPTION_TOIDLE; + huart->RxEventType = HAL_UART_RXEVENT_TC; status = UART_Start_Receive_IT(huart, pData, Size); @@ -1821,10 +1782,9 @@ HAL_StatusTypeDef HAL_UARTEx_ReceiveToIdle_DMA(UART_HandleTypeDef *huart, uint8_ return HAL_ERROR; } - __HAL_LOCK(huart); - /* Set Reception type to reception till IDLE Event*/ huart->ReceptionType = HAL_UART_RECEPTION_TOIDLE; + huart->RxEventType = HAL_UART_RXEVENT_TC; status = UART_Start_Receive_DMA(huart, pData, Size); @@ -1854,6 +1814,36 @@ HAL_StatusTypeDef HAL_UARTEx_ReceiveToIdle_DMA(UART_HandleTypeDef *huart, uint8_ } } +/** + * @brief Provide Rx Event type that has lead to RxEvent callback execution. + * @note When HAL_UARTEx_ReceiveToIdle_IT() or HAL_UARTEx_ReceiveToIdle_DMA() API are called, progress + * of reception process is provided to application through calls of Rx Event callback (either default one + * HAL_UARTEx_RxEventCallback() or user registered one). As several types of events could occur (IDLE event, + * Half Transfer, or Transfer Complete), this function allows to retrieve the Rx Event type that has lead + * to Rx Event callback execution. + * @note This function is expected to be called within the user implementation of Rx Event Callback, + * in order to provide the accurate value : + * In Interrupt Mode : + * - HAL_UART_RXEVENT_TC : when Reception has been completed (expected nb of data has been received) + * - HAL_UART_RXEVENT_IDLE : when Idle event occurred prior reception has been completed (nb of + * received data is lower than expected one) + * In DMA Mode : + * - HAL_UART_RXEVENT_TC : when Reception has been completed (expected nb of data has been received) + * - HAL_UART_RXEVENT_HT : when half of expected nb of data has been received + * - HAL_UART_RXEVENT_IDLE : when Idle event occurred prior reception has been completed (nb of + * received data is lower than expected one). + * In DMA mode, RxEvent callback could be called several times; + * When DMA is configured in Normal Mode, HT event does not stop Reception process; + * When DMA is configured in Circular Mode, HT, TC or IDLE events don't stop Reception process; + * @param huart UART handle. + * @retval Rx Event Type (returned value will be a value of @ref UART_RxEvent_Type_Values) + */ +HAL_UART_RxEventTypeTypeDef HAL_UARTEx_GetRxEventType(UART_HandleTypeDef *huart) +{ + /* Return Rx Event type value, as stored in UART handle */ + return(huart->RxEventType); +} + /** * @brief Abort ongoing transfers (blocking mode). * @param huart UART handle. @@ -2526,6 +2516,11 @@ void HAL_UART_IRQHandler(UART_HandleTypeDef *huart) /* Last bytes received, so no need as the abort is immediate */ (void)HAL_DMA_Abort(huart->hdmarx); } + + /* Initialize type of RxEvent that correspond to RxEvent callback execution; + In this case, Rx Event type is Idle Event */ + huart->RxEventType = HAL_UART_RXEVENT_IDLE; + #if (USE_HAL_UART_REGISTER_CALLBACKS == 1) /*Call registered Rx Event callback*/ huart->RxEventCallback(huart, (huart->RxXferSize - huart->RxXferCount)); @@ -2556,6 +2551,11 @@ void HAL_UART_IRQHandler(UART_HandleTypeDef *huart) huart->ReceptionType = HAL_UART_RECEPTION_STANDARD; ATOMIC_CLEAR_BIT(huart->Instance->CR1, USART_CR1_IDLEIE); + + /* Initialize type of RxEvent that correspond to RxEvent callback execution; + In this case, Rx Event type is Idle Event */ + huart->RxEventType = HAL_UART_RXEVENT_IDLE; + #if (USE_HAL_UART_REGISTER_CALLBACKS == 1) /*Call registered Rx complete callback*/ huart->RxEventCallback(huart, nb_rx_data); @@ -2791,6 +2791,7 @@ HAL_StatusTypeDef HAL_MultiProcessor_EnterMuteMode(UART_HandleTypeDef *huart) ATOMIC_SET_BIT(huart->Instance->CR1, USART_CR1_RWU); huart->gState = HAL_UART_STATE_READY; + huart->RxEventType = HAL_UART_RXEVENT_TC; /* Process Unlocked */ __HAL_UNLOCK(huart); @@ -2818,6 +2819,7 @@ HAL_StatusTypeDef HAL_MultiProcessor_ExitMuteMode(UART_HandleTypeDef *huart) ATOMIC_CLEAR_BIT(huart->Instance->CR1, USART_CR1_RWU); huart->gState = HAL_UART_STATE_READY; + huart->RxEventType = HAL_UART_RXEVENT_TC; /* Process Unlocked */ __HAL_UNLOCK(huart); @@ -2923,7 +2925,7 @@ HAL_StatusTypeDef HAL_HalfDuplex_EnableReceiver(UART_HandleTypeDef *huart) * the configuration information for the specified UART module. * @retval HAL state */ -HAL_UART_StateTypeDef HAL_UART_GetState(UART_HandleTypeDef *huart) +HAL_UART_StateTypeDef HAL_UART_GetState(const UART_HandleTypeDef *huart) { uint32_t temp1 = 0x00U, temp2 = 0x00U; temp1 = huart->gState; @@ -2938,7 +2940,7 @@ HAL_UART_StateTypeDef HAL_UART_GetState(UART_HandleTypeDef *huart) * the configuration information for the specified UART. * @retval UART Error Code */ -uint32_t HAL_UART_GetError(UART_HandleTypeDef *huart) +uint32_t HAL_UART_GetError(const UART_HandleTypeDef *huart) { return huart->ErrorCode; } @@ -3040,6 +3042,7 @@ static void UART_DMATxHalfCplt(DMA_HandleTypeDef *hdma) static void UART_DMAReceiveCplt(DMA_HandleTypeDef *hdma) { UART_HandleTypeDef *huart = (UART_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent; + /* DMA Normal mode*/ if ((hdma->Instance->CR & DMA_SxCR_CIRC) == 0U) { @@ -3063,6 +3066,10 @@ static void UART_DMAReceiveCplt(DMA_HandleTypeDef *hdma) } } + /* Initialize type of RxEvent that correspond to RxEvent callback execution; + In this case, Rx Event type is Transfer Complete */ + huart->RxEventType = HAL_UART_RXEVENT_TC; + /* Check current reception Mode : If Reception till IDLE event has been selected : use Rx Event callback */ if (huart->ReceptionType == HAL_UART_RECEPTION_TOIDLE) @@ -3098,6 +3105,10 @@ static void UART_DMARxHalfCplt(DMA_HandleTypeDef *hdma) { UART_HandleTypeDef *huart = (UART_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent; + /* Initialize type of RxEvent that correspond to RxEvent callback execution; + In this case, Rx Event type is Half Transfer */ + huart->RxEventType = HAL_UART_RXEVENT_HT; + /* Check current reception Mode : If Reception till IDLE event has been selected : use Rx Event callback */ if (huart->ReceptionType == HAL_UART_RECEPTION_TOIDLE) @@ -3180,19 +3191,31 @@ static HAL_StatusTypeDef UART_WaitOnFlagUntilTimeout(UART_HandleTypeDef *huart, /* Check for the Timeout */ if (Timeout != HAL_MAX_DELAY) { - if ((Timeout == 0U) || ((HAL_GetTick() - Tickstart) > Timeout)) + if (((HAL_GetTick() - Tickstart) > Timeout) || (Timeout == 0U)) { - /* Disable TXE, RXNE, PE and ERR (Frame error, noise error, overrun error) interrupts for the interrupt process */ - ATOMIC_CLEAR_BIT(huart->Instance->CR1, (USART_CR1_RXNEIE | USART_CR1_PEIE | USART_CR1_TXEIE)); - ATOMIC_CLEAR_BIT(huart->Instance->CR3, USART_CR3_EIE); - huart->gState = HAL_UART_STATE_READY; - huart->RxState = HAL_UART_STATE_READY; + return HAL_TIMEOUT; + } - /* Process Unlocked */ - __HAL_UNLOCK(huart); + if ((READ_BIT(huart->Instance->CR1, USART_CR1_RE) != 0U) && (Flag != UART_FLAG_TXE) && (Flag != UART_FLAG_TC)) + { + if (__HAL_UART_GET_FLAG(huart, UART_FLAG_ORE) == SET) + { + /* Clear Overrun Error flag*/ + __HAL_UART_CLEAR_OREFLAG(huart); - return HAL_TIMEOUT; + /* Blocking error : transfer is aborted + Set the UART state ready to be able to start again the process, + Disable Rx Interrupts if ongoing */ + UART_EndRxTransfer(huart); + + huart->ErrorCode = HAL_UART_ERROR_ORE; + + /* Process Unlocked */ + __HAL_UNLOCK(huart); + + return HAL_ERROR; + } } } } @@ -3219,9 +3242,6 @@ HAL_StatusTypeDef UART_Start_Receive_IT(UART_HandleTypeDef *huart, uint8_t *pDat huart->ErrorCode = HAL_UART_ERROR_NONE; huart->RxState = HAL_UART_STATE_BUSY_RX; - /* Process Unlocked */ - __HAL_UNLOCK(huart); - if (huart->Init.Parity != UART_PARITY_NONE) { /* Enable the UART Parity Error Interrupt */ @@ -3277,9 +3297,6 @@ HAL_StatusTypeDef UART_Start_Receive_DMA(UART_HandleTypeDef *huart, uint8_t *pDa /* Clear the Overrun flag just before enabling the DMA Rx request: can be mandatory for the second transfer */ __HAL_UART_CLEAR_OREFLAG(huart); - /* Process Unlocked */ - __HAL_UNLOCK(huart); - if (huart->Init.Parity != UART_PARITY_NONE) { /* Enable the UART Parity Error Interrupt */ @@ -3619,6 +3636,9 @@ static HAL_StatusTypeDef UART_Receive_IT(UART_HandleTypeDef *huart) /* Rx process is completed, restore huart->RxState to Ready */ huart->RxState = HAL_UART_STATE_READY; + /* Initialize type of RxEvent to Transfer Complete */ + huart->RxEventType = HAL_UART_RXEVENT_TC; + /* Check current reception Mode : If Reception till IDLE event has been selected : */ if (huart->ReceptionType == HAL_UART_RECEPTION_TOIDLE) diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_usart.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_usart.c index 945a6469..50c1b817 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_usart.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_usart.c @@ -427,6 +427,8 @@ __weak void HAL_USART_MspDeInit(USART_HandleTypeDef *husart) /** * @brief Register a User USART Callback * To be used instead of the weak predefined callback + * @note The HAL_USART_RegisterCallback() may be called before HAL_USART_Init() in HAL_USART_STATE_RESET + * to register callbacks for HAL_USART_MSPINIT_CB_ID and HAL_USART_MSPDEINIT_CB_ID * @param husart usart handle * @param CallbackID ID of the callback to be registered * This parameter can be one of the following values: @@ -454,8 +456,6 @@ HAL_StatusTypeDef HAL_USART_RegisterCallback(USART_HandleTypeDef *husart, HAL_US return HAL_ERROR; } - /* Process locked */ - __HAL_LOCK(husart); if (husart->State == HAL_USART_STATE_READY) { @@ -536,15 +536,14 @@ HAL_StatusTypeDef HAL_USART_RegisterCallback(USART_HandleTypeDef *husart, HAL_US status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(husart); - return status; } /** * @brief Unregister an USART Callback * USART callaback is redirected to the weak predefined callback + * @note The HAL_USART_UnRegisterCallback() may be called before HAL_USART_Init() in HAL_USART_STATE_RESET + * to un-register callbacks for HAL_USART_MSPINIT_CB_ID and HAL_USART_MSPDEINIT_CB_ID * @param husart usart handle * @param CallbackID ID of the callback to be unregistered * This parameter can be one of the following values: @@ -563,9 +562,6 @@ HAL_StatusTypeDef HAL_USART_UnRegisterCallback(USART_HandleTypeDef *husart, HAL_ { HAL_StatusTypeDef status = HAL_OK; - /* Process locked */ - __HAL_LOCK(husart); - if (husart->State == HAL_USART_STATE_READY) { switch (CallbackID) @@ -645,9 +641,6 @@ HAL_StatusTypeDef HAL_USART_UnRegisterCallback(USART_HandleTypeDef *husart, HAL_ status = HAL_ERROR; } - /* Release Lock */ - __HAL_UNLOCK(husart); - return status; } #endif /* USE_HAL_USART_REGISTER_CALLBACKS */ @@ -2075,7 +2068,7 @@ __weak void HAL_USART_AbortCpltCallback(USART_HandleTypeDef *husart) * the configuration information for the specified USART module. * @retval HAL state */ -HAL_USART_StateTypeDef HAL_USART_GetState(USART_HandleTypeDef *husart) +HAL_USART_StateTypeDef HAL_USART_GetState(const USART_HandleTypeDef *husart) { return husart->State; } @@ -2086,11 +2079,14 @@ HAL_USART_StateTypeDef HAL_USART_GetState(USART_HandleTypeDef *husart) * the configuration information for the specified USART. * @retval USART Error Code */ -uint32_t HAL_USART_GetError(USART_HandleTypeDef *husart) +uint32_t HAL_USART_GetError(const USART_HandleTypeDef *husart) { return husart->ErrorCode; } +/** + * @} + */ /** * @} */ @@ -2819,10 +2815,6 @@ static void USART_SetConfig(USART_HandleTypeDef *husart) } } -/** - * @} - */ - /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_adc.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_adc.c index 2d6e0828..cdb750fe 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_adc.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_adc.c @@ -22,9 +22,9 @@ #include "stm32f4xx_ll_bus.h" #ifdef USE_FULL_ASSERT - #include "stm32_assert.h" +#include "stm32_assert.h" #else - #define assert_param(expr) ((void)0U) +#define assert_param(expr) ((void)0U) #endif /** @addtogroup STM32F4xx_LL_Driver @@ -326,7 +326,7 @@ ErrorStatus LL_ADC_CommonInit(ADC_Common_TypeDef *ADCxy_COMMON, LL_ADC_CommonIni #if defined(ADC_MULTIMODE_SUPPORT) assert_param(IS_LL_ADC_MULTI_MODE(ADC_CommonInitStruct->Multimode)); - if(ADC_CommonInitStruct->Multimode != LL_ADC_MULTI_INDEPENDENT) + if (ADC_CommonInitStruct->Multimode != LL_ADC_MULTI_INDEPENDENT) { assert_param(IS_LL_ADC_MULTI_DMA_TRANSFER(ADC_CommonInitStruct->MultiDMATransfer)); assert_param(IS_LL_ADC_MULTI_TWOSMP_DELAY(ADC_CommonInitStruct->MultiTwoSamplingDelay)); @@ -338,7 +338,7 @@ ErrorStatus LL_ADC_CommonInit(ADC_Common_TypeDef *ADCxy_COMMON, LL_ADC_CommonIni /* On this STM32 series, setting of these features is conditioned to */ /* ADC state: */ /* All ADC instances of the ADC common group must be disabled. */ - if(__LL_ADC_IS_ENABLED_ALL_COMMON_INSTANCE(ADCxy_COMMON) == 0UL) + if (__LL_ADC_IS_ENABLED_ALL_COMMON_INSTANCE(ADCxy_COMMON) == 0UL) { /* Configuration of ADC hierarchical scope: */ /* - common to several ADC */ @@ -350,16 +350,16 @@ ErrorStatus LL_ADC_CommonInit(ADC_Common_TypeDef *ADCxy_COMMON, LL_ADC_CommonIni /* - Set ADC multimode DMA transfer */ /* - Set ADC multimode: delay between 2 sampling phases */ #if defined(ADC_MULTIMODE_SUPPORT) - if(ADC_CommonInitStruct->Multimode != LL_ADC_MULTI_INDEPENDENT) + if (ADC_CommonInitStruct->Multimode != LL_ADC_MULTI_INDEPENDENT) { MODIFY_REG(ADCxy_COMMON->CCR, - ADC_CCR_ADCPRE + ADC_CCR_ADCPRE | ADC_CCR_MULTI | ADC_CCR_DMA | ADC_CCR_DDS | ADC_CCR_DELAY - , - ADC_CommonInitStruct->CommonClock + , + ADC_CommonInitStruct->CommonClock | ADC_CommonInitStruct->Multimode | ADC_CommonInitStruct->MultiDMATransfer | ADC_CommonInitStruct->MultiTwoSamplingDelay @@ -368,13 +368,13 @@ ErrorStatus LL_ADC_CommonInit(ADC_Common_TypeDef *ADCxy_COMMON, LL_ADC_CommonIni else { MODIFY_REG(ADCxy_COMMON->CCR, - ADC_CCR_ADCPRE + ADC_CCR_ADCPRE | ADC_CCR_MULTI | ADC_CCR_DMA | ADC_CCR_DDS | ADC_CCR_DELAY - , - ADC_CommonInitStruct->CommonClock + , + ADC_CommonInitStruct->CommonClock | LL_ADC_MULTI_INDEPENDENT ); } @@ -408,7 +408,7 @@ void LL_ADC_CommonStructInit(LL_ADC_CommonInitTypeDef *ADC_CommonInitStruct) #if defined(ADC_MULTIMODE_SUPPORT) /* Set fields of ADC multimode */ ADC_CommonInitStruct->Multimode = LL_ADC_MULTI_INDEPENDENT; - ADC_CommonInitStruct->MultiDMATransfer = LL_ADC_MULTI_REG_DMA_EACH_ADC; + ADC_CommonInitStruct->MultiDMATransfer = LL_ADC_MULTI_REG_DMA_EACH_ADC; ADC_CommonInitStruct->MultiTwoSamplingDelay = LL_ADC_MULTI_TWOSMP_DELAY_5CYCLES; #endif /* ADC_MULTIMODE_SUPPORT */ } @@ -431,7 +431,7 @@ ErrorStatus LL_ADC_DeInit(ADC_TypeDef *ADCx) assert_param(IS_ADC_ALL_INSTANCE(ADCx)); /* Disable ADC instance if not already disabled. */ - if(LL_ADC_IsEnabled(ADCx) == 1UL) + if (LL_ADC_IsEnabled(ADCx) == 1UL) { /* Set ADC group regular trigger source to SW start to ensure to not */ /* have an external trigger event occurring during the conversion stop */ @@ -449,48 +449,48 @@ ErrorStatus LL_ADC_DeInit(ADC_TypeDef *ADCx) /* Check whether ADC state is compliant with expected state */ /* (hardware requirements of bits state to reset registers below) */ - if(READ_BIT(ADCx->CR2, ADC_CR2_ADON) == 0UL) + if (READ_BIT(ADCx->CR2, ADC_CR2_ADON) == 0UL) { /* ========== Reset ADC registers ========== */ /* Reset register SR */ CLEAR_BIT(ADCx->SR, - ( LL_ADC_FLAG_STRT + (LL_ADC_FLAG_STRT | LL_ADC_FLAG_JSTRT | LL_ADC_FLAG_EOCS | LL_ADC_FLAG_OVR | LL_ADC_FLAG_JEOS - | LL_ADC_FLAG_AWD1 ) + | LL_ADC_FLAG_AWD1) ); /* Reset register CR1 */ CLEAR_BIT(ADCx->CR1, - ( ADC_CR1_OVRIE | ADC_CR1_RES | ADC_CR1_AWDEN + (ADC_CR1_OVRIE | ADC_CR1_RES | ADC_CR1_AWDEN | ADC_CR1_JAWDEN | ADC_CR1_DISCNUM | ADC_CR1_JDISCEN | ADC_CR1_DISCEN | ADC_CR1_JAUTO | ADC_CR1_AWDSGL | ADC_CR1_SCAN | ADC_CR1_JEOCIE | ADC_CR1_AWDIE | ADC_CR1_EOCIE - | ADC_CR1_AWDCH ) + | ADC_CR1_AWDCH) ); /* Reset register CR2 */ CLEAR_BIT(ADCx->CR2, - ( ADC_CR2_SWSTART | ADC_CR2_EXTEN | ADC_CR2_EXTSEL + (ADC_CR2_SWSTART | ADC_CR2_EXTEN | ADC_CR2_EXTSEL | ADC_CR2_JSWSTART | ADC_CR2_JEXTEN | ADC_CR2_JEXTSEL | ADC_CR2_ALIGN | ADC_CR2_EOCS | ADC_CR2_DDS | ADC_CR2_DMA - | ADC_CR2_CONT | ADC_CR2_ADON ) + | ADC_CR2_CONT | ADC_CR2_ADON) ); /* Reset register SMPR1 */ CLEAR_BIT(ADCx->SMPR1, - ( ADC_SMPR1_SMP18 | ADC_SMPR1_SMP17 | ADC_SMPR1_SMP16 + (ADC_SMPR1_SMP18 | ADC_SMPR1_SMP17 | ADC_SMPR1_SMP16 | ADC_SMPR1_SMP15 | ADC_SMPR1_SMP14 | ADC_SMPR1_SMP13 | ADC_SMPR1_SMP12 | ADC_SMPR1_SMP11 | ADC_SMPR1_SMP10) ); /* Reset register SMPR2 */ CLEAR_BIT(ADCx->SMPR2, - ( ADC_SMPR2_SMP9 + (ADC_SMPR2_SMP9 | ADC_SMPR2_SMP8 | ADC_SMPR2_SMP7 | ADC_SMPR2_SMP6 | ADC_SMPR2_SMP5 | ADC_SMPR2_SMP4 | ADC_SMPR2_SMP3 | ADC_SMPR2_SMP2 | ADC_SMPR2_SMP1 | ADC_SMPR2_SMP0) @@ -512,28 +512,28 @@ ErrorStatus LL_ADC_DeInit(ADC_TypeDef *ADCx) /* Reset register SQR1 */ CLEAR_BIT(ADCx->SQR1, - ( ADC_SQR1_L + (ADC_SQR1_L | ADC_SQR1_SQ16 | ADC_SQR1_SQ15 | ADC_SQR1_SQ14 | ADC_SQR1_SQ13) ); /* Reset register SQR2 */ CLEAR_BIT(ADCx->SQR2, - ( ADC_SQR2_SQ12 | ADC_SQR2_SQ11 | ADC_SQR2_SQ10 + (ADC_SQR2_SQ12 | ADC_SQR2_SQ11 | ADC_SQR2_SQ10 | ADC_SQR2_SQ9 | ADC_SQR2_SQ8 | ADC_SQR2_SQ7) ); /* Reset register SQR3 */ CLEAR_BIT(ADCx->SQR3, - ( ADC_SQR3_SQ6 | ADC_SQR3_SQ5 | ADC_SQR3_SQ4 + (ADC_SQR3_SQ6 | ADC_SQR3_SQ5 | ADC_SQR3_SQ4 | ADC_SQR3_SQ3 | ADC_SQR3_SQ2 | ADC_SQR3_SQ1) ); /* Reset register JSQR */ CLEAR_BIT(ADCx->JSQR, - ( ADC_JSQR_JL + (ADC_JSQR_JL | ADC_JSQR_JSQ4 | ADC_JSQR_JSQ3 - | ADC_JSQR_JSQ2 | ADC_JSQR_JSQ1 ) + | ADC_JSQR_JSQ2 | ADC_JSQR_JSQ1) ); /* Reset register DR */ @@ -595,24 +595,24 @@ ErrorStatus LL_ADC_Init(ADC_TypeDef *ADCx, LL_ADC_InitTypeDef *ADC_InitStruct) /* Note: Hardware constraint (refer to description of this function): */ /* ADC instance must be disabled. */ - if(LL_ADC_IsEnabled(ADCx) == 0UL) + if (LL_ADC_IsEnabled(ADCx) == 0UL) { /* Configuration of ADC hierarchical scope: */ /* - ADC instance */ /* - Set ADC data resolution */ /* - Set ADC conversion data alignment */ MODIFY_REG(ADCx->CR1, - ADC_CR1_RES + ADC_CR1_RES | ADC_CR1_SCAN - , - ADC_InitStruct->Resolution + , + ADC_InitStruct->Resolution | ADC_InitStruct->SequencersScanMode ); MODIFY_REG(ADCx->CR2, - ADC_CR2_ALIGN - , - ADC_InitStruct->DataAlignment + ADC_CR2_ALIGN + , + ADC_InitStruct->DataAlignment ); } @@ -685,7 +685,7 @@ ErrorStatus LL_ADC_REG_Init(ADC_TypeDef *ADCx, LL_ADC_REG_InitTypeDef *ADC_REG_I assert_param(IS_ADC_ALL_INSTANCE(ADCx)); assert_param(IS_LL_ADC_REG_TRIG_SOURCE(ADC_REG_InitStruct->TriggerSource)); assert_param(IS_LL_ADC_REG_SEQ_SCAN_LENGTH(ADC_REG_InitStruct->SequencerLength)); - if(ADC_REG_InitStruct->SequencerLength != LL_ADC_REG_SEQ_SCAN_DISABLE) + if (ADC_REG_InitStruct->SequencerLength != LL_ADC_REG_SEQ_SCAN_DISABLE) { assert_param(IS_LL_ADC_REG_SEQ_SCAN_DISCONT_MODE(ADC_REG_InitStruct->SequencerDiscont)); } @@ -699,7 +699,7 @@ ErrorStatus LL_ADC_REG_Init(ADC_TypeDef *ADCx, LL_ADC_REG_InitTypeDef *ADC_REG_I /* Note: Hardware constraint (refer to description of this function): */ /* ADC instance must be disabled. */ - if(LL_ADC_IsEnabled(ADCx) == 0UL) + if (LL_ADC_IsEnabled(ADCx) == 0UL) { /* Configuration of ADC hierarchical scope: */ /* - ADC group regular */ @@ -712,33 +712,33 @@ ErrorStatus LL_ADC_REG_Init(ADC_TypeDef *ADCx, LL_ADC_REG_InitTypeDef *ADC_REG_I /* Note: On this STM32 series, ADC trigger edge is set when starting */ /* ADC conversion. */ /* Refer to function @ref LL_ADC_REG_StartConversionExtTrig(). */ - if(ADC_REG_InitStruct->SequencerLength != LL_ADC_REG_SEQ_SCAN_DISABLE) + if (ADC_REG_InitStruct->SequencerLength != LL_ADC_REG_SEQ_SCAN_DISABLE) { MODIFY_REG(ADCx->CR1, - ADC_CR1_DISCEN + ADC_CR1_DISCEN | ADC_CR1_DISCNUM - , - ADC_REG_InitStruct->SequencerDiscont + , + ADC_REG_InitStruct->SequencerDiscont ); } else { MODIFY_REG(ADCx->CR1, - ADC_CR1_DISCEN + ADC_CR1_DISCEN | ADC_CR1_DISCNUM - , - LL_ADC_REG_SEQ_DISCONT_DISABLE + , + LL_ADC_REG_SEQ_DISCONT_DISABLE ); } MODIFY_REG(ADCx->CR2, - ADC_CR2_EXTSEL + ADC_CR2_EXTSEL | ADC_CR2_EXTEN | ADC_CR2_CONT | ADC_CR2_DMA | ADC_CR2_DDS - , - (ADC_REG_InitStruct->TriggerSource & ADC_CR2_EXTSEL) + , + (ADC_REG_InitStruct->TriggerSource & ADC_CR2_EXTSEL) | ADC_REG_InitStruct->ContinuousMode | ADC_REG_InitStruct->DMATransfer ); @@ -820,7 +820,7 @@ ErrorStatus LL_ADC_INJ_Init(ADC_TypeDef *ADCx, LL_ADC_INJ_InitTypeDef *ADC_INJ_I assert_param(IS_ADC_ALL_INSTANCE(ADCx)); assert_param(IS_LL_ADC_INJ_TRIG_SOURCE(ADC_INJ_InitStruct->TriggerSource)); assert_param(IS_LL_ADC_INJ_SEQ_SCAN_LENGTH(ADC_INJ_InitStruct->SequencerLength)); - if(ADC_INJ_InitStruct->SequencerLength != LL_ADC_INJ_SEQ_SCAN_DISABLE) + if (ADC_INJ_InitStruct->SequencerLength != LL_ADC_INJ_SEQ_SCAN_DISABLE) { assert_param(IS_LL_ADC_INJ_SEQ_SCAN_DISCONT_MODE(ADC_INJ_InitStruct->SequencerDiscont)); } @@ -828,7 +828,7 @@ ErrorStatus LL_ADC_INJ_Init(ADC_TypeDef *ADCx, LL_ADC_INJ_InitTypeDef *ADC_INJ_I /* Note: Hardware constraint (refer to description of this function): */ /* ADC instance must be disabled. */ - if(LL_ADC_IsEnabled(ADCx) == 0UL) + if (LL_ADC_IsEnabled(ADCx) == 0UL) { /* Configuration of ADC hierarchical scope: */ /* - ADC group injected */ @@ -840,32 +840,32 @@ ErrorStatus LL_ADC_INJ_Init(ADC_TypeDef *ADCx, LL_ADC_INJ_InitTypeDef *ADC_INJ_I /* Note: On this STM32 series, ADC trigger edge is set when starting */ /* ADC conversion. */ /* Refer to function @ref LL_ADC_INJ_StartConversionExtTrig(). */ - if(ADC_INJ_InitStruct->SequencerLength != LL_ADC_REG_SEQ_SCAN_DISABLE) + if (ADC_INJ_InitStruct->SequencerLength != LL_ADC_REG_SEQ_SCAN_DISABLE) { MODIFY_REG(ADCx->CR1, - ADC_CR1_JDISCEN + ADC_CR1_JDISCEN | ADC_CR1_JAUTO - , - ADC_INJ_InitStruct->SequencerDiscont + , + ADC_INJ_InitStruct->SequencerDiscont | ADC_INJ_InitStruct->TrigAuto ); } else { MODIFY_REG(ADCx->CR1, - ADC_CR1_JDISCEN + ADC_CR1_JDISCEN | ADC_CR1_JAUTO - , - LL_ADC_REG_SEQ_DISCONT_DISABLE + , + LL_ADC_REG_SEQ_DISCONT_DISABLE | ADC_INJ_InitStruct->TrigAuto ); } MODIFY_REG(ADCx->CR2, - ADC_CR2_JEXTSEL + ADC_CR2_JEXTSEL | ADC_CR2_JEXTEN - , - (ADC_INJ_InitStruct->TriggerSource & ADC_CR2_JEXTSEL) + , + (ADC_INJ_InitStruct->TriggerSource & ADC_CR2_JEXTSEL) ); /* Note: Hardware constraint (refer to description of this function): */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_crc.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_crc.c index 11387930..b64b4761 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_crc.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_crc.c @@ -59,7 +59,7 @@ * - SUCCESS: CRC registers are de-initialized * - ERROR: CRC registers are not de-initialized */ -ErrorStatus LL_CRC_DeInit(CRC_TypeDef *CRCx) +ErrorStatus LL_CRC_DeInit(const CRC_TypeDef *CRCx) { ErrorStatus status = SUCCESS; diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_dac.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_dac.c index 4514d918..77ce3587 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_dac.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_dac.c @@ -47,19 +47,16 @@ */ #if defined(DAC_CHANNEL2_SUPPORT) #define IS_LL_DAC_CHANNEL(__DACX__, __DAC_CHANNEL__) \ - ( \ - ((__DAC_CHANNEL__) == LL_DAC_CHANNEL_1) \ + (((__DAC_CHANNEL__) == LL_DAC_CHANNEL_1) \ || ((__DAC_CHANNEL__) == LL_DAC_CHANNEL_2) \ ) #else #define IS_LL_DAC_CHANNEL(__DACX__, __DAC_CHANNEL__) \ - ( \ - ((__DAC_CHANNEL__) == LL_DAC_CHANNEL_1) \ - ) + (((__DAC_CHANNEL__) == LL_DAC_CHANNEL_1)) #endif /* DAC_CHANNEL2_SUPPORT */ #define IS_LL_DAC_TRIGGER_SOURCE(__TRIGGER_SOURCE__) \ - ( ((__TRIGGER_SOURCE__) == LL_DAC_TRIG_SOFTWARE) \ + (((__TRIGGER_SOURCE__) == LL_DAC_TRIG_SOFTWARE) \ || ((__TRIGGER_SOURCE__) == LL_DAC_TRIG_EXT_TIM2_TRGO) \ || ((__TRIGGER_SOURCE__) == LL_DAC_TRIG_EXT_TIM4_TRGO) \ || ((__TRIGGER_SOURCE__) == LL_DAC_TRIG_EXT_TIM5_TRGO) \ @@ -70,45 +67,45 @@ ) #define IS_LL_DAC_WAVE_AUTO_GENER_MODE(__WAVE_AUTO_GENERATION_MODE__) \ - ( ((__WAVE_AUTO_GENERATION_MODE__) == LL_DAC_WAVE_AUTO_GENERATION_NONE) \ - || ((__WAVE_AUTO_GENERATION_MODE__) == LL_DAC_WAVE_AUTO_GENERATION_NOISE) \ - || ((__WAVE_AUTO_GENERATION_MODE__) == LL_DAC_WAVE_AUTO_GENERATION_TRIANGLE) \ + (((__WAVE_AUTO_GENERATION_MODE__) == LL_DAC_WAVE_AUTO_GENERATION_NONE) \ + || ((__WAVE_AUTO_GENERATION_MODE__) == LL_DAC_WAVE_AUTO_GENERATION_NOISE) \ + || ((__WAVE_AUTO_GENERATION_MODE__) == LL_DAC_WAVE_AUTO_GENERATION_TRIANGLE) \ ) #define IS_LL_DAC_WAVE_AUTO_GENER_CONFIG(__WAVE_AUTO_GENERATION_MODE__, __WAVE_AUTO_GENERATION_CONFIG__) \ ( (((__WAVE_AUTO_GENERATION_MODE__) == LL_DAC_WAVE_AUTO_GENERATION_NOISE) \ - && ( ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BIT0) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS1_0) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS2_0) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS3_0) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS4_0) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS5_0) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS6_0) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS7_0) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS8_0) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS9_0) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS10_0) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS11_0)) \ + && (((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BIT0) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS1_0) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS2_0) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS3_0) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS4_0) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS5_0) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS6_0) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS7_0) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS8_0) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS9_0) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS10_0) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_NOISE_LFSR_UNMASK_BITS11_0)) \ ) \ ||(((__WAVE_AUTO_GENERATION_MODE__) == LL_DAC_WAVE_AUTO_GENERATION_TRIANGLE) \ - && ( ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_1) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_3) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_7) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_15) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_31) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_63) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_127) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_255) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_511) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_1023) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_2047) \ - || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_4095)) \ + && (((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_1) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_3) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_7) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_15) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_31) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_63) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_127) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_255) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_511) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_1023) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_2047) \ + || ((__WAVE_AUTO_GENERATION_CONFIG__) == LL_DAC_TRIANGLE_AMPLITUDE_4095)) \ ) \ ) #define IS_LL_DAC_OUTPUT_BUFFER(__OUTPUT_BUFFER__) \ - ( ((__OUTPUT_BUFFER__) == LL_DAC_OUTPUT_BUFFER_ENABLE) \ - || ((__OUTPUT_BUFFER__) == LL_DAC_OUTPUT_BUFFER_DISABLE) \ + (((__OUTPUT_BUFFER__) == LL_DAC_OUTPUT_BUFFER_ENABLE) \ + || ((__OUTPUT_BUFFER__) == LL_DAC_OUTPUT_BUFFER_DISABLE) \ ) /** @@ -135,7 +132,7 @@ * - SUCCESS: DAC registers are de-initialized * - ERROR: not applicable */ -ErrorStatus LL_DAC_DeInit(DAC_TypeDef *DACx) +ErrorStatus LL_DAC_DeInit(const DAC_TypeDef *DACx) { /* Check the parameters */ assert_param(IS_DAC_ALL_INSTANCE(DACx)); @@ -170,14 +167,14 @@ ErrorStatus LL_DAC_DeInit(DAC_TypeDef *DACx) * @arg @ref LL_DAC_CHANNEL_1 * @arg @ref LL_DAC_CHANNEL_2 (1) * - * (1) On this STM32 serie, parameter not available on all devices. + * (1) On this STM32 series, parameter not available on all devices. * Refer to device datasheet for channels availability. * @param DAC_InitStruct Pointer to a @ref LL_DAC_InitTypeDef structure * @retval An ErrorStatus enumeration value: * - SUCCESS: DAC registers are initialized * - ERROR: DAC registers are not initialized */ -ErrorStatus LL_DAC_Init(DAC_TypeDef *DACx, uint32_t DAC_Channel, LL_DAC_InitTypeDef *DAC_InitStruct) +ErrorStatus LL_DAC_Init(DAC_TypeDef *DACx, uint32_t DAC_Channel, const LL_DAC_InitTypeDef *DAC_InitStruct) { ErrorStatus status = SUCCESS; @@ -277,4 +274,3 @@ void LL_DAC_StructInit(LL_DAC_InitTypeDef *DAC_InitStruct) */ #endif /* USE_FULL_LL_DRIVER */ - diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_fmc.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_fmc.c index 1bdf9dea..24312139 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_fmc.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_fmc.c @@ -61,7 +61,8 @@ /** @addtogroup STM32F4xx_HAL_Driver * @{ */ -#if defined(HAL_NOR_MODULE_ENABLED) || defined(HAL_SRAM_MODULE_ENABLED) || (defined(HAL_NAND_MODULE_ENABLED)) || defined(HAL_PCCARD_MODULE_ENABLED) || defined(HAL_SDRAM_MODULE_ENABLED) +#if defined(HAL_NOR_MODULE_ENABLED) || (defined(HAL_NAND_MODULE_ENABLED)) || defined(HAL_PCCARD_MODULE_ENABLED) || defined(HAL_SDRAM_MODULE_ENABLED)\ + || defined(HAL_SRAM_MODULE_ENABLED) /** @defgroup FMC_LL FMC Low Layer * @brief FMC driver modules @@ -1449,7 +1450,7 @@ HAL_StatusTypeDef FMC_SDRAM_SetAutoRefreshNumber(FMC_SDRAM_TypeDef *Device, * FMC_SDRAM_NORMAL_MODE, FMC_SDRAM_SELF_REFRESH_MODE or * FMC_SDRAM_POWER_DOWN_MODE. */ -uint32_t FMC_SDRAM_GetModeStatus(FMC_SDRAM_TypeDef *Device, uint32_t Bank) +uint32_t FMC_SDRAM_GetModeStatus(const FMC_SDRAM_TypeDef *Device, uint32_t Bank) { uint32_t tmpreg; diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_fmpi2c.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_fmpi2c.c index 04966ea0..94718b7b 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_fmpi2c.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_fmpi2c.c @@ -46,22 +46,22 @@ */ #define IS_LL_FMPI2C_PERIPHERAL_MODE(__VALUE__) (((__VALUE__) == LL_FMPI2C_MODE_I2C) || \ - ((__VALUE__) == LL_FMPI2C_MODE_SMBUS_HOST) || \ - ((__VALUE__) == LL_FMPI2C_MODE_SMBUS_DEVICE) || \ - ((__VALUE__) == LL_FMPI2C_MODE_SMBUS_DEVICE_ARP)) + ((__VALUE__) == LL_FMPI2C_MODE_SMBUS_HOST) || \ + ((__VALUE__) == LL_FMPI2C_MODE_SMBUS_DEVICE) || \ + ((__VALUE__) == LL_FMPI2C_MODE_SMBUS_DEVICE_ARP)) #define IS_LL_FMPI2C_ANALOG_FILTER(__VALUE__) (((__VALUE__) == LL_FMPI2C_ANALOGFILTER_ENABLE) || \ - ((__VALUE__) == LL_FMPI2C_ANALOGFILTER_DISABLE)) + ((__VALUE__) == LL_FMPI2C_ANALOGFILTER_DISABLE)) #define IS_LL_FMPI2C_DIGITAL_FILTER(__VALUE__) ((__VALUE__) <= 0x0000000FU) #define IS_LL_FMPI2C_OWN_ADDRESS1(__VALUE__) ((__VALUE__) <= 0x000003FFU) #define IS_LL_FMPI2C_TYPE_ACKNOWLEDGE(__VALUE__) (((__VALUE__) == LL_FMPI2C_ACK) || \ - ((__VALUE__) == LL_FMPI2C_NACK)) + ((__VALUE__) == LL_FMPI2C_NACK)) #define IS_LL_FMPI2C_OWN_ADDRSIZE(__VALUE__) (((__VALUE__) == LL_FMPI2C_OWNADDRESS1_7BIT) || \ - ((__VALUE__) == LL_FMPI2C_OWNADDRESS1_10BIT)) + ((__VALUE__) == LL_FMPI2C_OWNADDRESS1_10BIT)) /** * @} */ @@ -84,7 +84,7 @@ * - SUCCESS: FMPI2C registers are de-initialized * - ERROR: FMPI2C registers are not de-initialized */ -ErrorStatus LL_FMPI2C_DeInit(FMPI2C_TypeDef *FMPI2Cx) +ErrorStatus LL_FMPI2C_DeInit(const FMPI2C_TypeDef *FMPI2Cx) { ErrorStatus status = SUCCESS; @@ -115,7 +115,7 @@ ErrorStatus LL_FMPI2C_DeInit(FMPI2C_TypeDef *FMPI2Cx) * - SUCCESS: FMPI2C registers are initialized * - ERROR: Not applicable */ -ErrorStatus LL_FMPI2C_Init(FMPI2C_TypeDef *FMPI2Cx, LL_FMPI2C_InitTypeDef *FMPI2C_InitStruct) +ErrorStatus LL_FMPI2C_Init(FMPI2C_TypeDef *FMPI2Cx, const LL_FMPI2C_InitTypeDef *FMPI2C_InitStruct) { /* Check the FMPI2C Instance FMPI2Cx */ assert_param(IS_FMPI2C_ALL_INSTANCE(FMPI2Cx)); diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_fsmc.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_fsmc.c index 8172871f..ff791118 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_fsmc.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_fsmc.c @@ -59,7 +59,8 @@ /** @addtogroup STM32F4xx_HAL_Driver * @{ */ -#if defined(HAL_NOR_MODULE_ENABLED) || defined(HAL_SRAM_MODULE_ENABLED) || defined(HAL_NAND_MODULE_ENABLED) || defined(HAL_PCCARD_MODULE_ENABLED) +#if defined(HAL_NOR_MODULE_ENABLED) || defined(HAL_NAND_MODULE_ENABLED) || defined(HAL_PCCARD_MODULE_ENABLED) \ + || defined(HAL_SRAM_MODULE_ENABLED) /** @defgroup FSMC_LL FSMC Low Layer * @brief FSMC driver modules diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_lptim.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_lptim.c index 9336b5ca..bb75125d 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_lptim.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_lptim.c @@ -92,7 +92,7 @@ * - SUCCESS: LPTIMx registers are de-initialized * - ERROR: invalid LPTIMx instance */ -ErrorStatus LL_LPTIM_DeInit(LPTIM_TypeDef *LPTIMx) +ErrorStatus LL_LPTIM_DeInit(const LPTIM_TypeDef *LPTIMx) { ErrorStatus result = SUCCESS; @@ -137,7 +137,7 @@ void LL_LPTIM_StructInit(LL_LPTIM_InitTypeDef *LPTIM_InitStruct) * - SUCCESS: LPTIMx instance has been initialized * - ERROR: LPTIMx instance hasn't been initialized */ -ErrorStatus LL_LPTIM_Init(LPTIM_TypeDef *LPTIMx, LL_LPTIM_InitTypeDef *LPTIM_InitStruct) +ErrorStatus LL_LPTIM_Init(LPTIM_TypeDef *LPTIMx, const LL_LPTIM_InitTypeDef *LPTIM_InitStruct) { ErrorStatus result = SUCCESS; /* Check the parameters */ @@ -259,8 +259,7 @@ void LL_LPTIM_Disable(LPTIM_TypeDef *LPTIMx) do { rcc_clock.SYSCLK_Frequency--; /* Used for timeout */ - } - while (((LL_LPTIM_IsActiveFlag_ARROK(LPTIMx) != 1UL)) && ((rcc_clock.SYSCLK_Frequency) > 0UL)); + } while (((LL_LPTIM_IsActiveFlag_ARROK(LPTIMx) != 1UL)) && ((rcc_clock.SYSCLK_Frequency) > 0UL)); LL_LPTIM_ClearFlag_ARROK(LPTIMx); } diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_rng.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_rng.c index 333d63ca..fc1ea50b 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_rng.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_rng.c @@ -59,7 +59,7 @@ * - SUCCESS: RNG registers are de-initialized * - ERROR: not applicable */ -ErrorStatus LL_RNG_DeInit(RNG_TypeDef *RNGx) +ErrorStatus LL_RNG_DeInit(const RNG_TypeDef *RNGx) { ErrorStatus status = SUCCESS; @@ -77,7 +77,7 @@ ErrorStatus LL_RNG_DeInit(RNG_TypeDef *RNGx) /* Enable RNG reset state */ LL_AHB2_GRP1_ForceReset(LL_AHB2_GRP1_PERIPH_RNG); - /* Release RNG from reset state */ + /* Release RNG from reset state */ LL_AHB2_GRP1_ReleaseReset(LL_AHB2_GRP1_PERIPH_RNG); #endif /* !RCC_AHB2_SUPPORT */ } diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_rtc.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_rtc.c index f3ee3ce0..4d106da3 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_rtc.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_rtc.c @@ -6,7 +6,7 @@ ****************************************************************************** * @attention * - * Copyright (c) 2017 STMicroelectronics. + * Copyright (c) 2016 STMicroelectronics. * All rights reserved. * * This software is licensed under terms that can be found in the LICENSE file diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_tim.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_tim.c index b272b62d..49217db9 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_tim.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_tim.c @@ -137,14 +137,14 @@ /** @defgroup TIM_LL_Private_Functions TIM Private Functions * @{ */ -static ErrorStatus OC1Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCInitStruct); -static ErrorStatus OC2Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCInitStruct); -static ErrorStatus OC3Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCInitStruct); -static ErrorStatus OC4Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCInitStruct); -static ErrorStatus IC1Config(TIM_TypeDef *TIMx, LL_TIM_IC_InitTypeDef *TIM_ICInitStruct); -static ErrorStatus IC2Config(TIM_TypeDef *TIMx, LL_TIM_IC_InitTypeDef *TIM_ICInitStruct); -static ErrorStatus IC3Config(TIM_TypeDef *TIMx, LL_TIM_IC_InitTypeDef *TIM_ICInitStruct); -static ErrorStatus IC4Config(TIM_TypeDef *TIMx, LL_TIM_IC_InitTypeDef *TIM_ICInitStruct); +static ErrorStatus OC1Config(TIM_TypeDef *TIMx, const LL_TIM_OC_InitTypeDef *TIM_OCInitStruct); +static ErrorStatus OC2Config(TIM_TypeDef *TIMx, const LL_TIM_OC_InitTypeDef *TIM_OCInitStruct); +static ErrorStatus OC3Config(TIM_TypeDef *TIMx, const LL_TIM_OC_InitTypeDef *TIM_OCInitStruct); +static ErrorStatus OC4Config(TIM_TypeDef *TIMx, const LL_TIM_OC_InitTypeDef *TIM_OCInitStruct); +static ErrorStatus IC1Config(TIM_TypeDef *TIMx, const LL_TIM_IC_InitTypeDef *TIM_ICInitStruct); +static ErrorStatus IC2Config(TIM_TypeDef *TIMx, const LL_TIM_IC_InitTypeDef *TIM_ICInitStruct); +static ErrorStatus IC3Config(TIM_TypeDef *TIMx, const LL_TIM_IC_InitTypeDef *TIM_ICInitStruct); +static ErrorStatus IC4Config(TIM_TypeDef *TIMx, const LL_TIM_IC_InitTypeDef *TIM_ICInitStruct); /** * @} */ @@ -165,7 +165,7 @@ static ErrorStatus IC4Config(TIM_TypeDef *TIMx, LL_TIM_IC_InitTypeDef *TIM_ICIni * - SUCCESS: TIMx registers are de-initialized * - ERROR: invalid TIMx instance */ -ErrorStatus LL_TIM_DeInit(TIM_TypeDef *TIMx) +ErrorStatus LL_TIM_DeInit(const TIM_TypeDef *TIMx) { ErrorStatus result = SUCCESS; @@ -301,7 +301,7 @@ void LL_TIM_StructInit(LL_TIM_InitTypeDef *TIM_InitStruct) * - SUCCESS: TIMx registers are de-initialized * - ERROR: not applicable */ -ErrorStatus LL_TIM_Init(TIM_TypeDef *TIMx, LL_TIM_InitTypeDef *TIM_InitStruct) +ErrorStatus LL_TIM_Init(TIM_TypeDef *TIMx, const LL_TIM_InitTypeDef *TIM_InitStruct) { uint32_t tmpcr1; @@ -380,7 +380,7 @@ void LL_TIM_OC_StructInit(LL_TIM_OC_InitTypeDef *TIM_OC_InitStruct) * - SUCCESS: TIMx output channel is initialized * - ERROR: TIMx output channel is not initialized */ -ErrorStatus LL_TIM_OC_Init(TIM_TypeDef *TIMx, uint32_t Channel, LL_TIM_OC_InitTypeDef *TIM_OC_InitStruct) +ErrorStatus LL_TIM_OC_Init(TIM_TypeDef *TIMx, uint32_t Channel, const LL_TIM_OC_InitTypeDef *TIM_OC_InitStruct) { ErrorStatus result = ERROR; @@ -435,7 +435,7 @@ void LL_TIM_IC_StructInit(LL_TIM_IC_InitTypeDef *TIM_ICInitStruct) * - SUCCESS: TIMx output channel is initialized * - ERROR: TIMx output channel is not initialized */ -ErrorStatus LL_TIM_IC_Init(TIM_TypeDef *TIMx, uint32_t Channel, LL_TIM_IC_InitTypeDef *TIM_IC_InitStruct) +ErrorStatus LL_TIM_IC_Init(TIM_TypeDef *TIMx, uint32_t Channel, const LL_TIM_IC_InitTypeDef *TIM_IC_InitStruct) { ErrorStatus result = ERROR; @@ -489,7 +489,7 @@ void LL_TIM_ENCODER_StructInit(LL_TIM_ENCODER_InitTypeDef *TIM_EncoderInitStruct * - SUCCESS: TIMx registers are de-initialized * - ERROR: not applicable */ -ErrorStatus LL_TIM_ENCODER_Init(TIM_TypeDef *TIMx, LL_TIM_ENCODER_InitTypeDef *TIM_EncoderInitStruct) +ErrorStatus LL_TIM_ENCODER_Init(TIM_TypeDef *TIMx, const LL_TIM_ENCODER_InitTypeDef *TIM_EncoderInitStruct) { uint32_t tmpccmr1; uint32_t tmpccer; @@ -582,7 +582,7 @@ void LL_TIM_HALLSENSOR_StructInit(LL_TIM_HALLSENSOR_InitTypeDef *TIM_HallSensorI * - SUCCESS: TIMx registers are de-initialized * - ERROR: not applicable */ -ErrorStatus LL_TIM_HALLSENSOR_Init(TIM_TypeDef *TIMx, LL_TIM_HALLSENSOR_InitTypeDef *TIM_HallSensorInitStruct) +ErrorStatus LL_TIM_HALLSENSOR_Init(TIM_TypeDef *TIMx, const LL_TIM_HALLSENSOR_InitTypeDef *TIM_HallSensorInitStruct) { uint32_t tmpcr2; uint32_t tmpccmr1; @@ -687,7 +687,7 @@ void LL_TIM_BDTR_StructInit(LL_TIM_BDTR_InitTypeDef *TIM_BDTRInitStruct) * - SUCCESS: Break and Dead Time is initialized * - ERROR: not applicable */ -ErrorStatus LL_TIM_BDTR_Init(TIM_TypeDef *TIMx, LL_TIM_BDTR_InitTypeDef *TIM_BDTRInitStruct) +ErrorStatus LL_TIM_BDTR_Init(TIM_TypeDef *TIMx, const LL_TIM_BDTR_InitTypeDef *TIM_BDTRInitStruct) { uint32_t tmpbdtr = 0; @@ -711,7 +711,6 @@ ErrorStatus LL_TIM_BDTR_Init(TIM_TypeDef *TIMx, LL_TIM_BDTR_InitTypeDef *TIM_BDT MODIFY_REG(tmpbdtr, TIM_BDTR_BKE, TIM_BDTRInitStruct->BreakState); MODIFY_REG(tmpbdtr, TIM_BDTR_BKP, TIM_BDTRInitStruct->BreakPolarity); MODIFY_REG(tmpbdtr, TIM_BDTR_AOE, TIM_BDTRInitStruct->AutomaticOutput); - MODIFY_REG(tmpbdtr, TIM_BDTR_MOE, TIM_BDTRInitStruct->AutomaticOutput); /* Set TIMx_BDTR */ LL_TIM_WriteReg(TIMx, BDTR, tmpbdtr); @@ -738,7 +737,7 @@ ErrorStatus LL_TIM_BDTR_Init(TIM_TypeDef *TIMx, LL_TIM_BDTR_InitTypeDef *TIM_BDT * - SUCCESS: TIMx registers are de-initialized * - ERROR: not applicable */ -static ErrorStatus OC1Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCInitStruct) +static ErrorStatus OC1Config(TIM_TypeDef *TIMx, const LL_TIM_OC_InitTypeDef *TIM_OCInitStruct) { uint32_t tmpccmr1; uint32_t tmpccer; @@ -749,8 +748,6 @@ static ErrorStatus OC1Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCIni assert_param(IS_LL_TIM_OCMODE(TIM_OCInitStruct->OCMode)); assert_param(IS_LL_TIM_OCSTATE(TIM_OCInitStruct->OCState)); assert_param(IS_LL_TIM_OCPOLARITY(TIM_OCInitStruct->OCPolarity)); - assert_param(IS_LL_TIM_OCSTATE(TIM_OCInitStruct->OCNState)); - assert_param(IS_LL_TIM_OCPOLARITY(TIM_OCInitStruct->OCNPolarity)); /* Disable the Channel 1: Reset the CC1E Bit */ CLEAR_BIT(TIMx->CCER, TIM_CCER_CC1E); @@ -778,8 +775,10 @@ static ErrorStatus OC1Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCIni if (IS_TIM_BREAK_INSTANCE(TIMx)) { - assert_param(IS_LL_TIM_OCIDLESTATE(TIM_OCInitStruct->OCNIdleState)); assert_param(IS_LL_TIM_OCIDLESTATE(TIM_OCInitStruct->OCIdleState)); + assert_param(IS_LL_TIM_OCSTATE(TIM_OCInitStruct->OCNState)); + assert_param(IS_LL_TIM_OCPOLARITY(TIM_OCInitStruct->OCNPolarity)); + assert_param(IS_LL_TIM_OCIDLESTATE(TIM_OCInitStruct->OCNIdleState)); /* Set the complementary output Polarity */ MODIFY_REG(tmpccer, TIM_CCER_CC1NP, TIM_OCInitStruct->OCNPolarity << 2U); @@ -817,7 +816,7 @@ static ErrorStatus OC1Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCIni * - SUCCESS: TIMx registers are de-initialized * - ERROR: not applicable */ -static ErrorStatus OC2Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCInitStruct) +static ErrorStatus OC2Config(TIM_TypeDef *TIMx, const LL_TIM_OC_InitTypeDef *TIM_OCInitStruct) { uint32_t tmpccmr1; uint32_t tmpccer; @@ -828,8 +827,6 @@ static ErrorStatus OC2Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCIni assert_param(IS_LL_TIM_OCMODE(TIM_OCInitStruct->OCMode)); assert_param(IS_LL_TIM_OCSTATE(TIM_OCInitStruct->OCState)); assert_param(IS_LL_TIM_OCPOLARITY(TIM_OCInitStruct->OCPolarity)); - assert_param(IS_LL_TIM_OCSTATE(TIM_OCInitStruct->OCNState)); - assert_param(IS_LL_TIM_OCPOLARITY(TIM_OCInitStruct->OCNPolarity)); /* Disable the Channel 2: Reset the CC2E Bit */ CLEAR_BIT(TIMx->CCER, TIM_CCER_CC2E); @@ -857,8 +854,10 @@ static ErrorStatus OC2Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCIni if (IS_TIM_BREAK_INSTANCE(TIMx)) { - assert_param(IS_LL_TIM_OCIDLESTATE(TIM_OCInitStruct->OCNIdleState)); assert_param(IS_LL_TIM_OCIDLESTATE(TIM_OCInitStruct->OCIdleState)); + assert_param(IS_LL_TIM_OCSTATE(TIM_OCInitStruct->OCNState)); + assert_param(IS_LL_TIM_OCPOLARITY(TIM_OCInitStruct->OCNPolarity)); + assert_param(IS_LL_TIM_OCIDLESTATE(TIM_OCInitStruct->OCNIdleState)); /* Set the complementary output Polarity */ MODIFY_REG(tmpccer, TIM_CCER_CC2NP, TIM_OCInitStruct->OCNPolarity << 6U); @@ -896,7 +895,7 @@ static ErrorStatus OC2Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCIni * - SUCCESS: TIMx registers are de-initialized * - ERROR: not applicable */ -static ErrorStatus OC3Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCInitStruct) +static ErrorStatus OC3Config(TIM_TypeDef *TIMx, const LL_TIM_OC_InitTypeDef *TIM_OCInitStruct) { uint32_t tmpccmr2; uint32_t tmpccer; @@ -907,8 +906,6 @@ static ErrorStatus OC3Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCIni assert_param(IS_LL_TIM_OCMODE(TIM_OCInitStruct->OCMode)); assert_param(IS_LL_TIM_OCSTATE(TIM_OCInitStruct->OCState)); assert_param(IS_LL_TIM_OCPOLARITY(TIM_OCInitStruct->OCPolarity)); - assert_param(IS_LL_TIM_OCSTATE(TIM_OCInitStruct->OCNState)); - assert_param(IS_LL_TIM_OCPOLARITY(TIM_OCInitStruct->OCNPolarity)); /* Disable the Channel 3: Reset the CC3E Bit */ CLEAR_BIT(TIMx->CCER, TIM_CCER_CC3E); @@ -936,8 +933,10 @@ static ErrorStatus OC3Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCIni if (IS_TIM_BREAK_INSTANCE(TIMx)) { - assert_param(IS_LL_TIM_OCIDLESTATE(TIM_OCInitStruct->OCNIdleState)); assert_param(IS_LL_TIM_OCIDLESTATE(TIM_OCInitStruct->OCIdleState)); + assert_param(IS_LL_TIM_OCSTATE(TIM_OCInitStruct->OCNState)); + assert_param(IS_LL_TIM_OCPOLARITY(TIM_OCInitStruct->OCNPolarity)); + assert_param(IS_LL_TIM_OCIDLESTATE(TIM_OCInitStruct->OCNIdleState)); /* Set the complementary output Polarity */ MODIFY_REG(tmpccer, TIM_CCER_CC3NP, TIM_OCInitStruct->OCNPolarity << 10U); @@ -975,7 +974,7 @@ static ErrorStatus OC3Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCIni * - SUCCESS: TIMx registers are de-initialized * - ERROR: not applicable */ -static ErrorStatus OC4Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCInitStruct) +static ErrorStatus OC4Config(TIM_TypeDef *TIMx, const LL_TIM_OC_InitTypeDef *TIM_OCInitStruct) { uint32_t tmpccmr2; uint32_t tmpccer; @@ -986,8 +985,6 @@ static ErrorStatus OC4Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCIni assert_param(IS_LL_TIM_OCMODE(TIM_OCInitStruct->OCMode)); assert_param(IS_LL_TIM_OCSTATE(TIM_OCInitStruct->OCState)); assert_param(IS_LL_TIM_OCPOLARITY(TIM_OCInitStruct->OCPolarity)); - assert_param(IS_LL_TIM_OCPOLARITY(TIM_OCInitStruct->OCNPolarity)); - assert_param(IS_LL_TIM_OCSTATE(TIM_OCInitStruct->OCNState)); /* Disable the Channel 4: Reset the CC4E Bit */ CLEAR_BIT(TIMx->CCER, TIM_CCER_CC4E); @@ -1015,7 +1012,6 @@ static ErrorStatus OC4Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCIni if (IS_TIM_BREAK_INSTANCE(TIMx)) { - assert_param(IS_LL_TIM_OCIDLESTATE(TIM_OCInitStruct->OCNIdleState)); assert_param(IS_LL_TIM_OCIDLESTATE(TIM_OCInitStruct->OCIdleState)); /* Set the Output Idle state */ @@ -1037,7 +1033,6 @@ static ErrorStatus OC4Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCIni return SUCCESS; } - /** * @brief Configure the TIMx input channel 1. * @param TIMx Timer Instance @@ -1046,7 +1041,7 @@ static ErrorStatus OC4Config(TIM_TypeDef *TIMx, LL_TIM_OC_InitTypeDef *TIM_OCIni * - SUCCESS: TIMx registers are de-initialized * - ERROR: not applicable */ -static ErrorStatus IC1Config(TIM_TypeDef *TIMx, LL_TIM_IC_InitTypeDef *TIM_ICInitStruct) +static ErrorStatus IC1Config(TIM_TypeDef *TIMx, const LL_TIM_IC_InitTypeDef *TIM_ICInitStruct) { /* Check the parameters */ assert_param(IS_TIM_CC1_INSTANCE(TIMx)); @@ -1079,7 +1074,7 @@ static ErrorStatus IC1Config(TIM_TypeDef *TIMx, LL_TIM_IC_InitTypeDef *TIM_ICIni * - SUCCESS: TIMx registers are de-initialized * - ERROR: not applicable */ -static ErrorStatus IC2Config(TIM_TypeDef *TIMx, LL_TIM_IC_InitTypeDef *TIM_ICInitStruct) +static ErrorStatus IC2Config(TIM_TypeDef *TIMx, const LL_TIM_IC_InitTypeDef *TIM_ICInitStruct) { /* Check the parameters */ assert_param(IS_TIM_CC2_INSTANCE(TIMx)); @@ -1112,7 +1107,7 @@ static ErrorStatus IC2Config(TIM_TypeDef *TIMx, LL_TIM_IC_InitTypeDef *TIM_ICIni * - SUCCESS: TIMx registers are de-initialized * - ERROR: not applicable */ -static ErrorStatus IC3Config(TIM_TypeDef *TIMx, LL_TIM_IC_InitTypeDef *TIM_ICInitStruct) +static ErrorStatus IC3Config(TIM_TypeDef *TIMx, const LL_TIM_IC_InitTypeDef *TIM_ICInitStruct) { /* Check the parameters */ assert_param(IS_TIM_CC3_INSTANCE(TIMx)); @@ -1145,7 +1140,7 @@ static ErrorStatus IC3Config(TIM_TypeDef *TIMx, LL_TIM_IC_InitTypeDef *TIM_ICIni * - SUCCESS: TIMx registers are de-initialized * - ERROR: not applicable */ -static ErrorStatus IC4Config(TIM_TypeDef *TIMx, LL_TIM_IC_InitTypeDef *TIM_ICInitStruct) +static ErrorStatus IC4Config(TIM_TypeDef *TIMx, const LL_TIM_IC_InitTypeDef *TIM_ICInitStruct) { /* Check the parameters */ assert_param(IS_TIM_CC4_INSTANCE(TIMx)); @@ -1162,7 +1157,7 @@ static ErrorStatus IC4Config(TIM_TypeDef *TIMx, LL_TIM_IC_InitTypeDef *TIM_ICIni (TIM_CCMR2_CC4S | TIM_CCMR2_IC4F | TIM_CCMR2_IC4PSC), (TIM_ICInitStruct->ICActiveInput | TIM_ICInitStruct->ICFilter | TIM_ICInitStruct->ICPrescaler) >> 8U); - /* Select the Polarity and set the CC2E Bit */ + /* Select the Polarity and set the CC4E Bit */ MODIFY_REG(TIMx->CCER, (TIM_CCER_CC4P | TIM_CCER_CC4NP), ((TIM_ICInitStruct->ICPolarity << 12U) | TIM_CCER_CC4E)); diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_usart.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_usart.c index 3cf68e36..5b1ffa83 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_usart.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_usart.c @@ -121,7 +121,7 @@ * - SUCCESS: USART registers are de-initialized * - ERROR: USART registers are not de-initialized */ -ErrorStatus LL_USART_DeInit(USART_TypeDef *USARTx) +ErrorStatus LL_USART_DeInit(const USART_TypeDef *USARTx) { ErrorStatus status = SUCCESS; @@ -245,7 +245,7 @@ ErrorStatus LL_USART_DeInit(USART_TypeDef *USARTx) * - SUCCESS: USART registers are initialized according to USART_InitStruct content * - ERROR: Problem occurred during USART Registers initialization */ -ErrorStatus LL_USART_Init(USART_TypeDef *USARTx, LL_USART_InitTypeDef *USART_InitStruct) +ErrorStatus LL_USART_Init(USART_TypeDef *USARTx, const LL_USART_InitTypeDef *USART_InitStruct) { ErrorStatus status = ERROR; uint32_t periphclk = LL_RCC_PERIPH_FREQUENCY_NO; @@ -409,7 +409,7 @@ void LL_USART_StructInit(LL_USART_InitTypeDef *USART_InitStruct) * - SUCCESS: USART registers related to Clock settings are initialized according to USART_ClockInitStruct content * - ERROR: Problem occurred during USART Registers initialization */ -ErrorStatus LL_USART_ClockInit(USART_TypeDef *USARTx, LL_USART_ClockInitTypeDef *USART_ClockInitStruct) +ErrorStatus LL_USART_ClockInit(USART_TypeDef *USARTx, const LL_USART_ClockInitTypeDef *USART_ClockInitStruct) { ErrorStatus status = SUCCESS; diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_usb.c b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_usb.c index 1df5fcf1..2570d422 100644 --- a/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_usb.c +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_ll_usb.c @@ -27,7 +27,7 @@ ##### How to use this driver ##### ============================================================================== [..] - (#) Fill parameters of Init structure in USB_OTG_CfgTypeDef structure. + (#) Fill parameters of Init structure in USB_CfgTypeDef structure. (#) Call USB_CoreInit() API to initialize the USB Core peripheral. @@ -258,9 +258,9 @@ HAL_StatusTypeDef USB_SetCurrentMode(USB_OTG_GlobalTypeDef *USBx, USB_OTG_ModeTy do { - HAL_Delay(1U); - ms++; - } while ((USB_GetMode(USBx) != (uint32_t)USB_HOST_MODE) && (ms < 50U)); + HAL_Delay(10U); + ms += 10U; + } while ((USB_GetMode(USBx) != (uint32_t)USB_HOST_MODE) && (ms < HAL_USB_CURRENT_MODE_MAX_DELAY_MS)); } else if (mode == USB_DEVICE_MODE) { @@ -268,16 +268,16 @@ HAL_StatusTypeDef USB_SetCurrentMode(USB_OTG_GlobalTypeDef *USBx, USB_OTG_ModeTy do { - HAL_Delay(1U); - ms++; - } while ((USB_GetMode(USBx) != (uint32_t)USB_DEVICE_MODE) && (ms < 50U)); + HAL_Delay(10U); + ms += 10U; + } while ((USB_GetMode(USBx) != (uint32_t)USB_DEVICE_MODE) && (ms < HAL_USB_CURRENT_MODE_MAX_DELAY_MS)); } else { return HAL_ERROR; } - if (ms == 50U) + if (ms == HAL_USB_CURRENT_MODE_MAX_DELAY_MS) { return HAL_ERROR; } @@ -304,7 +304,9 @@ HAL_StatusTypeDef USB_DevInit(USB_OTG_GlobalTypeDef *USBx, USB_OTG_CfgTypeDef cf USBx->DIEPTXF[i] = 0U; } -#if defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) +#if defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) \ + || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) \ + || defined(STM32F423xx) /* VBUS Sensing setup */ if (cfg.vbus_sensing_enable == 0U) { @@ -341,14 +343,13 @@ HAL_StatusTypeDef USB_DevInit(USB_OTG_GlobalTypeDef *USBx, USB_OTG_CfgTypeDef cf USBx->GCCFG &= ~USB_OTG_GCCFG_NOVBUSSENS; USBx->GCCFG |= USB_OTG_GCCFG_VBUSBSEN; } -#endif /* defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) */ +#endif /* defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || + defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || + defined(STM32F423xx) */ /* Restart the Phy Clock */ USBx_PCGCCTL = 0U; - /* Device mode configuration */ - USBx_DEVICE->DCFG |= DCFG_FRAME_INTERVAL_80; - if (cfg.phy_itface == USB_OTG_ULPI_PHY) { if (cfg.speed == USBD_HS_SPEED) @@ -478,7 +479,7 @@ HAL_StatusTypeDef USB_FlushTxFifo(USB_OTG_GlobalTypeDef *USBx, uint32_t num) { count++; - if (count > 200000U) + if (count > HAL_USB_TIMEOUT) { return HAL_TIMEOUT; } @@ -492,7 +493,7 @@ HAL_StatusTypeDef USB_FlushTxFifo(USB_OTG_GlobalTypeDef *USBx, uint32_t num) { count++; - if (count > 200000U) + if (count > HAL_USB_TIMEOUT) { return HAL_TIMEOUT; } @@ -515,7 +516,7 @@ HAL_StatusTypeDef USB_FlushRxFifo(USB_OTG_GlobalTypeDef *USBx) { count++; - if (count > 200000U) + if (count > HAL_USB_TIMEOUT) { return HAL_TIMEOUT; } @@ -529,7 +530,7 @@ HAL_StatusTypeDef USB_FlushRxFifo(USB_OTG_GlobalTypeDef *USBx) { count++; - if (count > 200000U) + if (count > HAL_USB_TIMEOUT) { return HAL_TIMEOUT; } @@ -549,7 +550,7 @@ HAL_StatusTypeDef USB_FlushRxFifo(USB_OTG_GlobalTypeDef *USBx) * @arg USB_OTG_SPEED_FULL: Full speed mode * @retval Hal status */ -HAL_StatusTypeDef USB_SetDevSpeed(USB_OTG_GlobalTypeDef *USBx, uint8_t speed) +HAL_StatusTypeDef USB_SetDevSpeed(const USB_OTG_GlobalTypeDef *USBx, uint8_t speed) { uint32_t USBx_BASE = (uint32_t)USBx; @@ -565,7 +566,7 @@ HAL_StatusTypeDef USB_SetDevSpeed(USB_OTG_GlobalTypeDef *USBx, uint8_t speed) * @arg USBD_HS_SPEED: High speed mode * @arg USBD_FS_SPEED: Full speed mode */ -uint8_t USB_GetDevSpeed(USB_OTG_GlobalTypeDef *USBx) +uint8_t USB_GetDevSpeed(const USB_OTG_GlobalTypeDef *USBx) { uint32_t USBx_BASE = (uint32_t)USBx; uint8_t speed; @@ -594,7 +595,7 @@ uint8_t USB_GetDevSpeed(USB_OTG_GlobalTypeDef *USBx) * @param ep pointer to endpoint structure * @retval HAL status */ -HAL_StatusTypeDef USB_ActivateEndpoint(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep) +HAL_StatusTypeDef USB_ActivateEndpoint(const USB_OTG_GlobalTypeDef *USBx, const USB_OTG_EPTypeDef *ep) { uint32_t USBx_BASE = (uint32_t)USBx; uint32_t epnum = (uint32_t)ep->num; @@ -632,7 +633,7 @@ HAL_StatusTypeDef USB_ActivateEndpoint(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTy * @param ep pointer to endpoint structure * @retval HAL status */ -HAL_StatusTypeDef USB_ActivateDedicatedEndpoint(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep) +HAL_StatusTypeDef USB_ActivateDedicatedEndpoint(const USB_OTG_GlobalTypeDef *USBx, const USB_OTG_EPTypeDef *ep) { uint32_t USBx_BASE = (uint32_t)USBx; uint32_t epnum = (uint32_t)ep->num; @@ -671,7 +672,7 @@ HAL_StatusTypeDef USB_ActivateDedicatedEndpoint(USB_OTG_GlobalTypeDef *USBx, USB * @param ep pointer to endpoint structure * @retval HAL status */ -HAL_StatusTypeDef USB_DeactivateEndpoint(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep) +HAL_StatusTypeDef USB_DeactivateEndpoint(const USB_OTG_GlobalTypeDef *USBx, const USB_OTG_EPTypeDef *ep) { uint32_t USBx_BASE = (uint32_t)USBx; uint32_t epnum = (uint32_t)ep->num; @@ -718,7 +719,7 @@ HAL_StatusTypeDef USB_DeactivateEndpoint(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EP * @param ep pointer to endpoint structure * @retval HAL status */ -HAL_StatusTypeDef USB_DeactivateDedicatedEndpoint(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep) +HAL_StatusTypeDef USB_DeactivateDedicatedEndpoint(const USB_OTG_GlobalTypeDef *USBx, const USB_OTG_EPTypeDef *ep) { uint32_t USBx_BASE = (uint32_t)USBx; uint32_t epnum = (uint32_t)ep->num; @@ -785,8 +786,21 @@ HAL_StatusTypeDef USB_EPStartXfer(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef */ USBx_INEP(epnum)->DIEPTSIZ &= ~(USB_OTG_DIEPTSIZ_XFRSIZ); USBx_INEP(epnum)->DIEPTSIZ &= ~(USB_OTG_DIEPTSIZ_PKTCNT); - USBx_INEP(epnum)->DIEPTSIZ |= (USB_OTG_DIEPTSIZ_PKTCNT & - (((ep->xfer_len + ep->maxpacket - 1U) / ep->maxpacket) << 19)); + + if (epnum == 0U) + { + if (ep->xfer_len > ep->maxpacket) + { + ep->xfer_len = ep->maxpacket; + } + + USBx_INEP(epnum)->DIEPTSIZ |= (USB_OTG_DIEPTSIZ_PKTCNT & (1U << 19)); + } + else + { + USBx_INEP(epnum)->DIEPTSIZ |= (USB_OTG_DIEPTSIZ_PKTCNT & + (((ep->xfer_len + ep->maxpacket - 1U) / ep->maxpacket) << 19)); + } USBx_INEP(epnum)->DIEPTSIZ |= (USB_OTG_DIEPTSIZ_XFRSIZ & ep->xfer_len); @@ -856,18 +870,34 @@ HAL_StatusTypeDef USB_EPStartXfer(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef USBx_OUTEP(epnum)->DOEPTSIZ &= ~(USB_OTG_DOEPTSIZ_XFRSIZ); USBx_OUTEP(epnum)->DOEPTSIZ &= ~(USB_OTG_DOEPTSIZ_PKTCNT); - if (ep->xfer_len == 0U) + if (epnum == 0U) { - USBx_OUTEP(epnum)->DOEPTSIZ |= (USB_OTG_DOEPTSIZ_XFRSIZ & ep->maxpacket); + if (ep->xfer_len > 0U) + { + ep->xfer_len = ep->maxpacket; + } + + /* Store transfer size, for EP0 this is equal to endpoint max packet size */ + ep->xfer_size = ep->maxpacket; + + USBx_OUTEP(epnum)->DOEPTSIZ |= (USB_OTG_DOEPTSIZ_XFRSIZ & ep->xfer_size); USBx_OUTEP(epnum)->DOEPTSIZ |= (USB_OTG_DOEPTSIZ_PKTCNT & (1U << 19)); } else { - pktcnt = (uint16_t)((ep->xfer_len + ep->maxpacket - 1U) / ep->maxpacket); - ep->xfer_size = ep->maxpacket * pktcnt; + if (ep->xfer_len == 0U) + { + USBx_OUTEP(epnum)->DOEPTSIZ |= (USB_OTG_DOEPTSIZ_XFRSIZ & ep->maxpacket); + USBx_OUTEP(epnum)->DOEPTSIZ |= (USB_OTG_DOEPTSIZ_PKTCNT & (1U << 19)); + } + else + { + pktcnt = (uint16_t)((ep->xfer_len + ep->maxpacket - 1U) / ep->maxpacket); + ep->xfer_size = ep->maxpacket * pktcnt; - USBx_OUTEP(epnum)->DOEPTSIZ |= USB_OTG_DOEPTSIZ_PKTCNT & ((uint32_t)pktcnt << 19); - USBx_OUTEP(epnum)->DOEPTSIZ |= USB_OTG_DOEPTSIZ_XFRSIZ & ep->xfer_size; + USBx_OUTEP(epnum)->DOEPTSIZ |= USB_OTG_DOEPTSIZ_PKTCNT & ((uint32_t)pktcnt << 19); + USBx_OUTEP(epnum)->DOEPTSIZ |= USB_OTG_DOEPTSIZ_XFRSIZ & ep->xfer_size; + } } if (dma == 1U) @@ -896,106 +926,6 @@ HAL_StatusTypeDef USB_EPStartXfer(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef return HAL_OK; } -/** - * @brief USB_EP0StartXfer : setup and starts a transfer over the EP 0 - * @param USBx Selected device - * @param ep pointer to endpoint structure - * @param dma USB dma enabled or disabled - * This parameter can be one of these values: - * 0 : DMA feature not used - * 1 : DMA feature used - * @retval HAL status - */ -HAL_StatusTypeDef USB_EP0StartXfer(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep, uint8_t dma) -{ - uint32_t USBx_BASE = (uint32_t)USBx; - uint32_t epnum = (uint32_t)ep->num; - - /* IN endpoint */ - if (ep->is_in == 1U) - { - /* Zero Length Packet? */ - if (ep->xfer_len == 0U) - { - USBx_INEP(epnum)->DIEPTSIZ &= ~(USB_OTG_DIEPTSIZ_PKTCNT); - USBx_INEP(epnum)->DIEPTSIZ |= (USB_OTG_DIEPTSIZ_PKTCNT & (1U << 19)); - USBx_INEP(epnum)->DIEPTSIZ &= ~(USB_OTG_DIEPTSIZ_XFRSIZ); - } - else - { - /* Program the transfer size and packet count - * as follows: xfersize = N * maxpacket + - * short_packet pktcnt = N + (short_packet - * exist ? 1 : 0) - */ - USBx_INEP(epnum)->DIEPTSIZ &= ~(USB_OTG_DIEPTSIZ_XFRSIZ); - USBx_INEP(epnum)->DIEPTSIZ &= ~(USB_OTG_DIEPTSIZ_PKTCNT); - - if (ep->xfer_len > ep->maxpacket) - { - ep->xfer_len = ep->maxpacket; - } - USBx_INEP(epnum)->DIEPTSIZ |= (USB_OTG_DIEPTSIZ_PKTCNT & (1U << 19)); - USBx_INEP(epnum)->DIEPTSIZ |= (USB_OTG_DIEPTSIZ_XFRSIZ & ep->xfer_len); - } - - if (dma == 1U) - { - if ((uint32_t)ep->dma_addr != 0U) - { - USBx_INEP(epnum)->DIEPDMA = (uint32_t)(ep->dma_addr); - } - - /* EP enable, IN data in FIFO */ - USBx_INEP(epnum)->DIEPCTL |= (USB_OTG_DIEPCTL_CNAK | USB_OTG_DIEPCTL_EPENA); - } - else - { - /* EP enable, IN data in FIFO */ - USBx_INEP(epnum)->DIEPCTL |= (USB_OTG_DIEPCTL_CNAK | USB_OTG_DIEPCTL_EPENA); - - /* Enable the Tx FIFO Empty Interrupt for this EP */ - if (ep->xfer_len > 0U) - { - USBx_DEVICE->DIEPEMPMSK |= 1UL << (ep->num & EP_ADDR_MSK); - } - } - } - else /* OUT endpoint */ - { - /* Program the transfer size and packet count as follows: - * pktcnt = N - * xfersize = N * maxpacket - */ - USBx_OUTEP(epnum)->DOEPTSIZ &= ~(USB_OTG_DOEPTSIZ_XFRSIZ); - USBx_OUTEP(epnum)->DOEPTSIZ &= ~(USB_OTG_DOEPTSIZ_PKTCNT); - - if (ep->xfer_len > 0U) - { - ep->xfer_len = ep->maxpacket; - } - - /* Store transfer size, for EP0 this is equal to endpoint max packet size */ - ep->xfer_size = ep->maxpacket; - - USBx_OUTEP(epnum)->DOEPTSIZ |= (USB_OTG_DOEPTSIZ_PKTCNT & (1U << 19)); - USBx_OUTEP(epnum)->DOEPTSIZ |= (USB_OTG_DOEPTSIZ_XFRSIZ & ep->xfer_size); - - if (dma == 1U) - { - if ((uint32_t)ep->xfer_buff != 0U) - { - USBx_OUTEP(epnum)->DOEPDMA = (uint32_t)(ep->xfer_buff); - } - } - - /* EP enable */ - USBx_OUTEP(epnum)->DOEPCTL |= (USB_OTG_DOEPCTL_CNAK | USB_OTG_DOEPCTL_EPENA); - } - - return HAL_OK; -} - /** * @brief USB_EPStoptXfer Stop transfer on an EP @@ -1003,7 +933,7 @@ HAL_StatusTypeDef USB_EP0StartXfer(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDe * @param ep pointer to endpoint structure * @retval HAL status */ -HAL_StatusTypeDef USB_EPStopXfer(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep) +HAL_StatusTypeDef USB_EPStopXfer(const USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep) { __IO uint32_t count = 0U; HAL_StatusTypeDef ret = HAL_OK; @@ -1067,7 +997,7 @@ HAL_StatusTypeDef USB_EPStopXfer(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef * 1 : DMA feature used * @retval HAL status */ -HAL_StatusTypeDef USB_WritePacket(USB_OTG_GlobalTypeDef *USBx, uint8_t *src, +HAL_StatusTypeDef USB_WritePacket(const USB_OTG_GlobalTypeDef *USBx, uint8_t *src, uint8_t ch_ep_num, uint16_t len, uint8_t dma) { uint32_t USBx_BASE = (uint32_t)USBx; @@ -1098,7 +1028,7 @@ HAL_StatusTypeDef USB_WritePacket(USB_OTG_GlobalTypeDef *USBx, uint8_t *src, * @param len Number of bytes to read * @retval pointer to destination buffer */ -void *USB_ReadPacket(USB_OTG_GlobalTypeDef *USBx, uint8_t *dest, uint16_t len) +void *USB_ReadPacket(const USB_OTG_GlobalTypeDef *USBx, uint8_t *dest, uint16_t len) { uint32_t USBx_BASE = (uint32_t)USBx; uint8_t *pDest = dest; @@ -1140,7 +1070,7 @@ void *USB_ReadPacket(USB_OTG_GlobalTypeDef *USBx, uint8_t *dest, uint16_t len) * @param ep pointer to endpoint structure * @retval HAL status */ -HAL_StatusTypeDef USB_EPSetStall(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep) +HAL_StatusTypeDef USB_EPSetStall(const USB_OTG_GlobalTypeDef *USBx, const USB_OTG_EPTypeDef *ep) { uint32_t USBx_BASE = (uint32_t)USBx; uint32_t epnum = (uint32_t)ep->num; @@ -1171,7 +1101,7 @@ HAL_StatusTypeDef USB_EPSetStall(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef * @param ep pointer to endpoint structure * @retval HAL status */ -HAL_StatusTypeDef USB_EPClearStall(USB_OTG_GlobalTypeDef *USBx, USB_OTG_EPTypeDef *ep) +HAL_StatusTypeDef USB_EPClearStall(const USB_OTG_GlobalTypeDef *USBx, const USB_OTG_EPTypeDef *ep) { uint32_t USBx_BASE = (uint32_t)USBx; uint32_t epnum = (uint32_t)ep->num; @@ -1241,7 +1171,7 @@ HAL_StatusTypeDef USB_StopDevice(USB_OTG_GlobalTypeDef *USBx) * This parameter can be a value from 0 to 255 * @retval HAL status */ -HAL_StatusTypeDef USB_SetDevAddress(USB_OTG_GlobalTypeDef *USBx, uint8_t address) +HAL_StatusTypeDef USB_SetDevAddress(const USB_OTG_GlobalTypeDef *USBx, uint8_t address) { uint32_t USBx_BASE = (uint32_t)USBx; @@ -1256,7 +1186,7 @@ HAL_StatusTypeDef USB_SetDevAddress(USB_OTG_GlobalTypeDef *USBx, uint8_t addres * @param USBx Selected device * @retval HAL status */ -HAL_StatusTypeDef USB_DevConnect(USB_OTG_GlobalTypeDef *USBx) +HAL_StatusTypeDef USB_DevConnect(const USB_OTG_GlobalTypeDef *USBx) { uint32_t USBx_BASE = (uint32_t)USBx; @@ -1273,7 +1203,7 @@ HAL_StatusTypeDef USB_DevConnect(USB_OTG_GlobalTypeDef *USBx) * @param USBx Selected device * @retval HAL status */ -HAL_StatusTypeDef USB_DevDisconnect(USB_OTG_GlobalTypeDef *USBx) +HAL_StatusTypeDef USB_DevDisconnect(const USB_OTG_GlobalTypeDef *USBx) { uint32_t USBx_BASE = (uint32_t)USBx; @@ -1288,9 +1218,9 @@ HAL_StatusTypeDef USB_DevDisconnect(USB_OTG_GlobalTypeDef *USBx) /** * @brief USB_ReadInterrupts: return the global USB interrupt status * @param USBx Selected device - * @retval HAL status + * @retval USB Global Interrupt status */ -uint32_t USB_ReadInterrupts(USB_OTG_GlobalTypeDef *USBx) +uint32_t USB_ReadInterrupts(USB_OTG_GlobalTypeDef const *USBx) { uint32_t tmpreg; @@ -1300,12 +1230,29 @@ uint32_t USB_ReadInterrupts(USB_OTG_GlobalTypeDef *USBx) return tmpreg; } +/** + * @brief USB_ReadChInterrupts: return USB channel interrupt status + * @param USBx Selected device + * @param chnum Channel number + * @retval USB Channel Interrupt status + */ +uint32_t USB_ReadChInterrupts(const USB_OTG_GlobalTypeDef *USBx, uint8_t chnum) +{ + uint32_t USBx_BASE = (uint32_t)USBx; + uint32_t tmpreg; + + tmpreg = USBx_HC(chnum)->HCINT; + tmpreg &= USBx_HC(chnum)->HCINTMSK; + + return tmpreg; +} + /** * @brief USB_ReadDevAllOutEpInterrupt: return the USB device OUT endpoints interrupt status * @param USBx Selected device - * @retval HAL status + * @retval USB Device OUT EP interrupt status */ -uint32_t USB_ReadDevAllOutEpInterrupt(USB_OTG_GlobalTypeDef *USBx) +uint32_t USB_ReadDevAllOutEpInterrupt(const USB_OTG_GlobalTypeDef *USBx) { uint32_t USBx_BASE = (uint32_t)USBx; uint32_t tmpreg; @@ -1319,9 +1266,9 @@ uint32_t USB_ReadDevAllOutEpInterrupt(USB_OTG_GlobalTypeDef *USBx) /** * @brief USB_ReadDevAllInEpInterrupt: return the USB device IN endpoints interrupt status * @param USBx Selected device - * @retval HAL status + * @retval USB Device IN EP interrupt status */ -uint32_t USB_ReadDevAllInEpInterrupt(USB_OTG_GlobalTypeDef *USBx) +uint32_t USB_ReadDevAllInEpInterrupt(const USB_OTG_GlobalTypeDef *USBx) { uint32_t USBx_BASE = (uint32_t)USBx; uint32_t tmpreg; @@ -1339,7 +1286,7 @@ uint32_t USB_ReadDevAllInEpInterrupt(USB_OTG_GlobalTypeDef *USBx) * This parameter can be a value from 0 to 15 * @retval Device OUT EP Interrupt register */ -uint32_t USB_ReadDevOutEPInterrupt(USB_OTG_GlobalTypeDef *USBx, uint8_t epnum) +uint32_t USB_ReadDevOutEPInterrupt(const USB_OTG_GlobalTypeDef *USBx, uint8_t epnum) { uint32_t USBx_BASE = (uint32_t)USBx; uint32_t tmpreg; @@ -1357,7 +1304,7 @@ uint32_t USB_ReadDevOutEPInterrupt(USB_OTG_GlobalTypeDef *USBx, uint8_t epnum) * This parameter can be a value from 0 to 15 * @retval Device IN EP Interrupt register */ -uint32_t USB_ReadDevInEPInterrupt(USB_OTG_GlobalTypeDef *USBx, uint8_t epnum) +uint32_t USB_ReadDevInEPInterrupt(const USB_OTG_GlobalTypeDef *USBx, uint8_t epnum) { uint32_t USBx_BASE = (uint32_t)USBx; uint32_t tmpreg; @@ -1380,7 +1327,7 @@ uint32_t USB_ReadDevInEPInterrupt(USB_OTG_GlobalTypeDef *USBx, uint8_t epnum) */ void USB_ClearInterrupts(USB_OTG_GlobalTypeDef *USBx, uint32_t interrupt) { - USBx->GINTSTS |= interrupt; + USBx->GINTSTS &= interrupt; } /** @@ -1391,7 +1338,7 @@ void USB_ClearInterrupts(USB_OTG_GlobalTypeDef *USBx, uint32_t interrupt) * 0 : Host * 1 : Device */ -uint32_t USB_GetMode(USB_OTG_GlobalTypeDef *USBx) +uint32_t USB_GetMode(const USB_OTG_GlobalTypeDef *USBx) { return ((USBx->GINTSTS) & 0x1U); } @@ -1401,7 +1348,7 @@ uint32_t USB_GetMode(USB_OTG_GlobalTypeDef *USBx) * @param USBx Selected device * @retval HAL status */ -HAL_StatusTypeDef USB_ActivateSetup(USB_OTG_GlobalTypeDef *USBx) +HAL_StatusTypeDef USB_ActivateSetup(const USB_OTG_GlobalTypeDef *USBx) { uint32_t USBx_BASE = (uint32_t)USBx; @@ -1423,10 +1370,10 @@ HAL_StatusTypeDef USB_ActivateSetup(USB_OTG_GlobalTypeDef *USBx) * @param psetup pointer to setup packet * @retval HAL status */ -HAL_StatusTypeDef USB_EP0_OutStart(USB_OTG_GlobalTypeDef *USBx, uint8_t dma, uint8_t *psetup) +HAL_StatusTypeDef USB_EP0_OutStart(const USB_OTG_GlobalTypeDef *USBx, uint8_t dma, const uint8_t *psetup) { uint32_t USBx_BASE = (uint32_t)USBx; - uint32_t gSNPSiD = *(__IO uint32_t *)(&USBx->CID + 0x1U); + uint32_t gSNPSiD = *(__IO const uint32_t *)(&USBx->CID + 0x1U); if (gSNPSiD > USB_OTG_CORE_ID_300A) { @@ -1465,7 +1412,7 @@ static HAL_StatusTypeDef USB_CoreReset(USB_OTG_GlobalTypeDef *USBx) { count++; - if (count > 200000U) + if (count > HAL_USB_TIMEOUT) { return HAL_TIMEOUT; } @@ -1479,7 +1426,7 @@ static HAL_StatusTypeDef USB_CoreReset(USB_OTG_GlobalTypeDef *USBx) { count++; - if (count > 200000U) + if (count > HAL_USB_TIMEOUT) { return HAL_TIMEOUT; } @@ -1505,7 +1452,9 @@ HAL_StatusTypeDef USB_HostInit(USB_OTG_GlobalTypeDef *USBx, USB_OTG_CfgTypeDef c /* Restart the Phy Clock */ USBx_PCGCCTL = 0U; -#if defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) +#if defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) \ + || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) \ + || defined(STM32F423xx) /* Disable HW VBUS sensing */ USBx->GCCFG &= ~(USB_OTG_GCCFG_VBDEN); #else @@ -1516,13 +1465,17 @@ HAL_StatusTypeDef USB_HostInit(USB_OTG_GlobalTypeDef *USBx, USB_OTG_CfgTypeDef c USBx->GCCFG |= USB_OTG_GCCFG_NOVBUSSENS; USBx->GCCFG &= ~USB_OTG_GCCFG_VBUSBSEN; USBx->GCCFG &= ~USB_OTG_GCCFG_VBUSASEN; -#endif /* defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) */ -#if defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) +#endif /* defined(STM32F446xx) || defined(STM32F469xx) || defined(STM32F479xx) || defined(STM32F412Zx) || + defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || + defined(STM32F423xx) */ +#if defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) \ + || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) /* Disable Battery chargin detector */ USBx->GCCFG &= ~(USB_OTG_GCCFG_BCDEN); -#endif /* defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) */ +#endif /* defined(STM32F412Zx) || defined(STM32F412Vx) || defined(STM32F412Rx) || + defined(STM32F412Cx) || defined(STM32F413xx) || defined(STM32F423xx) */ - if ((USBx->CID & (0x1U << 8)) != 0U) + if ((USBx->GUSBCFG & USB_OTG_GUSBCFG_PHYSEL) == 0U) { if (cfg.speed == USBH_FSLS_SPEED) { @@ -1555,7 +1508,7 @@ HAL_StatusTypeDef USB_HostInit(USB_OTG_GlobalTypeDef *USBx, USB_OTG_CfgTypeDef c /* Clear all pending HC Interrupts */ for (i = 0U; i < cfg.Host_channels; i++) { - USBx_HC(i)->HCINT = 0xFFFFFFFFU; + USBx_HC(i)->HCINT = CLEAR_INTERRUPT_MASK; USBx_HC(i)->HCINTMSK = 0U; } @@ -1563,9 +1516,9 @@ HAL_StatusTypeDef USB_HostInit(USB_OTG_GlobalTypeDef *USBx, USB_OTG_CfgTypeDef c USBx->GINTMSK = 0U; /* Clear any pending interrupts */ - USBx->GINTSTS = 0xFFFFFFFFU; - - if ((USBx->CID & (0x1U << 8)) != 0U) + USBx->GINTSTS = CLEAR_INTERRUPT_MASK; +#if defined (USB_OTG_HS) + if (USBx == USB_OTG_HS) { /* set Rx FIFO size */ USBx->GRXFSIZ = 0x200U; @@ -1573,6 +1526,7 @@ HAL_StatusTypeDef USB_HostInit(USB_OTG_GlobalTypeDef *USBx, USB_OTG_CfgTypeDef c USBx->HPTXFSIZ = (uint32_t)(((0xE0U << 16) & USB_OTG_HPTXFSIZ_PTXFD) | 0x300U); } else +#endif /* defined (USB_OTG_HS) */ { /* set Rx FIFO size */ USBx->GRXFSIZ = 0x80U; @@ -1604,7 +1558,7 @@ HAL_StatusTypeDef USB_HostInit(USB_OTG_GlobalTypeDef *USBx, USB_OTG_CfgTypeDef c * HCFG_6_MHZ : Low Speed 6 MHz Clock * @retval HAL status */ -HAL_StatusTypeDef USB_InitFSLSPClkSel(USB_OTG_GlobalTypeDef *USBx, uint8_t freq) +HAL_StatusTypeDef USB_InitFSLSPClkSel(const USB_OTG_GlobalTypeDef *USBx, uint8_t freq) { uint32_t USBx_BASE = (uint32_t)USBx; @@ -1613,15 +1567,15 @@ HAL_StatusTypeDef USB_InitFSLSPClkSel(USB_OTG_GlobalTypeDef *USBx, uint8_t freq) if (freq == HCFG_48_MHZ) { - USBx_HOST->HFIR = 48000U; + USBx_HOST->HFIR = HFIR_48_MHZ; } else if (freq == HCFG_6_MHZ) { - USBx_HOST->HFIR = 6000U; + USBx_HOST->HFIR = HFIR_6_MHZ; } else { - /* ... */ + return HAL_ERROR; } return HAL_OK; @@ -1634,7 +1588,7 @@ HAL_StatusTypeDef USB_InitFSLSPClkSel(USB_OTG_GlobalTypeDef *USBx, uint8_t freq) * @note (1)The application must wait at least 10 ms * before clearing the reset bit. */ -HAL_StatusTypeDef USB_ResetPort(USB_OTG_GlobalTypeDef *USBx) +HAL_StatusTypeDef USB_ResetPort(const USB_OTG_GlobalTypeDef *USBx) { uint32_t USBx_BASE = (uint32_t)USBx; @@ -1661,7 +1615,7 @@ HAL_StatusTypeDef USB_ResetPort(USB_OTG_GlobalTypeDef *USBx) * 1 : Activate VBUS * @retval HAL status */ -HAL_StatusTypeDef USB_DriveVbus(USB_OTG_GlobalTypeDef *USBx, uint8_t state) +HAL_StatusTypeDef USB_DriveVbus(const USB_OTG_GlobalTypeDef *USBx, uint8_t state) { uint32_t USBx_BASE = (uint32_t)USBx; __IO uint32_t hprt0 = 0U; @@ -1691,7 +1645,7 @@ HAL_StatusTypeDef USB_DriveVbus(USB_OTG_GlobalTypeDef *USBx, uint8_t state) * @arg HCD_SPEED_FULL: Full speed mode * @arg HCD_SPEED_LOW: Low speed mode */ -uint32_t USB_GetHostSpeed(USB_OTG_GlobalTypeDef *USBx) +uint32_t USB_GetHostSpeed(USB_OTG_GlobalTypeDef const *USBx) { uint32_t USBx_BASE = (uint32_t)USBx; __IO uint32_t hprt0 = 0U; @@ -1705,7 +1659,7 @@ uint32_t USB_GetHostSpeed(USB_OTG_GlobalTypeDef *USBx) * @param USBx Selected device * @retval current frame number */ -uint32_t USB_GetCurrentFrame(USB_OTG_GlobalTypeDef *USBx) +uint32_t USB_GetCurrentFrame(USB_OTG_GlobalTypeDef const *USBx) { uint32_t USBx_BASE = (uint32_t)USBx; @@ -1747,7 +1701,7 @@ HAL_StatusTypeDef USB_HC_Init(USB_OTG_GlobalTypeDef *USBx, uint8_t ch_num, uint32_t HostCoreSpeed; /* Clear old interrupt conditions for this host channel. */ - USBx_HC((uint32_t)ch_num)->HCINT = 0xFFFFFFFFU; + USBx_HC((uint32_t)ch_num)->HCINT = CLEAR_INTERRUPT_MASK; /* Enable channel interrupts required for this transfer. */ switch (ep_type) @@ -1767,11 +1721,13 @@ HAL_StatusTypeDef USB_HC_Init(USB_OTG_GlobalTypeDef *USBx, uint8_t ch_num, } else { - if ((USBx->CID & (0x1U << 8)) != 0U) +#if defined (USB_OTG_HS) + if (USBx == USB_OTG_HS) { USBx_HC((uint32_t)ch_num)->HCINTMSK |= USB_OTG_HCINTMSK_NYET | USB_OTG_HCINTMSK_ACKM; } +#endif /* defined (USB_OTG_HS) */ } break; @@ -1808,6 +1764,9 @@ HAL_StatusTypeDef USB_HC_Init(USB_OTG_GlobalTypeDef *USBx, uint8_t ch_num, break; } + /* Clear Hub Start Split transaction */ + USBx_HC((uint32_t)ch_num)->HCSPLT = 0U; + /* Enable host channel Halt interrupt */ USBx_HC((uint32_t)ch_num)->HCINTMSK |= USB_OTG_HCINTMSK_CHHM; @@ -1842,7 +1801,8 @@ HAL_StatusTypeDef USB_HC_Init(USB_OTG_GlobalTypeDef *USBx, uint8_t ch_num, USBx_HC((uint32_t)ch_num)->HCCHAR = (((uint32_t)dev_address << 22) & USB_OTG_HCCHAR_DAD) | ((((uint32_t)epnum & 0x7FU) << 11) & USB_OTG_HCCHAR_EPNUM) | (((uint32_t)ep_type << 18) & USB_OTG_HCCHAR_EPTYP) | - ((uint32_t)mps & USB_OTG_HCCHAR_MPSIZ) | HCcharEpDir | HCcharLowSpeed; + ((uint32_t)mps & USB_OTG_HCCHAR_MPSIZ) | + USB_OTG_HCCHAR_MC_0 | HCcharEpDir | HCcharLowSpeed; if ((ep_type == EP_TYPE_INTR) || (ep_type == EP_TYPE_ISOC)) { @@ -1870,53 +1830,118 @@ HAL_StatusTypeDef USB_HC_StartXfer(USB_OTG_GlobalTypeDef *USBx, USB_OTG_HCTypeDe uint8_t is_oddframe; uint16_t len_words; uint16_t num_packets; - uint16_t max_hc_pkt_count = 256U; + uint16_t max_hc_pkt_count = HC_MAX_PKT_CNT; - if (((USBx->CID & (0x1U << 8)) != 0U) && (hc->speed == USBH_HS_SPEED)) +#if defined (USB_OTG_HS) + if (USBx == USB_OTG_HS) { - /* in DMA mode host Core automatically issues ping in case of NYET/NAK */ - if ((dma == 1U) && ((hc->ep_type == EP_TYPE_CTRL) || (hc->ep_type == EP_TYPE_BULK))) + /* in DMA mode host Core automatically issues ping in case of NYET/NAK */ + if (dma == 1U) { - USBx_HC((uint32_t)ch_num)->HCINTMSK &= ~(USB_OTG_HCINTMSK_NYET | - USB_OTG_HCINTMSK_ACKM | - USB_OTG_HCINTMSK_NAKM); - } + if (((hc->ep_type == EP_TYPE_CTRL) || (hc->ep_type == EP_TYPE_BULK)) && (hc->do_ssplit == 0U)) + { - if ((dma == 0U) && (hc->do_ping == 1U)) + USBx_HC((uint32_t)ch_num)->HCINTMSK &= ~(USB_OTG_HCINTMSK_NYET | + USB_OTG_HCINTMSK_ACKM | + USB_OTG_HCINTMSK_NAKM); + } + } + else { - (void)USB_DoPing(USBx, hc->ch_num); - return HAL_OK; + if ((hc->speed == USBH_HS_SPEED) && (hc->do_ping == 1U)) + { + (void)USB_DoPing(USBx, hc->ch_num); + return HAL_OK; + } } - } +#endif /* defined (USB_OTG_HS) */ - /* Compute the expected number of packets associated to the transfer */ - if (hc->xfer_len > 0U) + if (hc->do_ssplit == 1U) { - num_packets = (uint16_t)((hc->xfer_len + hc->max_packet - 1U) / hc->max_packet); + /* Set number of packet to 1 for Split transaction */ + num_packets = 1U; - if (num_packets > max_hc_pkt_count) + if (hc->ep_is_in != 0U) { - num_packets = max_hc_pkt_count; hc->XferSize = (uint32_t)num_packets * hc->max_packet; } - } - else - { - num_packets = 1U; - } + else + { + if (hc->ep_type == EP_TYPE_ISOC) + { + if (hc->xfer_len > ISO_SPLT_MPS) + { + /* Isochrone Max Packet Size for Split mode */ + hc->XferSize = hc->max_packet; + hc->xfer_len = hc->XferSize; - /* - * For IN channel HCTSIZ.XferSize is expected to be an integer multiple of - * max_packet size. - */ - if (hc->ep_is_in != 0U) - { - hc->XferSize = (uint32_t)num_packets * hc->max_packet; + if ((hc->iso_splt_xactPos == HCSPLT_BEGIN) || (hc->iso_splt_xactPos == HCSPLT_MIDDLE)) + { + hc->iso_splt_xactPos = HCSPLT_MIDDLE; + } + else + { + hc->iso_splt_xactPos = HCSPLT_BEGIN; + } + } + else + { + hc->XferSize = hc->xfer_len; + + if ((hc->iso_splt_xactPos != HCSPLT_BEGIN) && (hc->iso_splt_xactPos != HCSPLT_MIDDLE)) + { + hc->iso_splt_xactPos = HCSPLT_FULL; + } + else + { + hc->iso_splt_xactPos = HCSPLT_END; + } + } + } + else + { + if ((dma == 1U) && (hc->xfer_len > hc->max_packet)) + { + hc->XferSize = (uint32_t)num_packets * hc->max_packet; + } + else + { + hc->XferSize = hc->xfer_len; + } + } + } } else { - hc->XferSize = hc->xfer_len; + /* Compute the expected number of packets associated to the transfer */ + if (hc->xfer_len > 0U) + { + num_packets = (uint16_t)((hc->xfer_len + hc->max_packet - 1U) / hc->max_packet); + + if (num_packets > max_hc_pkt_count) + { + num_packets = max_hc_pkt_count; + hc->XferSize = (uint32_t)num_packets * hc->max_packet; + } + } + else + { + num_packets = 1U; + } + + /* + * For IN channel HCTSIZ.XferSize is expected to be an integer multiple of + * max_packet size. + */ + if (hc->ep_is_in != 0U) + { + hc->XferSize = (uint32_t)num_packets * hc->max_packet; + } + else + { + hc->XferSize = hc->xfer_len; + } } /* Initialize the HCTSIZn register */ @@ -1934,6 +1959,65 @@ HAL_StatusTypeDef USB_HC_StartXfer(USB_OTG_GlobalTypeDef *USBx, USB_OTG_HCTypeDe USBx_HC(ch_num)->HCCHAR &= ~USB_OTG_HCCHAR_ODDFRM; USBx_HC(ch_num)->HCCHAR |= (uint32_t)is_oddframe << 29; + if (hc->do_ssplit == 1U) + { + /* Set Hub start Split transaction */ + USBx_HC((uint32_t)ch_num)->HCSPLT = ((uint32_t)hc->hub_addr << USB_OTG_HCSPLT_HUBADDR_Pos) | + (uint32_t)hc->hub_port_nbr | USB_OTG_HCSPLT_SPLITEN; + + /* unmask ack & nyet for IN/OUT transactions */ + USBx_HC((uint32_t)ch_num)->HCINTMSK |= (USB_OTG_HCINTMSK_ACKM | + USB_OTG_HCINTMSK_NYET); + + if ((hc->do_csplit == 1U) && (hc->ep_is_in == 0U)) + { + USBx_HC((uint32_t)ch_num)->HCSPLT |= USB_OTG_HCSPLT_COMPLSPLT; + USBx_HC((uint32_t)ch_num)->HCINTMSK |= USB_OTG_HCINTMSK_NYET; + } + + if (((hc->ep_type == EP_TYPE_ISOC) || (hc->ep_type == EP_TYPE_INTR)) && + (hc->do_csplit == 1U) && (hc->ep_is_in == 1U)) + { + USBx_HC((uint32_t)ch_num)->HCSPLT |= USB_OTG_HCSPLT_COMPLSPLT; + } + + /* Position management for iso out transaction on split mode */ + if ((hc->ep_type == EP_TYPE_ISOC) && (hc->ep_is_in == 0U)) + { + /* Set data payload position */ + switch (hc->iso_splt_xactPos) + { + case HCSPLT_BEGIN: + /* First data payload for OUT Transaction */ + USBx_HC((uint32_t)ch_num)->HCSPLT |= USB_OTG_HCSPLT_XACTPOS_1; + break; + + case HCSPLT_MIDDLE: + /* Middle data payload for OUT Transaction */ + USBx_HC((uint32_t)ch_num)->HCSPLT |= USB_OTG_HCSPLT_XACTPOS_Pos; + break; + + case HCSPLT_END: + /* End data payload for OUT Transaction */ + USBx_HC((uint32_t)ch_num)->HCSPLT |= USB_OTG_HCSPLT_XACTPOS_0; + break; + + case HCSPLT_FULL: + /* Entire data payload for OUT Transaction */ + USBx_HC((uint32_t)ch_num)->HCSPLT |= USB_OTG_HCSPLT_XACTPOS; + break; + + default: + break; + } + } + } + else + { + /* Clear Hub Start Split transaction */ + USBx_HC((uint32_t)ch_num)->HCSPLT = 0U; + } + /* Set host channel enable */ tmpreg = USBx_HC(ch_num)->HCCHAR; tmpreg &= ~USB_OTG_HCCHAR_CHDIS; @@ -1955,7 +2039,7 @@ HAL_StatusTypeDef USB_HC_StartXfer(USB_OTG_GlobalTypeDef *USBx, USB_OTG_HCTypeDe return HAL_OK; } - if ((hc->ep_is_in == 0U) && (hc->xfer_len > 0U)) + if ((hc->ep_is_in == 0U) && (hc->xfer_len > 0U) && (hc->do_csplit == 0U)) { switch (hc->ep_type) { @@ -2001,7 +2085,7 @@ HAL_StatusTypeDef USB_HC_StartXfer(USB_OTG_GlobalTypeDef *USBx, USB_OTG_HCTypeDe * @param USBx Selected device * @retval HAL state */ -uint32_t USB_HC_ReadInterrupt(USB_OTG_GlobalTypeDef *USBx) +uint32_t USB_HC_ReadInterrupt(const USB_OTG_GlobalTypeDef *USBx) { uint32_t USBx_BASE = (uint32_t)USBx; @@ -2015,16 +2099,21 @@ uint32_t USB_HC_ReadInterrupt(USB_OTG_GlobalTypeDef *USBx) * This parameter can be a value from 1 to 15 * @retval HAL state */ -HAL_StatusTypeDef USB_HC_Halt(USB_OTG_GlobalTypeDef *USBx, uint8_t hc_num) +HAL_StatusTypeDef USB_HC_Halt(const USB_OTG_GlobalTypeDef *USBx, uint8_t hc_num) { uint32_t USBx_BASE = (uint32_t)USBx; uint32_t hcnum = (uint32_t)hc_num; __IO uint32_t count = 0U; uint32_t HcEpType = (USBx_HC(hcnum)->HCCHAR & USB_OTG_HCCHAR_EPTYP) >> 18; uint32_t ChannelEna = (USBx_HC(hcnum)->HCCHAR & USB_OTG_HCCHAR_CHENA) >> 31; + uint32_t SplitEna = (USBx_HC(hcnum)->HCSPLT & USB_OTG_HCSPLT_SPLITEN) >> 31; + + /* In buffer DMA, Channel disable must not be programmed for non-split periodic channels. + At the end of the next uframe/frame (in the worst case), the core generates a channel halted + and disables the channel automatically. */ - if (((USBx->GAHBCFG & USB_OTG_GAHBCFG_DMAEN) == USB_OTG_GAHBCFG_DMAEN) && - (ChannelEna == 0U)) + if ((((USBx->GAHBCFG & USB_OTG_GAHBCFG_DMAEN) == USB_OTG_GAHBCFG_DMAEN) && (SplitEna == 0U)) && + ((ChannelEna == 0U) || (((HcEpType == HCCHAR_ISOC) || (HcEpType == HCCHAR_INTR))))) { return HAL_OK; } @@ -2055,6 +2144,10 @@ HAL_StatusTypeDef USB_HC_Halt(USB_OTG_GlobalTypeDef *USBx, uint8_t hc_num) USBx_HC(hcnum)->HCCHAR |= USB_OTG_HCCHAR_CHENA; } } + else + { + USBx_HC(hcnum)->HCCHAR |= USB_OTG_HCCHAR_CHENA; + } } else { @@ -2090,7 +2183,7 @@ HAL_StatusTypeDef USB_HC_Halt(USB_OTG_GlobalTypeDef *USBx, uint8_t hc_num) * This parameter can be a value from 1 to 15 * @retval HAL state */ -HAL_StatusTypeDef USB_DoPing(USB_OTG_GlobalTypeDef *USBx, uint8_t ch_num) +HAL_StatusTypeDef USB_DoPing(const USB_OTG_GlobalTypeDef *USBx, uint8_t ch_num) { uint32_t USBx_BASE = (uint32_t)USBx; uint32_t chnum = (uint32_t)ch_num; @@ -2166,8 +2259,8 @@ HAL_StatusTypeDef USB_StopHost(USB_OTG_GlobalTypeDef *USBx) } /* Clear any pending Host interrupts */ - USBx_HOST->HAINT = 0xFFFFFFFFU; - USBx->GINTSTS = 0xFFFFFFFFU; + USBx_HOST->HAINT = CLEAR_INTERRUPT_MASK; + USBx->GINTSTS = CLEAR_INTERRUPT_MASK; (void)USB_EnableGlobalInt(USBx); @@ -2179,7 +2272,7 @@ HAL_StatusTypeDef USB_StopHost(USB_OTG_GlobalTypeDef *USBx) * @param USBx Selected device * @retval HAL status */ -HAL_StatusTypeDef USB_ActivateRemoteWakeup(USB_OTG_GlobalTypeDef *USBx) +HAL_StatusTypeDef USB_ActivateRemoteWakeup(const USB_OTG_GlobalTypeDef *USBx) { uint32_t USBx_BASE = (uint32_t)USBx; @@ -2197,7 +2290,7 @@ HAL_StatusTypeDef USB_ActivateRemoteWakeup(USB_OTG_GlobalTypeDef *USBx) * @param USBx Selected device * @retval HAL status */ -HAL_StatusTypeDef USB_DeActivateRemoteWakeup(USB_OTG_GlobalTypeDef *USBx) +HAL_StatusTypeDef USB_DeActivateRemoteWakeup(const USB_OTG_GlobalTypeDef *USBx) { uint32_t USBx_BASE = (uint32_t)USBx; @@ -2208,7 +2301,6 @@ HAL_StatusTypeDef USB_DeActivateRemoteWakeup(USB_OTG_GlobalTypeDef *USBx) } #endif /* defined (USB_OTG_FS) || defined (USB_OTG_HS) */ - /** * @} */ diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/favicon.png b/stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/favicon.png new file mode 100644 index 0000000000000000000000000000000000000000..06713eec4974e141c6e9b4156d34e61e89f282ca GIT binary patch literal 4126 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 literal 0 HcmV?d00001 diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/mini-st_2020.css b/stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/mini-st_2020.css new file mode 100644 index 00000000..db8b406a --- /dev/null +++ b/stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/mini-st_2020.css @@ -0,0 +1,1711 @@ +@charset "UTF-8"; +/* + Flavor name: Custom (mini-custom) + Generated online - https://minicss.org/flavors + mini.css version: v3.0.1 +*/ +/* + Browsers resets and base typography. +*/ +/* Core module CSS variable definitions */ +:root { + --fore-color: #03234b; + --secondary-fore-color: #03234b; + --back-color: #ffffff; + --secondary-back-color: #ffffff; + --blockquote-color: #e6007e; + --pre-color: #e6007e; + --border-color: #3cb4e6; + --secondary-border-color: #3cb4e6; + --heading-ratio: 1.2; + --universal-margin: 0.5rem; + --universal-padding: 0.25rem; + --universal-border-radius: 0.075rem; + --background-margin: 1.5%; + --a-link-color: #3cb4e6; + --a-visited-color: #8c0078; } + +html { + font-size: 13.5px; } + +a, b, del, em, i, ins, q, span, strong, u { + font-size: 1em; } + +html, * { + font-family: -apple-system, BlinkMacSystemFont, Helvetica, arial, sans-serif; + line-height: 1.25; + -webkit-text-size-adjust: 100%; } + +* { + font-size: 1rem; } + +body { + margin: 0; + color: var(--fore-color); + @background: var(--back-color); + background: var(--back-color) linear-gradient(#ffd200, #ffd200) repeat-y left top; + background-size: var(--background-margin); + } + +details { + display: block; } + +summary { + display: list-item; } + +abbr[title] { + border-bottom: none; + text-decoration: underline dotted; } + +input { + overflow: visible; } + +img { + max-width: 100%; + height: auto; } + +h1, h2, h3, h4, h5, h6 { + line-height: 1.25; + margin: calc(1.5 * var(--universal-margin)) var(--universal-margin); + font-weight: 400; } + h1 small, h2 small, h3 small, h4 small, h5 small, h6 small { + color: var(--secondary-fore-color); + display: block; + margin-top: -0.25rem; } + +h1 { + font-size: calc(1rem * var(--heading-ratio) * var(--heading-ratio) * var(--heading-ratio)); } + +h2 { + font-size: calc(1rem * var(--heading-ratio) * var(--heading-ratio) ); + border-style: none none solid none ; + border-width: thin; + border-color: var(--border-color); } +h3 { + font-size: calc(1rem * var(--heading-ratio) ); } + +h4 { + font-size: calc(1rem * var(--heading-ratio)); } + +h5 { + font-size: 1rem; } + +h6 { + font-size: calc(1rem / var(--heading-ratio)); } + +p { + margin: var(--universal-margin); } + +ol, ul { + margin: var(--universal-margin); + padding-left: calc(3 * var(--universal-margin)); } + +b, strong { + font-weight: 700; } + +hr { + box-sizing: content-box; + border: 0; + line-height: 1.25em; + margin: var(--universal-margin); + height: 0.0714285714rem; + background: linear-gradient(to right, transparent, var(--border-color) 20%, var(--border-color) 80%, transparent); } + +blockquote { + display: block; + position: relative; + font-style: italic; + color: var(--secondary-fore-color); + margin: var(--universal-margin); + padding: calc(3 * var(--universal-padding)); + border: 0.0714285714rem solid var(--secondary-border-color); + border-left: 0.3rem solid var(--blockquote-color); + border-radius: 0 var(--universal-border-radius) var(--universal-border-radius) 0; } + blockquote:before { + position: absolute; + top: calc(0rem - var(--universal-padding)); + left: 0; + font-family: sans-serif; + font-size: 2rem; + font-weight: 800; + content: "\201c"; + color: var(--blockquote-color); } + blockquote[cite]:after { + font-style: normal; + font-size: 0.75em; + font-weight: 700; + content: "\a— " attr(cite); + white-space: pre; } + +code, kbd, pre, samp { + font-family: Menlo, Consolas, monospace; + font-size: 0.85em; } + +code { + background: var(--secondary-back-color); + border-radius: var(--universal-border-radius); + padding: calc(var(--universal-padding) / 4) calc(var(--universal-padding) / 2); } + +kbd { + background: var(--fore-color); + color: var(--back-color); + border-radius: var(--universal-border-radius); + padding: calc(var(--universal-padding) / 4) calc(var(--universal-padding) / 2); } + +pre { + overflow: auto; + background: var(--secondary-back-color); + padding: calc(1.5 * var(--universal-padding)); + margin: var(--universal-margin); + border: 0.0714285714rem solid var(--secondary-border-color); + border-left: 0.2857142857rem solid var(--pre-color); + border-radius: 0 var(--universal-border-radius) var(--universal-border-radius) 0; } + +sup, sub, code, kbd { + line-height: 0; + position: relative; + vertical-align: baseline; } + +small, sup, sub, figcaption { + font-size: 0.75em; } + +sup { + top: -0.5em; } + +sub { + bottom: -0.25em; } + +figure { + margin: var(--universal-margin); } + +figcaption { + color: var(--secondary-fore-color); } + +a { + text-decoration: none; } + a:link { + color: var(--a-link-color); } + a:visited { + color: var(--a-visited-color); } + a:hover, a:focus { + text-decoration: underline; } + +/* + Definitions for the grid system, cards and containers. +*/ +.container { + margin: 0 auto; + padding: 0 calc(1.5 * var(--universal-padding)); } + +.row { + box-sizing: border-box; + display: flex; + flex: 0 1 auto; + flex-flow: row wrap; + margin: 0 0 0 var(--background-margin); } + +.col-sm, +[class^='col-sm-'], +[class^='col-sm-offset-'], +.row[class*='cols-sm-'] > * { + box-sizing: border-box; + flex: 0 0 auto; + padding: 0 calc(var(--universal-padding) / 2); } + +.col-sm, +.row.cols-sm > * { + max-width: 100%; + flex-grow: 1; + flex-basis: 0; } + +.col-sm-1, +.row.cols-sm-1 > * { + max-width: 8.3333333333%; + flex-basis: 8.3333333333%; } + +.col-sm-offset-0 { + margin-left: 0; } + +.col-sm-2, +.row.cols-sm-2 > * { + max-width: 16.6666666667%; + flex-basis: 16.6666666667%; } + +.col-sm-offset-1 { + margin-left: 8.3333333333%; } + +.col-sm-3, +.row.cols-sm-3 > * { + max-width: 25%; + flex-basis: 25%; } + +.col-sm-offset-2 { + margin-left: 16.6666666667%; } + +.col-sm-4, +.row.cols-sm-4 > * { + max-width: 33.3333333333%; + flex-basis: 33.3333333333%; } + +.col-sm-offset-3 { + margin-left: 25%; } + +.col-sm-5, +.row.cols-sm-5 > * { + max-width: 41.6666666667%; + flex-basis: 41.6666666667%; } + +.col-sm-offset-4 { + margin-left: 33.3333333333%; } + +.col-sm-6, +.row.cols-sm-6 > * { + max-width: 50%; + flex-basis: 50%; } + +.col-sm-offset-5 { + margin-left: 41.6666666667%; } + +.col-sm-7, +.row.cols-sm-7 > * { + max-width: 58.3333333333%; + flex-basis: 58.3333333333%; } + +.col-sm-offset-6 { + margin-left: 50%; } + +.col-sm-8, +.row.cols-sm-8 > * { + max-width: 66.6666666667%; + flex-basis: 66.6666666667%; } + +.col-sm-offset-7 { + margin-left: 58.3333333333%; } + +.col-sm-9, +.row.cols-sm-9 > * { + max-width: 75%; + flex-basis: 75%; } + +.col-sm-offset-8 { + margin-left: 66.6666666667%; } + +.col-sm-10, +.row.cols-sm-10 > * { + max-width: 83.3333333333%; + flex-basis: 83.3333333333%; } + +.col-sm-offset-9 { + margin-left: 75%; } + +.col-sm-11, +.row.cols-sm-11 > * { + max-width: 91.6666666667%; + flex-basis: 91.6666666667%; } + +.col-sm-offset-10 { + margin-left: 83.3333333333%; } + +.col-sm-12, +.row.cols-sm-12 > * { + max-width: 100%; + flex-basis: 100%; } + +.col-sm-offset-11 { + margin-left: 91.6666666667%; } + +.col-sm-normal { + order: initial; } + +.col-sm-first { + order: -999; } + +.col-sm-last { + order: 999; } + +@media screen and (min-width: 500px) { + .col-md, + [class^='col-md-'], + [class^='col-md-offset-'], + .row[class*='cols-md-'] > * { + box-sizing: border-box; + flex: 0 0 auto; + padding: 0 calc(var(--universal-padding) / 2); } + + .col-md, + .row.cols-md > * { + max-width: 100%; + flex-grow: 1; + flex-basis: 0; } + + .col-md-1, + .row.cols-md-1 > * { + max-width: 8.3333333333%; + flex-basis: 8.3333333333%; } + + .col-md-offset-0 { + margin-left: 0; } + + .col-md-2, + .row.cols-md-2 > * { + max-width: 16.6666666667%; + flex-basis: 16.6666666667%; } + + .col-md-offset-1 { + margin-left: 8.3333333333%; } + + .col-md-3, + .row.cols-md-3 > * { + max-width: 25%; + flex-basis: 25%; } + + .col-md-offset-2 { + margin-left: 16.6666666667%; } + + .col-md-4, + .row.cols-md-4 > * { + max-width: 33.3333333333%; + flex-basis: 33.3333333333%; } + + .col-md-offset-3 { + margin-left: 25%; } + + .col-md-5, + .row.cols-md-5 > * { + max-width: 41.6666666667%; + flex-basis: 41.6666666667%; } + + .col-md-offset-4 { + margin-left: 33.3333333333%; } + + .col-md-6, + .row.cols-md-6 > * { + max-width: 50%; + flex-basis: 50%; } + + .col-md-offset-5 { + margin-left: 41.6666666667%; } + + .col-md-7, + .row.cols-md-7 > * { + max-width: 58.3333333333%; + flex-basis: 58.3333333333%; } + + .col-md-offset-6 { + margin-left: 50%; } + + .col-md-8, + .row.cols-md-8 > * { + max-width: 66.6666666667%; + flex-basis: 66.6666666667%; } + + .col-md-offset-7 { + margin-left: 58.3333333333%; } + + .col-md-9, + .row.cols-md-9 > * { + max-width: 75%; + flex-basis: 75%; } + + .col-md-offset-8 { + margin-left: 66.6666666667%; } + + .col-md-10, + .row.cols-md-10 > * { + max-width: 83.3333333333%; + flex-basis: 83.3333333333%; } + + .col-md-offset-9 { + margin-left: 75%; } + + .col-md-11, + .row.cols-md-11 > * { + max-width: 91.6666666667%; + flex-basis: 91.6666666667%; } + + .col-md-offset-10 { + margin-left: 83.3333333333%; } + + .col-md-12, + .row.cols-md-12 > * { + max-width: 100%; + flex-basis: 100%; } + + .col-md-offset-11 { + margin-left: 91.6666666667%; } + + .col-md-normal { + order: initial; } + + .col-md-first { + order: -999; } + + .col-md-last { + order: 999; } } +@media screen and (min-width: 1280px) { + .col-lg, + [class^='col-lg-'], + [class^='col-lg-offset-'], + .row[class*='cols-lg-'] > * { + box-sizing: border-box; + flex: 0 0 auto; + padding: 0 calc(var(--universal-padding) / 2); } + + .col-lg, + .row.cols-lg > * { + max-width: 100%; + flex-grow: 1; + flex-basis: 0; } + + .col-lg-1, + .row.cols-lg-1 > * { + max-width: 8.3333333333%; + flex-basis: 8.3333333333%; } + + .col-lg-offset-0 { + margin-left: 0; } + + .col-lg-2, + .row.cols-lg-2 > * { + max-width: 16.6666666667%; + flex-basis: 16.6666666667%; } + + .col-lg-offset-1 { + margin-left: 8.3333333333%; } + + .col-lg-3, + .row.cols-lg-3 > * { + max-width: 25%; + flex-basis: 25%; } + + .col-lg-offset-2 { + margin-left: 16.6666666667%; } + + .col-lg-4, + .row.cols-lg-4 > * { + max-width: 33.3333333333%; + flex-basis: 33.3333333333%; } + + .col-lg-offset-3 { + margin-left: 25%; } + + .col-lg-5, + .row.cols-lg-5 > * { + max-width: 41.6666666667%; + flex-basis: 41.6666666667%; } + + .col-lg-offset-4 { + margin-left: 33.3333333333%; } + + .col-lg-6, + .row.cols-lg-6 > * { + max-width: 50%; + flex-basis: 50%; } + + .col-lg-offset-5 { + margin-left: 41.6666666667%; } + + .col-lg-7, + .row.cols-lg-7 > * { + max-width: 58.3333333333%; + flex-basis: 58.3333333333%; } + + .col-lg-offset-6 { + margin-left: 50%; } + + .col-lg-8, + .row.cols-lg-8 > * { + max-width: 66.6666666667%; + flex-basis: 66.6666666667%; } + + .col-lg-offset-7 { + margin-left: 58.3333333333%; } + + .col-lg-9, + .row.cols-lg-9 > * { + max-width: 75%; + flex-basis: 75%; } + + .col-lg-offset-8 { + margin-left: 66.6666666667%; } + + .col-lg-10, + .row.cols-lg-10 > * { + max-width: 83.3333333333%; + flex-basis: 83.3333333333%; } + + .col-lg-offset-9 { + margin-left: 75%; } + + .col-lg-11, + .row.cols-lg-11 > * { + max-width: 91.6666666667%; + flex-basis: 91.6666666667%; } + + .col-lg-offset-10 { + margin-left: 83.3333333333%; } + + .col-lg-12, + .row.cols-lg-12 > * { + max-width: 100%; + flex-basis: 100%; } + + .col-lg-offset-11 { + margin-left: 91.6666666667%; } + + .col-lg-normal { + order: initial; } + + .col-lg-first { + order: -999; } + + .col-lg-last { + order: 999; } } +/* Card component CSS variable definitions */ +:root { + --card-back-color: #3cb4e6; + --card-fore-color: #03234b; + --card-border-color: #03234b; } + +.card { + display: flex; + flex-direction: column; + justify-content: space-between; + align-self: center; + position: relative; + width: 100%; + background: var(--card-back-color); + color: var(--card-fore-color); + border: 0.0714285714rem solid var(--card-border-color); + border-radius: var(--universal-border-radius); + margin: var(--universal-margin); + overflow: hidden; } + @media screen and (min-width: 320px) { + .card { + max-width: 320px; } } + .card > .sectione { + background: var(--card-back-color); + color: var(--card-fore-color); + box-sizing: border-box; + margin: 0; + border: 0; + border-radius: 0; + border-bottom: 0.0714285714rem solid var(--card-border-color); + padding: var(--universal-padding); + width: 100%; } + .card > .sectione.media { + height: 200px; + padding: 0; + -o-object-fit: cover; + object-fit: cover; } + .card > .sectione:last-child { + border-bottom: 0; } + +/* + Custom elements for card elements. +*/ +@media screen and (min-width: 240px) { + .card.small { + max-width: 240px; } } +@media screen and (min-width: 480px) { + .card.large { + max-width: 480px; } } +.card.fluid { + max-width: 100%; + width: auto; } + +.card.warning { + --card-back-color: #e5b8b7; + --card-fore-color: #3b234b; + --card-border-color: #8c0078; } + +.card.error { + --card-back-color: #464650; + --card-fore-color: #ffffff; + --card-border-color: #8c0078; } + +.card > .sectione.dark { + --card-back-color: #3b234b; + --card-fore-color: #ffffff; } + +.card > .sectione.double-padded { + padding: calc(1.5 * var(--universal-padding)); } + +/* + Definitions for forms and input elements. +*/ +/* Input_control module CSS variable definitions */ +:root { + --form-back-color: #ffe97f; + --form-fore-color: #03234b; + --form-border-color: #3cb4e6; + --input-back-color: #ffffff; + --input-fore-color: #03234b; + --input-border-color: #3cb4e6; + --input-focus-color: #0288d1; + --input-invalid-color: #d32f2f; + --button-back-color: #e2e2e2; + --button-hover-back-color: #dcdcdc; + --button-fore-color: #212121; + --button-border-color: transparent; + --button-hover-border-color: transparent; + --button-group-border-color: rgba(124, 124, 124, 0.54); } + +form { + background: var(--form-back-color); + color: var(--form-fore-color); + border: 0.0714285714rem solid var(--form-border-color); + border-radius: var(--universal-border-radius); + margin: var(--universal-margin); + padding: calc(2 * var(--universal-padding)) var(--universal-padding); } + +fieldset { + border: 0.0714285714rem solid var(--form-border-color); + border-radius: var(--universal-border-radius); + margin: calc(var(--universal-margin) / 4); + padding: var(--universal-padding); } + +legend { + box-sizing: border-box; + display: table; + max-width: 100%; + white-space: normal; + font-weight: 500; + padding: calc(var(--universal-padding) / 2); } + +label { + padding: calc(var(--universal-padding) / 2) var(--universal-padding); } + +.input-group { + display: inline-block; } + .input-group.fluid { + display: flex; + align-items: center; + justify-content: center; } + .input-group.fluid > input { + max-width: 100%; + flex-grow: 1; + flex-basis: 0px; } + @media screen and (max-width: 499px) { + .input-group.fluid { + align-items: stretch; + flex-direction: column; } } + .input-group.vertical { + display: flex; + align-items: stretch; + flex-direction: column; } + .input-group.vertical > input { + max-width: 100%; + flex-grow: 1; + flex-basis: 0px; } + +[type="number"]::-webkit-inner-spin-button, [type="number"]::-webkit-outer-spin-button { + height: auto; } + +[type="search"] { + -webkit-appearance: textfield; + outline-offset: -2px; } + +[type="search"]::-webkit-search-cancel-button, +[type="search"]::-webkit-search-decoration { + -webkit-appearance: none; } + +input:not([type]), [type="text"], [type="email"], [type="number"], [type="search"], +[type="password"], [type="url"], [type="tel"], [type="checkbox"], [type="radio"], textarea, select { + box-sizing: border-box; + background: var(--input-back-color); + color: var(--input-fore-color); + border: 0.0714285714rem solid var(--input-border-color); + border-radius: var(--universal-border-radius); + margin: calc(var(--universal-margin) / 2); + padding: var(--universal-padding) calc(1.5 * var(--universal-padding)); } + +input:not([type="button"]):not([type="submit"]):not([type="reset"]):hover, input:not([type="button"]):not([type="submit"]):not([type="reset"]):focus, textarea:hover, textarea:focus, select:hover, select:focus { + border-color: var(--input-focus-color); + box-shadow: none; } +input:not([type="button"]):not([type="submit"]):not([type="reset"]):invalid, input:not([type="button"]):not([type="submit"]):not([type="reset"]):focus:invalid, textarea:invalid, textarea:focus:invalid, select:invalid, select:focus:invalid { + border-color: var(--input-invalid-color); + box-shadow: none; } +input:not([type="button"]):not([type="submit"]):not([type="reset"])[readonly], textarea[readonly], select[readonly] { + background: var(--secondary-back-color); } + +select { + max-width: 100%; } + +option { + overflow: hidden; + text-overflow: ellipsis; } + +[type="checkbox"], [type="radio"] { + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; + position: relative; + height: calc(1rem + var(--universal-padding) / 2); + width: calc(1rem + var(--universal-padding) / 2); + vertical-align: text-bottom; + padding: 0; + flex-basis: calc(1rem + var(--universal-padding) / 2) !important; + flex-grow: 0 !important; } + [type="checkbox"]:checked:before, [type="radio"]:checked:before { + position: absolute; } + +[type="checkbox"]:checked:before { + content: '\2713'; + font-family: sans-serif; + font-size: calc(1rem + var(--universal-padding) / 2); + top: calc(0rem - var(--universal-padding)); + left: calc(var(--universal-padding) / 4); } + +[type="radio"] { + border-radius: 100%; } + [type="radio"]:checked:before { + border-radius: 100%; + content: ''; + top: calc(0.0714285714rem + var(--universal-padding) / 2); + left: calc(0.0714285714rem + var(--universal-padding) / 2); + background: var(--input-fore-color); + width: 0.5rem; + height: 0.5rem; } + +:placeholder-shown { + color: var(--input-fore-color); } + +::-ms-placeholder { + color: var(--input-fore-color); + opacity: 0.54; } + +button::-moz-focus-inner, [type="button"]::-moz-focus-inner, [type="reset"]::-moz-focus-inner, [type="submit"]::-moz-focus-inner { + border-style: none; + padding: 0; } + +button, html [type="button"], [type="reset"], [type="submit"] { + -webkit-appearance: button; } + +button { + overflow: visible; + text-transform: none; } + +button, [type="button"], [type="submit"], [type="reset"], +a.button, label.button, .button, +a[role="button"], label[role="button"], [role="button"] { + display: inline-block; + background: var(--button-back-color); + color: var(--button-fore-color); + border: 0.0714285714rem solid var(--button-border-color); + border-radius: var(--universal-border-radius); + padding: var(--universal-padding) calc(1.5 * var(--universal-padding)); + margin: var(--universal-margin); + text-decoration: none; + cursor: pointer; + transition: background 0.3s; } + button:hover, button:focus, [type="button"]:hover, [type="button"]:focus, [type="submit"]:hover, [type="submit"]:focus, [type="reset"]:hover, [type="reset"]:focus, + a.button:hover, + a.button:focus, label.button:hover, label.button:focus, .button:hover, .button:focus, + a[role="button"]:hover, + a[role="button"]:focus, label[role="button"]:hover, label[role="button"]:focus, [role="button"]:hover, [role="button"]:focus { + background: var(--button-hover-back-color); + border-color: var(--button-hover-border-color); } + +input:disabled, input[disabled], textarea:disabled, textarea[disabled], select:disabled, select[disabled], button:disabled, button[disabled], .button:disabled, .button[disabled], [role="button"]:disabled, [role="button"][disabled] { + cursor: not-allowed; + opacity: 0.75; } + +.button-group { + display: flex; + border: 0.0714285714rem solid var(--button-group-border-color); + border-radius: var(--universal-border-radius); + margin: var(--universal-margin); } + .button-group > button, .button-group [type="button"], .button-group > [type="submit"], .button-group > [type="reset"], .button-group > .button, .button-group > [role="button"] { + margin: 0; + max-width: 100%; + flex: 1 1 auto; + text-align: center; + border: 0; + border-radius: 0; + box-shadow: none; } + .button-group > :not(:first-child) { + border-left: 0.0714285714rem solid var(--button-group-border-color); } + @media screen and (max-width: 499px) { + .button-group { + flex-direction: column; } + .button-group > :not(:first-child) { + border: 0; + border-top: 0.0714285714rem solid var(--button-group-border-color); } } + +/* + Custom elements for forms and input elements. +*/ +button.primary, [type="button"].primary, [type="submit"].primary, [type="reset"].primary, .button.primary, [role="button"].primary { + --button-back-color: #1976d2; + --button-fore-color: #f8f8f8; } + button.primary:hover, button.primary:focus, [type="button"].primary:hover, [type="button"].primary:focus, [type="submit"].primary:hover, [type="submit"].primary:focus, [type="reset"].primary:hover, [type="reset"].primary:focus, .button.primary:hover, .button.primary:focus, [role="button"].primary:hover, [role="button"].primary:focus { + --button-hover-back-color: #1565c0; } + +button.secondary, [type="button"].secondary, [type="submit"].secondary, [type="reset"].secondary, .button.secondary, [role="button"].secondary { + --button-back-color: #d32f2f; + --button-fore-color: #f8f8f8; } + button.secondary:hover, button.secondary:focus, [type="button"].secondary:hover, [type="button"].secondary:focus, [type="submit"].secondary:hover, [type="submit"].secondary:focus, [type="reset"].secondary:hover, [type="reset"].secondary:focus, .button.secondary:hover, .button.secondary:focus, [role="button"].secondary:hover, [role="button"].secondary:focus { + --button-hover-back-color: #c62828; } + +button.tertiary, [type="button"].tertiary, [type="submit"].tertiary, [type="reset"].tertiary, .button.tertiary, [role="button"].tertiary { + --button-back-color: #308732; + --button-fore-color: #f8f8f8; } + button.tertiary:hover, button.tertiary:focus, [type="button"].tertiary:hover, [type="button"].tertiary:focus, [type="submit"].tertiary:hover, [type="submit"].tertiary:focus, [type="reset"].tertiary:hover, [type="reset"].tertiary:focus, .button.tertiary:hover, .button.tertiary:focus, [role="button"].tertiary:hover, [role="button"].tertiary:focus { + --button-hover-back-color: #277529; } + +button.inverse, [type="button"].inverse, [type="submit"].inverse, [type="reset"].inverse, .button.inverse, [role="button"].inverse { + --button-back-color: #212121; + --button-fore-color: #f8f8f8; } + button.inverse:hover, button.inverse:focus, [type="button"].inverse:hover, [type="button"].inverse:focus, [type="submit"].inverse:hover, [type="submit"].inverse:focus, [type="reset"].inverse:hover, [type="reset"].inverse:focus, .button.inverse:hover, .button.inverse:focus, [role="button"].inverse:hover, [role="button"].inverse:focus { + --button-hover-back-color: #111; } + +button.small, [type="button"].small, [type="submit"].small, [type="reset"].small, .button.small, [role="button"].small { + padding: calc(0.5 * var(--universal-padding)) calc(0.75 * var(--universal-padding)); + margin: var(--universal-margin); } + +button.large, [type="button"].large, [type="submit"].large, [type="reset"].large, .button.large, [role="button"].large { + padding: calc(1.5 * var(--universal-padding)) calc(2 * var(--universal-padding)); + margin: var(--universal-margin); } + +/* + Definitions for navigation elements. +*/ +/* Navigation module CSS variable definitions */ +:root { + --header-back-color: #03234b; + --header-hover-back-color: #ffd200; + --header-fore-color: #ffffff; + --header-border-color: #3cb4e6; + --nav-back-color: #ffffff; + --nav-hover-back-color: #ffe97f; + --nav-fore-color: #e6007e; + --nav-border-color: #3cb4e6; + --nav-link-color: #3cb4e6; + --footer-fore-color: #ffffff; + --footer-back-color: #03234b; + --footer-border-color: #3cb4e6; + --footer-link-color: #3cb4e6; + --drawer-back-color: #ffffff; + --drawer-hover-back-color: #ffe97f; + --drawer-border-color: #3cb4e6; + --drawer-close-color: #e6007e; } + +header { + height: 2.75rem; + background: var(--header-back-color); + color: var(--header-fore-color); + border-bottom: 0.0714285714rem solid var(--header-border-color); + padding: calc(var(--universal-padding) / 4) 0; + white-space: nowrap; + overflow-x: auto; + overflow-y: hidden; } + header.row { + box-sizing: content-box; } + header .logo { + color: var(--header-fore-color); + font-size: 1.75rem; + padding: var(--universal-padding) calc(2 * var(--universal-padding)); + text-decoration: none; } + header button, header [type="button"], header .button, header [role="button"] { + box-sizing: border-box; + position: relative; + top: calc(0rem - var(--universal-padding) / 4); + height: calc(3.1875rem + var(--universal-padding) / 2); + background: var(--header-back-color); + line-height: calc(3.1875rem - var(--universal-padding) * 1.5); + text-align: center; + color: var(--header-fore-color); + border: 0; + border-radius: 0; + margin: 0; + text-transform: uppercase; } + header button:hover, header button:focus, header [type="button"]:hover, header [type="button"]:focus, header .button:hover, header .button:focus, header [role="button"]:hover, header [role="button"]:focus { + background: var(--header-hover-back-color); } + +nav { + background: var(--nav-back-color); + color: var(--nav-fore-color); + border: 0.0714285714rem solid var(--nav-border-color); + border-radius: var(--universal-border-radius); + margin: var(--universal-margin); } + nav * { + padding: var(--universal-padding) calc(1.5 * var(--universal-padding)); } + nav a, nav a:visited { + display: block; + color: var(--nav-link-color); + border-radius: var(--universal-border-radius); + transition: background 0.3s; } + nav a:hover, nav a:focus, nav a:visited:hover, nav a:visited:focus { + text-decoration: none; + background: var(--nav-hover-back-color); } + nav .sublink-1 { + position: relative; + margin-left: calc(2 * var(--universal-padding)); } + nav .sublink-1:before { + position: absolute; + left: calc(var(--universal-padding) - 1 * var(--universal-padding)); + top: -0.0714285714rem; + content: ''; + height: 100%; + border: 0.0714285714rem solid var(--nav-border-color); + border-left: 0; } + nav .sublink-2 { + position: relative; + margin-left: calc(4 * var(--universal-padding)); } + nav .sublink-2:before { + position: absolute; + left: calc(var(--universal-padding) - 3 * var(--universal-padding)); + top: -0.0714285714rem; + content: ''; + height: 100%; + border: 0.0714285714rem solid var(--nav-border-color); + border-left: 0; } + +footer { + background: var(--footer-back-color); + color: var(--footer-fore-color); + border-top: 0.0714285714rem solid var(--footer-border-color); + padding: calc(2 * var(--universal-padding)) var(--universal-padding); + font-size: 0.875rem; } + footer a, footer a:visited { + color: var(--footer-link-color); } + +header.sticky { + position: -webkit-sticky; + position: sticky; + z-index: 1101; + top: 0; } + +footer.sticky { + position: -webkit-sticky; + position: sticky; + z-index: 1101; + bottom: 0; } + +.drawer-toggle:before { + display: inline-block; + position: relative; + vertical-align: bottom; + content: '\00a0\2261\00a0'; + font-family: sans-serif; + font-size: 1.5em; } +@media screen and (min-width: 500px) { + .drawer-toggle:not(.persistent) { + display: none; } } + +[type="checkbox"].drawer { + height: 1px; + width: 1px; + margin: -1px; + overflow: hidden; + position: absolute; + clip: rect(0 0 0 0); + -webkit-clip-path: inset(100%); + clip-path: inset(100%); } + [type="checkbox"].drawer + * { + display: block; + box-sizing: border-box; + position: fixed; + top: 0; + width: 320px; + height: 100vh; + overflow-y: auto; + background: var(--drawer-back-color); + border: 0.0714285714rem solid var(--drawer-border-color); + border-radius: 0; + margin: 0; + z-index: 1110; + right: -320px; + transition: right 0.3s; } + [type="checkbox"].drawer + * .drawer-close { + position: absolute; + top: var(--universal-margin); + right: var(--universal-margin); + z-index: 1111; + width: 2rem; + height: 2rem; + border-radius: var(--universal-border-radius); + padding: var(--universal-padding); + margin: 0; + cursor: pointer; + transition: background 0.3s; } + [type="checkbox"].drawer + * .drawer-close:before { + display: block; + content: '\00D7'; + color: var(--drawer-close-color); + position: relative; + font-family: sans-serif; + font-size: 2rem; + line-height: 1; + text-align: center; } + [type="checkbox"].drawer + * .drawer-close:hover, [type="checkbox"].drawer + * .drawer-close:focus { + background: var(--drawer-hover-back-color); } + @media screen and (max-width: 320px) { + [type="checkbox"].drawer + * { + width: 100%; } } + [type="checkbox"].drawer:checked + * { + right: 0; } + @media screen and (min-width: 500px) { + [type="checkbox"].drawer:not(.persistent) + * { + position: static; + height: 100%; + z-index: 1100; } + [type="checkbox"].drawer:not(.persistent) + * .drawer-close { + display: none; } } + +/* + Definitions for the responsive table component. +*/ +/* Table module CSS variable definitions. */ +:root { + --table-border-color: #03234b; + --table-border-separator-color: #03234b; + --table-head-back-color: #03234b; + --table-head-fore-color: #ffffff; + --table-body-back-color: #ffffff; + --table-body-fore-color: #03234b; + --table-body-alt-back-color: #f4f4f4; } + +table { + border-collapse: separate; + border-spacing: 0; + margin: 0; + display: flex; + flex: 0 1 auto; + flex-flow: row wrap; + padding: var(--universal-padding); + padding-top: 0; } + table caption { + font-size: 1rem; + margin: calc(2 * var(--universal-margin)) 0; + max-width: 100%; + flex: 0 0 100%; } + table thead, table tbody { + display: flex; + flex-flow: row wrap; + border: 0.0714285714rem solid var(--table-border-color); } + table thead { + z-index: 999; + border-radius: var(--universal-border-radius) var(--universal-border-radius) 0 0; + border-bottom: 0.0714285714rem solid var(--table-border-separator-color); } + table tbody { + border-top: 0; + margin-top: calc(0 - var(--universal-margin)); + border-radius: 0 0 var(--universal-border-radius) var(--universal-border-radius); } + table tr { + display: flex; + padding: 0; } + table th, table td { + padding: calc(0.5 * var(--universal-padding)); + font-size: 0.9rem; } + table th { + text-align: left; + background: var(--table-head-back-color); + color: var(--table-head-fore-color); } + table td { + background: var(--table-body-back-color); + color: var(--table-body-fore-color); + border-top: 0.0714285714rem solid var(--table-border-color); } + +table:not(.horizontal) { + overflow: auto; + max-height: 100%; } + table:not(.horizontal) thead, table:not(.horizontal) tbody { + max-width: 100%; + flex: 0 0 100%; } + table:not(.horizontal) tr { + flex-flow: row wrap; + flex: 0 0 100%; } + table:not(.horizontal) th, table:not(.horizontal) td { + flex: 1 0 0%; + overflow: hidden; + text-overflow: ellipsis; } + table:not(.horizontal) thead { + position: sticky; + top: 0; } + table:not(.horizontal) tbody tr:first-child td { + border-top: 0; } + +table.horizontal { + border: 0; } + table.horizontal thead, table.horizontal tbody { + border: 0; + flex: .2 0 0; + flex-flow: row nowrap; } + table.horizontal tbody { + overflow: auto; + justify-content: space-between; + flex: .8 0 0; + margin-left: 0; + padding-bottom: calc(var(--universal-padding) / 4); } + table.horizontal tr { + flex-direction: column; + flex: 1 0 auto; } + table.horizontal th, table.horizontal td { + width: auto; + border: 0; + border-bottom: 0.0714285714rem solid var(--table-border-color); } + table.horizontal th:not(:first-child), table.horizontal td:not(:first-child) { + border-top: 0; } + table.horizontal th { + text-align: right; + border-left: 0.0714285714rem solid var(--table-border-color); + border-right: 0.0714285714rem solid var(--table-border-separator-color); } + table.horizontal thead tr:first-child { + padding-left: 0; } + table.horizontal th:first-child, table.horizontal td:first-child { + border-top: 0.0714285714rem solid var(--table-border-color); } + table.horizontal tbody tr:last-child td { + border-right: 0.0714285714rem solid var(--table-border-color); } + table.horizontal tbody tr:last-child td:first-child { + border-top-right-radius: 0.25rem; } + table.horizontal tbody tr:last-child td:last-child { + border-bottom-right-radius: 0.25rem; } + table.horizontal thead tr:first-child th:first-child { + border-top-left-radius: 0.25rem; } + table.horizontal thead tr:first-child th:last-child { + border-bottom-left-radius: 0.25rem; } + +@media screen and (max-width: 499px) { + table, table.horizontal { + border-collapse: collapse; + border: 0; + width: 100%; + display: table; } + table thead, table th, table.horizontal thead, table.horizontal th { + border: 0; + height: 1px; + width: 1px; + margin: -1px; + overflow: hidden; + padding: 0; + position: absolute; + clip: rect(0 0 0 0); + -webkit-clip-path: inset(100%); + clip-path: inset(100%); } + table tbody, table.horizontal tbody { + border: 0; + display: table-row-group; } + table tr, table.horizontal tr { + display: block; + border: 0.0714285714rem solid var(--table-border-color); + border-radius: var(--universal-border-radius); + background: #ffffff; + padding: var(--universal-padding); + margin: var(--universal-margin); + margin-bottom: calc(1 * var(--universal-margin)); } + table th, table td, table.horizontal th, table.horizontal td { + width: auto; } + table td, table.horizontal td { + display: block; + border: 0; + text-align: right; } + table td:before, table.horizontal td:before { + content: attr(data-label); + float: left; + font-weight: 600; } + table th:first-child, table td:first-child, table.horizontal th:first-child, table.horizontal td:first-child { + border-top: 0; } + table tbody tr:last-child td, table.horizontal tbody tr:last-child td { + border-right: 0; } } +table tr:nth-of-type(2n) > td { + background: var(--table-body-alt-back-color); } + +@media screen and (max-width: 500px) { + table tr:nth-of-type(2n) { + background: var(--table-body-alt-back-color); } } +:root { + --table-body-hover-back-color: #90caf9; } + +table.hoverable tr:hover, table.hoverable tr:hover > td, table.hoverable tr:focus, table.hoverable tr:focus > td { + background: var(--table-body-hover-back-color); } + +@media screen and (max-width: 500px) { + table.hoverable tr:hover, table.hoverable tr:hover > td, table.hoverable tr:focus, table.hoverable tr:focus > td { + background: var(--table-body-hover-back-color); } } +/* + Definitions for contextual background elements, toasts and tooltips. +*/ +/* Contextual module CSS variable definitions */ +:root { + --mark-back-color: #3cb4e6; + --mark-fore-color: #ffffff; } + +mark { + background: var(--mark-back-color); + color: var(--mark-fore-color); + font-size: 0.95em; + line-height: 1em; + border-radius: var(--universal-border-radius); + padding: calc(var(--universal-padding) / 4) var(--universal-padding); } + mark.inline-block { + display: inline-block; + font-size: 1em; + line-height: 1.4; + padding: calc(var(--universal-padding) / 2) var(--universal-padding); } + +:root { + --toast-back-color: #424242; + --toast-fore-color: #fafafa; } + +.toast { + position: fixed; + bottom: calc(var(--universal-margin) * 3); + left: 50%; + transform: translate(-50%, -50%); + z-index: 1111; + color: var(--toast-fore-color); + background: var(--toast-back-color); + border-radius: calc(var(--universal-border-radius) * 16); + padding: var(--universal-padding) calc(var(--universal-padding) * 3); } + +:root { + --tooltip-back-color: #212121; + --tooltip-fore-color: #fafafa; } + +.tooltip { + position: relative; + display: inline-block; } + .tooltip:before, .tooltip:after { + position: absolute; + opacity: 0; + clip: rect(0 0 0 0); + -webkit-clip-path: inset(100%); + clip-path: inset(100%); + transition: all 0.3s; + z-index: 1010; + left: 50%; } + .tooltip:not(.bottom):before, .tooltip:not(.bottom):after { + bottom: 75%; } + .tooltip.bottom:before, .tooltip.bottom:after { + top: 75%; } + .tooltip:hover:before, .tooltip:hover:after, .tooltip:focus:before, .tooltip:focus:after { + opacity: 1; + clip: auto; + -webkit-clip-path: inset(0%); + clip-path: inset(0%); } + .tooltip:before { + content: ''; + background: transparent; + border: var(--universal-margin) solid transparent; + left: calc(50% - var(--universal-margin)); } + .tooltip:not(.bottom):before { + border-top-color: #212121; } + .tooltip.bottom:before { + border-bottom-color: #212121; } + .tooltip:after { + content: attr(aria-label); + color: var(--tooltip-fore-color); + background: var(--tooltip-back-color); + border-radius: var(--universal-border-radius); + padding: var(--universal-padding); + white-space: nowrap; + transform: translateX(-50%); } + .tooltip:not(.bottom):after { + margin-bottom: calc(2 * var(--universal-margin)); } + .tooltip.bottom:after { + margin-top: calc(2 * var(--universal-margin)); } + +:root { + --modal-overlay-color: rgba(0, 0, 0, 0.45); + --modal-close-color: #e6007e; + --modal-close-hover-color: #ffe97f; } + +[type="checkbox"].modal { + height: 1px; + width: 1px; + margin: -1px; + overflow: hidden; + position: absolute; + clip: rect(0 0 0 0); + -webkit-clip-path: inset(100%); + clip-path: inset(100%); } + [type="checkbox"].modal + div { + position: fixed; + top: 0; + left: 0; + display: none; + width: 100vw; + height: 100vh; + background: var(--modal-overlay-color); } + [type="checkbox"].modal + div .card { + margin: 0 auto; + max-height: 50vh; + overflow: auto; } + [type="checkbox"].modal + div .card .modal-close { + position: absolute; + top: 0; + right: 0; + width: 1.75rem; + height: 1.75rem; + border-radius: var(--universal-border-radius); + padding: var(--universal-padding); + margin: 0; + cursor: pointer; + transition: background 0.3s; } + [type="checkbox"].modal + div .card .modal-close:before { + display: block; + content: '\00D7'; + color: var(--modal-close-color); + position: relative; + font-family: sans-serif; + font-size: 1.75rem; + line-height: 1; + text-align: center; } + [type="checkbox"].modal + div .card .modal-close:hover, [type="checkbox"].modal + div .card .modal-close:focus { + background: var(--modal-close-hover-color); } + [type="checkbox"].modal:checked + div { + display: flex; + flex: 0 1 auto; + z-index: 1200; } + [type="checkbox"].modal:checked + div .card .modal-close { + z-index: 1211; } + +:root { + --collapse-label-back-color: #03234b; + --collapse-label-fore-color: #ffffff; + --collapse-label-hover-back-color: #3cb4e6; + --collapse-selected-label-back-color: #3cb4e6; + --collapse-border-color: var(--collapse-label-back-color); + --collapse-selected-border-color: #ceecf8; + --collapse-content-back-color: #ffffff; + --collapse-selected-label-border-color: #3cb4e6; } + +.collapse { + width: calc(100% - 2 * var(--universal-margin)); + opacity: 1; + display: flex; + flex-direction: column; + margin: var(--universal-margin); + border-radius: var(--universal-border-radius); } + .collapse > [type="radio"], .collapse > [type="checkbox"] { + height: 1px; + width: 1px; + margin: -1px; + overflow: hidden; + position: absolute; + clip: rect(0 0 0 0); + -webkit-clip-path: inset(100%); + clip-path: inset(100%); } + .collapse > label { + flex-grow: 1; + display: inline-block; + height: 1.25rem; + cursor: pointer; + transition: background 0.2s; + color: var(--collapse-label-fore-color); + background: var(--collapse-label-back-color); + border: 0.0714285714rem solid var(--collapse-selected-border-color); + padding: calc(1.25 * var(--universal-padding)); } + .collapse > label:hover, .collapse > label:focus { + background: var(--collapse-label-hover-back-color); } + .collapse > label + div { + flex-basis: auto; + height: 1px; + width: 1px; + margin: -1px; + overflow: hidden; + position: absolute; + clip: rect(0 0 0 0); + -webkit-clip-path: inset(100%); + clip-path: inset(100%); + transition: max-height 0.3s; + max-height: 1px; } + .collapse > :checked + label { + background: var(--collapse-selected-label-back-color); + border-color: var(--collapse-selected-label-border-color); } + .collapse > :checked + label + div { + box-sizing: border-box; + position: relative; + width: 100%; + height: auto; + overflow: auto; + margin: 0; + background: var(--collapse-content-back-color); + border: 0.0714285714rem solid var(--collapse-selected-border-color); + border-top: 0; + padding: var(--universal-padding); + clip: auto; + -webkit-clip-path: inset(0%); + clip-path: inset(0%); + max-height: 100%; } + .collapse > label:not(:first-of-type) { + border-top: 0; } + .collapse > label:first-of-type { + border-radius: var(--universal-border-radius) var(--universal-border-radius) 0 0; } + .collapse > label:last-of-type:not(:first-of-type) { + border-radius: 0 0 var(--universal-border-radius) var(--universal-border-radius); } + .collapse > label:last-of-type:first-of-type { + border-radius: var(--universal-border-radius); } + .collapse > :checked:last-of-type:not(:first-of-type) + label { + border-radius: 0; } + .collapse > :checked:last-of-type + label + div { + border-radius: 0 0 var(--universal-border-radius) var(--universal-border-radius); } + +/* + Custom elements for contextual background elements, toasts and tooltips. +*/ +mark.tertiary { + --mark-back-color: #3cb4e6; } + +mark.tag { + padding: calc(var(--universal-padding)/2) var(--universal-padding); + border-radius: 1em; } + +/* + Definitions for progress elements and spinners. +*/ +/* Progress module CSS variable definitions */ +:root { + --progress-back-color: #3cb4e6; + --progress-fore-color: #555; } + +progress { + display: block; + vertical-align: baseline; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; + height: 0.75rem; + width: calc(100% - 2 * var(--universal-margin)); + margin: var(--universal-margin); + border: 0; + border-radius: calc(2 * var(--universal-border-radius)); + background: var(--progress-back-color); + color: var(--progress-fore-color); } + progress::-webkit-progress-value { + background: var(--progress-fore-color); + border-top-left-radius: calc(2 * var(--universal-border-radius)); + border-bottom-left-radius: calc(2 * var(--universal-border-radius)); } + progress::-webkit-progress-bar { + background: var(--progress-back-color); } + progress::-moz-progress-bar { + background: var(--progress-fore-color); + border-top-left-radius: calc(2 * var(--universal-border-radius)); + border-bottom-left-radius: calc(2 * var(--universal-border-radius)); } + progress[value="1000"]::-webkit-progress-value { + border-radius: calc(2 * var(--universal-border-radius)); } + progress[value="1000"]::-moz-progress-bar { + border-radius: calc(2 * var(--universal-border-radius)); } + progress.inline { + display: inline-block; + vertical-align: middle; + width: 60%; } + +:root { + --spinner-back-color: #ddd; + --spinner-fore-color: #555; } + +@keyframes spinner-donut-anim { + 0% { + transform: rotate(0deg); } + 100% { + transform: rotate(360deg); } } +.spinner { + display: inline-block; + margin: var(--universal-margin); + border: 0.25rem solid var(--spinner-back-color); + border-left: 0.25rem solid var(--spinner-fore-color); + border-radius: 50%; + width: 1.25rem; + height: 1.25rem; + animation: spinner-donut-anim 1.2s linear infinite; } + +/* + Custom elements for progress bars and spinners. +*/ +progress.primary { + --progress-fore-color: #1976d2; } + +progress.secondary { + --progress-fore-color: #d32f2f; } + +progress.tertiary { + --progress-fore-color: #308732; } + +.spinner.primary { + --spinner-fore-color: #1976d2; } + +.spinner.secondary { + --spinner-fore-color: #d32f2f; } + +.spinner.tertiary { + --spinner-fore-color: #308732; } + +/* + Definitions for icons - powered by Feather (https://feathericons.com/). +*/ +span[class^='icon-'] { + display: inline-block; + height: 1em; + width: 1em; + vertical-align: -0.125em; + background-size: contain; + margin: 0 calc(var(--universal-margin) / 4); } + span[class^='icon-'].secondary { + -webkit-filter: invert(25%); + filter: invert(25%); } + span[class^='icon-'].inverse { + -webkit-filter: invert(100%); + filter: invert(100%); } + +span.icon-alert { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='8' x2='12' y2='12'%3E%3C/line%3E%3Cline x1='12' y1='16' x2='12' y2='16'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-bookmark { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M19 21l-7-5-7 5V5a2 2 0 0 1 2-2h10a2 2 0 0 1 2 2z'%3E%3C/path%3E%3C/svg%3E"); } +span.icon-calendar { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='3' y='4' width='18' height='18' rx='2' ry='2'%3E%3C/rect%3E%3Cline x1='16' y1='2' x2='16' y2='6'%3E%3C/line%3E%3Cline x1='8' y1='2' x2='8' y2='6'%3E%3C/line%3E%3Cline x1='3' y1='10' x2='21' y2='10'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-credit { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='1' y='4' width='22' height='16' rx='2' ry='2'%3E%3C/rect%3E%3Cline x1='1' y1='10' x2='23' y2='10'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-edit { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M20 14.66V20a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V6a2 2 0 0 1 2-2h5.34'%3E%3C/path%3E%3Cpolygon points='18 2 22 6 12 16 8 16 8 12 18 2'%3E%3C/polygon%3E%3C/svg%3E"); } +span.icon-link { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6'%3E%3C/path%3E%3Cpolyline points='15 3 21 3 21 9'%3E%3C/polyline%3E%3Cline x1='10' y1='14' x2='21' y2='3'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-help { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M9.09 9a3 3 0 0 1 5.83 1c0 2-3 3-3 3'%3E%3C/path%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='17' x2='12' y2='17'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-home { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M3 9l9-7 9 7v11a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2z'%3E%3C/path%3E%3Cpolyline points='9 22 9 12 15 12 15 22'%3E%3C/polyline%3E%3C/svg%3E"); } +span.icon-info { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='16' x2='12' y2='12'%3E%3C/line%3E%3Cline x1='12' y1='8' x2='12' y2='8'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-lock { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='3' y='11' width='18' height='11' rx='2' ry='2'%3E%3C/rect%3E%3Cpath d='M7 11V7a5 5 0 0 1 10 0v4'%3E%3C/path%3E%3C/svg%3E"); } +span.icon-mail { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M4 4h16c1.1 0 2 .9 2 2v12c0 1.1-.9 2-2 2H4c-1.1 0-2-.9-2-2V6c0-1.1.9-2 2-2z'%3E%3C/path%3E%3Cpolyline points='22,6 12,13 2,6'%3E%3C/polyline%3E%3C/svg%3E"); } +span.icon-location { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M21 10c0 7-9 13-9 13s-9-6-9-13a9 9 0 0 1 18 0z'%3E%3C/path%3E%3Ccircle cx='12' cy='10' r='3'%3E%3C/circle%3E%3C/svg%3E"); } +span.icon-phone { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M22 16.92v3a2 2 0 0 1-2.18 2 19.79 19.79 0 0 1-8.63-3.07 19.5 19.5 0 0 1-6-6 19.79 19.79 0 0 1-3.07-8.67A2 2 0 0 1 4.11 2h3a2 2 0 0 1 2 1.72 12.84 12.84 0 0 0 .7 2.81 2 2 0 0 1-.45 2.11L8.09 9.91a16 16 0 0 0 6 6l1.27-1.27a2 2 0 0 1 2.11-.45 12.84 12.84 0 0 0 2.81.7A2 2 0 0 1 22 16.92z'%3E%3C/path%3E%3C/svg%3E"); } +span.icon-rss { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M4 11a9 9 0 0 1 9 9'%3E%3C/path%3E%3Cpath d='M4 4a16 16 0 0 1 16 16'%3E%3C/path%3E%3Ccircle cx='5' cy='19' r='1'%3E%3C/circle%3E%3C/svg%3E"); } +span.icon-search { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='11' cy='11' r='8'%3E%3C/circle%3E%3Cline x1='21' y1='21' x2='16.65' y2='16.65'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-settings { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='3'%3E%3C/circle%3E%3Cpath d='M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z'%3E%3C/path%3E%3C/svg%3E"); } +span.icon-share { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='18' cy='5' r='3'%3E%3C/circle%3E%3Ccircle cx='6' cy='12' r='3'%3E%3C/circle%3E%3Ccircle cx='18' cy='19' r='3'%3E%3C/circle%3E%3Cline x1='8.59' y1='13.51' x2='15.42' y2='17.49'%3E%3C/line%3E%3Cline x1='15.41' y1='6.51' x2='8.59' y2='10.49'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-cart { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='9' cy='21' r='1'%3E%3C/circle%3E%3Ccircle cx='20' cy='21' r='1'%3E%3C/circle%3E%3Cpath d='M1 1h4l2.68 13.39a2 2 0 0 0 2 1.61h9.72a2 2 0 0 0 2-1.61L23 6H6'%3E%3C/path%3E%3C/svg%3E"); } +span.icon-upload { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4'%3E%3C/path%3E%3Cpolyline points='17 8 12 3 7 8'%3E%3C/polyline%3E%3Cline x1='12' y1='3' x2='12' y2='15'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-user { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M20 21v-2a4 4 0 0 0-4-4H8a4 4 0 0 0-4 4v2'%3E%3C/path%3E%3Ccircle cx='12' cy='7' r='4'%3E%3C/circle%3E%3C/svg%3E"); } + +/* + Definitions for STMicroelectronics icons (https://brandportal.st.com/document/26). +*/ +span.icon-st-update { + background-image: url("Update.svg"); } +span.icon-st-add { + background-image: url("Add button.svg"); } + +/* + Definitions for utilities and helper classes. +*/ +/* Utility module CSS variable definitions */ +:root { + --generic-border-color: rgba(0, 0, 0, 0.3); + --generic-box-shadow: 0 0.2857142857rem 0.2857142857rem 0 rgba(0, 0, 0, 0.125), 0 0.1428571429rem 0.1428571429rem -0.1428571429rem rgba(0, 0, 0, 0.125); } + +.hidden { + display: none !important; } + +.visually-hidden { + position: absolute !important; + width: 1px !important; + height: 1px !important; + margin: -1px !important; + border: 0 !important; + padding: 0 !important; + clip: rect(0 0 0 0) !important; + -webkit-clip-path: inset(100%) !important; + clip-path: inset(100%) !important; + overflow: hidden !important; } + +.bordered { + border: 0.0714285714rem solid var(--generic-border-color) !important; } + +.rounded { + border-radius: var(--universal-border-radius) !important; } + +.circular { + border-radius: 50% !important; } + +.shadowed { + box-shadow: var(--generic-box-shadow) !important; } + +.responsive-margin { + margin: calc(var(--universal-margin) / 4) !important; } + @media screen and (min-width: 500px) { + .responsive-margin { + margin: calc(var(--universal-margin) / 2) !important; } } + @media screen and (min-width: 1280px) { + .responsive-margin { + margin: var(--universal-margin) !important; } } + +.responsive-padding { + padding: calc(var(--universal-padding) / 4) !important; } + @media screen and (min-width: 500px) { + .responsive-padding { + padding: calc(var(--universal-padding) / 2) !important; } } + @media screen and (min-width: 1280px) { + .responsive-padding { + padding: var(--universal-padding) !important; } } + +@media screen and (max-width: 499px) { + .hidden-sm { + display: none !important; } } +@media screen and (min-width: 500px) and (max-width: 1279px) { + .hidden-md { + display: none !important; } } +@media screen and (min-width: 1280px) { + .hidden-lg { + display: none !important; } } +@media screen and (max-width: 499px) { + .visually-hidden-sm { + position: absolute !important; + width: 1px !important; + height: 1px !important; + margin: -1px !important; + border: 0 !important; + padding: 0 !important; + clip: rect(0 0 0 0) !important; + -webkit-clip-path: inset(100%) !important; + clip-path: inset(100%) !important; + overflow: hidden !important; } } +@media screen and (min-width: 500px) and (max-width: 1279px) { + .visually-hidden-md { + position: absolute !important; + width: 1px !important; + height: 1px !important; + margin: -1px !important; + border: 0 !important; + padding: 0 !important; + clip: rect(0 0 0 0) !important; + -webkit-clip-path: inset(100%) !important; + clip-path: inset(100%) !important; + overflow: hidden !important; } } +@media screen and (min-width: 1280px) { + .visually-hidden-lg { + position: absolute !important; + width: 1px !important; + height: 1px !important; + margin: -1px !important; + border: 0 !important; + padding: 0 !important; + clip: rect(0 0 0 0) !important; + -webkit-clip-path: inset(100%) !important; + clip-path: inset(100%) !important; + overflow: hidden !important; } } + +/*# sourceMappingURL=mini-custom.css.map */ + +img[alt="ST logo"] { display: block; margin: auto; width: 75%; max-width: 250px; min-width: 71px; } +img[alt="Cube logo"] { float: right; width: 30%; max-width: 10rem; min-width: 8rem; padding-right: 1rem;} + +.figure { + display: block; + margin-left: auto; + margin-right: auto; + text-align: center; +} \ No newline at end of file diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/st_logo.png b/stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/st_logo.png deleted file mode 100644 index 8b80057fd3a454a97de1c9d732b7fede82c83227..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 18616 zcmbTd^-~<*6D~X~?jgaQV8LAj0X_tm1Ydk1xVy{Z3GPmS;IP2r4oh%%cMl#Qcz~Pl zz5l>lZ`GVRHB&V|boY7A^z(F|Z=Y4=aIwg-006*MkpHOuZ?5<^0x;12-SsK9!v0Mt zmQpHG08kT${nrHb-!rC@ysj$%ki7ceKq56ESOEZeJ%x`_nqEey{^(v>eK${gL>pJ% zX8+KBAR_W-jhDrs{egi|sP<73DP`UFoa(>xj;8qknEx2bL~2@t%3k>}hnl@CWQrW@ zqfK>@e3$sL-m%ftg0YAkk!@=P!Ognuz(zhb|Tux{FeX<<7(5oLVU8=W*sUZ*$TqlSb6o1O0a zzeP#ZW!;?#>0N5v?0D|q?mzD8-<^@1V0FH{fY}2A9ooXbylcB6Y>PVo4nMxLi|AWA z8M(b#9`j|%0v7ktATOSzsh-T7%Wqa>t*x!29M*iDetE6#^`?iEoQW5F*w7rjcWYw>-UyKyDHetK@Im)qdu0o-zudq@gQN3)r z=(%XIh|%7(Y}2mODA6--)=u;7mi|lUCki50L@QOyZN@2N`Bwwn9et)BF?yQr9`Sn# ze!a;09%cuNiCJ+Hwx|5Sw&L`0rJvq<$7D5j#Y=O^YcW)1x!+MVRWRVHrXDj~g@40Q zBvp_niE6-dasJKX&t@%;X`7_R9QhT$w_Dv~zW73kCM;9WC z#^@^R#^^HZ#`rQ5ZjC*^uYUMgw=ae5*IV2JyEL@LlJ1k!yA8p=fmyQ={`Pjq&sK}Y>k9r>*Y-3njDRLc8z*D?su--n+y(fpV8FB zwS%vLw=L>F9>rMJzXaXgg5NRvaHPKO=qdV`%ecKE^q=CNs6^=Vl)5QG9h0>AKM-1F zvU-S)!Vnz~yg}XNmnaKSqm&}<1}#nOBCWZsLvn3_pkm8Z)~*KF8yv=yRk*!4rf$7T zT*ey^g`%>`O82HoVNPMCaM^5e_Eeop`^`Wsro=Q9SzJ-{LW5j1QdRH>Oq5bEX({TJ-TNGPvNBrk5{my=8FEQ%0fftv4 z)$FK)-usf%cyd|Y@=r@u!~HI3-5_Q=E%R!AkEqtv$Yv%Zit4K`i*n5tM!wdwLFM?% z@N0D&tLS9%TD>`41R~`%HzXtZS6pjo$}fsAA6cq`&Llq^TE@#ID4eU}(xZH$-0oa>g$RMe)N_S(=w@nXEL&?{|e zd%-=H@Ei^9kz3up?3!?QYr2O7^M9)q_E2E@^vESGQ&5WzDh<(QgQEd3BICrRm8O)S!fPO#z(h0}Vk) zolMw(Ecl!UD7xMUH0>?+9qzTMCMQxcM+Od*!L7F!tiwSSG>D@|J~*c~gu?`RewztA z1cO8*h9GGR{``zPp9t6vZJ81Ar<-bz38Jv-ro`wI#Mq&-k$*5tL<>Pk=)T1H_z8YhPJDWCuq5c#f&iDRo3$~XHhc-#T3{whJvB?;N^IKpX^H#=oYNa@u&^9He20t za7qlYKRH^S(Tj2{XC=lPI|MVMOVVX4V8cbx(9Ix%YK__iyN9E(k)118*aO-OzZNT# zbhE^f=Cze>bdhX>8xBFW70+=Tb@QnIyKKmQGt`}ZHXrVVWgxIT1k&eFDonM5iFh{^ z;FtT_qYo%x6$`ChDD~;i`c>h@T~X~pZ&-v==wrV4)ra@?=39Z}7c)OR&&9#@9uxU( z?hh)jyY_o}tH;1B>v%95XoGM@gDYB{I@;aJAn;N$2z~uDX|IL`uf-*Mm1ic21|E8c zQZWw`gvb==bz|iv=774j$zii$vlW@T4LDFEfea$Z+frqVA{<)qP_mhp2AbFqEE(0z zfCJgi{n&vKxpSY#-W)(E-Y3u@1KQGcnWN=qz;Nz2-6>bIL8wZk?oy8xe49zo9Evpm zI>QVA&&4C5*aCjxksX%9lfPpQNw|#TzMQ;YvC%Rx=uA#dmU{e@tzaW&rq}9N5VXBw z6Mff^1He^5U}j4TZD};Z7u2!LZ@OjGIPgR|MLZ*9%)E@0nE%K=W5s+NOT~n_{fBc9 z8DlU6un9om`MN~!FtpPXkJSq(+KPHqF&N23_vGeqphc*cEAF=okHGoFWHHWTm&R zAZXR)=q}Jv`jsvKCoL27h?ylNq0fz5xasR{P`5RW_7kzL^b_#T@e?r5nGKuMX?!lz zcEq|hYJscWj{YtO1of8Xi0jH z6s+!rS0;ag(Cml~|NKB+tNwwq9kl+8wc0!T$L$CFw95drNPiuZ3jOf4G_NXoM$sQj zZn*2v3^ISC(OoqO%W>m};%SHDOcD)D7%f&?jnrI9&1_u;6m(x2g#=wb zH$Cl!I6f#QI6iFo2i^nPy^8_Rt0g@Gzv3FoK629)r#wPie#!P^T*B)9JDi>Qta-Ee zyLS}t0#vL+3WcNfUo47o=g+h7Q(waq$0Fo`#^t+!ugP{n=lV`j6a9^vBl)I!L&VaI zK(10FWw?KM*=_ynJ3HIwyD^##=aKUk4u|yIYk$&C>^B?x{I5c+Il`m3RQ%_=Tq`!D zQw3HQ7dw%VR~rkqeqr+THi``YT){njI8j~%3VNWBl3EUyQ zx>y&BaDTkwjg$12&1?kD`IcCB_?j~8XMfHm4iQ(TCj7-)DOn-+%UzP)ab?nnNlfTA zh(FmGsK1tl`G8>eb=1j~9lDZPh<*?zhjW@Gx5%UjcH4 zbrrd<#%%JyFrW`_Loz= zP30^V%kIB;=&%K@{YbXT6@(|c>dXlNk~?15SVEmMX6`Mjv>+MN2M$^N?ju|1T-qoW zJQV;x5rIpTc>eCM*`;fq^U3U2uW>l1RVxe^4B$CEub2J}+bN)$=(gE92((ah@ar_) z+I|k<9;iL6@Dyhc+LX|pTR>r3{P!==s^guY!a#cZ5Ry6QtTzvk zUh~+ICB=TnC(!+~G1}X`=zKbJF=VNy60Le=gO@j5lEJet5>jc!PbM+D!ZlS$KuYx&pkm{S?k)BU1<65@ z({=ySGqzCiV-vc5qOJ z48y)rR(Ys{uWIjyQX*o`4?xK$K9nE1K!t$coI~(ku$IzWaVM`ocnY1)=&_o_R%I_2 zZ_{Cs>@7#7ktZS)0EENs++_HHh39c*#7z#Pyifk3+e!lsET`nm%a#Zp{hflp4Vw$+ zOju*)#0tN99xzE1;G}_c;Oj@<_%Z8;SCB3P74uOYE__wpp<3HB0g0wsxZ1toEwg)5 z23F}NQwRV%3UQi)GQQt^$a%zzV8w>aIl;CkQ!6h%=n!jXPZ;sfULBWNTi1QT%V~R| zdrjBQt+%&EcrjOO0&pO(SR|R1%nis?Q}KUl75Q=`bI5TGenEMls+QNXGp;Grr-EZVy`f(ovFSmI(u6D90n zU}rWOG+9F)ioe9yO)lx~AD<~|_xP=uVs4I z6w+kccIU+(Ltf0bDM$mvJrBdPzjnQ4w#L-qTZ+S6V5l=pqj|%(!m@K!R(Sm5G<;5V zXK~r#d34;M-;>*+VXbyWbw`4vdOanA^uK`Ag&w)G;7}_OpATxWe^GjFe%&*Ocx)w7 zwt4Bs4luF3C-9V+n~E!?(W3d6$CtEn7OZ{~I`6iW|1x;QzkF49GF&d=Wg#fC2^Vn?KLfW@n~pFc4gBpg!U$uFR0 z6`f||PCJat3glNlwW|z^j;^p%9oQc82S&N+!L>xWR*UT~JbFCj)0}2J6c-rV3iVO! z`IdFp zB0H{SvHRu;zx(EM(0%j9fA`HVZ|@5Oo0EGok@w*1K*{Sg3QERYynQ|7kzI{t_?~>T zQGQ|?TPR(EZYAFen;>d7>k zc`O4jwao>J?dp~fG@8l|SBHzOE5h7?Ba_OYs%93|;KP${8}j%VGb?LRi<;yffk06& zmc)TH`g@-+zt@fG!z|MO3057>Y}ppB{w8IS2o68)NnHSA-jKa+X$k+&Klw{5Ksly#ye_HBKV&h1zbIsIT-|0XRq)zWf_~s9{=n3BOfpPy7{f5RZzL^9tdzjj zr)R?-SV}4UX;&dWNKq={6q|g;FEbIjXC}?$K%uY_ur_MF+MkJ>-c@8l1|6F7^BR4N zf%t(1oJ!m zg^z<^ddW{6+A~!=F*1he)s`5=HR&3O@tjq)pn!{ zodn}X=d$=iUh-ibxQ>PQw|#fHTLppRwXG}*HyUkLKB?Vxf>#@2_z&V#B0Cjvmfka$ znI~k?Pp)A)OXy(kdOeH7nbmp9bNb|>|e%T7Dg>BKo&y=JzU)v zs{+P#O$)wko3MOQY!bv_78@Q%uABK!ZPIi<~iCxyQ>J*D53j_;0vks;+?UxqO^ z8)9k;>&t3F)oFofc_t(0cdCn(OIM;4fePgKSw+PKcigoQR9JV_C-y`&%By+|aMjTd z;$iN6>#`KNXtG+yNhfl+PYn(#cr;Nf>DZ1mRU`A-PFI}Scq~0EgRR31c4LZcz_w!3 zU&-x*oGPQoz`-m#bYEC;V<7tHiC(wn395M}YNU9p|6@2$$6(9N_DyMjuOwT6X&Cu> zXg1{_^+%NsBhDf;)3V~J5%bl|^XVjqRgu^moR2288%NOgcLoNBkN6t5F&l2`tPvao zfAbQy!&*Ln*uWc{tVDqwT1{Q>{s19S6+;c@2e$2eZd>zL~I~M}G^8w4Y2bnyq)>=S+L6j%|@%XWqbYm%+}R z%Jg=|X7Y&0*lujN6>tzy)?{CBuT|FT#I=sU+569+)8oyIH?8?{Y{Im(PMHAGs5_GI z>1wLl+yiE$+I28-c2!jx)_?k2nIm}7iH=O{X#yL$s@}hUPf^xece9Vi{DUPRKm%@= zI4q=C$Qla?I0{;1W!^-Bt)o=r>#KNZnZPW3piq_&q`~HLF~1_^MHlt66*62}BJqzu zM;g!LlycVJ?1ohPMvFHu3^-`<`sR(iyLG`EB|;bk%3GG!#?x`m5gx zWnZm7bb@UTrR9OXVs1t)?(5a%Yqq>?ivrob2S7W|CH$C|Kscw z=5hgFRsHTTA{lDQ(a0VW8vk$By+wL4Ao<5{Br)oU$x2pMfJKrlPqr@4P$Y9Nt_7R| zCx>hhMeHtjM0mJ|?T<(EIY{^^cAiA&R=2C=g&o@6vm!E&&86BrLOf18fr==x77OBH zdyOvB1fjqxDMa5;G9@=qu?tN_vB?)=#H^qB;g*jHrr^*ISGt+pLXyWcu+bAWNk&IG zl?zGxV&+)tmQ@d~T5Yypa4*^P5t*t6C($W-Y9zknsGLXPPDR^RF~`>QcV4iB%ltJg#%JgzSOl!L!d<7;Gfa5FAv zjVdBTD(TpZ3>zF8@VbIAM{aYtDv8fh>oAmOoV`*>G_abe#aOPM+6b%!IzPP2K{>A5U*>>2+^+79)a z;+jQ03qhGCNA7Yx7^lX9Ba9FuFHNen`s{buqNeEv)$x#QoePK6M~soRL17NVafu`4RB%F$`Pl z5~X9X{(zDkw(=x-=6pOllhfSrJCozywriAokKZ^VZ?epc?F2YfOmC=V98gW?oL=*# zC!4VJtdyAXwE6cHlNoijVy3KiZxeTrjL5AO4?|IT4#6gV63bUTC!(fd*MK@3^J@F! zOg&Y}^l`KyT>$RnH8O17_%?_PVh?o(+5L|_R7c|c+R_PRXb26L8QM&z+5MaH{wtOk zn}L=^TXs*WwrBLOJ6hDKim{LKAa3?WEiRefh;#TMZ3y1zA%QAUYh={Ux!GU!o~ zQNH$+pUp$BPoB27%q zF^6BflF{;t=SZSz+GrMJ3q~ti7gQ;5SbjS`5!DFxQB8KOt1OQ(G%_V;vcdj>K_dXjNxb}0M?HyjDs(afDCVx%>+I2GAO;jMfy0Iwh$=Utfm z5snMAm4|C3O1?MDEQ%I@RL1I{SrN67(Q)b*7k&Ip+-THJr%-;ILx=v!SaW75@EH3` zUhVOn4CYZ>iZ!iaGNBq9Be`Mcq5Opf?{HZfcJM-VDr$qSCy^3Lij|O&UW{&ffZ&!( zaA9$H9_5lFs;vRx6|mmn{Ic~u%y*(_t~*m12^>%iUOQ9Ap<@`U;!iRpBZ5y=p}@B6 zSP;R6QS{hs7)q75Mgj7814d~Bae=<{A1Z5>;LN66N?m?;5pl?`*_wW1l4a8IBb4tyR6@^@^BOm`{tD6YyAv};)Te2G+K}4;<~T9 ztiHbWTlGjD1=omQ_viT9PJOR7GjZ^{`7u?a_$hGpx54G9Z4Uj-NJ+>3SA0ZSx1vXw zLxYWusP2Sm*#o~_#B)vb&lTfmtsonTnPHIvx!#}HYvp=bPcZe zcHOCWuo0{MxR+#P#Pz1PSlaT$g-HbB!hTlHpV_F!Ay^U-vb1-6W)!xh?3imeOv*Z3 z=D=Ij-4e>!J=_Q#nqT5Fkomgv(@3uQo!?=8R9Sw(0)&ni z2jsV8*xm^OAO91C)$^*!X=%ZHvh_G35URQ9mZ|{A0)E?gJcL0T$H-NA92s6VF$CYW z9RHBse3R!V%B}9#+)P1_9L@j@2VcH-GZ=N2{$k05r?kj$KxpvthW zd7m|F4Ka%sEOHJC`oN z{Q9h2$S$VYkMHBEw7ybMx&7`nIaMLI5n~s)u5f7_tg^|2p4eFF&|6C45|-}T zY2bbCicJ7u0b>nvzMSvbBTOChoOAKvC$b5)Y}lT;{a-@oZBJ!oQNfsC36M4qtjvVR zX;Qkn$Pw56!sOMyw2f6>a4-#^ zy$1D*lt}-KofQ^atUig?;uYP;un=4nq7RPpS6+7^7eT`a+9Hs&(5Wu`IyLv0kJINP zH{2$kHb`Me^3C!975F7KG!qcJ%Ot-tp1f*bJffu1KR9B1lQ=XYBq15?hlJ33*QN-~ z25i$#OI}x{k+-P3EKo3v2XVk4?t;KE4nj1dk!Zo@w6D?!o#k^~T|3?;an*{_dc}rZ zWWWrKbdBu0k$7Zn5A%~0$lei$vU1P?CE&!L*!t%`ziuxu= z$+Xt=qUvFYn;a&JSK-D!mWnDWtF|5q!R|hT$Hv!*O-Hv$ zFMd5*W#~$3AJN-2|IVd@2bWN6TIfD_0uz(~vS50vn&4k2seimRF5`Q+1IS}!NNHN| zuWuQz50#5kO>f(wTSg+{VKXLrOZR$Gm~DhS1f%%-9{FGG$s*ZrqKZL|g5VaRU11N3WB;tGWJx5jj1rPZ1}$YE7~gsu zE25FmauDeN0tjmI!T8LA_@Jktp-r4gQRI3~pz@ext*^u56U%RNNACtB2^N&i&Zkq_ z`%gV|mr`$f?Rog-De|tRlA$9w&gIG-7Zqk}`K~S#ez0!r0TA4$*?1vW^S1eRHim+x~x!Fuo?ZZGGykdj`C(v!pIX!M7^#v%t*g zcznI+6jSi4g8knZOJ2XD^*-Nu8++1xNL67@Dpa}id>w3=oC<2l|TauHqSGbyr z9Lb=M3fe$ymZM2IcIy2$WhWPLfA8YEy!~$2XHICgk})!EbwTa@re-=DC1|8#7fNFq6gJ2K}GKAX`f_@q32jY5x4yTSxUH;`}j*L?c8b@JA9D(4X1n>r5 zmjA{5zUzqX9?77@2f4TGSC#Gv z>RXD%m8Sx#GLz`?10nyLA3f`rKtm)2mp8 z2WUMD#ZK*6rx@tHUO&Z&$15&*p$9S&RarVs7nI?jWCTx!i z0n`(39&^Y>ScN)8+_K-B#JBi}jEM2qqgbCqWKx*4*ll_rs)9n)b|4=f&23 zGJ5Ub{5j_`P?1;gHXtz{3VvNPjI4v63M z7VR-O|JQRM-E&ZagmZ6Y#+`oTU{Zdpg*T>rA?e2lXyimlx-MsB_vpS!^2jDQhm%@q z{n8XwoaYQc8y7Itb%2)$a=$~0tev`)%-s+AXZ8I@XV4DuPx#4Z3^R?1Q&1e*!{+@j zwy0-{m|^s)xqlSU>jQk{owo@5+inF)-p_24DlAw`pUe~G8ATB<-h>G97|FK_kfkQlN-!Xir7CB=dF)cJj`)++W>CeZ z0KpG5Ul%&-7q_N%mRtvtM37+jS>A#7p`RadxDFCIFsAEA)28 zRc#)^^3Z1>`W_P8_n+_5l5pGfayTk_=7^k}d#ir!c>8mR4k$J+> z7$;sN^3k#e1A<-CaO6F6V7^1u(puc4hVnfPK2u$wSE_XF>^Bp?OAv{2Y8)b{(a(2LFQfe!w)T1x>k{ZpuhTF(Y6rhpZbrH!ElxM! z5seXw{2(-vFEyNn8P2QzldxYgR;$=9Va+n>oR-HQXL;u7|E|m|OuX!t) z=Y4P{a-kdSJHXaCvpi=8=DW$Bomevgq&Ys4T71MX_~k_QpcOJ7j|>5e z8fKax8KCNY#00?1+;-F_`mYl6?wiA0M9-%AWH7g{~~uALu>r1q7;w|*!aJIeE{mR8WtR@KBhs8TcC2jA=CW|Xy-ycIi>d)c7Okmo?_;IS6kWJ z(`FLRj~hxiQw>hGi`}`RB+q+jpRWZ9z114q7dyj#>yMG?n=NfcSz}CGOi5Bt#D4u( zFREX`PCs3=cqxne=H=$udT;=|-YI7ij;hPlH)3oXm z`Zikh-OIS^*V9YKw;%r4iW?YA#ppM%LKP=jnMYQ)JEBqy1t4U@E<8VwMW2U*KvaS5 zNDwVyHjTg6hvcbS>{N7lJu=~^Ut)S#sq~v9%#hIV2H~>o^9=!kEGypac0E4e6TQIW zr~+Bn`Sb4k*0*Zts;f;Vq@fsZn1hLBQyIO8W(13u0211vHK)RMC5neH4xx7?6jMVOl3i-ENH1NU{ z-FW1hXwfmWi;TOg`k_dSL1ckNlukjE5IiKg=2DaEcWG#qTCd+ts`vavz;Wye>fPE6 zy5Y~H#6~R#r29XgZcKEUWF`#TkPjT0Tb$nr`$rM*rO!0=z{AwY-%*%Y>1iy07;xo= zlqRRR7Oc25bnNStf}IG@3`}b^k0oTD!zg(19YJjRnXs}9jracK>Fw6_hgpNk9M$d_ zY;%@p@*94vn6~^S;rS|c_SBN9%41Y5CNDz~xgJ>zs5bOlC^*0Hm`3d+UdEAQlhAJ~ z9rS!JpiEjf-g5TxWc*_}=Uu;kRBG#hg)R{HVt_KfnWZwXW)vK%qN^F`Uk1yRWlJX^%Xv zrk4pFBKoY0c4V8}-7;k5jeHn#no6bE=CpUiQ*YjAXr&^e4Ji=kd5l#`F`6lq$7V{v z3HxGM@4$C!_rCJ0-}}J#b+>i@#M5T@ zDq!my3QKfc?}%tQt*O2KZN233YvPN6nJ}^KNmAv>Z%4u&!~ecZRVXA}Vl6Juc1QC% z^+u0V1RbM%wwc6J;|v%G|8k{t}#XaV3b2aS>;{E0?a{QN?D zjap1}Foj*+4gOfLe03+j+-fGX6EVmh%q%{kCs18^=Y$ttM`Ru~Sih(@mxvo*(|OHJwq(zE2(ex%#gkzo*Y14gL&0 zb&R`Soa5K^wB%jo6cc>zQGL@J1IWOVy&G6nrZ5tClv8t|5cv^+Gb2^+T0kC3kdVb= zzt>d9Y8%qhJjVP{A;^*2E;@stxE=CCM8#hlN3jEzVQ}z~l*fFX-3jF?-%dnrKMp>* z+*ojsjy{>@Jvb5ZmHokSc4fmUNZRBEvkDd^(WV&AoGicLZM&xx+F?MzT8H=FtNK9| zS}XSejv}P(R*P5=IL)L^{d8bx{SC>9DDxXj4@z-n^Hya-p}k%LC>kvh2A}eK-{n8P z{ymeI^r5$}WuJ`hTT7y&m(wGugFoqC45jML$-|3L7JDo`mbG@4AeOa9^F5Xfc~AdJ z6z*HExRMYeE;qZsGE(eCPFCa$fMk$Uzn)5Lqpt$(K3(+J)whl&sJ0{&+hDO7rV zmH=Vx#~{t)BZI;GL9NP4eoCJAPi}V8s2_pM0^Qn!dLjeT+!j52$p%MSaS9-1=VIXE zZZI?CV3-Z~UNNk|?P_bEXiaFvcS$(=j(imNA_Txz*qk*3Zt> zNTsgN3vU6G(NEuWibkSSE-gZ&wr@}`tuvHEIJGFQY)vT7_Sn%Zf>;noCdR{II*9Uy zi1DPT!QZt9edc?XCO_%vF)Vha6tK-jiPV+wdZr2-8Z+moIE4fA9Um2wrmprd`ujDw zA4$!<#8*6C%(UP!wX!r@9XeCS{UX~rhBT6- z&m5@`REID~K)qRRLN40)>Fz=?P=C-jXZA1}lMo#Lic@|(zYtC?Sr$}gjz;wX-)dH; z>kQvsjFQ|FEvL5r4GE`Vi>HJ+qxMkQH`jx)M#C81t{fBmVaUEu2p_>}$^Lp*OiKYZg_C_ycw2+?0OT`)la$oyQwx zn_edD@HInp4-Gny;i{I~SnCp_RpFSS_!Eo_CI3DYHotlBCu`)~d17BV58M;K#oqAY zMpX+Xw9;xj#wpOozs(lT<+Th^5&14m(|Q*%;z`vKh4SNgAVBe}N~g2sLPrFC2|fE< zFpnnM-xp>{8@7DssTYKd@0S%KXilVkqrjiHGyiM<4X=4ToUoPe$O?bRyn$W!y*w+D z6&Dp2t9Ct*jrJO53Vv$UzniUP=-;pr=_NhmXKlFLRkmbSfW7QwHhvWb87Y|_ zx8ovSSXKm9h{zGnW$Hh-iI?ZMHSbjn*3Sh{-$#hX$;rQovTb9bL)q_$Wc zZmKiDhCM5p5vXSn($(MVPz`Tl^8Dq9O!MXzxdIh}Yi;I?zh>o(TXxwNlF}fbbJWC- z#GcWxTx796z)2UUjk&XWZFb3^oh-r)7Kkx{urkexT2D1!HLjPN~zvz2X#hz4#kSWLV*CW#DJu#do;exLU5E*Yb2H*HhXE&}5w)`L0O>xl{F?nRCT2 z*sv_q70&aZdR}eGSdA;#MccWyIlME%-v<$!Uv*^qnA&%(krwShZthK$iyit6H#l;> zK-^@!-w;mtEMfj7rnxx}?MKV=JHn^z-cHiGPN(d-mV0j(9hnwwg#l4%su_AWn&D=e zjR-cx9)55a@TwJcUi!8R@A2vD&T99g^diZcn-!n?8)u3269>8(cQRcMciiUGO^eip z5B)0E8kXbcz#sx*&|^TUl$Lb)lb&Ip>#TdtDfUcwzE~nzmuQ7EmTjAgdgUiGuSuNa zpCb6rE6(O5o(^pW-+RuE)g@nrZK=PFeQcL58r8o>9J$FQ<9+2A1d*DBdQ!b*dT;;4 z$Xo4EWN=S2^E$tAy9hSL=6Vn#bHD2g;0=sNhjJ6d)KUocZ)+A6o6_A*qTK}$*h#RS zyk#XkuOO@^1ht8v-%9N{Y9oewzu$e7L(scb^mXW2_TiW*-y)vNyH`OadIrI^Y>*Zd zp?=ROXFoq0Kk^tpwCFt$B)QKsZPM$&nJ*fs2;Xd)FtPd@FMUTnfVUp;sJHFaw;TuBTKR%BOW_}ClL_Bhz{A0l{Qgc%@tjIWj2ys8T z-56z(;=%E*LE!6!#2)6$>Eq4>1p;7`)Z_NSc1X=l%@0`gB7usIOR#p2{Cap%H#@u+ z`w+GL;VMer0DCjGMC|TGF_;&EgwZvSq=Q8@4}X7rF+n51h%CM@hl5WX$J z1a?I~km{+qh|RA-3+BNxgHjmg>KA!Bo!rA$QbB?cckI}KdkcLRox3JZd`fkXjx#A+ z_&En<1xc&Qmnoz0c*OV_guW?$J#uUHP(jS@beks0sZ#) z21ebzv6U?Wp@^S4Wn-$u_zmK3cE*C1Mlc5xAi|J_lu9>vY@H z+=VfBpk=&5g2V=pY;m2PHSN1`4hDAzs43VInEYm~-~S`AxRI%f?TU84wXtx z=s<1xk#OUIW)~ZG_2?E}ncAz?RlZ%Nu{wqJtc71aL~G>$Y^@Cl^I zh)|w&6EwGxERMm32{6|adN{lmCnO=?!|jUP3Ws1;e!SWGzjeq)Lvs!ZTTq&ie5vo- z`1p%Yqwt8KsRfc+Zbj`#L-1}(Bwi~Ax5qO&ZU@{ejQ+Hp4mt4VPoV_VeCr(6zF z9UR1ae&+2iX+s6E2V}Lxc6ZM+-8S6$a@?&Cn^C~=sPX~d#JLm;5Qw1n%IW*&PBV?q z09O(5{}gEc5xG_jOowcjF=x4y(&YamY5r}Y`?S#80Bh&J&-}>XgL{roRVEZo{x*i~ ziq&;TCj2%^Ju@%&4lTnyhe)5-5PDrQb*+9kAHW!EOaiu61g8cl_=CS1bA@HjhP}H5 zEBJUSKy2WF;ua_T{{-d-8TdvHidCA`BXq&j4cFtL z^yXVy20#nD1@%y@Y5U4sF1MvXa8K;F7B|Z;gH>tspveGY5S|}@U_A#|Imi?6GS1f%=ROP|BEkV#WqVG3b_;n2 z;H#;^adfh%ovD>w5Gs4>tI$7iJW3x%2mWus`fl%IFZf2qhN?JgWZYM_WBdsAyZ9Ln zRkEUt($@b`?c4fgl`7mn2lzu)}t zF)QPs=rMRr?Dp9+=yMv@`)?NKswHtVMS+34S>A@W)D9NFirDEhF)P8UhG0LzO-*O0 zw~iYtAHX;-bhAs~r#R<26~a<=Te-BB1z_}yavF7s_X>@Au~8kI-fv?*ch&2-MEDeRpn$| zQs#J6{sP}E#c@zKLH{=n*1NNgxp^;34)cyq+y$_nMaXHdPefdQB&ZYuaBF&F+#jI) z5iI(HZ*=0~V#^Xg^oqt{LGBS3`Mzzz-b6=qrl1#6B|u? z)MRjg9LIM9!?@uFajP;=#Ssg@2~wUs91pUhTWF1+X;!z;#!7zZ!HA3(S&VVh0-H-7)D5Ez?jhb5*13LRK%!y+ z0JbakM=Tfr@d$}P-7SM{#QqrU2pOeg#laPR_u*ECoxGxwD+5qp7mJFAC4KD`kx<@y z!H-TwF(`nXfja!2zxynS|Kfw?Nv{=+iYwx~iR_4 zsDFPJT72Tn&;L~mWIpqIHR?q6{H5=03xogjIQ00LT=Sm?Yu??dTo^X%GTU3y3 z5U%wt^lQ~lI;@oqpCR=JSG?o&&sGC)JkTBL$iPQn)gVhj=u1Ww=)nAbnfA|CTF1W} zHDFT%X57(fTIQ+HQ=ZLM-4b?z)=H^8gSHr jqXrx`;HZHtT?79Qd=?ufS>7*000000NkvXXu0mjfyH5ns diff --git a/stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/st_logo_2020.png b/stm32/system/Drivers/STM32F4xx_HAL_Driver/_htmresc/st_logo_2020.png new file mode 100644 index 0000000000000000000000000000000000000000..d6cebb5ac70e0594cbe37a6b60036a80ef3bed37 GIT binary patch literal 7520 zcmcI|WmKC@*KTl!K!Z~t6qn*&DDF_CK%q#B6QmH_DHL}I6b%;K-J$eBN|2xh0+bea zmtyV5bG|?CpZBcu=iF<}z4q*xz1LhL*PcBwx;m;Pgmi=e0DweYO-UaBz)*UWZ}4#+ zCC-V!SC16}H#HLv0Dy?%--0o{5_}H;Jf%=ql7H=+dziOk_)KzUSaiQ9L<^g!49k}} z?4y$V zrsn3Ul^TY`00G_J_8;F7Sr5c3Y)f-yOYl{fS0avfKJg7R%r!y-ohcBkr6XBZBkE)( z_t+tVV2}G1_CN!)yWes*wa-AGwk31N_T1`)4COlfz+o(9S90e?6jHV4)!>`j+oRqp z)gb?rtSdw<#&c?q!OKB+Ncn!kr5JR2EYan8HAPXBw*Oh-F(6GmaC!`$p7fl!_Cdu> z5@PUMR_K5&jgevDepWqepZVdMHR$q>ga=7k0ltW$HHVn>HRa!#(+BW_jN$5rx^MtR zzWT7tNWQeFzEp+86jOYI=I)*GZYEiXlf-Mw%tJ^ptL%2)%#PmUjxC4wx@%KxQ)8kO?|7E1 zd@5gd9_$*6nx?fj*iF_EJT*PYLFP}d8*Fw>cu3@&nm%V_C}V7HMmg|Nl#1J1^#)uz zZag|X>{6j-myvL}3(H8o3resNBq_jcuJuyT!QuXnNN&zG^tG? z>y*dy1#XfwyCE#UiLT(~LYAVwp)epErJLsHBA|l0xo)kxLAPob+7tFlr{7x8(#v|O zknK}QC6|~=cTJ0;x9MWMKua;LdTKIHX>35a@7{5y0MQ|zqsP64mM~`gJ&&R{pSUBA zf3Y>k>d|xdzBFztjtO~{y%@LUdwSgzEj=69hH3-NNWx7<|49%2%lvTEeE3_|$~iDd zlktnxf|NB>Bui)yY;%TXgWA;rhODxR!-4GzxKd473I)3;j zS4-Vun<^vqZ4)HbwVzq%5%)S!>Q(Y&PkF ze1ab_oy?_$PZT-mKJ3K^;6MrB*Wc$lkYJ_460AFYEq{AjlDxm;5GRwOsm>doJy3Z;a$FSu-q`X)Hw&o~?_`Dnpx4dOj6#)bGOdZbj4hD`l|&#v~K+WrGO zpLCVQg=*h=4H1sK)D*Pxtb#p`c+g-eK65k!CEZjLIrfm}5<4@KVqZ#8(S$vs{?Xv) zN(}CKeeLR%I)W!p(l-`utKmBbJk-On{)ciqUN5%?TS8uwKU(8}Mxhsb&qHL~DDcB@qi} z4z%efV#K69a#lShPt5}*7ME@^xunFR{k;|qp z`_Hgdj_Qle?%Ydlr3f^SdRkUncIj7qTiu85tA$v`#419Nlof8tjqUjO z!Rwyq$Krfh`(@MiE+)yie&jU^crprYV&?zN!2b33S2qtW+zB$nahDhVZiQ12bmcwYBcT8KS>v`jsuI@X zy^k^7>TM5Ndp@}pSl6vmkk*u`@C`z(gq||K2>y=Y#w1i`Nk2zAP7{_^ACq;bQmT<4 z;b~K3=?mhZX#~jQnZjdU={NFp^HRr~;^PXQ*UIHzkq}Xsn!56*ZCOOEt|^ zawUwGqs;zCEGKTz&(a(@h#0sCE+@`3Y<&}9!(;Pw&`C+d#D?`##&!@*ERaH0fksrh zfk2svh3!d~h{SC6BNKN2j6(lzC>S^_*eh{@gR zP$rV4$nqV1y|zYpd;nQr-`Vlp(5sQXx9AibC?xc7pUV)v2cWfHu3{KLbfP;;;`I&)mWqLcXt6s+5fps9R?K>hGR$5;=( zo#;zzQ5!w+(99p1_gh^?F#$tpMd+4%G|K`XG_#@A>30WD2L9!0a>wRASg`M?^z#)| zifb$d^KUD&3qLsr-@}r_7p${gdmRqhy819bj?yI-v({I?0o3_8d>CZmCzqH4{{QtD z|6dvgm<^~668G`6wLw4C4y-o9E9ZG3RiH^&;rrKSBT@bMR+!TlYo*9P;?Z>xORMV3ndQp9EyNn!!~j&x&$gDVke|t#!{QoegHm+sT7cSjwSu z)jZlx1pHrrx0&YCZWJ4xC&U%r_IwfJOxnYqYMcZn&_hgnKuls z>(Y$qjIhcweo^A_VVq_#UR&urF%UbI^><5L5zbV9owMQviM*_{@i|dke!!v8a_yG! zl%#ay{(6U_N3-)yeif3tx6>aQSf1r zO~OuNW<8a~bi6Xby0@tKw0@f1R!@+3S2nf#F&j~l0mrNCtjoZdPYob?g!Vx4F-uA2 z4uS>8eR?dA9i>C|cm-m5lO0m@$f(DB6Z<9Pe07xR`l4$ks3eZ@v5}1?^YNQy-tB(` zgUNZrGVf#b~`}jew;MQG1O~VDk_i*<)p9DfFfd;vLy4QZT zMYh|DxJg3-!{szUN*uo&`&DUkzq4qG2`Lj;Ar46DYUymcviS{Unhzmx z=LwZ-Lr{t1z?9Oip%!z{j+Z%_@u?C78I`V4fC`icG1c|2;`EoLK$ z)9|wrAV&V1Vgts~tQCw0X?X{74W"NJQ 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 b^}%0^qYmp2WD(pA|JtZ4>nPPKybJpu3@X?! literal 0 HcmV?d00001 diff --git a/stm32/system/Middlewares/OpenAMP/libmetal/.checkpatch.conf b/stm32/system/Middlewares/OpenAMP/libmetal/.checkpatch.conf index 96d956af..f147ba30 100644 --- a/stm32/system/Middlewares/OpenAMP/libmetal/.checkpatch.conf +++ b/stm32/system/Middlewares/OpenAMP/libmetal/.checkpatch.conf @@ -1,23 +1,23 @@ ---emacs ---no-tree ---summary-file ---show-types ---max-line-length=80 ---min-conf-desc-length=1 - ---ignore BRACES ---ignore PRINTK_WITHOUT_KERN_LEVEL ---ignore SPLIT_STRING ---ignore VOLATILE ---ignore CONFIG_EXPERIMENTAL ---ignore AVOID_EXTERNS ---ignore NETWORKING_BLOCK_COMMENT_STYLE ---ignore DATE_TIME ---ignore MINMAX ---ignore CONST_STRUCT ---ignore FILE_PATH_CHANGES ---ignore BIT_MACRO ---ignore PREFER_KERNEL_TYPES ---ignore NEW_TYPEDEFS ---ignore ARRAY_SIZE +--emacs +--no-tree +--summary-file +--show-types +--max-line-length=80 +--min-conf-desc-length=1 + +--ignore BRACES +--ignore PRINTK_WITHOUT_KERN_LEVEL +--ignore SPLIT_STRING +--ignore VOLATILE +--ignore CONFIG_EXPERIMENTAL +--ignore AVOID_EXTERNS +--ignore NETWORKING_BLOCK_COMMENT_STYLE +--ignore DATE_TIME +--ignore MINMAX +--ignore CONST_STRUCT +--ignore FILE_PATH_CHANGES +--ignore BIT_MACRO +--ignore PREFER_KERNEL_TYPES +--ignore NEW_TYPEDEFS +--ignore ARRAY_SIZE --ignore MACRO_ARG_REUSE \ No newline at end of file diff --git a/stm32/system/Middlewares/OpenAMP/libmetal/.gitlint b/stm32/system/Middlewares/OpenAMP/libmetal/.gitlint index 102cb75b..92d9e1b8 100644 --- a/stm32/system/Middlewares/OpenAMP/libmetal/.gitlint +++ b/stm32/system/Middlewares/OpenAMP/libmetal/.gitlint @@ -1,99 +1,99 @@ -# All these sections are optional, edit this file as you like. -[general] -# Ignore certain rules, you can reference them by their id or by their full name -ignore=title-trailing-punctuation, T3, title-max-length, T1, body-hard-tab, B1, B3 - -# verbosity should be a value between 1 and 3, the commandline -v flags take precedence over this -verbosity = 3 - -# By default gitlint will ignore merge commits. Set to 'false' to disable. -ignore-merge-commits=true - -# By default gitlint will ignore fixup commits. Set to 'false' to disable. -# ignore-fixup-commits=false - -# By default gitlint will ignore squash commits. Set to 'false' to disable. -# ignore-squash-commits=true - -# Ignore any data send to gitlint via stdin -# ignore-stdin=true - -# Enable debug mode (prints more output). Disabled by default. -debug=true - -# Enable community contributed rules -# See http://jorisroovers.github.io/gitlint/contrib_rules for details -# contrib=contrib-title-conventional-commits,CC1 - -# Set the extra-path where gitlint will search for user defined rules -# See http://jorisroovers.github.io/gitlint/user_defined_rules for details -extra-path=scripts/gitlint - -[title-max-length] -line-length=75 - -[body-min-line-count] -min-line-count=1 - -[body-max-line-count] -max-line-count=200 - -[title-must-not-contain-word] -# Comma-separated list of words that should not occur in the title. Matching is case -# insensitive. It's fine if the keyword occurs as part of a larger word (so "WIPING" -# will not cause a violation, but "WIP: my title" will. -words=wip - -# [title-match-regex] -# python like regex (https://docs.python.org/2/library/re.html) that the -# commit-msg title must be matched to. -# Note that the regex can contradict with other rules if not used correctly -# (e.g. title-must-not-contain-word). -# regex=^US[0-9]* - -[body-max-line-length] -# B1 = body-max-line-length -line-length=80 - -[body-min-length] -min-length=3 - -[body-is-missing] -# Whether to ignore this rule on merge commits (which typically only have a title) -# default = True -ignore-merge-commits=false - -# [body-changed-file-mention] -# List of files that need to be explicitly mentioned in the body when they are changed -# This is useful for when developers often erroneously edit certain files or git submodules. -# By specifying this rule, developers can only change the file when they explicitly reference -# it in the commit message. -# files=gitlint/rules.py,README.md - -# [author-valid-email] -# python like regex (https://docs.python.org/2/library/re.html) that the -# commit author email address should be matched to -# For example, use the following regex if you only want to allow email addresses from foo.com -# regex=[^@]+@foo.com - -# [ignore-by-title] -# Ignore certain rules for commits of which the title matches a regex -# E.g. Match commit titles that start with "Release" -# regex=^Release(.*) -# -# Ignore certain rules, you can reference them by their id or by their full name -# Use 'all' to ignore all rules -# ignore=T1,body-min-length - -# [ignore-by-body] -# Ignore certain rules for commits of which the body has a line that matches a regex -# E.g. Match bodies that have a line that that contain "release" -# regex=(.*)release(.*) -# -# Ignore certain rules, you can reference them by their id or by their full name -# Use 'all' to ignore all rules -# ignore=T1,body-min-length - -# [contrib-title-conventional-commits] -# Specify allowed commit types. For details see: https://www.conventionalcommits.org/ +# All these sections are optional, edit this file as you like. +[general] +# Ignore certain rules, you can reference them by their id or by their full name +ignore=title-trailing-punctuation, T3, title-max-length, T1, body-hard-tab, B1, B3 + +# verbosity should be a value between 1 and 3, the commandline -v flags take precedence over this +verbosity = 3 + +# By default gitlint will ignore merge commits. Set to 'false' to disable. +ignore-merge-commits=true + +# By default gitlint will ignore fixup commits. Set to 'false' to disable. +# ignore-fixup-commits=false + +# By default gitlint will ignore squash commits. Set to 'false' to disable. +# ignore-squash-commits=true + +# Ignore any data send to gitlint via stdin +# ignore-stdin=true + +# Enable debug mode (prints more output). Disabled by default. +debug=true + +# Enable community contributed rules +# See http://jorisroovers.github.io/gitlint/contrib_rules for details +# contrib=contrib-title-conventional-commits,CC1 + +# Set the extra-path where gitlint will search for user defined rules +# See http://jorisroovers.github.io/gitlint/user_defined_rules for details +extra-path=scripts/gitlint + +[title-max-length] +line-length=75 + +[body-min-line-count] +min-line-count=1 + +[body-max-line-count] +max-line-count=200 + +[title-must-not-contain-word] +# Comma-separated list of words that should not occur in the title. Matching is case +# insensitive. It's fine if the keyword occurs as part of a larger word (so "WIPING" +# will not cause a violation, but "WIP: my title" will. +words=wip + +# [title-match-regex] +# python like regex (https://docs.python.org/2/library/re.html) that the +# commit-msg title must be matched to. +# Note that the regex can contradict with other rules if not used correctly +# (e.g. title-must-not-contain-word). +# regex=^US[0-9]* + +[body-max-line-length] +# B1 = body-max-line-length +line-length=80 + +[body-min-length] +min-length=3 + +[body-is-missing] +# Whether to ignore this rule on merge commits (which typically only have a title) +# default = True +ignore-merge-commits=false + +# [body-changed-file-mention] +# List of files that need to be explicitly mentioned in the body when they are changed +# This is useful for when developers often erroneously edit certain files or git submodules. +# By specifying this rule, developers can only change the file when they explicitly reference +# it in the commit message. +# files=gitlint/rules.py,README.md + +# [author-valid-email] +# python like regex (https://docs.python.org/2/library/re.html) that the +# commit author email address should be matched to +# For example, use the following regex if you only want to allow email addresses from foo.com +# regex=[^@]+@foo.com + +# [ignore-by-title] +# Ignore certain rules for commits of which the title matches a regex +# E.g. Match commit titles that start with "Release" +# regex=^Release(.*) +# +# Ignore certain rules, you can reference them by their id or by their full name +# Use 'all' to ignore all rules +# ignore=T1,body-min-length + +# [ignore-by-body] +# Ignore certain rules for commits of which the body has a line that matches a regex +# E.g. Match bodies that have a line that that contain "release" +# regex=(.*)release(.*) +# +# Ignore certain rules, you can reference them by their id or by their full name +# Use 'all' to ignore all rules +# ignore=T1,body-min-length + +# [contrib-title-conventional-commits] +# Specify allowed commit types. For details see: https://www.conventionalcommits.org/ # types = bugfix,user-story,epic \ No newline at end of file diff --git a/stm32/system/Middlewares/OpenAMP/libmetal/VERSION b/stm32/system/Middlewares/OpenAMP/libmetal/VERSION index d2fe03a5..61df8f44 100644 --- a/stm32/system/Middlewares/OpenAMP/libmetal/VERSION +++ b/stm32/system/Middlewares/OpenAMP/libmetal/VERSION @@ -1,3 +1,3 @@ -VERSION_MAJOR = 1 -VERSION_MINOR = 1 -VERSION_PATCH = 0 +VERSION_MAJOR = 1 +VERSION_MINOR = 1 +VERSION_PATCH = 0 diff --git a/stm32/system/Middlewares/OpenAMP/libmetal/scripts/checkpatch.pl b/stm32/system/Middlewares/OpenAMP/libmetal/scripts/checkpatch.pl old mode 100755 new mode 100644 index 9190fa67..954c4626 --- a/stm32/system/Middlewares/OpenAMP/libmetal/scripts/checkpatch.pl +++ b/stm32/system/Middlewares/OpenAMP/libmetal/scripts/checkpatch.pl @@ -1,6494 +1,6494 @@ -#!/usr/bin/env perl -# (c) 2001, Dave Jones. (the file handling bit) -# (c) 2005, Joel Schopp (the ugly bit) -# (c) 2007,2008, Andy Whitcroft (new conditions, test suite) -# (c) 2008-2010 Andy Whitcroft -# Licensed under the terms of the GNU GPL License version 2 - -use strict; -use warnings; -use POSIX; -use File::Basename; -use Cwd 'abs_path'; -use Term::ANSIColor qw(:constants); - -my $P = $0; -my $D = dirname(abs_path($P)); - -my $V = '0.32'; - -use Getopt::Long qw(:config no_auto_abbrev); - -my $quiet = 0; -my $tree = 1; -my $chk_signoff = 1; -my $chk_patch = 1; -my $tst_only; -my $emacs = 0; -my $terse = 0; -my $showfile = 0; -my $file = 0; -my $git = 0; -my %git_commits = (); -my $check = 0; -my $check_orig = 0; -my $summary = 1; -my $mailback = 0; -my $summary_file = 0; -my $show_types = 0; -my $list_types = 0; -my $fix = 0; -my $fix_inplace = 0; -my $root; -my %debug; -my %camelcase = (); -my %use_type = (); -my @use = (); -my %ignore_type = (); -my @ignore = (); -my @exclude = (); -my $help = 0; -my $configuration_file = ".checkpatch.conf"; -my $max_line_length = 80; -my $ignore_perl_version = 0; -my $minimum_perl_version = 5.10.0; -my $min_conf_desc_length = 4; -my $spelling_file = "$D/spelling.txt"; -my $codespell = 0; -my $codespellfile = "/usr/share/codespell/dictionary.txt"; -my $conststructsfile = "$D/const_structs.checkpatch"; -my $typedefsfile = ""; -my $color = "auto"; -my $allow_c99_comments = 0; - -sub help { - my ($exitcode) = @_; - - print << "EOM"; -Usage: $P [OPTION]... [FILE]... -Version: $V - -Options: - -q, --quiet quiet - --no-tree run without a kernel tree - --no-signoff do not check for 'Signed-off-by' line - --patch treat FILE as patchfile (default) - --emacs emacs compile window format - --terse one line per report - --showfile emit diffed file position, not input file position - -g, --git treat FILE as a single commit or git revision range - single git commit with: - - ^ - ~n - multiple git commits with: - .. - ... - - - git merges are ignored - -f, --file treat FILE as regular source file - --subjective, --strict enable more subjective tests - --list-types list the possible message types - --types TYPE(,TYPE2...) show only these comma separated message types - --ignore TYPE(,TYPE2...) ignore various comma separated message types - --exclude DIR(,DIR22...) exclude directories - --show-types show the specific message type in the output - --max-line-length=n set the maximum line length, if exceeded, warn - --min-conf-desc-length=n set the min description length, if shorter, warn - --root=PATH PATH to the kernel tree root - --no-summary suppress the per-file summary - --mailback only produce a report in case of warnings/errors - --summary-file include the filename in summary - --debug KEY=[0|1] turn on/off debugging of KEY, where KEY is one of - 'values', 'possible', 'type', and 'attr' (default - is all off) - --test-only=WORD report only warnings/errors containing WORD - literally - --fix EXPERIMENTAL - may create horrible results - If correctable single-line errors exist, create - ".EXPERIMENTAL-checkpatch-fixes" - with potential errors corrected to the preferred - checkpatch style - --fix-inplace EXPERIMENTAL - may create horrible results - Is the same as --fix, but overwrites the input - file. It's your fault if there's no backup or git - --ignore-perl-version override checking of perl version. expect - runtime errors. - --codespell Use the codespell dictionary for spelling/typos - (default:/usr/share/codespell/dictionary.txt) - --codespellfile Use this codespell dictionary - --typedefsfile Read additional types from this file - --color[=WHEN] Use colors 'always', 'never', or only when output - is a terminal ('auto'). Default is 'auto'. - -h, --help, --version display this help and exit - -When FILE is - read standard input. -EOM - - exit($exitcode); -} - -sub uniq { - my %seen; - return grep { !$seen{$_}++ } @_; -} - -sub list_types { - my ($exitcode) = @_; - - my $count = 0; - - local $/ = undef; - - open(my $script, '<', abs_path($P)) or - die "$P: Can't read '$P' $!\n"; - - my $text = <$script>; - close($script); - - my @types = (); - # Also catch when type or level is passed through a variable - for ($text =~ /(?:(?:\bCHK|\bWARN|\bERROR|&\{\$msg_level})\s*\(|\$msg_type\s*=)\s*"([^"]+)"/g) { - push (@types, $_); - } - @types = sort(uniq(@types)); - print("#\tMessage type\n\n"); - foreach my $type (@types) { - print(++$count . "\t" . $type . "\n"); - } - - exit($exitcode); -} - -my $conf = which_conf($configuration_file); -if (-f $conf) { - my @conf_args; - open(my $conffile, '<', "$conf") - or warn "$P: Can't find a readable $configuration_file file $!\n"; - - while (<$conffile>) { - my $line = $_; - - $line =~ s/\s*\n?$//g; - $line =~ s/^\s*//g; - $line =~ s/\s+/ /g; - - next if ($line =~ m/^\s*#/); - next if ($line =~ m/^\s*$/); - - my @words = split(" ", $line); - foreach my $word (@words) { - last if ($word =~ m/^#/); - push (@conf_args, $word); - } - } - close($conffile); - unshift(@ARGV, @conf_args) if @conf_args; -} - -# Perl's Getopt::Long allows options to take optional arguments after a space. -# Prevent --color by itself from consuming other arguments -foreach (@ARGV) { - if ($_ eq "--color" || $_ eq "-color") { - $_ = "--color=$color"; - } -} - -GetOptions( - 'q|quiet+' => \$quiet, - 'tree!' => \$tree, - 'signoff!' => \$chk_signoff, - 'patch!' => \$chk_patch, - 'emacs!' => \$emacs, - 'terse!' => \$terse, - 'showfile!' => \$showfile, - 'f|file!' => \$file, - 'g|git!' => \$git, - 'subjective!' => \$check, - 'strict!' => \$check, - 'ignore=s' => \@ignore, - 'exclude=s' => \@exclude, - 'types=s' => \@use, - 'show-types!' => \$show_types, - 'list-types!' => \$list_types, - 'max-line-length=i' => \$max_line_length, - 'min-conf-desc-length=i' => \$min_conf_desc_length, - 'root=s' => \$root, - 'summary!' => \$summary, - 'mailback!' => \$mailback, - 'summary-file!' => \$summary_file, - 'fix!' => \$fix, - 'fix-inplace!' => \$fix_inplace, - 'ignore-perl-version!' => \$ignore_perl_version, - 'debug=s' => \%debug, - 'test-only=s' => \$tst_only, - 'codespell!' => \$codespell, - 'codespellfile=s' => \$codespellfile, - 'typedefsfile=s' => \$typedefsfile, - 'color=s' => \$color, - 'no-color' => \$color, #keep old behaviors of -nocolor - 'nocolor' => \$color, #keep old behaviors of -nocolor - 'h|help' => \$help, - 'version' => \$help -) or help(1); - -help(0) if ($help); - -list_types(0) if ($list_types); - -$fix = 1 if ($fix_inplace); -$check_orig = $check; - -my $exit = 0; - -if ($^V && $^V lt $minimum_perl_version) { - printf "$P: requires at least perl version %vd\n", $minimum_perl_version; - if (!$ignore_perl_version) { - exit(1); - } -} - -#if no filenames are given, push '-' to read patch from stdin -if ($#ARGV < 0) { - push(@ARGV, '-'); -} - -if ($color =~ /^[01]$/) { - $color = !$color; -} elsif ($color =~ /^always$/i) { - $color = 1; -} elsif ($color =~ /^never$/i) { - $color = 0; -} elsif ($color =~ /^auto$/i) { - $color = (-t STDOUT); -} else { - die "Invalid color mode: $color\n"; -} - -sub hash_save_array_words { - my ($hashRef, $arrayRef) = @_; - - my @array = split(/,/, join(',', @$arrayRef)); - foreach my $word (@array) { - $word =~ s/\s*\n?$//g; - $word =~ s/^\s*//g; - $word =~ s/\s+/ /g; - $word =~ tr/[a-z]/[A-Z]/; - - next if ($word =~ m/^\s*#/); - next if ($word =~ m/^\s*$/); - - $hashRef->{$word}++; - } -} - -sub hash_show_words { - my ($hashRef, $prefix) = @_; - - if (keys %$hashRef) { - print "\nNOTE: $prefix message types:"; - foreach my $word (sort keys %$hashRef) { - print " $word"; - } - print "\n"; - } -} - -hash_save_array_words(\%ignore_type, \@ignore); -hash_save_array_words(\%use_type, \@use); - -my $dbg_values = 0; -my $dbg_possible = 0; -my $dbg_type = 0; -my $dbg_attr = 0; -for my $key (keys %debug) { - ## no critic - eval "\${dbg_$key} = '$debug{$key}';"; - die "$@" if ($@); -} - -my $rpt_cleaners = 0; - -if ($terse) { - $emacs = 1; - $quiet++; -} - -if ($tree) { - if (defined $root) { - if (!top_of_kernel_tree($root)) { - die "$P: $root: --root does not point at a valid tree\n"; - } - } else { - if (top_of_kernel_tree('.')) { - $root = '.'; - } elsif ($0 =~ m@(.*)/scripts/[^/]*$@ && - top_of_kernel_tree($1)) { - $root = $1; - } - } - - if (!defined $root) { - print "Must be run from the top-level dir. of a kernel tree\n"; - exit(2); - } -} - -my $emitted_corrupt = 0; - -our $Ident = qr{ - [A-Za-z_][A-Za-z\d_]* - (?:\s*\#\#\s*[A-Za-z_][A-Za-z\d_]*)* - }x; -our $Storage = qr{extern|static|asmlinkage}; -our $Sparse = qr{ - __user| - __force| - __iomem| - __must_check| - __init_refok| - __kprobes| - __ref| - __rcu| - __private - }x; -our $InitAttributePrefix = qr{__(?:mem|cpu|dev|net_|)}; -our $InitAttributeData = qr{$InitAttributePrefix(?:initdata\b)}; -our $InitAttributeConst = qr{$InitAttributePrefix(?:initconst\b)}; -our $InitAttributeInit = qr{$InitAttributePrefix(?:init\b)}; -our $InitAttribute = qr{$InitAttributeData|$InitAttributeConst|$InitAttributeInit}; - -# Notes to $Attribute: -# We need \b after 'init' otherwise 'initconst' will cause a false positive in a check -our $Attribute = qr{ - const| - __percpu| - __nocast| - __safe| - __bitwise| - __packed__| - __packed2__| - __naked| - __maybe_unused| - __always_unused| - __noreturn| - __used| - __cold| - __pure| - __noclone| - __deprecated| - __read_mostly| - __kprobes| - $InitAttribute| - ____cacheline_aligned| - ____cacheline_aligned_in_smp| - ____cacheline_internodealigned_in_smp| - __weak| - __syscall| - __syscall_inline - }x; -our $Modifier; -our $Inline = qr{inline|__always_inline|noinline|__inline|__inline__}; -our $Member = qr{->$Ident|\.$Ident|\[[^]]*\]}; -our $Lval = qr{$Ident(?:$Member)*}; - -our $Int_type = qr{(?i)llu|ull|ll|lu|ul|l|u}; -our $Binary = qr{(?i)0b[01]+$Int_type?}; -our $Hex = qr{(?i)0x[0-9a-f]+$Int_type?}; -our $Int = qr{[0-9]+$Int_type?}; -our $Octal = qr{0[0-7]+$Int_type?}; -our $String = qr{"[X\t]*"}; -our $Float_hex = qr{(?i)0x[0-9a-f]+p-?[0-9]+[fl]?}; -our $Float_dec = qr{(?i)(?:[0-9]+\.[0-9]*|[0-9]*\.[0-9]+)(?:e-?[0-9]+)?[fl]?}; -our $Float_int = qr{(?i)[0-9]+e-?[0-9]+[fl]?}; -our $Float = qr{$Float_hex|$Float_dec|$Float_int}; -our $Constant = qr{$Float|$Binary|$Octal|$Hex|$Int}; -our $Assignment = qr{\*\=|/=|%=|\+=|-=|<<=|>>=|&=|\^=|\|=|=}; -our $Compare = qr{<=|>=|==|!=|<|(?}; -our $Arithmetic = qr{\+|-|\*|\/|%}; -our $Operators = qr{ - <=|>=|==|!=| - =>|->|<<|>>|<|>|!|~| - &&|\|\||,|\^|\+\+|--|&|\||$Arithmetic - }x; - -our $c90_Keywords = qr{do|for|while|if|else|return|goto|continue|switch|default|case|break}x; - -our $BasicType; -our $NonptrType; -our $NonptrTypeMisordered; -our $NonptrTypeWithAttr; -our $Type; -our $TypeMisordered; -our $Declare; -our $DeclareMisordered; - -our $NON_ASCII_UTF8 = qr{ - [\xC2-\xDF][\x80-\xBF] # non-overlong 2-byte - | \xE0[\xA0-\xBF][\x80-\xBF] # excluding overlongs - | [\xE1-\xEC\xEE\xEF][\x80-\xBF]{2} # straight 3-byte - | \xED[\x80-\x9F][\x80-\xBF] # excluding surrogates - | \xF0[\x90-\xBF][\x80-\xBF]{2} # planes 1-3 - | [\xF1-\xF3][\x80-\xBF]{3} # planes 4-15 - | \xF4[\x80-\x8F][\x80-\xBF]{2} # plane 16 -}x; - -our $UTF8 = qr{ - [\x09\x0A\x0D\x20-\x7E] # ASCII - | $NON_ASCII_UTF8 -}x; - -our $typeC99Typedefs = qr{(?:__)?(?:[us]_?)?int_?(?:8|16|32|64)_t}; -our $typeOtherOSTypedefs = qr{(?x: - u_(?:char|short|int|long) | # bsd - u(?:nchar|short|int|long) # sysv -)}; -our $typeKernelTypedefs = qr{(?x: - (?:__)?(?:u|s|be|le)(?:8|16|32|64)_t| - atomic_t -)}; -our $typeTypedefs = qr{(?x: - $typeC99Typedefs\b| - $typeOtherOSTypedefs\b| - $typeKernelTypedefs\b -)}; - -our $zero_initializer = qr{(?:(?:0[xX])?0+$Int_type?|NULL|false)\b}; - -our $logFunctions = qr{(?x: - printk(?:_ratelimited|_once|_deferred_once|_deferred|)| - (?:[a-z0-9]+_){1,2}(?:printk|emerg|alert|crit|err|warning|warn|notice|info|debug|dbg|vdbg|devel|cont|WARN)(?:_ratelimited|_once|)| - WARN(?:_RATELIMIT|_ONCE|)| - panic| - MODULE_[A-Z_]+| - seq_vprintf|seq_printf|seq_puts -)}; - -our $signature_tags = qr{(?xi: - Signed-off-by:| - Acked-by:| - Tested-by:| - Reviewed-by:| - Reported-by:| - Suggested-by:| - To:| - Cc: -)}; - -our @typeListMisordered = ( - qr{char\s+(?:un)?signed}, - qr{int\s+(?:(?:un)?signed\s+)?short\s}, - qr{int\s+short(?:\s+(?:un)?signed)}, - qr{short\s+int(?:\s+(?:un)?signed)}, - qr{(?:un)?signed\s+int\s+short}, - qr{short\s+(?:un)?signed}, - qr{long\s+int\s+(?:un)?signed}, - qr{int\s+long\s+(?:un)?signed}, - qr{long\s+(?:un)?signed\s+int}, - qr{int\s+(?:un)?signed\s+long}, - qr{int\s+(?:un)?signed}, - qr{int\s+long\s+long\s+(?:un)?signed}, - qr{long\s+long\s+int\s+(?:un)?signed}, - qr{long\s+long\s+(?:un)?signed\s+int}, - qr{long\s+long\s+(?:un)?signed}, - qr{long\s+(?:un)?signed}, -); - -our @typeList = ( - qr{void}, - qr{(?:(?:un)?signed\s+)?char}, - qr{(?:(?:un)?signed\s+)?short\s+int}, - qr{(?:(?:un)?signed\s+)?short}, - qr{(?:(?:un)?signed\s+)?int}, - qr{(?:(?:un)?signed\s+)?long\s+int}, - qr{(?:(?:un)?signed\s+)?long\s+long\s+int}, - qr{(?:(?:un)?signed\s+)?long\s+long}, - qr{(?:(?:un)?signed\s+)?long}, - qr{(?:un)?signed}, - qr{float}, - qr{double}, - qr{bool}, - qr{struct\s+$Ident}, - qr{union\s+$Ident}, - qr{enum\s+$Ident}, - qr{${Ident}_t}, - qr{${Ident}_handler}, - qr{${Ident}_handler_fn}, - @typeListMisordered, -); - -our $C90_int_types = qr{(?x: - long\s+long\s+int\s+(?:un)?signed| - long\s+long\s+(?:un)?signed\s+int| - long\s+long\s+(?:un)?signed| - (?:(?:un)?signed\s+)?long\s+long\s+int| - (?:(?:un)?signed\s+)?long\s+long| - int\s+long\s+long\s+(?:un)?signed| - int\s+(?:(?:un)?signed\s+)?long\s+long| - - long\s+int\s+(?:un)?signed| - long\s+(?:un)?signed\s+int| - long\s+(?:un)?signed| - (?:(?:un)?signed\s+)?long\s+int| - (?:(?:un)?signed\s+)?long| - int\s+long\s+(?:un)?signed| - int\s+(?:(?:un)?signed\s+)?long| - - int\s+(?:un)?signed| - (?:(?:un)?signed\s+)?int -)}; - -our @typeListFile = (); -our @typeListWithAttr = ( - @typeList, - qr{struct\s+$InitAttribute\s+$Ident}, - qr{union\s+$InitAttribute\s+$Ident}, -); - -our @modifierList = ( - qr{fastcall}, -); -our @modifierListFile = (); - -our @mode_permission_funcs = ( - ["module_param", 3], - ["module_param_(?:array|named|string)", 4], - ["module_param_array_named", 5], - ["debugfs_create_(?:file|u8|u16|u32|u64|x8|x16|x32|x64|size_t|atomic_t|bool|blob|regset32|u32_array)", 2], - ["proc_create(?:_data|)", 2], - ["(?:CLASS|DEVICE|SENSOR|SENSOR_DEVICE|IIO_DEVICE)_ATTR", 2], - ["IIO_DEV_ATTR_[A-Z_]+", 1], - ["SENSOR_(?:DEVICE_|)ATTR_2", 2], - ["SENSOR_TEMPLATE(?:_2|)", 3], - ["__ATTR", 2], -); - -#Create a search pattern for all these functions to speed up a loop below -our $mode_perms_search = ""; -foreach my $entry (@mode_permission_funcs) { - $mode_perms_search .= '|' if ($mode_perms_search ne ""); - $mode_perms_search .= $entry->[0]; -} - -our $mode_perms_world_writable = qr{ - S_IWUGO | - S_IWOTH | - S_IRWXUGO | - S_IALLUGO | - 0[0-7][0-7][2367] -}x; - -our %mode_permission_string_types = ( - "S_IRWXU" => 0700, - "S_IRUSR" => 0400, - "S_IWUSR" => 0200, - "S_IXUSR" => 0100, - "S_IRWXG" => 0070, - "S_IRGRP" => 0040, - "S_IWGRP" => 0020, - "S_IXGRP" => 0010, - "S_IRWXO" => 0007, - "S_IROTH" => 0004, - "S_IWOTH" => 0002, - "S_IXOTH" => 0001, - "S_IRWXUGO" => 0777, - "S_IRUGO" => 0444, - "S_IWUGO" => 0222, - "S_IXUGO" => 0111, -); - -#Create a search pattern for all these strings to speed up a loop below -our $mode_perms_string_search = ""; -foreach my $entry (keys %mode_permission_string_types) { - $mode_perms_string_search .= '|' if ($mode_perms_string_search ne ""); - $mode_perms_string_search .= $entry; -} - -our $allowed_asm_includes = qr{(?x: - irq| - memory| - time| - reboot -)}; -# memory.h: ARM has a custom one - -# Load common spelling mistakes and build regular expression list. -my $misspellings; -my %spelling_fix; - -if (open(my $spelling, '<', $spelling_file)) { - while (<$spelling>) { - my $line = $_; - - $line =~ s/\s*\n?$//g; - $line =~ s/^\s*//g; - - next if ($line =~ m/^\s*#/); - next if ($line =~ m/^\s*$/); - - my ($suspect, $fix) = split(/\|\|/, $line); - - $spelling_fix{$suspect} = $fix; - } - close($spelling); -} else { - warn "No typos will be found - file '$spelling_file': $!\n"; -} - -if ($codespell) { - if (open(my $spelling, '<', $codespellfile)) { - while (<$spelling>) { - my $line = $_; - - $line =~ s/\s*\n?$//g; - $line =~ s/^\s*//g; - - next if ($line =~ m/^\s*#/); - next if ($line =~ m/^\s*$/); - next if ($line =~ m/, disabled/i); - - $line =~ s/,.*$//; - - my ($suspect, $fix) = split(/->/, $line); - - $spelling_fix{$suspect} = $fix; - } - close($spelling); - } else { - warn "No codespell typos will be found - file '$codespellfile': $!\n"; - } -} - -$misspellings = join("|", sort keys %spelling_fix) if keys %spelling_fix; - -sub read_words { - my ($wordsRef, $file) = @_; - - if (open(my $words, '<', $file)) { - while (<$words>) { - my $line = $_; - - $line =~ s/\s*\n?$//g; - $line =~ s/^\s*//g; - - next if ($line =~ m/^\s*#/); - next if ($line =~ m/^\s*$/); - if ($line =~ /\s/) { - print("$file: '$line' invalid - ignored\n"); - next; - } - - $$wordsRef .= '|' if ($$wordsRef ne ""); - $$wordsRef .= $line; - } - close($file); - return 1; - } - - return 0; -} - -my $const_structs = ""; -#read_words(\$const_structs, $conststructsfile) -# or warn "No structs that should be const will be found - file '$conststructsfile': $!\n"; - -my $typeOtherTypedefs = ""; -if (length($typedefsfile)) { - read_words(\$typeOtherTypedefs, $typedefsfile) - or warn "No additional types will be considered - file '$typedefsfile': $!\n"; -} -$typeTypedefs .= '|' . $typeOtherTypedefs if ($typeOtherTypedefs ne ""); - -sub build_types { - my $mods = "(?x: \n" . join("|\n ", (@modifierList, @modifierListFile)) . "\n)"; - my $all = "(?x: \n" . join("|\n ", (@typeList, @typeListFile)) . "\n)"; - my $Misordered = "(?x: \n" . join("|\n ", @typeListMisordered) . "\n)"; - my $allWithAttr = "(?x: \n" . join("|\n ", @typeListWithAttr) . "\n)"; - $Modifier = qr{(?:$Attribute|$Sparse|$mods)}; - $BasicType = qr{ - (?:$typeTypedefs\b)| - (?:${all}\b) - }x; - $NonptrType = qr{ - (?:$Modifier\s+|const\s+)* - (?: - (?:typeof|__typeof__)\s*\([^\)]*\)| - (?:$typeTypedefs\b)| - (?:${all}\b) - ) - (?:\s+$Modifier|\s+const)* - }x; - $NonptrTypeMisordered = qr{ - (?:$Modifier\s+|const\s+)* - (?: - (?:${Misordered}\b) - ) - (?:\s+$Modifier|\s+const)* - }x; - $NonptrTypeWithAttr = qr{ - (?:$Modifier\s+|const\s+)* - (?: - (?:typeof|__typeof__)\s*\([^\)]*\)| - (?:$typeTypedefs\b)| - (?:${allWithAttr}\b) - ) - (?:\s+$Modifier|\s+const)* - }x; - $Type = qr{ - $NonptrType - (?:(?:\s|\*|\[\])+\s*const|(?:\s|\*\s*(?:const\s*)?|\[\])+|(?:\s*\[\s*\])+)? - (?:\s+$Inline|\s+$Modifier)* - }x; - $TypeMisordered = qr{ - $NonptrTypeMisordered - (?:(?:\s|\*|\[\])+\s*const|(?:\s|\*\s*(?:const\s*)?|\[\])+|(?:\s*\[\s*\])+)? - (?:\s+$Inline|\s+$Modifier)* - }x; - $Declare = qr{(?:$Storage\s+(?:$Inline\s+)?)?$Type}; - $DeclareMisordered = qr{(?:$Storage\s+(?:$Inline\s+)?)?$TypeMisordered}; -} -build_types(); - -our $Typecast = qr{\s*(\(\s*$NonptrType\s*\)){0,1}\s*}; - -# Using $balanced_parens, $LvalOrFunc, or $FuncArg -# requires at least perl version v5.10.0 -# Any use must be runtime checked with $^V - -our $balanced_parens = qr/(\((?:[^\(\)]++|(?-1))*\))/; -our $LvalOrFunc = qr{((?:[\&\*]\s*)?$Lval)\s*($balanced_parens{0,1})\s*}; -our $FuncArg = qr{$Typecast{0,1}($LvalOrFunc|$Constant|$String)}; - -our $declaration_macros = qr{(?x: - (?:$Storage\s+)?(?:[A-Z_][A-Z0-9]*_){0,2}(?:DEFINE|DECLARE)(?:_[A-Z0-9]+){1,6}\s*\(| - (?:$Storage\s+)?[HLP]?LIST_HEAD\s*\(| - (?:$Storage\s+)?${Type}\s+uninitialized_var\s*\( -)}; - -sub deparenthesize { - my ($string) = @_; - return "" if (!defined($string)); - - while ($string =~ /^\s*\(.*\)\s*$/) { - $string =~ s@^\s*\(\s*@@; - $string =~ s@\s*\)\s*$@@; - } - - $string =~ s@\s+@ @g; - - return $string; -} - -sub seed_camelcase_file { - my ($file) = @_; - - return if (!(-f $file)); - - local $/; - - open(my $include_file, '<', "$file") - or warn "$P: Can't read '$file' $!\n"; - my $text = <$include_file>; - close($include_file); - - my @lines = split('\n', $text); - - foreach my $line (@lines) { - next if ($line !~ /(?:[A-Z][a-z]|[a-z][A-Z])/); - if ($line =~ /^[ \t]*(?:#[ \t]*define|typedef\s+$Type)\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)/) { - $camelcase{$1} = 1; - } elsif ($line =~ /^\s*$Declare\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)\s*[\(\[,;]/) { - $camelcase{$1} = 1; - } elsif ($line =~ /^\s*(?:union|struct|enum)\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)\s*[;\{]/) { - $camelcase{$1} = 1; - } - } -} - -sub is_maintained_obsolete { - my ($filename) = @_; - - return 0 if (!$tree || !(-e "$root/scripts/get_maintainer.pl")); - - my $status = `perl $root/scripts/get_maintainer.pl --status --nom --nol --nogit --nogit-fallback -f $filename 2>&1`; - - return $status =~ /obsolete/i; -} - -my $camelcase_seeded = 0; -sub seed_camelcase_includes { - return if ($camelcase_seeded); - - my $files; - my $camelcase_cache = ""; - my @include_files = (); - - $camelcase_seeded = 1; - - if (-e ".git") { - my $git_last_include_commit = `git log --no-merges --pretty=format:"%h%n" -1 -- include`; - chomp $git_last_include_commit; - $camelcase_cache = ".checkpatch-camelcase.git.$git_last_include_commit"; - } else { - my $last_mod_date = 0; - $files = `find $root/include -name "*.h"`; - @include_files = split('\n', $files); - foreach my $file (@include_files) { - my $date = POSIX::strftime("%Y%m%d%H%M", - localtime((stat $file)[9])); - $last_mod_date = $date if ($last_mod_date < $date); - } - $camelcase_cache = ".checkpatch-camelcase.date.$last_mod_date"; - } - - if ($camelcase_cache ne "" && -f $camelcase_cache) { - open(my $camelcase_file, '<', "$camelcase_cache") - or warn "$P: Can't read '$camelcase_cache' $!\n"; - while (<$camelcase_file>) { - chomp; - $camelcase{$_} = 1; - } - close($camelcase_file); - - return; - } - - if (-e ".git") { - $files = `git ls-files "include/*.h"`; - @include_files = split('\n', $files); - } - - foreach my $file (@include_files) { - seed_camelcase_file($file); - } - - if ($camelcase_cache ne "") { - unlink glob ".checkpatch-camelcase.*"; - open(my $camelcase_file, '>', "$camelcase_cache") - or warn "$P: Can't write '$camelcase_cache' $!\n"; - foreach (sort { lc($a) cmp lc($b) } keys(%camelcase)) { - print $camelcase_file ("$_\n"); - } - close($camelcase_file); - } -} - -sub git_commit_info { - my ($commit, $id, $desc) = @_; - - return ($id, $desc) if ((which("git") eq "") || !(-e ".git")); - - my $output = `git log --no-color --format='%H %s' -1 $commit 2>&1`; - $output =~ s/^\s*//gm; - my @lines = split("\n", $output); - - return ($id, $desc) if ($#lines < 0); - - if ($lines[0] =~ /^error: short SHA1 $commit is ambiguous\./) { -# Maybe one day convert this block of bash into something that returns -# all matching commit ids, but it's very slow... -# -# echo "checking commits $1..." -# git rev-list --remotes | grep -i "^$1" | -# while read line ; do -# git log --format='%H %s' -1 $line | -# echo "commit $(cut -c 1-12,41-)" -# done - } elsif ($lines[0] =~ /^fatal: ambiguous argument '$commit': unknown revision or path not in the working tree\./) { - $id = undef; - } else { - $id = substr($lines[0], 0, 12); - $desc = substr($lines[0], 41); - } - - return ($id, $desc); -} - -$chk_signoff = 0 if ($file); - -my @rawlines = (); -my @lines = (); -my @fixed = (); -my @fixed_inserted = (); -my @fixed_deleted = (); -my $fixlinenr = -1; - -# If input is git commits, extract all commits from the commit expressions. -# For example, HEAD-3 means we need check 'HEAD, HEAD~1, HEAD~2'. -die "$P: No git repository found\n" if ($git && !-e ".git"); - -if ($git) { - my @commits = (); - foreach my $commit_expr (@ARGV) { - my $git_range; - if ($commit_expr =~ m/^(.*)-(\d+)$/) { - $git_range = "-$2 $1"; - } elsif ($commit_expr =~ m/\.\./) { - $git_range = "$commit_expr"; - } else { - $git_range = "-1 $commit_expr"; - } - my $lines = `git log --no-color --no-merges --pretty=format:'%H %s' $git_range`; - foreach my $line (split(/\n/, $lines)) { - $line =~ /^([0-9a-fA-F]{40,40}) (.*)$/; - next if (!defined($1) || !defined($2)); - my $sha1 = $1; - my $subject = $2; - unshift(@commits, $sha1); - $git_commits{$sha1} = $subject; - } - } - die "$P: no git commits after extraction!\n" if (@commits == 0); - @ARGV = @commits; -} - -my $vname; -for my $filename (@ARGV) { - my $FILE; - if ($git) { - open($FILE, '-|', "git format-patch -M --stdout -1 $filename") || - die "$P: $filename: git format-patch failed - $!\n"; - } elsif ($file) { - open($FILE, '-|', "diff -u /dev/null $filename") || - die "$P: $filename: diff failed - $!\n"; - } elsif ($filename eq '-') { - open($FILE, '<&STDIN'); - } else { - open($FILE, '<', "$filename") || - die "$P: $filename: open failed - $!\n"; - } - if ($filename eq '-') { - $vname = 'Your patch'; - } elsif ($git) { - $vname = "Commit " . substr($filename, 0, 12) . ' ("' . $git_commits{$filename} . '")'; - } else { - $vname = $filename; - } - while (<$FILE>) { - chomp; - push(@rawlines, $_); - } - close($FILE); - - if ($#ARGV > 0 && $quiet == 0) { - print '-' x length($vname) . "\n"; - print "$vname\n"; - print '-' x length($vname) . "\n"; - } - - if (!process($filename)) { - $exit = 1; - } - @rawlines = (); - @lines = (); - @fixed = (); - @fixed_inserted = (); - @fixed_deleted = (); - $fixlinenr = -1; - @modifierListFile = (); - @typeListFile = (); - build_types(); -} - -if (!$quiet) { - hash_show_words(\%use_type, "Used"); - hash_show_words(\%ignore_type, "Ignored"); - - if ($^V lt 5.10.0) { - print << "EOM" - -NOTE: perl $^V is not modern enough to detect all possible issues. - An upgrade to at least perl v5.10.0 is suggested. -EOM - } - if ($exit) { - print << "EOM" - -NOTE: If any of the errors are false positives, please report - them to the maintainers. -EOM - } -} - -exit($exit); - -sub top_of_kernel_tree { - my ($root) = @_; - - my @tree_check = ( - "LICENSE", "CODEOWNERS", "Kconfig", "Makefile", - "README.rst", "doc", "arch", "include", "drivers", - "boards", "kernel", "lib", "scripts", - ); - - foreach my $check (@tree_check) { - if (! -e $root . '/' . $check) { - return 0; - } - } - return 1; -} - -sub parse_email { - my ($formatted_email) = @_; - - my $name = ""; - my $address = ""; - my $comment = ""; - - if ($formatted_email =~ /^(.*)<(\S+\@\S+)>(.*)$/) { - $name = $1; - $address = $2; - $comment = $3 if defined $3; - } elsif ($formatted_email =~ /^\s*<(\S+\@\S+)>(.*)$/) { - $address = $1; - $comment = $2 if defined $2; - } elsif ($formatted_email =~ /(\S+\@\S+)(.*)$/) { - $address = $1; - $comment = $2 if defined $2; - $formatted_email =~ s/$address.*$//; - $name = $formatted_email; - $name = trim($name); - $name =~ s/^\"|\"$//g; - # If there's a name left after stripping spaces and - # leading quotes, and the address doesn't have both - # leading and trailing angle brackets, the address - # is invalid. ie: - # "joe smith joe@smith.com" bad - # "joe smith ]+>$/) { - $name = ""; - $address = ""; - $comment = ""; - } - } - - $name = trim($name); - $name =~ s/^\"|\"$//g; - $address = trim($address); - $address =~ s/^\<|\>$//g; - - if ($name =~ /[^\w \-]/i) { ##has "must quote" chars - $name =~ s/(?"; - } - - return $formatted_email; -} - -sub which { - my ($bin) = @_; - - foreach my $path (split(/:/, $ENV{PATH})) { - if (-e "$path/$bin") { - return "$path/$bin"; - } - } - - return ""; -} - -sub which_conf { - my ($conf) = @_; - - foreach my $path (split(/:/, ".:$ENV{HOME}:.scripts")) { - if (-e "$path/$conf") { - return "$path/$conf"; - } - } - - return ""; -} - -sub expand_tabs { - my ($str) = @_; - - my $res = ''; - my $n = 0; - for my $c (split(//, $str)) { - if ($c eq "\t") { - $res .= ' '; - $n++; - for (; ($n % 8) != 0; $n++) { - $res .= ' '; - } - next; - } - $res .= $c; - $n++; - } - - return $res; -} -sub copy_spacing { - (my $res = shift) =~ tr/\t/ /c; - return $res; -} - -sub line_stats { - my ($line) = @_; - - # Drop the diff line leader and expand tabs - $line =~ s/^.//; - $line = expand_tabs($line); - - # Pick the indent from the front of the line. - my ($white) = ($line =~ /^(\s*)/); - - return (length($line), length($white)); -} - -my $sanitise_quote = ''; - -sub sanitise_line_reset { - my ($in_comment) = @_; - - if ($in_comment) { - $sanitise_quote = '*/'; - } else { - $sanitise_quote = ''; - } -} -sub sanitise_line { - my ($line) = @_; - - my $res = ''; - my $l = ''; - - my $qlen = 0; - my $off = 0; - my $c; - - # Always copy over the diff marker. - $res = substr($line, 0, 1); - - for ($off = 1; $off < length($line); $off++) { - $c = substr($line, $off, 1); - - # Comments we are wacking completly including the begin - # and end, all to $;. - if ($sanitise_quote eq '' && substr($line, $off, 2) eq '/*') { - $sanitise_quote = '*/'; - - substr($res, $off, 2, "$;$;"); - $off++; - next; - } - if ($sanitise_quote eq '*/' && substr($line, $off, 2) eq '*/') { - $sanitise_quote = ''; - substr($res, $off, 2, "$;$;"); - $off++; - next; - } - if ($sanitise_quote eq '' && substr($line, $off, 2) eq '//') { - $sanitise_quote = '//'; - - substr($res, $off, 2, $sanitise_quote); - $off++; - next; - } - - # A \ in a string means ignore the next character. - if (($sanitise_quote eq "'" || $sanitise_quote eq '"') && - $c eq "\\") { - substr($res, $off, 2, 'XX'); - $off++; - next; - } - # Regular quotes. - if ($c eq "'" || $c eq '"') { - if ($sanitise_quote eq '') { - $sanitise_quote = $c; - - substr($res, $off, 1, $c); - next; - } elsif ($sanitise_quote eq $c) { - $sanitise_quote = ''; - } - } - - #print "c<$c> SQ<$sanitise_quote>\n"; - if ($off != 0 && $sanitise_quote eq '*/' && $c ne "\t") { - substr($res, $off, 1, $;); - } elsif ($off != 0 && $sanitise_quote eq '//' && $c ne "\t") { - substr($res, $off, 1, $;); - } elsif ($off != 0 && $sanitise_quote && $c ne "\t") { - substr($res, $off, 1, 'X'); - } else { - substr($res, $off, 1, $c); - } - } - - if ($sanitise_quote eq '//') { - $sanitise_quote = ''; - } - - # The pathname on a #include may be surrounded by '<' and '>'. - if ($res =~ /^.\s*\#\s*include\s+\<(.*)\>/) { - my $clean = 'X' x length($1); - $res =~ s@\<.*\>@<$clean>@; - - # The whole of a #error is a string. - } elsif ($res =~ /^.\s*\#\s*(?:error|warning)\s+(.*)\b/) { - my $clean = 'X' x length($1); - $res =~ s@(\#\s*(?:error|warning)\s+).*@$1$clean@; - } - - if ($allow_c99_comments && $res =~ m@(//.*$)@) { - my $match = $1; - $res =~ s/\Q$match\E/"$;" x length($match)/e; - } - - return $res; -} - -sub get_quoted_string { - my ($line, $rawline) = @_; - - return "" if ($line !~ m/($String)/g); - return substr($rawline, $-[0], $+[0] - $-[0]); -} - -sub ctx_statement_block { - my ($linenr, $remain, $off) = @_; - my $line = $linenr - 1; - my $blk = ''; - my $soff = $off; - my $coff = $off - 1; - my $coff_set = 0; - - my $loff = 0; - - my $type = ''; - my $level = 0; - my @stack = (); - my $p; - my $c; - my $len = 0; - - my $remainder; - while (1) { - @stack = (['', 0]) if ($#stack == -1); - - #warn "CSB: blk<$blk> remain<$remain>\n"; - # If we are about to drop off the end, pull in more - # context. - if ($off >= $len) { - for (; $remain > 0; $line++) { - last if (!defined $lines[$line]); - next if ($lines[$line] =~ /^-/); - $remain--; - $loff = $len; - $blk .= $lines[$line] . "\n"; - $len = length($blk); - $line++; - last; - } - # Bail if there is no further context. - #warn "CSB: blk<$blk> off<$off> len<$len>\n"; - if ($off >= $len) { - last; - } - if ($level == 0 && substr($blk, $off) =~ /^.\s*#\s*define/) { - $level++; - $type = '#'; - } - } - $p = $c; - $c = substr($blk, $off, 1); - $remainder = substr($blk, $off); - - #warn "CSB: c<$c> type<$type> level<$level> remainder<$remainder> coff_set<$coff_set>\n"; - - # Handle nested #if/#else. - if ($remainder =~ /^#\s*(?:ifndef|ifdef|if)\s/) { - push(@stack, [ $type, $level ]); - } elsif ($remainder =~ /^#\s*(?:else|elif)\b/) { - ($type, $level) = @{$stack[$#stack - 1]}; - } elsif ($remainder =~ /^#\s*endif\b/) { - ($type, $level) = @{pop(@stack)}; - } - - # Statement ends at the ';' or a close '}' at the - # outermost level. - if ($level == 0 && $c eq ';') { - last; - } - - # An else is really a conditional as long as its not else if - if ($level == 0 && $coff_set == 0 && - (!defined($p) || $p =~ /(?:\s|\}|\+)/) && - $remainder =~ /^(else)(?:\s|{)/ && - $remainder !~ /^else\s+if\b/) { - $coff = $off + length($1) - 1; - $coff_set = 1; - #warn "CSB: mark coff<$coff> soff<$soff> 1<$1>\n"; - #warn "[" . substr($blk, $soff, $coff - $soff + 1) . "]\n"; - } - - if (($type eq '' || $type eq '(') && $c eq '(') { - $level++; - $type = '('; - } - if ($type eq '(' && $c eq ')') { - $level--; - $type = ($level != 0)? '(' : ''; - - if ($level == 0 && $coff < $soff) { - $coff = $off; - $coff_set = 1; - #warn "CSB: mark coff<$coff>\n"; - } - } - if (($type eq '' || $type eq '{') && $c eq '{') { - $level++; - $type = '{'; - } - if ($type eq '{' && $c eq '}') { - $level--; - $type = ($level != 0)? '{' : ''; - - if ($level == 0) { - if (substr($blk, $off + 1, 1) eq ';') { - $off++; - } - last; - } - } - # Preprocessor commands end at the newline unless escaped. - if ($type eq '#' && $c eq "\n" && $p ne "\\") { - $level--; - $type = ''; - $off++; - last; - } - $off++; - } - # We are truly at the end, so shuffle to the next line. - if ($off == $len) { - $loff = $len + 1; - $line++; - $remain--; - } - - my $statement = substr($blk, $soff, $off - $soff + 1); - my $condition = substr($blk, $soff, $coff - $soff + 1); - - #warn "STATEMENT<$statement>\n"; - #warn "CONDITION<$condition>\n"; - - #print "coff<$coff> soff<$off> loff<$loff>\n"; - - return ($statement, $condition, - $line, $remain + 1, $off - $loff + 1, $level); -} - -sub statement_lines { - my ($stmt) = @_; - - # Strip the diff line prefixes and rip blank lines at start and end. - $stmt =~ s/(^|\n)./$1/g; - $stmt =~ s/^\s*//; - $stmt =~ s/\s*$//; - - my @stmt_lines = ($stmt =~ /\n/g); - - return $#stmt_lines + 2; -} - -sub statement_rawlines { - my ($stmt) = @_; - - my @stmt_lines = ($stmt =~ /\n/g); - - return $#stmt_lines + 2; -} - -sub statement_block_size { - my ($stmt) = @_; - - $stmt =~ s/(^|\n)./$1/g; - $stmt =~ s/^\s*{//; - $stmt =~ s/}\s*$//; - $stmt =~ s/^\s*//; - $stmt =~ s/\s*$//; - - my @stmt_lines = ($stmt =~ /\n/g); - my @stmt_statements = ($stmt =~ /;/g); - - my $stmt_lines = $#stmt_lines + 2; - my $stmt_statements = $#stmt_statements + 1; - - if ($stmt_lines > $stmt_statements) { - return $stmt_lines; - } else { - return $stmt_statements; - } -} - -sub ctx_statement_full { - my ($linenr, $remain, $off) = @_; - my ($statement, $condition, $level); - - my (@chunks); - - # Grab the first conditional/block pair. - ($statement, $condition, $linenr, $remain, $off, $level) = - ctx_statement_block($linenr, $remain, $off); - #print "F: c<$condition> s<$statement> remain<$remain>\n"; - push(@chunks, [ $condition, $statement ]); - if (!($remain > 0 && $condition =~ /^\s*(?:\n[+-])?\s*(?:if|else|do)\b/s)) { - return ($level, $linenr, @chunks); - } - - # Pull in the following conditional/block pairs and see if they - # could continue the statement. - for (;;) { - ($statement, $condition, $linenr, $remain, $off, $level) = - ctx_statement_block($linenr, $remain, $off); - #print "C: c<$condition> s<$statement> remain<$remain>\n"; - last if (!($remain > 0 && $condition =~ /^(?:\s*\n[+-])*\s*(?:else|do)\b/s)); - #print "C: push\n"; - push(@chunks, [ $condition, $statement ]); - } - - return ($level, $linenr, @chunks); -} - -sub ctx_block_get { - my ($linenr, $remain, $outer, $open, $close, $off) = @_; - my $line; - my $start = $linenr - 1; - my $blk = ''; - my @o; - my @c; - my @res = (); - - my $level = 0; - my @stack = ($level); - for ($line = $start; $remain > 0; $line++) { - next if ($rawlines[$line] =~ /^-/); - $remain--; - - $blk .= $rawlines[$line]; - - # Handle nested #if/#else. - if ($lines[$line] =~ /^.\s*#\s*(?:ifndef|ifdef|if)\s/) { - push(@stack, $level); - } elsif ($lines[$line] =~ /^.\s*#\s*(?:else|elif)\b/) { - $level = $stack[$#stack - 1]; - } elsif ($lines[$line] =~ /^.\s*#\s*endif\b/) { - $level = pop(@stack); - } - - foreach my $c (split(//, $lines[$line])) { - ##print "C<$c>L<$level><$open$close>O<$off>\n"; - if ($off > 0) { - $off--; - next; - } - - if ($c eq $close && $level > 0) { - $level--; - last if ($level == 0); - } elsif ($c eq $open) { - $level++; - } - } - - if (!$outer || $level <= 1) { - push(@res, $rawlines[$line]); - } - - last if ($level == 0); - } - - return ($level, @res); -} -sub ctx_block_outer { - my ($linenr, $remain) = @_; - - my ($level, @r) = ctx_block_get($linenr, $remain, 1, '{', '}', 0); - return @r; -} -sub ctx_block { - my ($linenr, $remain) = @_; - - my ($level, @r) = ctx_block_get($linenr, $remain, 0, '{', '}', 0); - return @r; -} -sub ctx_statement { - my ($linenr, $remain, $off) = @_; - - my ($level, @r) = ctx_block_get($linenr, $remain, 0, '(', ')', $off); - return @r; -} -sub ctx_block_level { - my ($linenr, $remain) = @_; - - return ctx_block_get($linenr, $remain, 0, '{', '}', 0); -} -sub ctx_statement_level { - my ($linenr, $remain, $off) = @_; - - return ctx_block_get($linenr, $remain, 0, '(', ')', $off); -} - -sub ctx_locate_comment { - my ($first_line, $end_line) = @_; - - # Catch a comment on the end of the line itself. - my ($current_comment) = ($rawlines[$end_line - 1] =~ m@.*(/\*.*\*/)\s*(?:\\\s*)?$@); - return $current_comment if (defined $current_comment); - - # Look through the context and try and figure out if there is a - # comment. - my $in_comment = 0; - $current_comment = ''; - for (my $linenr = $first_line; $linenr < $end_line; $linenr++) { - my $line = $rawlines[$linenr - 1]; - #warn " $line\n"; - if ($linenr == $first_line and $line =~ m@^.\s*\*@) { - $in_comment = 1; - } - if ($line =~ m@/\*@) { - $in_comment = 1; - } - if (!$in_comment && $current_comment ne '') { - $current_comment = ''; - } - $current_comment .= $line . "\n" if ($in_comment); - if ($line =~ m@\*/@) { - $in_comment = 0; - } - } - - chomp($current_comment); - return($current_comment); -} -sub ctx_has_comment { - my ($first_line, $end_line) = @_; - my $cmt = ctx_locate_comment($first_line, $end_line); - - ##print "LINE: $rawlines[$end_line - 1 ]\n"; - ##print "CMMT: $cmt\n"; - - return ($cmt ne ''); -} - -sub raw_line { - my ($linenr, $cnt) = @_; - - my $offset = $linenr - 1; - $cnt++; - - my $line; - while ($cnt) { - $line = $rawlines[$offset++]; - next if (defined($line) && $line =~ /^-/); - $cnt--; - } - - return $line; -} - -sub cat_vet { - my ($vet) = @_; - my ($res, $coded); - - $res = ''; - while ($vet =~ /([^[:cntrl:]]*)([[:cntrl:]]|$)/g) { - $res .= $1; - if ($2 ne '') { - $coded = sprintf("^%c", unpack('C', $2) + 64); - $res .= $coded; - } - } - $res =~ s/$/\$/; - - return $res; -} - -my $av_preprocessor = 0; -my $av_pending; -my @av_paren_type; -my $av_pend_colon; - -sub annotate_reset { - $av_preprocessor = 0; - $av_pending = '_'; - @av_paren_type = ('E'); - $av_pend_colon = 'O'; -} - -sub annotate_values { - my ($stream, $type) = @_; - - my $res; - my $var = '_' x length($stream); - my $cur = $stream; - - print "$stream\n" if ($dbg_values > 1); - - while (length($cur)) { - @av_paren_type = ('E') if ($#av_paren_type < 0); - print " <" . join('', @av_paren_type) . - "> <$type> <$av_pending>" if ($dbg_values > 1); - if ($cur =~ /^(\s+)/o) { - print "WS($1)\n" if ($dbg_values > 1); - if ($1 =~ /\n/ && $av_preprocessor) { - $type = pop(@av_paren_type); - $av_preprocessor = 0; - } - - } elsif ($cur =~ /^(\(\s*$Type\s*)\)/ && $av_pending eq '_') { - print "CAST($1)\n" if ($dbg_values > 1); - push(@av_paren_type, $type); - $type = 'c'; - - } elsif ($cur =~ /^($Type)\s*(?:$Ident|,|\)|\(|\s*$)/) { - print "DECLARE($1)\n" if ($dbg_values > 1); - $type = 'T'; - - } elsif ($cur =~ /^($Modifier)\s*/) { - print "MODIFIER($1)\n" if ($dbg_values > 1); - $type = 'T'; - - } elsif ($cur =~ /^(\#\s*define\s*$Ident)(\(?)/o) { - print "DEFINE($1,$2)\n" if ($dbg_values > 1); - $av_preprocessor = 1; - push(@av_paren_type, $type); - if ($2 ne '') { - $av_pending = 'N'; - } - $type = 'E'; - - } elsif ($cur =~ /^(\#\s*(?:undef\s*$Ident|include\b))/o) { - print "UNDEF($1)\n" if ($dbg_values > 1); - $av_preprocessor = 1; - push(@av_paren_type, $type); - - } elsif ($cur =~ /^(\#\s*(?:ifdef|ifndef|if))/o) { - print "PRE_START($1)\n" if ($dbg_values > 1); - $av_preprocessor = 1; - - push(@av_paren_type, $type); - push(@av_paren_type, $type); - $type = 'E'; - - } elsif ($cur =~ /^(\#\s*(?:else|elif))/o) { - print "PRE_RESTART($1)\n" if ($dbg_values > 1); - $av_preprocessor = 1; - - push(@av_paren_type, $av_paren_type[$#av_paren_type]); - - $type = 'E'; - - } elsif ($cur =~ /^(\#\s*(?:endif))/o) { - print "PRE_END($1)\n" if ($dbg_values > 1); - - $av_preprocessor = 1; - - # Assume all arms of the conditional end as this - # one does, and continue as if the #endif was not here. - pop(@av_paren_type); - push(@av_paren_type, $type); - $type = 'E'; - - } elsif ($cur =~ /^(\\\n)/o) { - print "PRECONT($1)\n" if ($dbg_values > 1); - - } elsif ($cur =~ /^(__attribute__)\s*\(?/o) { - print "ATTR($1)\n" if ($dbg_values > 1); - $av_pending = $type; - $type = 'N'; - - } elsif ($cur =~ /^(sizeof)\s*(\()?/o) { - print "SIZEOF($1)\n" if ($dbg_values > 1); - if (defined $2) { - $av_pending = 'V'; - } - $type = 'N'; - - } elsif ($cur =~ /^(if|while|for)\b/o) { - print "COND($1)\n" if ($dbg_values > 1); - $av_pending = 'E'; - $type = 'N'; - - } elsif ($cur =~/^(case)/o) { - print "CASE($1)\n" if ($dbg_values > 1); - $av_pend_colon = 'C'; - $type = 'N'; - - } elsif ($cur =~/^(return|else|goto|typeof|__typeof__)\b/o) { - print "KEYWORD($1)\n" if ($dbg_values > 1); - $type = 'N'; - - } elsif ($cur =~ /^(\()/o) { - print "PAREN('$1')\n" if ($dbg_values > 1); - push(@av_paren_type, $av_pending); - $av_pending = '_'; - $type = 'N'; - - } elsif ($cur =~ /^(\))/o) { - my $new_type = pop(@av_paren_type); - if ($new_type ne '_') { - $type = $new_type; - print "PAREN('$1') -> $type\n" - if ($dbg_values > 1); - } else { - print "PAREN('$1')\n" if ($dbg_values > 1); - } - - } elsif ($cur =~ /^($Ident)\s*\(/o) { - print "FUNC($1)\n" if ($dbg_values > 1); - $type = 'V'; - $av_pending = 'V'; - - } elsif ($cur =~ /^($Ident\s*):(?:\s*\d+\s*(,|=|;))?/) { - if (defined $2 && $type eq 'C' || $type eq 'T') { - $av_pend_colon = 'B'; - } elsif ($type eq 'E') { - $av_pend_colon = 'L'; - } - print "IDENT_COLON($1,$type>$av_pend_colon)\n" if ($dbg_values > 1); - $type = 'V'; - - } elsif ($cur =~ /^($Ident|$Constant)/o) { - print "IDENT($1)\n" if ($dbg_values > 1); - $type = 'V'; - - } elsif ($cur =~ /^($Assignment)/o) { - print "ASSIGN($1)\n" if ($dbg_values > 1); - $type = 'N'; - - } elsif ($cur =~/^(;|{|})/) { - print "END($1)\n" if ($dbg_values > 1); - $type = 'E'; - $av_pend_colon = 'O'; - - } elsif ($cur =~/^(,)/) { - print "COMMA($1)\n" if ($dbg_values > 1); - $type = 'C'; - - } elsif ($cur =~ /^(\?)/o) { - print "QUESTION($1)\n" if ($dbg_values > 1); - $type = 'N'; - - } elsif ($cur =~ /^(:)/o) { - print "COLON($1,$av_pend_colon)\n" if ($dbg_values > 1); - - substr($var, length($res), 1, $av_pend_colon); - if ($av_pend_colon eq 'C' || $av_pend_colon eq 'L') { - $type = 'E'; - } else { - $type = 'N'; - } - $av_pend_colon = 'O'; - - } elsif ($cur =~ /^(\[)/o) { - print "CLOSE($1)\n" if ($dbg_values > 1); - $type = 'N'; - - } elsif ($cur =~ /^(-(?![->])|\+(?!\+)|\*|\&\&|\&)/o) { - my $variant; - - print "OPV($1)\n" if ($dbg_values > 1); - if ($type eq 'V') { - $variant = 'B'; - } else { - $variant = 'U'; - } - - substr($var, length($res), 1, $variant); - $type = 'N'; - - } elsif ($cur =~ /^($Operators)/o) { - print "OP($1)\n" if ($dbg_values > 1); - if ($1 ne '++' && $1 ne '--') { - $type = 'N'; - } - - } elsif ($cur =~ /(^.)/o) { - print "C($1)\n" if ($dbg_values > 1); - } - if (defined $1) { - $cur = substr($cur, length($1)); - $res .= $type x length($1); - } - } - - return ($res, $var); -} - -sub possible { - my ($possible, $line) = @_; - my $notPermitted = qr{(?: - ^(?: - $Modifier| - $Storage| - $Type| - DEFINE_\S+ - )$| - ^(?: - goto| - return| - case| - else| - asm|__asm__| - do| - \#| - \#\#| - )(?:\s|$)| - ^(?:typedef|struct|enum)\b - )}x; - warn "CHECK<$possible> ($line)\n" if ($dbg_possible > 2); - if ($possible !~ $notPermitted) { - # Check for modifiers. - $possible =~ s/\s*$Storage\s*//g; - $possible =~ s/\s*$Sparse\s*//g; - if ($possible =~ /^\s*$/) { - - } elsif ($possible =~ /\s/) { - $possible =~ s/\s*$Type\s*//g; - for my $modifier (split(' ', $possible)) { - if ($modifier !~ $notPermitted) { - warn "MODIFIER: $modifier ($possible) ($line)\n" if ($dbg_possible); - push(@modifierListFile, $modifier); - } - } - - } else { - warn "POSSIBLE: $possible ($line)\n" if ($dbg_possible); - push(@typeListFile, $possible); - } - build_types(); - } else { - warn "NOTPOSS: $possible ($line)\n" if ($dbg_possible > 1); - } -} - -my $prefix = ''; - -sub show_type { - my ($type) = @_; - - $type =~ tr/[a-z]/[A-Z]/; - - return defined $use_type{$type} if (scalar keys %use_type > 0); - - return !defined $ignore_type{$type}; -} - -sub report { - my ($level, $type, $msg) = @_; - - if (!show_type($type) || - (defined $tst_only && $msg !~ /\Q$tst_only\E/)) { - return 0; - } - my $output = ''; - if ($color) { - if ($level eq 'ERROR') { - $output .= RED; - } elsif ($level eq 'WARNING') { - $output .= YELLOW; - } else { - $output .= GREEN; - } - } - $output .= $prefix . $level . ':'; - if ($show_types) { - $output .= BLUE if ($color); - $output .= "$type:"; - } - $output .= RESET if ($color); - $output .= ' ' . $msg . "\n"; - - if ($showfile) { - my @lines = split("\n", $output, -1); - splice(@lines, 1, 1); - $output = join("\n", @lines); - } - $output = (split('\n', $output))[0] . "\n" if ($terse); - - push(our @report, $output); - - return 1; -} - -sub report_dump { - our @report; -} - -sub fixup_current_range { - my ($lineRef, $offset, $length) = @_; - - if ($$lineRef =~ /^\@\@ -\d+,\d+ \+(\d+),(\d+) \@\@/) { - my $o = $1; - my $l = $2; - my $no = $o + $offset; - my $nl = $l + $length; - $$lineRef =~ s/\+$o,$l \@\@/\+$no,$nl \@\@/; - } -} - -sub fix_inserted_deleted_lines { - my ($linesRef, $insertedRef, $deletedRef) = @_; - - my $range_last_linenr = 0; - my $delta_offset = 0; - - my $old_linenr = 0; - my $new_linenr = 0; - - my $next_insert = 0; - my $next_delete = 0; - - my @lines = (); - - my $inserted = @{$insertedRef}[$next_insert++]; - my $deleted = @{$deletedRef}[$next_delete++]; - - foreach my $old_line (@{$linesRef}) { - my $save_line = 1; - my $line = $old_line; #don't modify the array - if ($line =~ /^(?:\+\+\+|\-\-\-)\s+\S+/) { #new filename - $delta_offset = 0; - } elsif ($line =~ /^\@\@ -\d+,\d+ \+\d+,\d+ \@\@/) { #new hunk - $range_last_linenr = $new_linenr; - fixup_current_range(\$line, $delta_offset, 0); - } - - while (defined($deleted) && ${$deleted}{'LINENR'} == $old_linenr) { - $deleted = @{$deletedRef}[$next_delete++]; - $save_line = 0; - fixup_current_range(\$lines[$range_last_linenr], $delta_offset--, -1); - } - - while (defined($inserted) && ${$inserted}{'LINENR'} == $old_linenr) { - push(@lines, ${$inserted}{'LINE'}); - $inserted = @{$insertedRef}[$next_insert++]; - $new_linenr++; - fixup_current_range(\$lines[$range_last_linenr], $delta_offset++, 1); - } - - if ($save_line) { - push(@lines, $line); - $new_linenr++; - } - - $old_linenr++; - } - - return @lines; -} - -sub fix_insert_line { - my ($linenr, $line) = @_; - - my $inserted = { - LINENR => $linenr, - LINE => $line, - }; - push(@fixed_inserted, $inserted); -} - -sub fix_delete_line { - my ($linenr, $line) = @_; - - my $deleted = { - LINENR => $linenr, - LINE => $line, - }; - - push(@fixed_deleted, $deleted); -} - -sub ERROR { - my ($type, $msg) = @_; - - if (report("ERROR", $type, $msg)) { - our $clean = 0; - our $cnt_error++; - return 1; - } - return 0; -} -sub WARN { - my ($type, $msg) = @_; - - if (report("WARNING", $type, $msg)) { - our $clean = 0; - our $cnt_warn++; - return 1; - } - return 0; -} -sub CHK { - my ($type, $msg) = @_; - - if ($check && report("CHECK", $type, $msg)) { - our $clean = 0; - our $cnt_chk++; - return 1; - } - return 0; -} - -sub check_absolute_file { - my ($absolute, $herecurr) = @_; - my $file = $absolute; - - ##print "absolute<$absolute>\n"; - - # See if any suffix of this path is a path within the tree. - while ($file =~ s@^[^/]*/@@) { - if (-f "$root/$file") { - ##print "file<$file>\n"; - last; - } - } - if (! -f _) { - return 0; - } - - # It is, so see if the prefix is acceptable. - my $prefix = $absolute; - substr($prefix, -length($file)) = ''; - - ##print "prefix<$prefix>\n"; - if ($prefix ne ".../") { - WARN("USE_RELATIVE_PATH", - "use relative pathname instead of absolute in changelog text\n" . $herecurr); - } -} - -sub trim { - my ($string) = @_; - - $string =~ s/^\s+|\s+$//g; - - return $string; -} - -sub ltrim { - my ($string) = @_; - - $string =~ s/^\s+//; - - return $string; -} - -sub rtrim { - my ($string) = @_; - - $string =~ s/\s+$//; - - return $string; -} - -sub string_find_replace { - my ($string, $find, $replace) = @_; - - $string =~ s/$find/$replace/g; - - return $string; -} - -sub tabify { - my ($leading) = @_; - - my $source_indent = 8; - my $max_spaces_before_tab = $source_indent - 1; - my $spaces_to_tab = " " x $source_indent; - - #convert leading spaces to tabs - 1 while $leading =~ s@^([\t]*)$spaces_to_tab@$1\t@g; - #Remove spaces before a tab - 1 while $leading =~ s@^([\t]*)( {1,$max_spaces_before_tab})\t@$1\t@g; - - return "$leading"; -} - -sub pos_last_openparen { - my ($line) = @_; - - my $pos = 0; - - my $opens = $line =~ tr/\(/\(/; - my $closes = $line =~ tr/\)/\)/; - - my $last_openparen = 0; - - if (($opens == 0) || ($closes >= $opens)) { - return -1; - } - - my $len = length($line); - - for ($pos = 0; $pos < $len; $pos++) { - my $string = substr($line, $pos); - if ($string =~ /^($FuncArg|$balanced_parens)/) { - $pos += length($1) - 1; - } elsif (substr($line, $pos, 1) eq '(') { - $last_openparen = $pos; - } elsif (index($string, '(') == -1) { - last; - } - } - - return length(expand_tabs(substr($line, 0, $last_openparen))) + 1; -} - -sub process { - my $filename = shift; - - my $linenr=0; - my $prevline=""; - my $prevrawline=""; - my $stashline=""; - my $stashrawline=""; - - my $length; - my $indent; - my $previndent=0; - my $stashindent=0; - - our $clean = 1; - my $signoff = 0; - my $is_patch = 0; - my $in_header_lines = $file ? 0 : 1; - my $in_commit_log = 0; #Scanning lines before patch - my $has_commit_log = 0; #Encountered lines before patch - my $commit_log_possible_stack_dump = 0; - my $commit_log_long_line = 0; - my $commit_log_has_diff = 0; - my $reported_maintainer_file = 0; - my $non_utf8_charset = 0; - - my $last_blank_line = 0; - my $last_coalesced_string_linenr = -1; - - our @report = (); - our $cnt_lines = 0; - our $cnt_error = 0; - our $cnt_warn = 0; - our $cnt_chk = 0; - - # Trace the real file/line as we go. - my $realfile = ''; - my $realline = 0; - my $realcnt = 0; - my $here = ''; - my $context_function; #undef'd unless there's a known function - my $in_comment = 0; - my $comment_edge = 0; - my $first_line = 0; - my $p1_prefix = ''; - - my $prev_values = 'E'; - - # suppression flags - my %suppress_ifbraces; - my %suppress_whiletrailers; - my %suppress_export; - my $suppress_statement = 0; - - my %signatures = (); - - # Pre-scan the patch sanitizing the lines. - # Pre-scan the patch looking for any __setup documentation. - # - my @setup_docs = (); - my $setup_docs = 0; - - my $camelcase_file_seeded = 0; - - sanitise_line_reset(); - my $line; - foreach my $rawline (@rawlines) { - $linenr++; - $line = $rawline; - - push(@fixed, $rawline) if ($fix); - - if ($rawline=~/^\+\+\+\s+(\S+)/) { - $setup_docs = 0; - if ($1 =~ m@Documentation/admin-guide/kernel-parameters.rst$@) { - $setup_docs = 1; - } - #next; - } - if ($rawline =~ /^\@\@ -\d+(?:,\d+)? \+(\d+)(,(\d+))? \@\@/) { - $realline=$1-1; - if (defined $2) { - $realcnt=$3+1; - } else { - $realcnt=1+1; - } - $in_comment = 0; - - # Guestimate if this is a continuing comment. Run - # the context looking for a comment "edge". If this - # edge is a close comment then we must be in a comment - # at context start. - my $edge; - my $cnt = $realcnt; - for (my $ln = $linenr + 1; $cnt > 0; $ln++) { - next if (defined $rawlines[$ln - 1] && - $rawlines[$ln - 1] =~ /^-/); - $cnt--; - #print "RAW<$rawlines[$ln - 1]>\n"; - last if (!defined $rawlines[$ln - 1]); - if ($rawlines[$ln - 1] =~ m@(/\*|\*/)@ && - $rawlines[$ln - 1] !~ m@"[^"]*(?:/\*|\*/)[^"]*"@) { - ($edge) = $1; - last; - } - } - if (defined $edge && $edge eq '*/') { - $in_comment = 1; - } - - # Guestimate if this is a continuing comment. If this - # is the start of a diff block and this line starts - # ' *' then it is very likely a comment. - if (!defined $edge && - $rawlines[$linenr] =~ m@^.\s*(?:\*\*+| \*)(?:\s|$)@) - { - $in_comment = 1; - } - - ##print "COMMENT:$in_comment edge<$edge> $rawline\n"; - sanitise_line_reset($in_comment); - - } elsif ($realcnt && $rawline =~ /^(?:\+| |$)/) { - # Standardise the strings and chars within the input to - # simplify matching -- only bother with positive lines. - $line = sanitise_line($rawline); - } - push(@lines, $line); - - if ($realcnt > 1) { - $realcnt-- if ($line =~ /^(?:\+| |$)/); - } else { - $realcnt = 0; - } - - #print "==>$rawline\n"; - #print "-->$line\n"; - - if ($setup_docs && $line =~ /^\+/) { - push(@setup_docs, $line); - } - } - - $prefix = ''; - - $realcnt = 0; - $linenr = 0; - $fixlinenr = -1; - foreach my $line (@lines) { - $linenr++; - $fixlinenr++; - my $sline = $line; #copy of $line - $sline =~ s/$;/ /g; #with comments as spaces - - my $rawline = $rawlines[$linenr - 1]; - -#extract the line range in the file after the patch is applied - if (!$in_commit_log && - $line =~ /^\@\@ -\d+(?:,\d+)? \+(\d+)(,(\d+))? \@\@(.*)/) { - my $context = $4; - $is_patch = 1; - $first_line = $linenr + 1; - $realline=$1-1; - if (defined $2) { - $realcnt=$3+1; - } else { - $realcnt=1+1; - } - annotate_reset(); - $prev_values = 'E'; - - %suppress_ifbraces = (); - %suppress_whiletrailers = (); - %suppress_export = (); - $suppress_statement = 0; - if ($context =~ /\b(\w+)\s*\(/) { - $context_function = $1; - } else { - undef $context_function; - } - next; - -# track the line number as we move through the hunk, note that -# new versions of GNU diff omit the leading space on completely -# blank context lines so we need to count that too. - } elsif ($line =~ /^( |\+|$)/) { - $realline++; - $realcnt-- if ($realcnt != 0); - - # Measure the line length and indent. - ($length, $indent) = line_stats($rawline); - - # Track the previous line. - ($prevline, $stashline) = ($stashline, $line); - ($previndent, $stashindent) = ($stashindent, $indent); - ($prevrawline, $stashrawline) = ($stashrawline, $rawline); - - #warn "line<$line>\n"; - - } elsif ($realcnt == 1) { - $realcnt--; - } - - my $hunk_line = ($realcnt != 0); - - $here = "#$linenr: " if (!$file); - $here = "#$realline: " if ($file); - - my $found_file = 0; - # extract the filename as it passes - if ($line =~ /^diff --git.*?(\S+)$/) { - $realfile = $1; - $realfile =~ s@^([^/]*)/@@ if (!$file); - $in_commit_log = 0; - $found_file = 1; - } elsif ($line =~ /^\+\+\+\s+(\S+)/) { - $realfile = $1; - $realfile =~ s@^([^/]*)/@@ if (!$file); - $in_commit_log = 0; - - $p1_prefix = $1; - if (!$file && $tree && $p1_prefix ne '' && - -e "$root/$p1_prefix") { - WARN("PATCH_PREFIX", - "patch prefix '$p1_prefix' exists, appears to be a -p0 patch\n"); - } - - if ($realfile =~ m@^include/asm/@) { - ERROR("MODIFIED_INCLUDE_ASM", - "do not modify files in include/asm, change architecture specific files in include/asm-\n" . "$here$rawline\n"); - } - $found_file = 1; - } - my $skipme = 0; - foreach (@exclude) { - if ($realfile =~ m@^(?:$_/)@) { - $skipme = 1; - } - } - if ($skipme) { - next; - } - -#make up the handle for any error we report on this line - if ($showfile) { - $prefix = "$realfile:$realline: " - } elsif ($emacs) { - if ($file) { - $prefix = "$filename:$realline: "; - } else { - $prefix = "$filename:$linenr: "; - } - } - - if ($found_file) { - if (is_maintained_obsolete($realfile)) { - WARN("OBSOLETE", - "$realfile is marked as 'obsolete' in the MAINTAINERS hierarchy. No unnecessary modifications please.\n"); - } - if ($realfile =~ m@^(?:drivers/net/|net/|drivers/staging/)@) { - $check = 1; - } else { - $check = $check_orig; - } - next; - } - - $here .= "FILE: $realfile:$realline:" if ($realcnt != 0); - - my $hereline = "$here\n$rawline\n"; - my $herecurr = "$here\n$rawline\n"; - my $hereprev = "$here\n$prevrawline\n$rawline\n"; - - $cnt_lines++ if ($realcnt != 0); - -# Check if the commit log has what seems like a diff which can confuse patch - if ($in_commit_log && !$commit_log_has_diff && - (($line =~ m@^\s+diff\b.*a/[\w/]+@ && - $line =~ m@^\s+diff\b.*a/([\w/]+)\s+b/$1\b@) || - $line =~ m@^\s*(?:\-\-\-\s+a/|\+\+\+\s+b/)@ || - $line =~ m/^\s*\@\@ \-\d+,\d+ \+\d+,\d+ \@\@/)) { - ERROR("DIFF_IN_COMMIT_MSG", - "Avoid using diff content in the commit message - patch(1) might not work\n" . $herecurr); - $commit_log_has_diff = 1; - } - -# Check for incorrect file permissions - if ($line =~ /^new (file )?mode.*[7531]\d{0,2}$/) { - my $permhere = $here . "FILE: $realfile\n"; - if ($realfile !~ m@scripts/@ && - $realfile !~ /\.(py|pl|awk|sh)$/) { - ERROR("EXECUTE_PERMISSIONS", - "do not set execute permissions for source files\n" . $permhere); - } - } - -# Check the patch for a signoff: - if ($line =~ /^\s*signed-off-by:/i) { - $signoff++; - $in_commit_log = 0; - } - -# Check if CODEOWNERS is being updated. If so, there's probably no need to -# emit the "does CODEOWNERS need updating?" message on file add/move/delete - if ($line =~ /^\s*CODEOWNERS\s*\|/) { - $reported_maintainer_file = 1; - } - -# Check signature styles - if (!$in_header_lines && - $line =~ /^(\s*)([a-z0-9_-]+by:|$signature_tags)(\s*)(.*)/i) { - my $space_before = $1; - my $sign_off = $2; - my $space_after = $3; - my $email = $4; - my $ucfirst_sign_off = ucfirst(lc($sign_off)); - - if ($sign_off !~ /$signature_tags/) { - WARN("BAD_SIGN_OFF", - "Non-standard signature: $sign_off\n" . $herecurr); - } - if (defined $space_before && $space_before ne "") { - if (WARN("BAD_SIGN_OFF", - "Do not use whitespace before $ucfirst_sign_off\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] = - "$ucfirst_sign_off $email"; - } - } - if ($sign_off =~ /-by:$/i && $sign_off ne $ucfirst_sign_off) { - if (WARN("BAD_SIGN_OFF", - "'$ucfirst_sign_off' is the preferred signature form\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] = - "$ucfirst_sign_off $email"; - } - - } - if (!defined $space_after || $space_after ne " ") { - if (WARN("BAD_SIGN_OFF", - "Use a single space after $ucfirst_sign_off\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] = - "$ucfirst_sign_off $email"; - } - } - - my ($email_name, $email_address, $comment) = parse_email($email); - my $suggested_email = format_email(($email_name, $email_address)); - if ($suggested_email eq "") { - ERROR("BAD_SIGN_OFF", - "Unrecognized email address: '$email'\n" . $herecurr); - } else { - my $dequoted = $suggested_email; - $dequoted =~ s/^"//; - $dequoted =~ s/" $comment" ne $email && - "$suggested_email$comment" ne $email) { - WARN("BAD_SIGN_OFF", - "email address '$email' might be better as '$suggested_email$comment'\n" . $herecurr); - } - } - -# Check for duplicate signatures - my $sig_nospace = $line; - $sig_nospace =~ s/\s//g; - $sig_nospace = lc($sig_nospace); - if (defined $signatures{$sig_nospace}) { - WARN("BAD_SIGN_OFF", - "Duplicate signature\n" . $herecurr); - } else { - $signatures{$sig_nospace} = 1; - } - } - -# Check email subject for common tools that don't need to be mentioned - if ($in_header_lines && - $line =~ /^Subject:.*\b(?:checkpatch|sparse|smatch)\b[^:]/i) { - WARN("EMAIL_SUBJECT", - "A patch subject line should describe the change not the tool that found it\n" . $herecurr); - } - -# Check for old stable address - if ($line =~ /^\s*cc:\s*.*?.*$/i) { - ERROR("STABLE_ADDRESS", - "The 'stable' address should be 'stable\@vger.kernel.org'\n" . $herecurr); - } - -# Check for unwanted Gerrit info - if ($in_commit_log && $line =~ /^\s*change-id:/i) { - ERROR("GERRIT_CHANGE_ID", - "Remove Gerrit Change-Id's before submitting upstream.\n" . $herecurr); - } - -# Check if the commit log is in a possible stack dump - if ($in_commit_log && !$commit_log_possible_stack_dump && - ($line =~ /^\s*(?:WARNING:|BUG:)/ || - $line =~ /^\s*\[\s*\d+\.\d{6,6}\s*\]/ || - # timestamp - $line =~ /^\s*\[\<[0-9a-fA-F]{8,}\>\]/)) { - # stack dump address - $commit_log_possible_stack_dump = 1; - } - -# Check for line lengths > 75 in commit log, warn once - if ($in_commit_log && !$commit_log_long_line && - length($line) > 75 && - !($line =~ /^\s*[a-zA-Z0-9_\/\.]+\s+\|\s+\d+/ || - # file delta changes - $line =~ /^\s*(?:[\w\.\-]+\/)++[\w\.\-]+:/ || - # filename then : - $line =~ /^\s*(?:Fixes:|Link:)/i || - # A Fixes: or Link: line - $commit_log_possible_stack_dump)) { - WARN("COMMIT_LOG_LONG_LINE", - "Possible unwrapped commit description (prefer a maximum 75 chars per line)\n" . $herecurr); - $commit_log_long_line = 1; - } - -# Reset possible stack dump if a blank line is found - if ($in_commit_log && $commit_log_possible_stack_dump && - $line =~ /^\s*$/) { - $commit_log_possible_stack_dump = 0; - } - -# Check for git id commit length and improperly formed commit descriptions - if ($in_commit_log && !$commit_log_possible_stack_dump && - $line !~ /^\s*(?:Link|Patchwork|http|https|BugLink):/i && - $line !~ /^This reverts commit [0-9a-f]{7,40}/ && - ($line =~ /\bcommit\s+[0-9a-f]{5,}\b/i || - ($line =~ /(?:\s|^)[0-9a-f]{12,40}(?:[\s"'\(\[]|$)/i && - $line !~ /[\<\[][0-9a-f]{12,40}[\>\]]/i && - $line !~ /\bfixes:\s*[0-9a-f]{12,40}/i))) { - my $init_char = "c"; - my $orig_commit = ""; - my $short = 1; - my $long = 0; - my $case = 1; - my $space = 1; - my $hasdesc = 0; - my $hasparens = 0; - my $id = '0123456789ab'; - my $orig_desc = "commit description"; - my $description = ""; - - if ($line =~ /\b(c)ommit\s+([0-9a-f]{5,})\b/i) { - $init_char = $1; - $orig_commit = lc($2); - } elsif ($line =~ /\b([0-9a-f]{12,40})\b/i) { - $orig_commit = lc($1); - } - - $short = 0 if ($line =~ /\bcommit\s+[0-9a-f]{12,40}/i); - $long = 1 if ($line =~ /\bcommit\s+[0-9a-f]{41,}/i); - $space = 0 if ($line =~ /\bcommit [0-9a-f]/i); - $case = 0 if ($line =~ /\b[Cc]ommit\s+[0-9a-f]{5,40}[^A-F]/); - if ($line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("([^"]+)"\)/i) { - $orig_desc = $1; - $hasparens = 1; - } elsif ($line =~ /\bcommit\s+[0-9a-f]{5,}\s*$/i && - defined $rawlines[$linenr] && - $rawlines[$linenr] =~ /^\s*\("([^"]+)"\)/) { - $orig_desc = $1; - $hasparens = 1; - } elsif ($line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("[^"]+$/i && - defined $rawlines[$linenr] && - $rawlines[$linenr] =~ /^\s*[^"]+"\)/) { - $line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("([^"]+)$/i; - $orig_desc = $1; - $rawlines[$linenr] =~ /^\s*([^"]+)"\)/; - $orig_desc .= " " . $1; - $hasparens = 1; - } - - ($id, $description) = git_commit_info($orig_commit, - $id, $orig_desc); - - if (defined($id) && - ($short || $long || $space || $case || ($orig_desc ne $description) || !$hasparens)) { - ERROR("GIT_COMMIT_ID", - "Please use git commit description style 'commit <12+ chars of sha1> (\"\")' - ie: '${init_char}ommit $id (\"$description\")'\n" . $herecurr); - } - } - -# Check for added, moved or deleted files - if (!$reported_maintainer_file && !$in_commit_log && - ($line =~ /^(?:new|deleted) file mode\s*\d+\s*$/ || - $line =~ /^rename (?:from|to) [\w\/\.\-]+\s*$/ || - ($line =~ /\{\s*([\w\/\.\-]*)\s*\=\>\s*([\w\/\.\-]*)\s*\}/ && - (defined($1) || defined($2))))) { - $is_patch = 1; - $reported_maintainer_file = 1; - WARN("FILE_PATH_CHANGES", - "added, moved or deleted file(s), does CODEOWNERS need updating?\n" . $herecurr); - } - -# Check for wrappage within a valid hunk of the file - if ($realcnt != 0 && $line !~ m{^(?:\+|-| |\\ No newline|$)}) { - ERROR("CORRUPTED_PATCH", - "patch seems to be corrupt (line wrapped?)\n" . - $herecurr) if (!$emitted_corrupt++); - } - -# UTF-8 regex found at http://www.w3.org/International/questions/qa-forms-utf-8.en.php - if (($realfile =~ /^$/ || $line =~ /^\+/) && - $rawline !~ m/^$UTF8*$/) { - my ($utf8_prefix) = ($rawline =~ /^($UTF8*)/); - - my $blank = copy_spacing($rawline); - my $ptr = substr($blank, 0, length($utf8_prefix)) . "^"; - my $hereptr = "$hereline$ptr\n"; - - CHK("INVALID_UTF8", - "Invalid UTF-8, patch and commit message should be encoded in UTF-8\n" . $hereptr); - } - -# Check if it's the start of a commit log -# (not a header line and we haven't seen the patch filename) - if ($in_header_lines && $realfile =~ /^$/ && - !($rawline =~ /^\s+(?:\S|$)/ || - $rawline =~ /^(?:commit\b|from\b|[\w-]+:)/i)) { - $in_header_lines = 0; - $in_commit_log = 1; - $has_commit_log = 1; - } - -# Check if there is UTF-8 in a commit log when a mail header has explicitly -# declined it, i.e defined some charset where it is missing. - if ($in_header_lines && - $rawline =~ /^Content-Type:.+charset="(.+)".*$/ && - $1 !~ /utf-8/i) { - $non_utf8_charset = 1; - } - - if ($in_commit_log && $non_utf8_charset && $realfile =~ /^$/ && - $rawline =~ /$NON_ASCII_UTF8/) { - WARN("UTF8_BEFORE_PATCH", - "8-bit UTF-8 used in possible commit log\n" . $herecurr); - } - -# Check for absolute kernel paths in commit message - if ($tree && $in_commit_log) { - while ($line =~ m{(?:^|\s)(/\S*)}g) { - my $file = $1; - - if ($file =~ m{^(.*?)(?::\d+)+:?$} && - check_absolute_file($1, $herecurr)) { - # - } else { - check_absolute_file($file, $herecurr); - } - } - } - -# Check for various typo / spelling mistakes - if (defined($misspellings) && - ($in_commit_log || $line =~ /^(?:\+|Subject:)/i)) { - while ($rawline =~ /(?:^|[^a-z@])($misspellings)(?:\b|$|[^a-z@])/gi) { - my $typo = $1; - my $typo_fix = $spelling_fix{lc($typo)}; - $typo_fix = ucfirst($typo_fix) if ($typo =~ /^[A-Z]/); - $typo_fix = uc($typo_fix) if ($typo =~ /^[A-Z]+$/); - my $msg_level = \&WARN; - $msg_level = \&CHK if ($file); - if (&{$msg_level}("TYPO_SPELLING", - "'$typo' may be misspelled - perhaps '$typo_fix'?\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/(^|[^A-Za-z@])($typo)($|[^A-Za-z@])/$1$typo_fix$3/; - } - } - } - -# ignore non-hunk lines and lines being removed - next if (!$hunk_line || $line =~ /^-/); - -#trailing whitespace - if ($line =~ /^\+.*\015/) { - my $herevet = "$here\n" . cat_vet($rawline) . "\n"; - if (ERROR("DOS_LINE_ENDINGS", - "DOS line endings\n" . $herevet) && - $fix) { - $fixed[$fixlinenr] =~ s/[\s\015]+$//; - } - } elsif ($rawline =~ /^\+.*\S\s+$/ || $rawline =~ /^\+\s+$/) { - my $herevet = "$here\n" . cat_vet($rawline) . "\n"; - if (ERROR("TRAILING_WHITESPACE", - "trailing whitespace\n" . $herevet) && - $fix) { - $fixed[$fixlinenr] =~ s/\s+$//; - } - - $rpt_cleaners = 1; - } - -# Check for FSF mailing addresses. - if ($rawline =~ /\bwrite to the Free/i || - $rawline =~ /\b675\s+Mass\s+Ave/i || - $rawline =~ /\b59\s+Temple\s+Pl/i || - $rawline =~ /\b51\s+Franklin\s+St/i) { - my $herevet = "$here\n" . cat_vet($rawline) . "\n"; - my $msg_level = \&ERROR; - $msg_level = \&CHK if ($file); - &{$msg_level}("FSF_MAILING_ADDRESS", - "Do not include the paragraph about writing to the Free Software Foundation's mailing address from the sample GPL notice. The FSF has changed addresses in the past, and may do so again. Linux already includes a copy of the GPL.\n" . $herevet) - } - -# check for Kconfig help text having a real description -# Only applies when adding the entry originally, after that we do not have -# sufficient context to determine whether it is indeed long enough. - if ($realfile =~ /Kconfig/ && - $line =~ /^\+\s*config\s+/) { - my $length = 0; - my $cnt = $realcnt; - my $ln = $linenr + 1; - my $f; - my $is_start = 0; - my $is_end = 0; - for (; $cnt > 0 && defined $lines[$ln - 1]; $ln++) { - $f = $lines[$ln - 1]; - $cnt-- if ($lines[$ln - 1] !~ /^-/); - $is_end = $lines[$ln - 1] =~ /^\+/; - - next if ($f =~ /^-/); - last if (!$file && $f =~ /^\@\@/); - - if ($lines[$ln - 1] =~ /^\+\s*(?:bool|tristate)\s*\"/) { - $is_start = 1; - } elsif ($lines[$ln - 1] =~ /^\+\s*(?:---)?help(?:---)?$/) { - $length = -1; - } - - $f =~ s/^.//; - $f =~ s/#.*//; - $f =~ s/^\s+//; - next if ($f =~ /^$/); - if ($f =~ /^\s*config\s/) { - $is_end = 1; - last; - } - $length++; - } - if ($is_start && $is_end && $length < $min_conf_desc_length) { - WARN("CONFIG_DESCRIPTION", - "please write a paragraph that describes the config symbol fully\n" . $herecurr); - } - #print "is_start<$is_start> is_end<$is_end> length<$length>\n"; - } - -# check for MAINTAINERS entries that don't have the right form - if ($realfile =~ /^MAINTAINERS$/ && - $rawline =~ /^\+[A-Z]:/ && - $rawline !~ /^\+[A-Z]:\t\S/) { - if (WARN("MAINTAINERS_STYLE", - "MAINTAINERS entries use one tab after TYPE:\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/^(\+[A-Z]):\s*/$1:\t/; - } - } - -# discourage the use of boolean for type definition attributes of Kconfig options - if ($realfile =~ /Kconfig/ && - $line =~ /^\+\s*\bboolean\b/) { - WARN("CONFIG_TYPE_BOOLEAN", - "Use of boolean is deprecated, please use bool instead.\n" . $herecurr); - } - - if (($realfile =~ /Makefile.*/ || $realfile =~ /Kbuild.*/) && - ($line =~ /\+(EXTRA_[A-Z]+FLAGS).*/)) { - my $flag = $1; - my $replacement = { - 'EXTRA_AFLAGS' => 'asflags-y', - 'EXTRA_CFLAGS' => 'ccflags-y', - 'EXTRA_CPPFLAGS' => 'cppflags-y', - 'EXTRA_LDFLAGS' => 'ldflags-y', - }; - - WARN("DEPRECATED_VARIABLE", - "Use of $flag is deprecated, please use \`$replacement->{$flag} instead.\n" . $herecurr) if ($replacement->{$flag}); - } -# Kconfig use tabs and no spaces in line - if ($realfile =~ /Kconfig/ && $rawline =~ /^\+ /) { - my $herevet = "$here\n" . cat_vet($rawline) . "\n"; - WARN("LEADING_SPACE", - "please, no spaces at the start of a line\n" . $herevet); - } - -# check for DT compatible documentation - if (defined $root && - (($realfile =~ /\.dtsi?$/ && $line =~ /^\+\s*compatible\s*=\s*\"/) || - ($realfile =~ /\.[ch]$/ && $line =~ /^\+.*\.compatible\s*=\s*\"/))) { - - my @compats = $rawline =~ /\"([a-zA-Z0-9\-\,\.\+_]+)\"/g; - - my $dt_path = $root . "/dts/bindings/"; - my $vp_file = $dt_path . "vendor-prefixes.txt"; - - foreach my $compat (@compats) { - my $compat2 = $compat; - $compat2 =~ s/\,[a-zA-Z0-9]*\-/\,<\.\*>\-/; - my $compat3 = $compat; - $compat3 =~ s/\,([a-z]*)[0-9]*\-/\,$1<\.\*>\-/; - `grep -Erq "$compat|$compat2|$compat3" $dt_path`; - if ( $? >> 8 ) { - WARN("UNDOCUMENTED_DT_STRING", - "DT compatible string \"$compat\" appears un-documented -- check $dt_path\n" . $herecurr); - } - - next if $compat !~ /^([a-zA-Z0-9\-]+)\,/; - my $vendor = $1; - `grep -Eq "^$vendor\\b" $vp_file`; - if ( $? >> 8 ) { - WARN("UNDOCUMENTED_DT_STRING", - "DT compatible string vendor \"$vendor\" appears un-documented -- check $vp_file\n" . $herecurr); - } - } - } - -# check we are in a valid source file if not then ignore this hunk - next if ($realfile !~ /\.(h|c|s|S|sh|dtsi|dts)$/); - -# line length limit (with some exclusions) -# -# There are a few types of lines that may extend beyond $max_line_length: -# logging functions like pr_info that end in a string -# lines with a single string -# #defines that are a single string -# -# There are 3 different line length message types: -# LONG_LINE_COMMENT a comment starts before but extends beyond $max_line_length -# LONG_LINE_STRING a string starts before but extends beyond $max_line_length -# LONG_LINE all other lines longer than $max_line_length -# -# if LONG_LINE is ignored, the other 2 types are also ignored -# - - if ($line =~ /^\+/ && $length > $max_line_length) { - my $msg_type = "LONG_LINE"; - - # Check the allowed long line types first - - # logging functions that end in a string that starts - # before $max_line_length - if ($line =~ /^\+\s*$logFunctions\s*\(\s*(?:(?:KERN_\S+\s*|[^"]*))?($String\s*(?:|,|\)\s*;)\s*)$/ && - length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) { - $msg_type = ""; - - # lines with only strings (w/ possible termination) - # #defines with only strings - } elsif ($line =~ /^\+\s*$String\s*(?:\s*|,|\)\s*;)\s*$/ || - $line =~ /^\+\s*#\s*define\s+\w+\s+$String$/) { - $msg_type = ""; - - # EFI_GUID is another special case - } elsif ($line =~ /^\+.*\bEFI_GUID\s*\(/) { - $msg_type = ""; - - # Otherwise set the alternate message types - - # a comment starts before $max_line_length - } elsif ($line =~ /($;[\s$;]*)$/ && - length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) { - $msg_type = "LONG_LINE_COMMENT" - - # a quoted string starts before $max_line_length - } elsif ($sline =~ /\s*($String(?:\s*(?:\\|,\s*|\)\s*;\s*))?)$/ && - length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) { - $msg_type = "LONG_LINE_STRING" - } - - if ($msg_type ne "" && - (show_type("LONG_LINE") || show_type($msg_type))) { - WARN($msg_type, - "line over $max_line_length characters\n" . $herecurr); - } - } - -# check for adding lines without a newline. - if ($line =~ /^\+/ && defined $lines[$linenr] && $lines[$linenr] =~ /^\\ No newline at end of file/) { - WARN("MISSING_EOF_NEWLINE", - "adding a line without newline at end of file\n" . $herecurr); - } - -# Blackfin: use hi/lo macros - if ($realfile =~ m@arch/blackfin/.*\.S$@) { - if ($line =~ /\.[lL][[:space:]]*=.*&[[:space:]]*0x[fF][fF][fF][fF]/) { - my $herevet = "$here\n" . cat_vet($line) . "\n"; - ERROR("LO_MACRO", - "use the LO() macro, not (... & 0xFFFF)\n" . $herevet); - } - if ($line =~ /\.[hH][[:space:]]*=.*>>[[:space:]]*16/) { - my $herevet = "$here\n" . cat_vet($line) . "\n"; - ERROR("HI_MACRO", - "use the HI() macro, not (... >> 16)\n" . $herevet); - } - } - -# check we are in a valid source file C or perl if not then ignore this hunk - next if ($realfile !~ /\.(h|c|pl|dtsi|dts)$/); - -# at the beginning of a line any tabs must come first and anything -# more than 8 must use tabs. - if ($rawline =~ /^\+\s* \t\s*\S/ || - $rawline =~ /^\+\s* \s*/) { - my $herevet = "$here\n" . cat_vet($rawline) . "\n"; - $rpt_cleaners = 1; - if (ERROR("CODE_INDENT", - "code indent should use tabs where possible\n" . $herevet) && - $fix) { - $fixed[$fixlinenr] =~ s/^\+([ \t]+)/"\+" . tabify($1)/e; - } - } - -# check for space before tabs. - if ($rawline =~ /^\+/ && $rawline =~ / \t/) { - my $herevet = "$here\n" . cat_vet($rawline) . "\n"; - if (WARN("SPACE_BEFORE_TAB", - "please, no space before tabs\n" . $herevet) && - $fix) { - while ($fixed[$fixlinenr] =~ - s/(^\+.*) {8,8}\t/$1\t\t/) {} - while ($fixed[$fixlinenr] =~ - s/(^\+.*) +\t/$1\t/) {} - } - } - -# check for && or || at the start of a line - if ($rawline =~ /^\+\s*(&&|\|\|)/) { - CHK("LOGICAL_CONTINUATIONS", - "Logical continuations should be on the previous line\n" . $hereprev); - } - -# check indentation starts on a tab stop - if ($^V && $^V ge 5.10.0 && - $sline =~ /^\+\t+( +)(?:$c90_Keywords\b|\{\s*$|\}\s*(?:else\b|while\b|\s*$))/) { - my $indent = length($1); - if ($indent % 8) { - if (WARN("TABSTOP", - "Statements should start on a tabstop\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s@(^\+\t+) +@$1 . "\t" x ($indent/8)@e; - } - } - } - -# check multi-line statement indentation matches previous line - if ($^V && $^V ge 5.10.0 && - $prevline =~ /^\+([ \t]*)((?:$c90_Keywords(?:\s+if)\s*)|(?:$Declare\s*)?(?:$Ident|\(\s*\*\s*$Ident\s*\))\s*|(?:\*\s*)*$Lval\s*=\s*$Ident\s*)\(.*(\&\&|\|\||,)\s*$/) { - $prevline =~ /^\+(\t*)(.*)$/; - my $oldindent = $1; - my $rest = $2; - - my $pos = pos_last_openparen($rest); - if ($pos >= 0) { - $line =~ /^(\+| )([ \t]*)/; - my $newindent = $2; - - my $goodtabindent = $oldindent . - "\t" x ($pos / 8) . - " " x ($pos % 8); - my $goodspaceindent = $oldindent . " " x $pos; - - if ($newindent ne $goodtabindent && - $newindent ne $goodspaceindent) { - - if (CHK("PARENTHESIS_ALIGNMENT", - "Alignment should match open parenthesis\n" . $hereprev) && - $fix && $line =~ /^\+/) { - $fixed[$fixlinenr] =~ - s/^\+[ \t]*/\+$goodtabindent/; - } - } - } - } - -# check for space after cast like "(int) foo" or "(struct foo) bar" -# avoid checking a few false positives: -# "sizeof(<type>)" or "__alignof__(<type>)" -# function pointer declarations like "(*foo)(int) = bar;" -# structure definitions like "(struct foo) { 0 };" -# multiline macros that define functions -# known attributes or the __attribute__ keyword - if ($line =~ /^\+(.*)\(\s*$Type\s*\)([ \t]++)((?![={]|\\$|$Attribute|__attribute__))/ && - (!defined($1) || $1 !~ /\b(?:sizeof|__alignof__)\s*$/)) { - if (CHK("SPACING", - "No space is necessary after a cast\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/(\(\s*$Type\s*\))[ \t]+/$1/; - } - } - -# Block comment styles -# Networking with an initial /* - if ($realfile =~ m@^(drivers/net/|net/)@ && - $prevrawline =~ /^\+[ \t]*\/\*[ \t]*$/ && - $rawline =~ /^\+[ \t]*\*/ && - $realline > 2) { - WARN("NETWORKING_BLOCK_COMMENT_STYLE", - "networking block comments don't use an empty /* line, use /* Comment...\n" . $hereprev); - } - -# Block comments use * on subsequent lines - if ($prevline =~ /$;[ \t]*$/ && #ends in comment - $prevrawline =~ /^\+.*?\/\*/ && #starting /* - $prevrawline !~ /\*\/[ \t]*$/ && #no trailing */ - $rawline =~ /^\+/ && #line is new - $rawline !~ /^\+[ \t]*\*/) { #no leading * - WARN("BLOCK_COMMENT_STYLE", - "Block comments use * on subsequent lines\n" . $hereprev); - } - -# Block comments use */ on trailing lines - if ($rawline !~ m@^\+[ \t]*\*/[ \t]*$@ && #trailing */ - $rawline !~ m@^\+.*/\*.*\*/[ \t]*$@ && #inline /*...*/ - $rawline !~ m@^\+.*\*{2,}/[ \t]*$@ && #trailing **/ - $rawline =~ m@^\+[ \t]*.+\*\/[ \t]*$@) { #non blank */ - WARN("BLOCK_COMMENT_STYLE", - "Block comments use a trailing */ on a separate line\n" . $herecurr); - } - -# Block comment * alignment - if ($prevline =~ /$;[ \t]*$/ && #ends in comment - $line =~ /^\+[ \t]*$;/ && #leading comment - $rawline =~ /^\+[ \t]*\*/ && #leading * - (($prevrawline =~ /^\+.*?\/\*/ && #leading /* - $prevrawline !~ /\*\/[ \t]*$/) || #no trailing */ - $prevrawline =~ /^\+[ \t]*\*/)) { #leading * - my $oldindent; - $prevrawline =~ m@^\+([ \t]*/?)\*@; - if (defined($1)) { - $oldindent = expand_tabs($1); - } else { - $prevrawline =~ m@^\+(.*/?)\*@; - $oldindent = expand_tabs($1); - } - $rawline =~ m@^\+([ \t]*)\*@; - my $newindent = $1; - $newindent = expand_tabs($newindent); - if (length($oldindent) ne length($newindent)) { - WARN("BLOCK_COMMENT_STYLE", - "Block comments should align the * on each line\n" . $hereprev); - } - } - -# check for missing blank lines after struct/union declarations -# with exceptions for various attributes and macros - if ($prevline =~ /^[\+ ]};?\s*$/ && - $line =~ /^\+/ && - !($line =~ /^\+\s*$/ || - $line =~ /^\+\s*EXPORT_SYMBOL/ || - $line =~ /^\+\s*MODULE_/i || - $line =~ /^\+\s*\#\s*(?:end|elif|else)/ || - $line =~ /^\+[a-z_]*init/ || - $line =~ /^\+\s*(?:static\s+)?[A-Z_]*ATTR/ || - $line =~ /^\+\s*DECLARE/ || - $line =~ /^\+\s*__setup/)) { - if (CHK("LINE_SPACING", - "Please use a blank line after function/struct/union/enum declarations\n" . $hereprev) && - $fix) { - fix_insert_line($fixlinenr, "\+"); - } - } - -# check for multiple consecutive blank lines - if ($prevline =~ /^[\+ ]\s*$/ && - $line =~ /^\+\s*$/ && - $last_blank_line != ($linenr - 1)) { - if (CHK("LINE_SPACING", - "Please don't use multiple blank lines\n" . $hereprev) && - $fix) { - fix_delete_line($fixlinenr, $rawline); - } - - $last_blank_line = $linenr; - } - -# check for missing blank lines after declarations - if ($sline =~ /^\+\s+\S/ && #Not at char 1 - # actual declarations - ($prevline =~ /^\+\s+$Declare\s*$Ident\s*[=,;:\[]/ || - # function pointer declarations - $prevline =~ /^\+\s+$Declare\s*\(\s*\*\s*$Ident\s*\)\s*[=,;:\[\(]/ || - # foo bar; where foo is some local typedef or #define - $prevline =~ /^\+\s+$Ident(?:\s+|\s*\*\s*)$Ident\s*[=,;\[]/ || - # known declaration macros - $prevline =~ /^\+\s+$declaration_macros/) && - # for "else if" which can look like "$Ident $Ident" - !($prevline =~ /^\+\s+$c90_Keywords\b/ || - # other possible extensions of declaration lines - $prevline =~ /(?:$Compare|$Assignment|$Operators)\s*$/ || - # not starting a section or a macro "\" extended line - $prevline =~ /(?:\{\s*|\\)$/) && - # looks like a declaration - !($sline =~ /^\+\s+$Declare\s*$Ident\s*[=,;:\[]/ || - # function pointer declarations - $sline =~ /^\+\s+$Declare\s*\(\s*\*\s*$Ident\s*\)\s*[=,;:\[\(]/ || - # foo bar; where foo is some local typedef or #define - $sline =~ /^\+\s+(?:volatile\s+)?$Ident(?:\s+|\s*\*\s*)$Ident\s*[=,;\[]/ || - # known declaration macros - $sline =~ /^\+\s+$declaration_macros/ || - # start of struct or union or enum - $sline =~ /^\+\s+(?:union|struct|enum|typedef)\b/ || - # start or end of block or continuation of declaration - $sline =~ /^\+\s+(?:$|[\{\}\.\#\"\?\:\(\[])/ || - # bitfield continuation - $sline =~ /^\+\s+$Ident\s*:\s*\d+\s*[,;]/ || - # other possible extensions of declaration lines - $sline =~ /^\+\s+\(?\s*(?:$Compare|$Assignment|$Operators)/) && - # indentation of previous and current line are the same - (($prevline =~ /\+(\s+)\S/) && $sline =~ /^\+$1\S/)) { - if (WARN("LINE_SPACING", - "Missing a blank line after declarations\n" . $hereprev) && - $fix) { - fix_insert_line($fixlinenr, "\+"); - } - } - -# check for spaces at the beginning of a line. -# Exceptions: -# 1) within comments -# 2) indented preprocessor commands -# 3) hanging labels - if ($rawline =~ /^\+ / && $line !~ /^\+ *(?:$;|#|$Ident:)/) { - my $herevet = "$here\n" . cat_vet($rawline) . "\n"; - if (WARN("LEADING_SPACE", - "please, no spaces at the start of a line\n" . $herevet) && - $fix) { - $fixed[$fixlinenr] =~ s/^\+([ \t]+)/"\+" . tabify($1)/e; - } - } - -# check we are in a valid C source file if not then ignore this hunk - next if ($realfile !~ /\.(h|c)$/); - -# check if this appears to be the start function declaration, save the name - if ($sline =~ /^\+\{\s*$/ && - $prevline =~ /^\+(?:(?:(?:$Storage|$Inline)\s*)*\s*$Type\s*)?($Ident)\(/) { - $context_function = $1; - } - -# check if this appears to be the end of function declaration - if ($sline =~ /^\+\}\s*$/) { - undef $context_function; - } - -# check indentation of any line with a bare else -# (but not if it is a multiple line "if (foo) return bar; else return baz;") -# if the previous line is a break or return and is indented 1 tab more... - if ($sline =~ /^\+([\t]+)(?:}[ \t]*)?else(?:[ \t]*{)?\s*$/) { - my $tabs = length($1) + 1; - if ($prevline =~ /^\+\t{$tabs,$tabs}break\b/ || - ($prevline =~ /^\+\t{$tabs,$tabs}return\b/ && - defined $lines[$linenr] && - $lines[$linenr] !~ /^[ \+]\t{$tabs,$tabs}return/)) { - WARN("UNNECESSARY_ELSE", - "else is not generally useful after a break or return\n" . $hereprev); - } - } - -# check indentation of a line with a break; -# if the previous line is a goto or return and is indented the same # of tabs - if ($sline =~ /^\+([\t]+)break\s*;\s*$/) { - my $tabs = $1; - if ($prevline =~ /^\+$tabs(?:goto|return)\b/) { - WARN("UNNECESSARY_BREAK", - "break is not useful after a goto or return\n" . $hereprev); - } - } - -# check for RCS/CVS revision markers - if ($rawline =~ /^\+.*\$(Revision|Log|Id)(?:\$|)/) { - WARN("CVS_KEYWORD", - "CVS style keyword markers, these will _not_ be updated\n". $herecurr); - } - -# Blackfin: don't use __builtin_bfin_[cs]sync - if ($line =~ /__builtin_bfin_csync/) { - my $herevet = "$here\n" . cat_vet($line) . "\n"; - ERROR("CSYNC", - "use the CSYNC() macro in asm/blackfin.h\n" . $herevet); - } - if ($line =~ /__builtin_bfin_ssync/) { - my $herevet = "$here\n" . cat_vet($line) . "\n"; - ERROR("SSYNC", - "use the SSYNC() macro in asm/blackfin.h\n" . $herevet); - } - -# check for old HOTPLUG __dev<foo> section markings - if ($line =~ /\b(__dev(init|exit)(data|const|))\b/) { - WARN("HOTPLUG_SECTION", - "Using $1 is unnecessary\n" . $herecurr); - } - -# Check for potential 'bare' types - my ($stat, $cond, $line_nr_next, $remain_next, $off_next, - $realline_next); -#print "LINE<$line>\n"; - if ($linenr > $suppress_statement && - $realcnt && $sline =~ /.\s*\S/) { - ($stat, $cond, $line_nr_next, $remain_next, $off_next) = - ctx_statement_block($linenr, $realcnt, 0); - $stat =~ s/\n./\n /g; - $cond =~ s/\n./\n /g; - -#print "linenr<$linenr> <$stat>\n"; - # If this statement has no statement boundaries within - # it there is no point in retrying a statement scan - # until we hit end of it. - my $frag = $stat; $frag =~ s/;+\s*$//; - if ($frag !~ /(?:{|;)/) { -#print "skip<$line_nr_next>\n"; - $suppress_statement = $line_nr_next; - } - - # Find the real next line. - $realline_next = $line_nr_next; - if (defined $realline_next && - (!defined $lines[$realline_next - 1] || - substr($lines[$realline_next - 1], $off_next) =~ /^\s*$/)) { - $realline_next++; - } - - my $s = $stat; - $s =~ s/{.*$//s; - - # Ignore goto labels. - if ($s =~ /$Ident:\*$/s) { - - # Ignore functions being called - } elsif ($s =~ /^.\s*$Ident\s*\(/s) { - - } elsif ($s =~ /^.\s*else\b/s) { - - # declarations always start with types - } elsif ($prev_values eq 'E' && $s =~ /^.\s*(?:$Storage\s+)?(?:$Inline\s+)?(?:const\s+)?((?:\s*$Ident)+?)\b(?:\s+$Sparse)?\s*\**\s*(?:$Ident|\(\*[^\)]*\))(?:\s*$Modifier)?\s*(?:;|=|,|\()/s) { - my $type = $1; - $type =~ s/\s+/ /g; - possible($type, "A:" . $s); - - # definitions in global scope can only start with types - } elsif ($s =~ /^.(?:$Storage\s+)?(?:$Inline\s+)?(?:const\s+)?($Ident)\b\s*(?!:)/s) { - possible($1, "B:" . $s); - } - - # any (foo ... *) is a pointer cast, and foo is a type - while ($s =~ /\(($Ident)(?:\s+$Sparse)*[\s\*]+\s*\)/sg) { - possible($1, "C:" . $s); - } - - # Check for any sort of function declaration. - # int foo(something bar, other baz); - # void (*store_gdt)(x86_descr_ptr *); - if ($prev_values eq 'E' && $s =~ /^(.(?:typedef\s*)?(?:(?:$Storage|$Inline)\s*)*\s*$Type\s*(?:\b$Ident|\(\*\s*$Ident\))\s*)\(/s) { - my ($name_len) = length($1); - - my $ctx = $s; - substr($ctx, 0, $name_len + 1, ''); - $ctx =~ s/\)[^\)]*$//; - - for my $arg (split(/\s*,\s*/, $ctx)) { - if ($arg =~ /^(?:const\s+)?($Ident)(?:\s+$Sparse)*\s*\**\s*(:?\b$Ident)?$/s || $arg =~ /^($Ident)$/s) { - - possible($1, "D:" . $s); - } - } - } - - } - -# -# Checks which may be anchored in the context. -# - -# Check for switch () and associated case and default -# statements should be at the same indent. - if ($line=~/\bswitch\s*\(.*\)/) { - my $err = ''; - my $sep = ''; - my @ctx = ctx_block_outer($linenr, $realcnt); - shift(@ctx); - for my $ctx (@ctx) { - my ($clen, $cindent) = line_stats($ctx); - if ($ctx =~ /^\+\s*(case\s+|default:)/ && - $indent != $cindent) { - $err .= "$sep$ctx\n"; - $sep = ''; - } else { - $sep = "[...]\n"; - } - } - if ($err ne '') { - ERROR("SWITCH_CASE_INDENT_LEVEL", - "switch and case should be at the same indent\n$hereline$err"); - } - } - -# if/while/etc brace do not go on next line, unless defining a do while loop, -# or if that brace on the next line is for something else - if ($line =~ /(.*)\b((?:if|while|for|switch|(?:[a-z_]+|)for_each[a-z_]+)\s*\(|do\b|else\b)/ && $line !~ /^.\s*\#/) { - my $pre_ctx = "$1$2"; - - my ($level, @ctx) = ctx_statement_level($linenr, $realcnt, 0); - - if ($line =~ /^\+\t{6,}/) { - WARN("DEEP_INDENTATION", - "Too many leading tabs - consider code refactoring\n" . $herecurr); - } - - my $ctx_cnt = $realcnt - $#ctx - 1; - my $ctx = join("\n", @ctx); - - my $ctx_ln = $linenr; - my $ctx_skip = $realcnt; - - while ($ctx_skip > $ctx_cnt || ($ctx_skip == $ctx_cnt && - defined $lines[$ctx_ln - 1] && - $lines[$ctx_ln - 1] =~ /^-/)) { - ##print "SKIP<$ctx_skip> CNT<$ctx_cnt>\n"; - $ctx_skip-- if (!defined $lines[$ctx_ln - 1] || $lines[$ctx_ln - 1] !~ /^-/); - $ctx_ln++; - } - - #print "realcnt<$realcnt> ctx_cnt<$ctx_cnt>\n"; - #print "pre<$pre_ctx>\nline<$line>\nctx<$ctx>\nnext<$lines[$ctx_ln - 1]>\n"; - - if ($ctx !~ /{\s*/ && defined($lines[$ctx_ln - 1]) && $lines[$ctx_ln - 1] =~ /^\+\s*{/) { - ERROR("OPEN_BRACE", - "that open brace { should be on the previous line\n" . - "$here\n$ctx\n$rawlines[$ctx_ln - 1]\n"); - } - if ($level == 0 && $pre_ctx !~ /}\s*while\s*\($/ && - $ctx =~ /\)\s*\;\s*$/ && - defined $lines[$ctx_ln - 1]) - { - my ($nlength, $nindent) = line_stats($lines[$ctx_ln - 1]); - if ($nindent > $indent) { - WARN("TRAILING_SEMICOLON", - "trailing semicolon indicates no statements, indent implies otherwise\n" . - "$here\n$ctx\n$rawlines[$ctx_ln - 1]\n"); - } - } - } - -# Check relative indent for conditionals and blocks. - if ($line =~ /\b(?:(?:if|while|for|(?:[a-z_]+|)for_each[a-z_]+)\s*\(|(?:do|else)\b)/ && $line !~ /^.\s*#/ && $line !~ /\}\s*while\s*/) { - ($stat, $cond, $line_nr_next, $remain_next, $off_next) = - ctx_statement_block($linenr, $realcnt, 0) - if (!defined $stat); - my ($s, $c) = ($stat, $cond); - - substr($s, 0, length($c), ''); - - # remove inline comments - $s =~ s/$;/ /g; - $c =~ s/$;/ /g; - - # Find out how long the conditional actually is. - my @newlines = ($c =~ /\n/gs); - my $cond_lines = 1 + $#newlines; - - # Make sure we remove the line prefixes as we have - # none on the first line, and are going to readd them - # where necessary. - $s =~ s/\n./\n/gs; - while ($s =~ /\n\s+\\\n/) { - $cond_lines += $s =~ s/\n\s+\\\n/\n/g; - } - - # We want to check the first line inside the block - # starting at the end of the conditional, so remove: - # 1) any blank line termination - # 2) any opening brace { on end of the line - # 3) any do (...) { - my $continuation = 0; - my $check = 0; - $s =~ s/^.*\bdo\b//; - $s =~ s/^\s*{//; - if ($s =~ s/^\s*\\//) { - $continuation = 1; - } - if ($s =~ s/^\s*?\n//) { - $check = 1; - $cond_lines++; - } - - # Also ignore a loop construct at the end of a - # preprocessor statement. - if (($prevline =~ /^.\s*#\s*define\s/ || - $prevline =~ /\\\s*$/) && $continuation == 0) { - $check = 0; - } - - my $cond_ptr = -1; - $continuation = 0; - while ($cond_ptr != $cond_lines) { - $cond_ptr = $cond_lines; - - # If we see an #else/#elif then the code - # is not linear. - if ($s =~ /^\s*\#\s*(?:else|elif)/) { - $check = 0; - } - - # Ignore: - # 1) blank lines, they should be at 0, - # 2) preprocessor lines, and - # 3) labels. - if ($continuation || - $s =~ /^\s*?\n/ || - $s =~ /^\s*#\s*?/ || - $s =~ /^\s*$Ident\s*:/) { - $continuation = ($s =~ /^.*?\\\n/) ? 1 : 0; - if ($s =~ s/^.*?\n//) { - $cond_lines++; - } - } - } - - my (undef, $sindent) = line_stats("+" . $s); - my $stat_real = raw_line($linenr, $cond_lines); - - # Check if either of these lines are modified, else - # this is not this patch's fault. - if (!defined($stat_real) || - $stat !~ /^\+/ && $stat_real !~ /^\+/) { - $check = 0; - } - if (defined($stat_real) && $cond_lines > 1) { - $stat_real = "[...]\n$stat_real"; - } - - #print "line<$line> prevline<$prevline> indent<$indent> sindent<$sindent> check<$check> continuation<$continuation> s<$s> cond_lines<$cond_lines> stat_real<$stat_real> stat<$stat>\n"; - - if ($check && $s ne '' && - (($sindent % 8) != 0 || - ($sindent < $indent) || - ($sindent == $indent && - ($s !~ /^\s*(?:\}|\{|else\b)/)) || - ($sindent > $indent + 8))) { - WARN("SUSPECT_CODE_INDENT", - "suspect code indent for conditional statements ($indent, $sindent)\n" . $herecurr . "$stat_real\n"); - } - } - - # Track the 'values' across context and added lines. - my $opline = $line; $opline =~ s/^./ /; - my ($curr_values, $curr_vars) = - annotate_values($opline . "\n", $prev_values); - $curr_values = $prev_values . $curr_values; - if ($dbg_values) { - my $outline = $opline; $outline =~ s/\t/ /g; - print "$linenr > .$outline\n"; - print "$linenr > $curr_values\n"; - print "$linenr > $curr_vars\n"; - } - $prev_values = substr($curr_values, -1); - -#ignore lines not being added - next if ($line =~ /^[^\+]/); - -# check for dereferences that span multiple lines - if ($prevline =~ /^\+.*$Lval\s*(?:\.|->)\s*$/ && - $line =~ /^\+\s*(?!\#\s*(?!define\s+|if))\s*$Lval/) { - $prevline =~ /($Lval\s*(?:\.|->))\s*$/; - my $ref = $1; - $line =~ /^.\s*($Lval)/; - $ref .= $1; - $ref =~ s/\s//g; - WARN("MULTILINE_DEREFERENCE", - "Avoid multiple line dereference - prefer '$ref'\n" . $hereprev); - } - -# check for declarations of signed or unsigned without int - while ($line =~ m{\b($Declare)\s*(?!char\b|short\b|int\b|long\b)\s*($Ident)?\s*[=,;\[\)\(]}g) { - my $type = $1; - my $var = $2; - $var = "" if (!defined $var); - if ($type =~ /^(?:(?:$Storage|$Inline|$Attribute)\s+)*((?:un)?signed)((?:\s*\*)*)\s*$/) { - my $sign = $1; - my $pointer = $2; - - $pointer = "" if (!defined $pointer); - - if (WARN("UNSPECIFIED_INT", - "Prefer '" . trim($sign) . " int" . rtrim($pointer) . "' to bare use of '$sign" . rtrim($pointer) . "'\n" . $herecurr) && - $fix) { - my $decl = trim($sign) . " int "; - my $comp_pointer = $pointer; - $comp_pointer =~ s/\s//g; - $decl .= $comp_pointer; - $decl = rtrim($decl) if ($var eq ""); - $fixed[$fixlinenr] =~ s@\b$sign\s*\Q$pointer\E\s*$var\b@$decl$var@; - } - } - } - -# TEST: allow direct testing of the type matcher. - if ($dbg_type) { - if ($line =~ /^.\s*$Declare\s*$/) { - ERROR("TEST_TYPE", - "TEST: is type\n" . $herecurr); - } elsif ($dbg_type > 1 && $line =~ /^.+($Declare)/) { - ERROR("TEST_NOT_TYPE", - "TEST: is not type ($1 is)\n". $herecurr); - } - next; - } -# TEST: allow direct testing of the attribute matcher. - if ($dbg_attr) { - if ($line =~ /^.\s*$Modifier\s*$/) { - ERROR("TEST_ATTR", - "TEST: is attr\n" . $herecurr); - } elsif ($dbg_attr > 1 && $line =~ /^.+($Modifier)/) { - ERROR("TEST_NOT_ATTR", - "TEST: is not attr ($1 is)\n". $herecurr); - } - next; - } - -# check for initialisation to aggregates open brace on the next line - if ($line =~ /^.\s*{/ && - $prevline =~ /(?:^|[^=])=\s*$/) { - if (ERROR("OPEN_BRACE", - "that open brace { should be on the previous line\n" . $hereprev) && - $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { - fix_delete_line($fixlinenr - 1, $prevrawline); - fix_delete_line($fixlinenr, $rawline); - my $fixedline = $prevrawline; - $fixedline =~ s/\s*=\s*$/ = {/; - fix_insert_line($fixlinenr, $fixedline); - $fixedline = $line; - $fixedline =~ s/^(.\s*)\{\s*/$1/; - fix_insert_line($fixlinenr, $fixedline); - } - } - -# -# Checks which are anchored on the added line. -# - -# check for malformed paths in #include statements (uses RAW line) - if ($rawline =~ m{^.\s*\#\s*include\s+[<"](.*)[">]}) { - my $path = $1; - if ($path =~ m{//}) { - ERROR("MALFORMED_INCLUDE", - "malformed #include filename\n" . $herecurr); - } - if ($path =~ "^uapi/" && $realfile =~ m@\binclude/uapi/@) { - ERROR("UAPI_INCLUDE", - "No #include in ...include/uapi/... should use a uapi/ path prefix\n" . $herecurr); - } - } - -# no C99 // comments - if ($line =~ m{//}) { - if (ERROR("C99_COMMENTS", - "do not use C99 // comments\n" . $herecurr) && - $fix) { - my $line = $fixed[$fixlinenr]; - if ($line =~ /\/\/(.*)$/) { - my $comment = trim($1); - $fixed[$fixlinenr] =~ s@\/\/(.*)$@/\* $comment \*/@; - } - } - } - # Remove C99 comments. - $line =~ s@//.*@@; - $opline =~ s@//.*@@; - -# EXPORT_SYMBOL should immediately follow the thing it is exporting, consider -# the whole statement. -#print "APW <$lines[$realline_next - 1]>\n"; - if (defined $realline_next && - exists $lines[$realline_next - 1] && - !defined $suppress_export{$realline_next} && - ($lines[$realline_next - 1] =~ /EXPORT_SYMBOL.*\((.*)\)/ || - $lines[$realline_next - 1] =~ /EXPORT_UNUSED_SYMBOL.*\((.*)\)/)) { - # Handle definitions which produce identifiers with - # a prefix: - # XXX(foo); - # EXPORT_SYMBOL(something_foo); - my $name = $1; - if ($stat =~ /^(?:.\s*}\s*\n)?.([A-Z_]+)\s*\(\s*($Ident)/ && - $name =~ /^${Ident}_$2/) { -#print "FOO C name<$name>\n"; - $suppress_export{$realline_next} = 1; - - } elsif ($stat !~ /(?: - \n.}\s*$| - ^.DEFINE_$Ident\(\Q$name\E\)| - ^.DECLARE_$Ident\(\Q$name\E\)| - ^.LIST_HEAD\(\Q$name\E\)| - ^.(?:$Storage\s+)?$Type\s*\(\s*\*\s*\Q$name\E\s*\)\s*\(| - \b\Q$name\E(?:\s+$Attribute)*\s*(?:;|=|\[|\() - )/x) { -#print "FOO A<$lines[$realline_next - 1]> stat<$stat> name<$name>\n"; - $suppress_export{$realline_next} = 2; - } else { - $suppress_export{$realline_next} = 1; - } - } - if (!defined $suppress_export{$linenr} && - $prevline =~ /^.\s*$/ && - ($line =~ /EXPORT_SYMBOL.*\((.*)\)/ || - $line =~ /EXPORT_UNUSED_SYMBOL.*\((.*)\)/)) { -#print "FOO B <$lines[$linenr - 1]>\n"; - $suppress_export{$linenr} = 2; - } - if (defined $suppress_export{$linenr} && - $suppress_export{$linenr} == 2) { - WARN("EXPORT_SYMBOL", - "EXPORT_SYMBOL(foo); should immediately follow its function/variable\n" . $herecurr); - } - -# check for global initialisers. - if ($line =~ /^\+$Type\s*$Ident(?:\s+$Modifier)*\s*=\s*($zero_initializer)\s*;/) { - if (ERROR("GLOBAL_INITIALISERS", - "do not initialise globals to $1\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/(^.$Type\s*$Ident(?:\s+$Modifier)*)\s*=\s*$zero_initializer\s*;/$1;/; - } - } -# check for static initialisers. - if ($line =~ /^\+.*\bstatic\s.*=\s*($zero_initializer)\s*;/) { - if (ERROR("INITIALISED_STATIC", - "do not initialise statics to $1\n" . - $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/(\bstatic\s.*?)\s*=\s*$zero_initializer\s*;/$1;/; - } - } - -# check for misordered declarations of char/short/int/long with signed/unsigned - while ($sline =~ m{(\b$TypeMisordered\b)}g) { - my $tmp = trim($1); - WARN("MISORDERED_TYPE", - "type '$tmp' should be specified in [[un]signed] [short|int|long|long long] order\n" . $herecurr); - } - -# check for static const char * arrays. - if ($line =~ /\bstatic\s+const\s+char\s*\*\s*(\w+)\s*\[\s*\]\s*=\s*/) { - WARN("STATIC_CONST_CHAR_ARRAY", - "static const char * array should probably be static const char * const\n" . - $herecurr); - } - -# check for static char foo[] = "bar" declarations. - if ($line =~ /\bstatic\s+char\s+(\w+)\s*\[\s*\]\s*=\s*"/) { - WARN("STATIC_CONST_CHAR_ARRAY", - "static char array declaration should probably be static const char\n" . - $herecurr); - } - -# check for const <foo> const where <foo> is not a pointer or array type - if ($sline =~ /\bconst\s+($BasicType)\s+const\b/) { - my $found = $1; - if ($sline =~ /\bconst\s+\Q$found\E\s+const\b\s*\*/) { - WARN("CONST_CONST", - "'const $found const *' should probably be 'const $found * const'\n" . $herecurr); - } elsif ($sline !~ /\bconst\s+\Q$found\E\s+const\s+\w+\s*\[/) { - WARN("CONST_CONST", - "'const $found const' should probably be 'const $found'\n" . $herecurr); - } - } - -# check for non-global char *foo[] = {"bar", ...} declarations. - if ($line =~ /^.\s+(?:static\s+|const\s+)?char\s+\*\s*\w+\s*\[\s*\]\s*=\s*\{/) { - WARN("STATIC_CONST_CHAR_ARRAY", - "char * array declaration might be better as static const\n" . - $herecurr); - } - -# check for sizeof(foo)/sizeof(foo[0]) that could be ARRAY_SIZE(foo) - if ($line =~ m@\bsizeof\s*\(\s*($Lval)\s*\)@) { - my $array = $1; - if ($line =~ m@\b(sizeof\s*\(\s*\Q$array\E\s*\)\s*/\s*sizeof\s*\(\s*\Q$array\E\s*\[\s*0\s*\]\s*\))@) { - my $array_div = $1; - if (WARN("ARRAY_SIZE", - "Prefer ARRAY_SIZE($array)\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\Q$array_div\E/ARRAY_SIZE($array)/; - } - } - } - -# check for function declarations without arguments like "int foo()" - if ($line =~ /(\b$Type\s+$Ident)\s*\(\s*\)/) { - if (ERROR("FUNCTION_WITHOUT_ARGS", - "Bad function definition - $1() should probably be $1(void)\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/(\b($Type)\s+($Ident))\s*\(\s*\)/$2 $3(void)/; - } - } - -# check for new typedefs, only function parameters and sparse annotations -# make sense. - if ($line =~ /\btypedef\s/ && - $line !~ /\btypedef\s+$Type\s*\(\s*\*?$Ident\s*\)\s*\(/ && - $line !~ /\btypedef\s+$Type\s+$Ident\s*\(/ && - $line !~ /\b$typeTypedefs\b/ && - $line !~ /\b__bitwise\b/) { - WARN("NEW_TYPEDEFS", - "do not add new typedefs\n" . $herecurr); - } - -# * goes on variable not on type - # (char*[ const]) - while ($line =~ m{(\($NonptrType(\s*(?:$Modifier\b\s*|\*\s*)+)\))}g) { - #print "AA<$1>\n"; - my ($ident, $from, $to) = ($1, $2, $2); - - # Should start with a space. - $to =~ s/^(\S)/ $1/; - # Should not end with a space. - $to =~ s/\s+$//; - # '*'s should not have spaces between. - while ($to =~ s/\*\s+\*/\*\*/) { - } - -## print "1: from<$from> to<$to> ident<$ident>\n"; - if ($from ne $to) { - if (ERROR("POINTER_LOCATION", - "\"(foo$from)\" should be \"(foo$to)\"\n" . $herecurr) && - $fix) { - my $sub_from = $ident; - my $sub_to = $ident; - $sub_to =~ s/\Q$from\E/$to/; - $fixed[$fixlinenr] =~ - s@\Q$sub_from\E@$sub_to@; - } - } - } - while ($line =~ m{(\b$NonptrType(\s*(?:$Modifier\b\s*|\*\s*)+)($Ident))}g) { - #print "BB<$1>\n"; - my ($match, $from, $to, $ident) = ($1, $2, $2, $3); - - # Should start with a space. - $to =~ s/^(\S)/ $1/; - # Should not end with a space. - $to =~ s/\s+$//; - # '*'s should not have spaces between. - while ($to =~ s/\*\s+\*/\*\*/) { - } - # Modifiers should have spaces. - $to =~ s/(\b$Modifier$)/$1 /; - -## print "2: from<$from> to<$to> ident<$ident>\n"; - if ($from ne $to && $ident !~ /^$Modifier$/) { - if (ERROR("POINTER_LOCATION", - "\"foo${from}bar\" should be \"foo${to}bar\"\n" . $herecurr) && - $fix) { - - my $sub_from = $match; - my $sub_to = $match; - $sub_to =~ s/\Q$from\E/$to/; - $fixed[$fixlinenr] =~ - s@\Q$sub_from\E@$sub_to@; - } - } - } - -# avoid BUG() or BUG_ON() - if ($line =~ /\b(?:BUG|BUG_ON)\b/) { - my $msg_level = \&WARN; - $msg_level = \&CHK if ($file); - &{$msg_level}("AVOID_BUG", - "Avoid crashing the kernel - try using WARN_ON & recovery code rather than BUG() or BUG_ON()\n" . $herecurr); - } - -# avoid LINUX_VERSION_CODE - if ($line =~ /\bLINUX_VERSION_CODE\b/) { - WARN("LINUX_VERSION_CODE", - "LINUX_VERSION_CODE should be avoided, code should be for the version to which it is merged\n" . $herecurr); - } - -# check for uses of printk_ratelimit - if ($line =~ /\bprintk_ratelimit\s*\(/) { - WARN("PRINTK_RATELIMITED", - "Prefer printk_ratelimited or pr_<level>_ratelimited to printk_ratelimit\n" . $herecurr); - } - -# printk should use KERN_* levels. Note that follow on printk's on the -# same line do not need a level, so we use the current block context -# to try and find and validate the current printk. In summary the current -# printk includes all preceding printk's which have no newline on the end. -# we assume the first bad printk is the one to report. - if ($line =~ /\bprintk\((?!KERN_)\s*"/) { - my $ok = 0; - for (my $ln = $linenr - 1; $ln >= $first_line; $ln--) { - #print "CHECK<$lines[$ln - 1]\n"; - # we have a preceding printk if it ends - # with "\n" ignore it, else it is to blame - if ($lines[$ln - 1] =~ m{\bprintk\(}) { - if ($rawlines[$ln - 1] !~ m{\\n"}) { - $ok = 1; - } - last; - } - } - if ($ok == 0) { - WARN("PRINTK_WITHOUT_KERN_LEVEL", - "printk() should include KERN_ facility level\n" . $herecurr); - } - } - - if ($line =~ /\bprintk\s*\(\s*KERN_([A-Z]+)/) { - my $orig = $1; - my $level = lc($orig); - $level = "warn" if ($level eq "warning"); - my $level2 = $level; - $level2 = "dbg" if ($level eq "debug"); - WARN("PREFER_PR_LEVEL", - "Prefer [subsystem eg: netdev]_$level2([subsystem]dev, ... then dev_$level2(dev, ... then pr_$level(... to printk(KERN_$orig ...\n" . $herecurr); - } - - if ($line =~ /\bpr_warning\s*\(/) { - if (WARN("PREFER_PR_LEVEL", - "Prefer pr_warn(... to pr_warning(...\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/\bpr_warning\b/pr_warn/; - } - } - - if ($line =~ /\bdev_printk\s*\(\s*KERN_([A-Z]+)/) { - my $orig = $1; - my $level = lc($orig); - $level = "warn" if ($level eq "warning"); - $level = "dbg" if ($level eq "debug"); - WARN("PREFER_DEV_LEVEL", - "Prefer dev_$level(... to dev_printk(KERN_$orig, ...\n" . $herecurr); - } - -# ENOSYS means "bad syscall nr" and nothing else. This will have a small -# number of false positives, but assembly files are not checked, so at -# least the arch entry code will not trigger this warning. - if ($line =~ /\bENOSYS\b/) { - WARN("ENOSYS", - "ENOSYS means 'invalid syscall nr' and nothing else\n" . $herecurr); - } - -# function brace can't be on same line, except for #defines of do while, -# or if closed on same line - if (($line=~/$Type\s*$Ident\(.*\).*\s*{/) and - !($line=~/\#\s*define.*do\s\{/) and !($line=~/}/)) { - if (ERROR("OPEN_BRACE", - "open brace '{' following function declarations go on the next line\n" . $herecurr) && - $fix) { - fix_delete_line($fixlinenr, $rawline); - my $fixed_line = $rawline; - $fixed_line =~ /(^..*$Type\s*$Ident\(.*\)\s*){(.*)$/; - my $line1 = $1; - my $line2 = $2; - fix_insert_line($fixlinenr, ltrim($line1)); - fix_insert_line($fixlinenr, "\+{"); - if ($line2 !~ /^\s*$/) { - fix_insert_line($fixlinenr, "\+\t" . trim($line2)); - } - } - } - -# open braces for enum, union and struct go on the same line. - if ($line =~ /^.\s*{/ && - $prevline =~ /^.\s*(?:typedef\s+)?(enum|union|struct)(?:\s+$Ident)?\s*$/) { - if (ERROR("OPEN_BRACE", - "open brace '{' following $1 go on the same line\n" . $hereprev) && - $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { - fix_delete_line($fixlinenr - 1, $prevrawline); - fix_delete_line($fixlinenr, $rawline); - my $fixedline = rtrim($prevrawline) . " {"; - fix_insert_line($fixlinenr, $fixedline); - $fixedline = $rawline; - $fixedline =~ s/^(.\s*)\{\s*/$1\t/; - if ($fixedline !~ /^\+\s*$/) { - fix_insert_line($fixlinenr, $fixedline); - } - } - } - -# missing space after union, struct or enum definition - if ($line =~ /^.\s*(?:typedef\s+)?(enum|union|struct)(?:\s+$Ident){1,2}[=\{]/) { - if (WARN("SPACING", - "missing space after $1 definition\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/^(.\s*(?:typedef\s+)?(?:enum|union|struct)(?:\s+$Ident){1,2})([=\{])/$1 $2/; - } - } - -# Function pointer declarations -# check spacing between type, funcptr, and args -# canonical declaration is "type (*funcptr)(args...)" - if ($line =~ /^.\s*($Declare)\((\s*)\*(\s*)($Ident)(\s*)\)(\s*)\(/) { - my $declare = $1; - my $pre_pointer_space = $2; - my $post_pointer_space = $3; - my $funcname = $4; - my $post_funcname_space = $5; - my $pre_args_space = $6; - -# the $Declare variable will capture all spaces after the type -# so check it for a missing trailing missing space but pointer return types -# don't need a space so don't warn for those. - my $post_declare_space = ""; - if ($declare =~ /(\s+)$/) { - $post_declare_space = $1; - $declare = rtrim($declare); - } - if ($declare !~ /\*$/ && $post_declare_space =~ /^$/) { - WARN("SPACING", - "missing space after return type\n" . $herecurr); - $post_declare_space = " "; - } - -# unnecessary space "type (*funcptr)(args...)" -# This test is not currently implemented because these declarations are -# equivalent to -# int foo(int bar, ...) -# and this is form shouldn't/doesn't generate a checkpatch warning. -# -# elsif ($declare =~ /\s{2,}$/) { -# WARN("SPACING", -# "Multiple spaces after return type\n" . $herecurr); -# } - -# unnecessary space "type ( *funcptr)(args...)" - if (defined $pre_pointer_space && - $pre_pointer_space =~ /^\s/) { - WARN("SPACING", - "Unnecessary space after function pointer open parenthesis\n" . $herecurr); - } - -# unnecessary space "type (* funcptr)(args...)" - if (defined $post_pointer_space && - $post_pointer_space =~ /^\s/) { - WARN("SPACING", - "Unnecessary space before function pointer name\n" . $herecurr); - } - -# unnecessary space "type (*funcptr )(args...)" - if (defined $post_funcname_space && - $post_funcname_space =~ /^\s/) { - WARN("SPACING", - "Unnecessary space after function pointer name\n" . $herecurr); - } - -# unnecessary space "type (*funcptr) (args...)" - if (defined $pre_args_space && - $pre_args_space =~ /^\s/) { - WARN("SPACING", - "Unnecessary space before function pointer arguments\n" . $herecurr); - } - - if (show_type("SPACING") && $fix) { - $fixed[$fixlinenr] =~ - s/^(.\s*)$Declare\s*\(\s*\*\s*$Ident\s*\)\s*\(/$1 . $declare . $post_declare_space . '(*' . $funcname . ')('/ex; - } - } - -# check for spacing round square brackets; allowed: -# 1. with a type on the left -- int [] a; -# 2. at the beginning of a line for slice initialisers -- [0...10] = 5, -# 3. inside a curly brace -- = { [0...10] = 5 } - while ($line =~ /(.*?\s)\[/g) { - my ($where, $prefix) = ($-[1], $1); - if ($prefix !~ /$Type\s+$/ && - ($where != 0 || $prefix !~ /^.\s+$/) && - $prefix !~ /[{,]\s+$/ && - $prefix !~ /:\s+$/) { - if (ERROR("BRACKET_SPACE", - "space prohibited before open square bracket '['\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/^(\+.*?)\s+\[/$1\[/; - } - } - } - -# check for spaces between functions and their parentheses. - while ($line =~ /($Ident)\s+\(/g) { - my $name = $1; - my $ctx_before = substr($line, 0, $-[1]); - my $ctx = "$ctx_before$name"; - - # Ignore those directives where spaces _are_ permitted. - if ($name =~ /^(?: - if|for|while|switch|return|case| - volatile|__volatile__| - __attribute__|format|__extension__| - asm|__asm__)$/x) - { - # cpp #define statements have non-optional spaces, ie - # if there is a space between the name and the open - # parenthesis it is simply not a parameter group. - } elsif ($ctx_before =~ /^.\s*\#\s*define\s*$/) { - - # cpp #elif statement condition may start with a ( - } elsif ($ctx =~ /^.\s*\#\s*elif\s*$/) { - - # If this whole things ends with a type its most - # likely a typedef for a function. - } elsif ($ctx =~ /$Type$/) { - - } else { - if (WARN("SPACING", - "space prohibited between function name and open parenthesis '('\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/\b$name\s+\(/$name\(/; - } - } - } - -# Check operator spacing. - if (!($line=~/\#\s*include/)) { - my $fixed_line = ""; - my $line_fixed = 0; - - my $ops = qr{ - <<=|>>=|<=|>=|==|!=| - \+=|-=|\*=|\/=|%=|\^=|\|=|&=| - =>|->|<<|>>|<|>|=|!|~| - &&|\|\||,|\^|\+\+|--|&|\||\+|-|\*|\/|%| - \?:|\?|: - }x; - my @elements = split(/($ops|;)/, $opline); - -## print("element count: <" . $#elements . ">\n"); -## foreach my $el (@elements) { -## print("el: <$el>\n"); -## } - - my @fix_elements = (); - my $off = 0; - - foreach my $el (@elements) { - push(@fix_elements, substr($rawline, $off, length($el))); - $off += length($el); - } - - $off = 0; - - my $blank = copy_spacing($opline); - my $last_after = -1; - - for (my $n = 0; $n < $#elements; $n += 2) { - - my $good = $fix_elements[$n] . $fix_elements[$n + 1]; - -## print("n: <$n> good: <$good>\n"); - - $off += length($elements[$n]); - - # Pick up the preceding and succeeding characters. - my $ca = substr($opline, 0, $off); - my $cc = ''; - if (length($opline) >= ($off + length($elements[$n + 1]))) { - $cc = substr($opline, $off + length($elements[$n + 1])); - } - my $cb = "$ca$;$cc"; - - my $a = ''; - $a = 'V' if ($elements[$n] ne ''); - $a = 'W' if ($elements[$n] =~ /\s$/); - $a = 'C' if ($elements[$n] =~ /$;$/); - $a = 'B' if ($elements[$n] =~ /(\[|\()$/); - $a = 'O' if ($elements[$n] eq ''); - $a = 'E' if ($ca =~ /^\s*$/); - - my $op = $elements[$n + 1]; - - my $c = ''; - if (defined $elements[$n + 2]) { - $c = 'V' if ($elements[$n + 2] ne ''); - $c = 'W' if ($elements[$n + 2] =~ /^\s/); - $c = 'C' if ($elements[$n + 2] =~ /^$;/); - $c = 'B' if ($elements[$n + 2] =~ /^(\)|\]|;)/); - $c = 'O' if ($elements[$n + 2] eq ''); - $c = 'E' if ($elements[$n + 2] =~ /^\s*\\$/); - } else { - $c = 'E'; - } - - my $ctx = "${a}x${c}"; - - my $at = "(ctx:$ctx)"; - - my $ptr = substr($blank, 0, $off) . "^"; - my $hereptr = "$hereline$ptr\n"; - - # Pull out the value of this operator. - my $op_type = substr($curr_values, $off + 1, 1); - - # Get the full operator variant. - my $opv = $op . substr($curr_vars, $off, 1); - - # Ignore operators passed as parameters. - if ($op_type ne 'V' && - $ca =~ /\s$/ && $cc =~ /^\s*[,\)]/) { - -# # Ignore comments -# } elsif ($op =~ /^$;+$/) { - - # ; should have either the end of line or a space or \ after it - } elsif ($op eq ';') { - if ($ctx !~ /.x[WEBC]/ && - $cc !~ /^\\/ && $cc !~ /^;/) { - if (ERROR("SPACING", - "space required after that '$op' $at\n" . $hereptr)) { - $good = $fix_elements[$n] . trim($fix_elements[$n + 1]) . " "; - $line_fixed = 1; - } - } - - # // is a comment - } elsif ($op eq '//') { - - # : when part of a bitfield - } elsif ($opv eq ':B') { - # skip the bitfield test for now - - # No spaces for: - # -> - } elsif ($op eq '->') { - if ($ctx =~ /Wx.|.xW/) { - if (ERROR("SPACING", - "spaces prohibited around that '$op' $at\n" . $hereptr)) { - $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); - if (defined $fix_elements[$n + 2]) { - $fix_elements[$n + 2] =~ s/^\s+//; - } - $line_fixed = 1; - } - } - - # , must not have a space before and must have a space on the right. - } elsif ($op eq ',') { - my $rtrim_before = 0; - my $space_after = 0; - if ($ctx =~ /Wx./) { - if (ERROR("SPACING", - "space prohibited before that '$op' $at\n" . $hereptr)) { - $line_fixed = 1; - $rtrim_before = 1; - } - } - if ($ctx !~ /.x[WEC]/ && $cc !~ /^}/ && $cc !~ /^\)/) { - if (ERROR("SPACING", - "space required after that '$op' $at\n" . $hereptr)) { - $line_fixed = 1; - $last_after = $n; - $space_after = 1; - } - } - if ($rtrim_before || $space_after) { - if ($rtrim_before) { - $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); - } else { - $good = $fix_elements[$n] . trim($fix_elements[$n + 1]); - } - if ($space_after) { - $good .= " "; - } - } - - # '*' as part of a type definition -- reported already. - } elsif ($opv eq '*_') { - #warn "'*' is part of type\n"; - - # unary operators should have a space before and - # none after. May be left adjacent to another - # unary operator, or a cast - } elsif ($op eq '!' || $op eq '~' || - $opv eq '*U' || $opv eq '-U' || - $opv eq '&U' || $opv eq '&&U') { - if ($ctx !~ /[WEBC]x./ && $ca !~ /(?:\)|!|~|\*|-|\&|\||\+\+|\-\-|\{)$/) { - if (ERROR("SPACING", - "space required before that '$op' $at\n" . $hereptr)) { - if ($n != $last_after + 2) { - $good = $fix_elements[$n] . " " . ltrim($fix_elements[$n + 1]); - $line_fixed = 1; - } - } - } - if ($op eq '*' && $cc =~/\s*$Modifier\b/) { - # A unary '*' may be const - - } elsif ($ctx =~ /.xW/) { - if (ERROR("SPACING", - "space prohibited after that '$op' $at\n" . $hereptr)) { - $good = $fix_elements[$n] . rtrim($fix_elements[$n + 1]); - if (defined $fix_elements[$n + 2]) { - $fix_elements[$n + 2] =~ s/^\s+//; - } - $line_fixed = 1; - } - } - - # unary ++ and unary -- are allowed no space on one side. - } elsif ($op eq '++' or $op eq '--') { - if ($ctx !~ /[WEOBC]x[^W]/ && $ctx !~ /[^W]x[WOBEC]/) { - if (ERROR("SPACING", - "space required one side of that '$op' $at\n" . $hereptr)) { - $good = $fix_elements[$n] . trim($fix_elements[$n + 1]) . " "; - $line_fixed = 1; - } - } - if ($ctx =~ /Wx[BE]/ || - ($ctx =~ /Wx./ && $cc =~ /^;/)) { - if (ERROR("SPACING", - "space prohibited before that '$op' $at\n" . $hereptr)) { - $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); - $line_fixed = 1; - } - } - if ($ctx =~ /ExW/) { - if (ERROR("SPACING", - "space prohibited after that '$op' $at\n" . $hereptr)) { - $good = $fix_elements[$n] . trim($fix_elements[$n + 1]); - if (defined $fix_elements[$n + 2]) { - $fix_elements[$n + 2] =~ s/^\s+//; - } - $line_fixed = 1; - } - } - - # << and >> may either have or not have spaces both sides - } elsif ($op eq '<<' or $op eq '>>' or - $op eq '&' or $op eq '^' or $op eq '|' or - $op eq '+' or $op eq '-' or - $op eq '*' or $op eq '/' or - $op eq '%') - { - if ($check) { - if (defined $fix_elements[$n + 2] && $ctx !~ /[EW]x[EW]/) { - if (CHK("SPACING", - "spaces preferred around that '$op' $at\n" . $hereptr)) { - $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " "; - $fix_elements[$n + 2] =~ s/^\s+//; - $line_fixed = 1; - } - } elsif (!defined $fix_elements[$n + 2] && $ctx !~ /Wx[OE]/) { - if (CHK("SPACING", - "space preferred before that '$op' $at\n" . $hereptr)) { - $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]); - $line_fixed = 1; - } - } - } elsif ($ctx =~ /Wx[^WCE]|[^WCE]xW/) { - if (ERROR("SPACING", - "need consistent spacing around '$op' $at\n" . $hereptr)) { - $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " "; - if (defined $fix_elements[$n + 2]) { - $fix_elements[$n + 2] =~ s/^\s+//; - } - $line_fixed = 1; - } - } - - # A colon needs no spaces before when it is - # terminating a case value or a label. - } elsif ($opv eq ':C' || $opv eq ':L') { - if ($ctx =~ /Wx./) { - if (ERROR("SPACING", - "space prohibited before that '$op' $at\n" . $hereptr)) { - $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); - $line_fixed = 1; - } - } - - # All the others need spaces both sides. - } elsif ($ctx !~ /[EWC]x[CWE]/) { - my $ok = 0; - - # Ignore email addresses <foo@bar> - if (($op eq '<' && - $cc =~ /^\S+\@\S+>/) || - ($op eq '>' && - $ca =~ /<\S+\@\S+$/)) - { - $ok = 1; - } - - # for asm volatile statements - # ignore a colon with another - # colon immediately before or after - if (($op eq ':') && - ($ca =~ /:$/ || $cc =~ /^:/)) { - $ok = 1; - } - - # messages are ERROR, but ?: are CHK - if ($ok == 0) { - my $msg_level = \&ERROR; - $msg_level = \&CHK if (($op eq '?:' || $op eq '?' || $op eq ':') && $ctx =~ /VxV/); - - if (&{$msg_level}("SPACING", - "spaces required around that '$op' $at\n" . $hereptr)) { - $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " "; - if (defined $fix_elements[$n + 2]) { - $fix_elements[$n + 2] =~ s/^\s+//; - } - $line_fixed = 1; - } - } - } - $off += length($elements[$n + 1]); - -## print("n: <$n> GOOD: <$good>\n"); - - $fixed_line = $fixed_line . $good; - } - - if (($#elements % 2) == 0) { - $fixed_line = $fixed_line . $fix_elements[$#elements]; - } - - if ($fix && $line_fixed && $fixed_line ne $fixed[$fixlinenr]) { - $fixed[$fixlinenr] = $fixed_line; - } - - - } - -# check for whitespace before a non-naked semicolon - if ($line =~ /^\+.*\S\s+;\s*$/) { - if (WARN("SPACING", - "space prohibited before semicolon\n" . $herecurr) && - $fix) { - 1 while $fixed[$fixlinenr] =~ - s/^(\+.*\S)\s+;/$1;/; - } - } - -# check for multiple assignments - if ($line =~ /^.\s*$Lval\s*=\s*$Lval\s*=(?!=)/) { - CHK("MULTIPLE_ASSIGNMENTS", - "multiple assignments should be avoided\n" . $herecurr); - } - -## # check for multiple declarations, allowing for a function declaration -## # continuation. -## if ($line =~ /^.\s*$Type\s+$Ident(?:\s*=[^,{]*)?\s*,\s*$Ident.*/ && -## $line !~ /^.\s*$Type\s+$Ident(?:\s*=[^,{]*)?\s*,\s*$Type\s*$Ident.*/) { -## -## # Remove any bracketed sections to ensure we do not -## # falsly report the parameters of functions. -## my $ln = $line; -## while ($ln =~ s/\([^\(\)]*\)//g) { -## } -## if ($ln =~ /,/) { -## WARN("MULTIPLE_DECLARATION", -## "declaring multiple variables together should be avoided\n" . $herecurr); -## } -## } - -#need space before brace following if, while, etc - if (($line =~ /\(.*\)\{/ && $line !~ /\($Type\)\{/) || - $line =~ /do\{/) { - if (ERROR("SPACING", - "space required before the open brace '{'\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/^(\+.*(?:do|\)))\{/$1 {/; - } - } - -## # check for blank lines before declarations -## if ($line =~ /^.\t+$Type\s+$Ident(?:\s*=.*)?;/ && -## $prevrawline =~ /^.\s*$/) { -## WARN("SPACING", -## "No blank lines before declarations\n" . $hereprev); -## } -## - -# closing brace should have a space following it when it has anything -# on the line - if ($line =~ /}(?!(?:,|;|\)))\S/) { - if (ERROR("SPACING", - "space required after that close brace '}'\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/}((?!(?:,|;|\)))\S)/} $1/; - } - } - -# check spacing on square brackets - if ($line =~ /\[\s/ && $line !~ /\[\s*$/) { - if (ERROR("SPACING", - "space prohibited after that open square bracket '['\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/\[\s+/\[/; - } - } - if ($line =~ /\s\]/) { - if (ERROR("SPACING", - "space prohibited before that close square bracket ']'\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/\s+\]/\]/; - } - } - -# check spacing on parentheses - if ($line =~ /\(\s/ && $line !~ /\(\s*(?:\\)?$/ && - $line !~ /for\s*\(\s+;/) { - if (ERROR("SPACING", - "space prohibited after that open parenthesis '('\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/\(\s+/\(/; - } - } - if ($line =~ /(\s+)\)/ && $line !~ /^.\s*\)/ && - $line !~ /for\s*\(.*;\s+\)/ && - $line !~ /:\s+\)/) { - if (ERROR("SPACING", - "space prohibited before that close parenthesis ')'\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/\s+\)/\)/; - } - } - -# check unnecessary parentheses around addressof/dereference single $Lvals -# ie: &(foo->bar) should be &foo->bar and *(foo->bar) should be *foo->bar - - while ($line =~ /(?:[^&]&\s*|\*)\(\s*($Ident\s*(?:$Member\s*)+)\s*\)/g) { - my $var = $1; - if (CHK("UNNECESSARY_PARENTHESES", - "Unnecessary parentheses around $var\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\(\s*\Q$var\E\s*\)/$var/; - } - } - -# check for unnecessary parentheses around function pointer uses -# ie: (foo->bar)(); should be foo->bar(); -# but not "if (foo->bar) (" to avoid some false positives - if ($line =~ /(\bif\s*|)(\(\s*$Ident\s*(?:$Member\s*)+\))[ \t]*\(/ && $1 !~ /^if/) { - my $var = $2; - if (CHK("UNNECESSARY_PARENTHESES", - "Unnecessary parentheses around function pointer $var\n" . $herecurr) && - $fix) { - my $var2 = deparenthesize($var); - $var2 =~ s/\s//g; - $fixed[$fixlinenr] =~ s/\Q$var\E/$var2/; - } - } - -# check for unnecessary parentheses around comparisons in if uses - if ($^V && $^V ge 5.10.0 && defined($stat) && - $stat =~ /(^.\s*if\s*($balanced_parens))/) { - my $if_stat = $1; - my $test = substr($2, 1, -1); - my $herectx; - while ($test =~ /(?:^|[^\w\&\!\~])+\s*\(\s*([\&\!\~]?\s*$Lval\s*(?:$Compare\s*$FuncArg)?)\s*\)/g) { - my $match = $1; - # avoid parentheses around potential macro args - next if ($match =~ /^\s*\w+\s*$/); - if (!defined($herectx)) { - $herectx = $here . "\n"; - my $cnt = statement_rawlines($if_stat); - for (my $n = 0; $n < $cnt; $n++) { - my $rl = raw_line($linenr, $n); - $herectx .= $rl . "\n"; - last if $rl =~ /^[ \+].*\{/; - } - } - CHK("UNNECESSARY_PARENTHESES", - "Unnecessary parentheses around '$match'\n" . $herectx); - } - } - -#goto labels aren't indented, allow a single space however - if ($line=~/^.\s+[A-Za-z\d_]+:(?![0-9]+)/ and - !($line=~/^. [A-Za-z\d_]+:/) and !($line=~/^.\s+default:/)) { - if (WARN("INDENTED_LABEL", - "labels should not be indented\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/^(.)\s+/$1/; - } - } - -# return is not a function - if (defined($stat) && $stat =~ /^.\s*return(\s*)\(/s) { - my $spacing = $1; - if ($^V && $^V ge 5.10.0 && - $stat =~ /^.\s*return\s*($balanced_parens)\s*;\s*$/) { - my $value = $1; - $value = deparenthesize($value); - if ($value =~ m/^\s*$FuncArg\s*(?:\?|$)/) { - ERROR("RETURN_PARENTHESES", - "return is not a function, parentheses are not required\n" . $herecurr); - } - } elsif ($spacing !~ /\s+/) { - ERROR("SPACING", - "space required before the open parenthesis '('\n" . $herecurr); - } - } - -# unnecessary return in a void function -# at end-of-function, with the previous line a single leading tab, then return; -# and the line before that not a goto label target like "out:" - if ($sline =~ /^[ \+]}\s*$/ && - $prevline =~ /^\+\treturn\s*;\s*$/ && - $linenr >= 3 && - $lines[$linenr - 3] =~ /^[ +]/ && - $lines[$linenr - 3] !~ /^[ +]\s*$Ident\s*:/) { - WARN("RETURN_VOID", - "void function return statements are not generally useful\n" . $hereprev); - } - -# if statements using unnecessary parentheses - ie: if ((foo == bar)) - if ($^V && $^V ge 5.10.0 && - $line =~ /\bif\s*((?:\(\s*){2,})/) { - my $openparens = $1; - my $count = $openparens =~ tr@\(@\(@; - my $msg = ""; - if ($line =~ /\bif\s*(?:\(\s*){$count,$count}$LvalOrFunc\s*($Compare)\s*$LvalOrFunc(?:\s*\)){$count,$count}/) { - my $comp = $4; #Not $1 because of $LvalOrFunc - $msg = " - maybe == should be = ?" if ($comp eq "=="); - WARN("UNNECESSARY_PARENTHESES", - "Unnecessary parentheses$msg\n" . $herecurr); - } - } - -# comparisons with a constant or upper case identifier on the left -# avoid cases like "foo + BAR < baz" -# only fix matches surrounded by parentheses to avoid incorrect -# conversions like "FOO < baz() + 5" being "misfixed" to "baz() > FOO + 5" - if ($^V && $^V ge 5.10.0 && - $line =~ /^\+(.*)\b($Constant|[A-Z_][A-Z0-9_]*)\s*($Compare)\s*($LvalOrFunc)/) { - my $lead = $1; - my $const = $2; - my $comp = $3; - my $to = $4; - my $newcomp = $comp; - if ($lead !~ /(?:$Operators|\.)\s*$/ && - $to !~ /^(?:Constant|[A-Z_][A-Z0-9_]*)$/ && - WARN("CONSTANT_COMPARISON", - "Comparisons should place the constant on the right side of the test\n" . $herecurr) && - $fix) { - if ($comp eq "<") { - $newcomp = ">"; - } elsif ($comp eq "<=") { - $newcomp = ">="; - } elsif ($comp eq ">") { - $newcomp = "<"; - } elsif ($comp eq ">=") { - $newcomp = "<="; - } - $fixed[$fixlinenr] =~ s/\(\s*\Q$const\E\s*$Compare\s*\Q$to\E\s*\)/($to $newcomp $const)/; - } - } - -# Return of what appears to be an errno should normally be negative - if ($sline =~ /\breturn(?:\s*\(+\s*|\s+)(E[A-Z]+)(?:\s*\)+\s*|\s*)[;:,]/) { - my $name = $1; - if ($name ne 'EOF' && $name ne 'ERROR') { - WARN("USE_NEGATIVE_ERRNO", - "return of an errno should typically be negative (ie: return -$1)\n" . $herecurr); - } - } - -# Need a space before open parenthesis after if, while etc - if ($line =~ /\b(if|while|for|switch)\(/) { - if (ERROR("SPACING", - "space required before the open parenthesis '('\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/\b(if|while|for|switch)\(/$1 \(/; - } - } - -# Check for illegal assignment in if conditional -- and check for trailing -# statements after the conditional. - if ($line =~ /do\s*(?!{)/) { - ($stat, $cond, $line_nr_next, $remain_next, $off_next) = - ctx_statement_block($linenr, $realcnt, 0) - if (!defined $stat); - my ($stat_next) = ctx_statement_block($line_nr_next, - $remain_next, $off_next); - $stat_next =~ s/\n./\n /g; - ##print "stat<$stat> stat_next<$stat_next>\n"; - - if ($stat_next =~ /^\s*while\b/) { - # If the statement carries leading newlines, - # then count those as offsets. - my ($whitespace) = - ($stat_next =~ /^((?:\s*\n[+-])*\s*)/s); - my $offset = - statement_rawlines($whitespace) - 1; - - $suppress_whiletrailers{$line_nr_next + - $offset} = 1; - } - } - if (!defined $suppress_whiletrailers{$linenr} && - defined($stat) && defined($cond) && - $line =~ /\b(?:if|while|for)\s*\(/ && $line !~ /^.\s*#/) { - my ($s, $c) = ($stat, $cond); - - if ($c =~ /\bif\s*\(.*[^<>!=]=[^=].*/s) { - ERROR("ASSIGN_IN_IF", - "do not use assignment in if condition\n" . $herecurr); - } - - # Find out what is on the end of the line after the - # conditional. - substr($s, 0, length($c), ''); - $s =~ s/\n.*//g; - $s =~ s/$;//g; # Remove any comments - if (length($c) && $s !~ /^\s*{?\s*\\*\s*$/ && - $c !~ /}\s*while\s*/) - { - # Find out how long the conditional actually is. - my @newlines = ($c =~ /\n/gs); - my $cond_lines = 1 + $#newlines; - my $stat_real = ''; - - $stat_real = raw_line($linenr, $cond_lines) - . "\n" if ($cond_lines); - if (defined($stat_real) && $cond_lines > 1) { - $stat_real = "[...]\n$stat_real"; - } - - ERROR("TRAILING_STATEMENTS", - "trailing statements should be on next line\n" . $herecurr . $stat_real); - } - } - -# Check for bitwise tests written as boolean - if ($line =~ / - (?: - (?:\[|\(|\&\&|\|\|) - \s*0[xX][0-9]+\s* - (?:\&\&|\|\|) - | - (?:\&\&|\|\|) - \s*0[xX][0-9]+\s* - (?:\&\&|\|\||\)|\]) - )/x) - { - WARN("HEXADECIMAL_BOOLEAN_TEST", - "boolean test with hexadecimal, perhaps just 1 \& or \|?\n" . $herecurr); - } - -# if and else should not have general statements after it - if ($line =~ /^.\s*(?:}\s*)?else\b(.*)/) { - my $s = $1; - $s =~ s/$;//g; # Remove any comments - if ($s !~ /^\s*(?:\sif|(?:{|)\s*\\?\s*$)/) { - ERROR("TRAILING_STATEMENTS", - "trailing statements should be on next line\n" . $herecurr); - } - } -# if should not continue a brace - if ($line =~ /}\s*if\b/) { - ERROR("TRAILING_STATEMENTS", - "trailing statements should be on next line (or did you mean 'else if'?)\n" . - $herecurr); - } -# case and default should not have general statements after them - if ($line =~ /^.\s*(?:case\s*.*|default\s*):/g && - $line !~ /\G(?: - (?:\s*$;*)(?:\s*{)?(?:\s*$;*)(?:\s*\\)?\s*$| - \s*return\s+ - )/xg) - { - ERROR("TRAILING_STATEMENTS", - "trailing statements should be on next line\n" . $herecurr); - } - - # Check for }<nl>else {, these must be at the same - # indent level to be relevant to each other. - if ($prevline=~/}\s*$/ and $line=~/^.\s*else\s*/ && - $previndent == $indent) { - if (ERROR("ELSE_AFTER_BRACE", - "else should follow close brace '}'\n" . $hereprev) && - $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { - fix_delete_line($fixlinenr - 1, $prevrawline); - fix_delete_line($fixlinenr, $rawline); - my $fixedline = $prevrawline; - $fixedline =~ s/}\s*$//; - if ($fixedline !~ /^\+\s*$/) { - fix_insert_line($fixlinenr, $fixedline); - } - $fixedline = $rawline; - $fixedline =~ s/^(.\s*)else/$1} else/; - fix_insert_line($fixlinenr, $fixedline); - } - } - - if ($prevline=~/}\s*$/ and $line=~/^.\s*while\s*/ && - $previndent == $indent) { - my ($s, $c) = ctx_statement_block($linenr, $realcnt, 0); - - # Find out what is on the end of the line after the - # conditional. - substr($s, 0, length($c), ''); - $s =~ s/\n.*//g; - - if ($s =~ /^\s*;/) { - if (ERROR("WHILE_AFTER_BRACE", - "while should follow close brace '}'\n" . $hereprev) && - $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { - fix_delete_line($fixlinenr - 1, $prevrawline); - fix_delete_line($fixlinenr, $rawline); - my $fixedline = $prevrawline; - my $trailing = $rawline; - $trailing =~ s/^\+//; - $trailing = trim($trailing); - $fixedline =~ s/}\s*$/} $trailing/; - fix_insert_line($fixlinenr, $fixedline); - } - } - } - -#Specific variable tests - while ($line =~ m{($Constant|$Lval)}g) { - my $var = $1; - -#gcc binary extension - if ($var =~ /^$Binary$/) { - if (WARN("GCC_BINARY_CONSTANT", - "Avoid gcc v4.3+ binary constant extension: <$var>\n" . $herecurr) && - $fix) { - my $hexval = sprintf("0x%x", oct($var)); - $fixed[$fixlinenr] =~ - s/\b$var\b/$hexval/; - } - } - -#CamelCase - if ($var !~ /^$Constant$/ && - $var =~ /[A-Z][a-z]|[a-z][A-Z]/ && -#Ignore Page<foo> variants - $var !~ /^(?:Clear|Set|TestClear|TestSet|)Page[A-Z]/ && -#Ignore SI style variants like nS, mV and dB (ie: max_uV, regulator_min_uA_show) - $var !~ /^(?:[a-z_]*?)_?[a-z][A-Z](?:_[a-z_]+)?$/ && -#Ignore some three character SI units explicitly, like MiB and KHz - $var !~ /^(?:[a-z_]*?)_?(?:[KMGT]iB|[KMGT]?Hz)(?:_[a-z_]+)?$/) { - while ($var =~ m{($Ident)}g) { - my $word = $1; - next if ($word !~ /[A-Z][a-z]|[a-z][A-Z]/); - if ($check) { - seed_camelcase_includes(); - if (!$file && !$camelcase_file_seeded) { - seed_camelcase_file($realfile); - $camelcase_file_seeded = 1; - } - } - if (!defined $camelcase{$word}) { - $camelcase{$word} = 1; - CHK("CAMELCASE", - "Avoid CamelCase: <$word>\n" . $herecurr); - } - } - } - } - -#no spaces allowed after \ in define - if ($line =~ /\#\s*define.*\\\s+$/) { - if (WARN("WHITESPACE_AFTER_LINE_CONTINUATION", - "Whitespace after \\ makes next lines useless\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\s+$//; - } - } - -# warn if <asm/foo.h> is #included and <linux/foo.h> is available and includes -# itself <asm/foo.h> (uses RAW line) - if ($tree && $rawline =~ m{^.\s*\#\s*include\s*\<asm\/(.*)\.h\>}) { - my $file = "$1.h"; - my $checkfile = "include/linux/$file"; - if (-f "$root/$checkfile" && - $realfile ne $checkfile && - $1 !~ /$allowed_asm_includes/) - { - my $asminclude = `grep -Ec "#include\\s+<asm/$file>" $root/$checkfile`; - if ($asminclude > 0) { - if ($realfile =~ m{^arch/}) { - CHK("ARCH_INCLUDE_LINUX", - "Consider using #include <linux/$file> instead of <asm/$file>\n" . $herecurr); - } else { - WARN("INCLUDE_LINUX", - "Use #include <linux/$file> instead of <asm/$file>\n" . $herecurr); - } - } - } - } - -# multi-statement macros should be enclosed in a do while loop, grab the -# first statement and ensure its the whole macro if its not enclosed -# in a known good container - if ($realfile !~ m@/vmlinux.lds.h$@ && - $line =~ /^.\s*\#\s*define\s*$Ident(\()?/) { - my $ln = $linenr; - my $cnt = $realcnt; - my ($off, $dstat, $dcond, $rest); - my $ctx = ''; - my $has_flow_statement = 0; - my $has_arg_concat = 0; - ($dstat, $dcond, $ln, $cnt, $off) = - ctx_statement_block($linenr, $realcnt, 0); - $ctx = $dstat; - #print "dstat<$dstat> dcond<$dcond> cnt<$cnt> off<$off>\n"; - #print "LINE<$lines[$ln-1]> len<" . length($lines[$ln-1]) . "\n"; - - $has_flow_statement = 1 if ($ctx =~ /\b(goto|return)\b/); - $has_arg_concat = 1 if ($ctx =~ /\#\#/ && $ctx !~ /\#\#\s*(?:__VA_ARGS__|args)\b/); - - $dstat =~ s/^.\s*\#\s*define\s+$Ident(\([^\)]*\))?\s*//; - my $define_args = $1; - my $define_stmt = $dstat; - my @def_args = (); - - if (defined $define_args && $define_args ne "") { - $define_args = substr($define_args, 1, length($define_args) - 2); - $define_args =~ s/\s*//g; - @def_args = split(",", $define_args); - } - - $dstat =~ s/$;//g; - $dstat =~ s/\\\n.//g; - $dstat =~ s/^\s*//s; - $dstat =~ s/\s*$//s; - - # Flatten any parentheses and braces - while ($dstat =~ s/\([^\(\)]*\)/1/ || - $dstat =~ s/\{[^\{\}]*\}/1/ || - $dstat =~ s/.\[[^\[\]]*\]/1/) - { - } - - # Flatten any obvious string concatentation. - while ($dstat =~ s/($String)\s*$Ident/$1/ || - $dstat =~ s/$Ident\s*($String)/$1/) - { - } - - # Make asm volatile uses seem like a generic function - $dstat =~ s/\b_*asm_*\s+_*volatile_*\b/asm_volatile/g; - - my $exceptions = qr{ - $Declare| - module_param_named| - MODULE_PARM_DESC| - DECLARE_PER_CPU| - DEFINE_PER_CPU| - __typeof__\(| - union| - struct| - \.$Ident\s*=\s*| - ^\"|\"$| - ^\[ - }x; - #print "REST<$rest> dstat<$dstat> ctx<$ctx>\n"; - - $ctx =~ s/\n*$//; - my $herectx = $here . "\n"; - my $stmt_cnt = statement_rawlines($ctx); - - for (my $n = 0; $n < $stmt_cnt; $n++) { - $herectx .= raw_line($linenr, $n) . "\n"; - } - - if ($dstat ne '' && - $dstat !~ /^(?:$Ident|-?$Constant),$/ && # 10, // foo(), - $dstat !~ /^(?:$Ident|-?$Constant);$/ && # foo(); - $dstat !~ /^[!~-]?(?:$Lval|$Constant)$/ && # 10 // foo() // !foo // ~foo // -foo // foo->bar // foo.bar->baz - $dstat !~ /^'X'$/ && $dstat !~ /^'XX'$/ && # character constants - $dstat !~ /$exceptions/ && - $dstat !~ /^\.$Ident\s*=/ && # .foo = - $dstat !~ /^(?:\#\s*$Ident|\#\s*$Constant)\s*$/ && # stringification #foo - $dstat !~ /^do\s*$Constant\s*while\s*$Constant;?$/ && # do {...} while (...); // do {...} while (...) - $dstat !~ /^for\s*$Constant$/ && # for (...) - $dstat !~ /^for\s*$Constant\s+(?:$Ident|-?$Constant)$/ && # for (...) bar() - $dstat !~ /^do\s*{/ && # do {... - $dstat !~ /^\(\{/ && # ({... - $ctx !~ /^.\s*#\s*define\s+TRACE_(?:SYSTEM|INCLUDE_FILE|INCLUDE_PATH)\b/) - { - if ($dstat =~ /^\s*if\b/) { - ERROR("MULTISTATEMENT_MACRO_USE_DO_WHILE", - "Macros starting with if should be enclosed by a do - while loop to avoid possible if/else logic defects\n" . "$herectx"); - } elsif ($dstat =~ /;/) { - ERROR("MULTISTATEMENT_MACRO_USE_DO_WHILE", - "Macros with multiple statements should be enclosed in a do - while loop\n" . "$herectx"); - } else { - WARN("COMPLEX_MACRO", - "Macros with complex values should be enclosed in parentheses\n" . "$herectx"); - } - - } - - # Make $define_stmt single line, comment-free, etc - my @stmt_array = split('\n', $define_stmt); - my $first = 1; - $define_stmt = ""; - foreach my $l (@stmt_array) { - $l =~ s/\\$//; - if ($first) { - $define_stmt = $l; - $first = 0; - } elsif ($l =~ /^[\+ ]/) { - $define_stmt .= substr($l, 1); - } - } - $define_stmt =~ s/$;//g; - $define_stmt =~ s/\s+/ /g; - $define_stmt = trim($define_stmt); - -# check if any macro arguments are reused (ignore '...' and 'type') - foreach my $arg (@def_args) { - next if ($arg =~ /\.\.\./); - next if ($arg =~ /^type$/i); - my $tmp_stmt = $define_stmt; - $tmp_stmt =~ s/\b(typeof|__typeof__|__builtin\w+|typecheck\s*\(\s*$Type\s*,|\#+)\s*\(*\s*$arg\s*\)*\b//g; - $tmp_stmt =~ s/\#+\s*$arg\b//g; - $tmp_stmt =~ s/\b$arg\s*\#\#//g; - my $use_cnt = $tmp_stmt =~ s/\b$arg\b//g; - if ($use_cnt > 1) { - CHK("MACRO_ARG_REUSE", - "Macro argument reuse '$arg' - possible side-effects?\n" . "$herectx"); - } -# check if any macro arguments may have other precedence issues - if ($tmp_stmt =~ m/($Operators)?\s*\b$arg\b\s*($Operators)?/m && - ((defined($1) && $1 ne ',') || - (defined($2) && $2 ne ','))) { - CHK("MACRO_ARG_PRECEDENCE", - "Macro argument '$arg' may be better as '($arg)' to avoid precedence issues\n" . "$herectx"); - } - } - -# check for macros with flow control, but without ## concatenation -# ## concatenation is commonly a macro that defines a function so ignore those - if ($has_flow_statement && !$has_arg_concat) { - my $herectx = $here . "\n"; - my $cnt = statement_rawlines($ctx); - - for (my $n = 0; $n < $cnt; $n++) { - $herectx .= raw_line($linenr, $n) . "\n"; - } - WARN("MACRO_WITH_FLOW_CONTROL", - "Macros with flow control statements should be avoided\n" . "$herectx"); - } - -# check for line continuations outside of #defines, preprocessor #, and asm - - } else { - if ($prevline !~ /^..*\\$/ && - $line !~ /^\+\s*\#.*\\$/ && # preprocessor - $line !~ /^\+.*\b(__asm__|asm)\b.*\\$/ && # asm - $line =~ /^\+.*\\$/) { - WARN("LINE_CONTINUATIONS", - "Avoid unnecessary line continuations\n" . $herecurr); - } - } - -# do {} while (0) macro tests: -# single-statement macros do not need to be enclosed in do while (0) loop, -# macro should not end with a semicolon - if ($^V && $^V ge 5.10.0 && - $realfile !~ m@/vmlinux.lds.h$@ && - $line =~ /^.\s*\#\s*define\s+$Ident(\()?/) { - my $ln = $linenr; - my $cnt = $realcnt; - my ($off, $dstat, $dcond, $rest); - my $ctx = ''; - ($dstat, $dcond, $ln, $cnt, $off) = - ctx_statement_block($linenr, $realcnt, 0); - $ctx = $dstat; - - $dstat =~ s/\\\n.//g; - $dstat =~ s/$;/ /g; - - if ($dstat =~ /^\+\s*#\s*define\s+$Ident\s*${balanced_parens}\s*do\s*{(.*)\s*}\s*while\s*\(\s*0\s*\)\s*([;\s]*)\s*$/) { - my $stmts = $2; - my $semis = $3; - - $ctx =~ s/\n*$//; - my $cnt = statement_rawlines($ctx); - my $herectx = $here . "\n"; - - for (my $n = 0; $n < $cnt; $n++) { - $herectx .= raw_line($linenr, $n) . "\n"; - } - - if (($stmts =~ tr/;/;/) == 1 && - $stmts !~ /^\s*(if|while|for|switch)\b/) { - WARN("SINGLE_STATEMENT_DO_WHILE_MACRO", - "Single statement macros should not use a do {} while (0) loop\n" . "$herectx"); - } - if (defined $semis && $semis ne "") { - WARN("DO_WHILE_MACRO_WITH_TRAILING_SEMICOLON", - "do {} while (0) macros should not be semicolon terminated\n" . "$herectx"); - } - } elsif ($dstat =~ /^\+\s*#\s*define\s+$Ident.*;\s*$/) { - $ctx =~ s/\n*$//; - my $cnt = statement_rawlines($ctx); - my $herectx = $here . "\n"; - - for (my $n = 0; $n < $cnt; $n++) { - $herectx .= raw_line($linenr, $n) . "\n"; - } - - WARN("TRAILING_SEMICOLON", - "macros should not use a trailing semicolon\n" . "$herectx"); - } - } - -# make sure symbols are always wrapped with VMLINUX_SYMBOL() ... -# all assignments may have only one of the following with an assignment: -# . -# ALIGN(...) -# VMLINUX_SYMBOL(...) - if ($realfile eq 'vmlinux.lds.h' && $line =~ /(?:(?:^|\s)$Ident\s*=|=\s*$Ident(?:\s|$))/) { - WARN("MISSING_VMLINUX_SYMBOL", - "vmlinux.lds.h needs VMLINUX_SYMBOL() around C-visible symbols\n" . $herecurr); - } - -# check for redundant bracing round if etc - if ($line =~ /(^.*)\bif\b/ && $1 !~ /else\s*$/) { - my ($level, $endln, @chunks) = - ctx_statement_full($linenr, $realcnt, 1); - #print "chunks<$#chunks> linenr<$linenr> endln<$endln> level<$level>\n"; - #print "APW: <<$chunks[1][0]>><<$chunks[1][1]>>\n"; - if ($#chunks > 0 && $level == 0) { - my @allowed = (); - my $allow = 0; - my $seen = 0; - my $herectx = $here . "\n"; - my $ln = $linenr - 1; - for my $chunk (@chunks) { - my ($cond, $block) = @{$chunk}; - - # If the condition carries leading newlines, then count those as offsets. - my ($whitespace) = ($cond =~ /^((?:\s*\n[+-])*\s*)/s); - my $offset = statement_rawlines($whitespace) - 1; - - $allowed[$allow] = 0; - #print "COND<$cond> whitespace<$whitespace> offset<$offset>\n"; - - # We have looked at and allowed this specific line. - $suppress_ifbraces{$ln + $offset} = 1; - - $herectx .= "$rawlines[$ln + $offset]\n[...]\n"; - $ln += statement_rawlines($block) - 1; - - substr($block, 0, length($cond), ''); - - $seen++ if ($block =~ /^\s*{/); - - #print "cond<$cond> block<$block> allowed<$allowed[$allow]>\n"; - if (statement_lines($cond) > 1) { - #print "APW: ALLOWED: cond<$cond>\n"; - $allowed[$allow] = 1; - } - if ($block =~/\b(?:if|for|while)\b/) { - #print "APW: ALLOWED: block<$block>\n"; - $allowed[$allow] = 1; - } - if (statement_block_size($block) > 1) { - #print "APW: ALLOWED: lines block<$block>\n"; - $allowed[$allow] = 1; - } - $allow++; - } - if ($seen) { - my $sum_allowed = 0; - foreach (@allowed) { - $sum_allowed += $_; - } - if ($sum_allowed == 0) { - WARN("BRACES", - "braces {} are not necessary for any arm of this statement\n" . $herectx); - } elsif ($sum_allowed != $allow && - $seen != $allow) { - CHK("BRACES", - "braces {} should be used on all arms of this statement\n" . $herectx); - } - } - } - } - if (!defined $suppress_ifbraces{$linenr - 1} && - $line =~ /\b(if|while|for|else)\b/) { - my $allowed = 0; - - # Check the pre-context. - if (substr($line, 0, $-[0]) =~ /(\}\s*)$/) { - #print "APW: ALLOWED: pre<$1>\n"; - $allowed = 1; - } - - my ($level, $endln, @chunks) = - ctx_statement_full($linenr, $realcnt, $-[0]); - - # Check the condition. - my ($cond, $block) = @{$chunks[0]}; - #print "CHECKING<$linenr> cond<$cond> block<$block>\n"; - if (defined $cond) { - substr($block, 0, length($cond), ''); - } - if (statement_lines($cond) > 1) { - #print "APW: ALLOWED: cond<$cond>\n"; - $allowed = 1; - } - if ($block =~/\b(?:if|for|while)\b/) { - #print "APW: ALLOWED: block<$block>\n"; - $allowed = 1; - } - if (statement_block_size($block) > 1) { - #print "APW: ALLOWED: lines block<$block>\n"; - $allowed = 1; - } - # Check the post-context. - if (defined $chunks[1]) { - my ($cond, $block) = @{$chunks[1]}; - if (defined $cond) { - substr($block, 0, length($cond), ''); - } - if ($block =~ /^\s*\{/) { - #print "APW: ALLOWED: chunk-1 block<$block>\n"; - $allowed = 1; - } - } - if ($level == 0 && $block =~ /^\s*\{/ && !$allowed) { - my $herectx = $here . "\n"; - my $cnt = statement_rawlines($block); - - for (my $n = 0; $n < $cnt; $n++) { - $herectx .= raw_line($linenr, $n) . "\n"; - } - - WARN("BRACES", - "braces {} are not necessary for single statement blocks\n" . $herectx); - } - } - -# check for single line unbalanced braces - if ($sline =~ /^.\s*\}\s*else\s*$/ || - $sline =~ /^.\s*else\s*\{\s*$/) { - CHK("BRACES", "Unbalanced braces around else statement\n" . $herecurr); - } - -# check for unnecessary blank lines around braces - if (($line =~ /^.\s*}\s*$/ && $prevrawline =~ /^.\s*$/)) { - if (CHK("BRACES", - "Blank lines aren't necessary before a close brace '}'\n" . $hereprev) && - $fix && $prevrawline =~ /^\+/) { - fix_delete_line($fixlinenr - 1, $prevrawline); - } - } - if (($rawline =~ /^.\s*$/ && $prevline =~ /^..*{\s*$/)) { - if (CHK("BRACES", - "Blank lines aren't necessary after an open brace '{'\n" . $hereprev) && - $fix) { - fix_delete_line($fixlinenr, $rawline); - } - } - -# no volatiles please - my $asm_volatile = qr{\b(__asm__|asm)\s+(__volatile__|volatile)\b}; - if ($line =~ /\bvolatile\b/ && $line !~ /$asm_volatile/) { - WARN("VOLATILE", - "Use of volatile is usually wrong: see Documentation/process/volatile-considered-harmful.rst\n" . $herecurr); - } - -# Check for user-visible strings broken across lines, which breaks the ability -# to grep for the string. Make exceptions when the previous string ends in a -# newline (multiple lines in one string constant) or '\t', '\r', ';', or '{' -# (common in inline assembly) or is a octal \123 or hexadecimal \xaf value - if ($line =~ /^\+\s*$String/ && - $prevline =~ /"\s*$/ && - $prevrawline !~ /(?:\\(?:[ntr]|[0-7]{1,3}|x[0-9a-fA-F]{1,2})|;\s*|\{\s*)"\s*$/) { - if (WARN("SPLIT_STRING", - "quoted string split across lines\n" . $hereprev) && - $fix && - $prevrawline =~ /^\+.*"\s*$/ && - $last_coalesced_string_linenr != $linenr - 1) { - my $extracted_string = get_quoted_string($line, $rawline); - my $comma_close = ""; - if ($rawline =~ /\Q$extracted_string\E(\s*\)\s*;\s*$|\s*,\s*)/) { - $comma_close = $1; - } - - fix_delete_line($fixlinenr - 1, $prevrawline); - fix_delete_line($fixlinenr, $rawline); - my $fixedline = $prevrawline; - $fixedline =~ s/"\s*$//; - $fixedline .= substr($extracted_string, 1) . trim($comma_close); - fix_insert_line($fixlinenr - 1, $fixedline); - $fixedline = $rawline; - $fixedline =~ s/\Q$extracted_string\E\Q$comma_close\E//; - if ($fixedline !~ /\+\s*$/) { - fix_insert_line($fixlinenr, $fixedline); - } - $last_coalesced_string_linenr = $linenr; - } - } - -# check for missing a space in a string concatenation - if ($prevrawline =~ /[^\\]\w"$/ && $rawline =~ /^\+[\t ]+"\w/) { - WARN('MISSING_SPACE', - "break quoted strings at a space character\n" . $hereprev); - } - -# check for an embedded function name in a string when the function is known -# This does not work very well for -f --file checking as it depends on patch -# context providing the function name or a single line form for in-file -# function declarations - if ($line =~ /^\+.*$String/ && - defined($context_function) && - get_quoted_string($line, $rawline) =~ /\b$context_function\b/ && - length(get_quoted_string($line, $rawline)) != (length($context_function) + 2)) { - WARN("EMBEDDED_FUNCTION_NAME", - "Prefer using '\"%s...\", __func__' to using '$context_function', this function's name, in a string\n" . $herecurr); - } - -# check for spaces before a quoted newline - if ($rawline =~ /^.*\".*\s\\n/) { - if (WARN("QUOTED_WHITESPACE_BEFORE_NEWLINE", - "unnecessary whitespace before a quoted newline\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/^(\+.*\".*)\s+\\n/$1\\n/; - } - - } - -# concatenated string without spaces between elements - if ($line =~ /$String[A-Z_]/ || $line =~ /[A-Za-z0-9_]$String/) { - CHK("CONCATENATED_STRING", - "Concatenated strings should use spaces between elements\n" . $herecurr); - } - -# uncoalesced string fragments - if ($line =~ /$String\s*"/) { - WARN("STRING_FRAGMENTS", - "Consecutive strings are generally better as a single string\n" . $herecurr); - } - -# check for non-standard and hex prefixed decimal printf formats - my $show_L = 1; #don't show the same defect twice - my $show_Z = 1; - while ($line =~ /(?:^|")([X\t]*)(?:"|$)/g) { - my $string = substr($rawline, $-[1], $+[1] - $-[1]); - $string =~ s/%%/__/g; - # check for %L - if ($show_L && $string =~ /%[\*\d\.\$]*L([diouxX])/) { - WARN("PRINTF_L", - "\%L$1 is non-standard C, use %ll$1\n" . $herecurr); - $show_L = 0; - } - # check for %Z - if ($show_Z && $string =~ /%[\*\d\.\$]*Z([diouxX])/) { - WARN("PRINTF_Z", - "%Z$1 is non-standard C, use %z$1\n" . $herecurr); - $show_Z = 0; - } - # check for 0x<decimal> - if ($string =~ /0x%[\*\d\.\$\Llzth]*[diou]/) { - ERROR("PRINTF_0XDECIMAL", - "Prefixing 0x with decimal output is defective\n" . $herecurr); - } - } - -# check for line continuations in quoted strings with odd counts of " - if ($rawline =~ /\\$/ && $rawline =~ tr/"/"/ % 2) { - WARN("LINE_CONTINUATIONS", - "Avoid line continuations in quoted strings\n" . $herecurr); - } - -# warn about #if 0 - if ($line =~ /^.\s*\#\s*if\s+0\b/) { - CHK("REDUNDANT_CODE", - "if this code is redundant consider removing it\n" . - $herecurr); - } - -# check for needless "if (<foo>) fn(<foo>)" uses - if ($prevline =~ /\bif\s*\(\s*($Lval)\s*\)/) { - my $tested = quotemeta($1); - my $expr = '\s*\(\s*' . $tested . '\s*\)\s*;'; - if ($line =~ /\b(kfree|usb_free_urb|debugfs_remove(?:_recursive)?|(?:kmem_cache|mempool|dma_pool)_destroy)$expr/) { - my $func = $1; - if (WARN('NEEDLESS_IF', - "$func(NULL) is safe and this check is probably not required\n" . $hereprev) && - $fix) { - my $do_fix = 1; - my $leading_tabs = ""; - my $new_leading_tabs = ""; - if ($lines[$linenr - 2] =~ /^\+(\t*)if\s*\(\s*$tested\s*\)\s*$/) { - $leading_tabs = $1; - } else { - $do_fix = 0; - } - if ($lines[$linenr - 1] =~ /^\+(\t+)$func\s*\(\s*$tested\s*\)\s*;\s*$/) { - $new_leading_tabs = $1; - if (length($leading_tabs) + 1 ne length($new_leading_tabs)) { - $do_fix = 0; - } - } else { - $do_fix = 0; - } - if ($do_fix) { - fix_delete_line($fixlinenr - 1, $prevrawline); - $fixed[$fixlinenr] =~ s/^\+$new_leading_tabs/\+$leading_tabs/; - } - } - } - } - -# check for unnecessary "Out of Memory" messages - if ($line =~ /^\+.*\b$logFunctions\s*\(/ && - $prevline =~ /^[ \+]\s*if\s*\(\s*(\!\s*|NULL\s*==\s*)?($Lval)(\s*==\s*NULL\s*)?\s*\)/ && - (defined $1 || defined $3) && - $linenr > 3) { - my $testval = $2; - my $testline = $lines[$linenr - 3]; - - my ($s, $c) = ctx_statement_block($linenr - 3, $realcnt, 0); -# print("line: <$line>\nprevline: <$prevline>\ns: <$s>\nc: <$c>\n\n\n"); - - if ($s =~ /(?:^|\n)[ \+]\s*(?:$Type\s*)?\Q$testval\E\s*=\s*(?:\([^\)]*\)\s*)?\s*(?:devm_)?(?:[kv][czm]alloc(?:_node|_array)?\b|kstrdup|kmemdup|(?:dev_)?alloc_skb)/) { - WARN("OOM_MESSAGE", - "Possible unnecessary 'out of memory' message\n" . $hereprev); - } - } - -# check for logging functions with KERN_<LEVEL> - if ($line !~ /printk(?:_ratelimited|_once)?\s*\(/ && - $line =~ /\b$logFunctions\s*\(.*\b(KERN_[A-Z]+)\b/) { - my $level = $1; - if (WARN("UNNECESSARY_KERN_LEVEL", - "Possible unnecessary $level\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\s*$level\s*//; - } - } - -# check for logging continuations - if ($line =~ /\bprintk\s*\(\s*KERN_CONT\b|\bpr_cont\s*\(/) { - WARN("LOGGING_CONTINUATION", - "Avoid logging continuation uses where feasible\n" . $herecurr); - } - -# check for mask then right shift without a parentheses - if ($^V && $^V ge 5.10.0 && - $line =~ /$LvalOrFunc\s*\&\s*($LvalOrFunc)\s*>>/ && - $4 !~ /^\&/) { # $LvalOrFunc may be &foo, ignore if so - WARN("MASK_THEN_SHIFT", - "Possible precedence defect with mask then right shift - may need parentheses\n" . $herecurr); - } - -# check for pointer comparisons to NULL - if ($^V && $^V ge 5.10.0) { - while ($line =~ /\b$LvalOrFunc\s*(==|\!=)\s*NULL\b/g) { - my $val = $1; - my $equal = "!"; - $equal = "" if ($4 eq "!="); - if (CHK("COMPARISON_TO_NULL", - "Comparison to NULL could be written \"${equal}${val}\"\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\b\Q$val\E\s*(?:==|\!=)\s*NULL\b/$equal$val/; - } - } - } - -# check for bad placement of section $InitAttribute (e.g.: __initdata) - if ($line =~ /(\b$InitAttribute\b)/) { - my $attr = $1; - if ($line =~ /^\+\s*static\s+(?:const\s+)?(?:$attr\s+)?($NonptrTypeWithAttr)\s+(?:$attr\s+)?($Ident(?:\[[^]]*\])?)\s*[=;]/) { - my $ptr = $1; - my $var = $2; - if ((($ptr =~ /\b(union|struct)\s+$attr\b/ && - ERROR("MISPLACED_INIT", - "$attr should be placed after $var\n" . $herecurr)) || - ($ptr !~ /\b(union|struct)\s+$attr\b/ && - WARN("MISPLACED_INIT", - "$attr should be placed after $var\n" . $herecurr))) && - $fix) { - $fixed[$fixlinenr] =~ s/(\bstatic\s+(?:const\s+)?)(?:$attr\s+)?($NonptrTypeWithAttr)\s+(?:$attr\s+)?($Ident(?:\[[^]]*\])?)\s*([=;])\s*/"$1" . trim(string_find_replace($2, "\\s*$attr\\s*", " ")) . " " . trim(string_find_replace($3, "\\s*$attr\\s*", "")) . " $attr" . ("$4" eq ";" ? ";" : " = ")/e; - } - } - } - -# check for $InitAttributeData (ie: __initdata) with const - if ($line =~ /\bconst\b/ && $line =~ /($InitAttributeData)/) { - my $attr = $1; - $attr =~ /($InitAttributePrefix)(.*)/; - my $attr_prefix = $1; - my $attr_type = $2; - if (ERROR("INIT_ATTRIBUTE", - "Use of const init definition must use ${attr_prefix}initconst\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/$InitAttributeData/${attr_prefix}initconst/; - } - } - -# check for $InitAttributeConst (ie: __initconst) without const - if ($line !~ /\bconst\b/ && $line =~ /($InitAttributeConst)/) { - my $attr = $1; - if (ERROR("INIT_ATTRIBUTE", - "Use of $attr requires a separate use of const\n" . $herecurr) && - $fix) { - my $lead = $fixed[$fixlinenr] =~ - /(^\+\s*(?:static\s+))/; - $lead = rtrim($1); - $lead = "$lead " if ($lead !~ /^\+$/); - $lead = "${lead}const "; - $fixed[$fixlinenr] =~ s/(^\+\s*(?:static\s+))/$lead/; - } - } - -# check for __read_mostly with const non-pointer (should just be const) - if ($line =~ /\b__read_mostly\b/ && - $line =~ /($Type)\s*$Ident/ && $1 !~ /\*\s*$/ && $1 =~ /\bconst\b/) { - if (ERROR("CONST_READ_MOSTLY", - "Invalid use of __read_mostly with const type\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\s+__read_mostly\b//; - } - } - -# don't use __constant_<foo> functions outside of include/uapi/ - if ($realfile !~ m@^include/uapi/@ && - $line =~ /(__constant_(?:htons|ntohs|[bl]e(?:16|32|64)_to_cpu|cpu_to_[bl]e(?:16|32|64)))\s*\(/) { - my $constant_func = $1; - my $func = $constant_func; - $func =~ s/^__constant_//; - if (WARN("CONSTANT_CONVERSION", - "$constant_func should be $func\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\b$constant_func\b/$func/g; - } - } - -# prefer usleep_range over udelay - if ($line =~ /\budelay\s*\(\s*(\d+)\s*\)/) { - my $delay = $1; - # ignore udelay's < 10, however - if (! ($delay < 10) ) { - CHK("USLEEP_RANGE", - "usleep_range is preferred over udelay; see Documentation/timers/timers-howto.txt\n" . $herecurr); - } - if ($delay > 2000) { - WARN("LONG_UDELAY", - "long udelay - prefer mdelay; see arch/arm/include/asm/delay.h\n" . $herecurr); - } - } - -# warn about unexpectedly long msleep's - if ($line =~ /\bmsleep\s*\((\d+)\);/) { - if ($1 < 20) { - WARN("MSLEEP", - "msleep < 20ms can sleep for up to 20ms; see Documentation/timers/timers-howto.txt\n" . $herecurr); - } - } - -# check for comparisons of jiffies - if ($line =~ /\bjiffies\s*$Compare|$Compare\s*jiffies\b/) { - WARN("JIFFIES_COMPARISON", - "Comparing jiffies is almost always wrong; prefer time_after, time_before and friends\n" . $herecurr); - } - -# check for comparisons of get_jiffies_64() - if ($line =~ /\bget_jiffies_64\s*\(\s*\)\s*$Compare|$Compare\s*get_jiffies_64\s*\(\s*\)/) { - WARN("JIFFIES_COMPARISON", - "Comparing get_jiffies_64() is almost always wrong; prefer time_after64, time_before64 and friends\n" . $herecurr); - } - -# warn about #ifdefs in C files -# if ($line =~ /^.\s*\#\s*if(|n)def/ && ($realfile =~ /\.c$/)) { -# print "#ifdef in C files should be avoided\n"; -# print "$herecurr"; -# $clean = 0; -# } - -# warn about spacing in #ifdefs - if ($line =~ /^.\s*\#\s*(ifdef|ifndef|elif)\s\s+/) { - if (ERROR("SPACING", - "exactly one space required after that #$1\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/^(.\s*\#\s*(ifdef|ifndef|elif))\s{2,}/$1 /; - } - - } - -# check for spinlock_t definitions without a comment. - if ($line =~ /^.\s*(struct\s+mutex|spinlock_t)\s+\S+;/ || - $line =~ /^.\s*(DEFINE_MUTEX)\s*\(/) { - my $which = $1; - if (!ctx_has_comment($first_line, $linenr)) { - CHK("UNCOMMENTED_DEFINITION", - "$1 definition without comment\n" . $herecurr); - } - } -# check for memory barriers without a comment. - - my $barriers = qr{ - mb| - rmb| - wmb| - read_barrier_depends - }x; - my $barrier_stems = qr{ - mb__before_atomic| - mb__after_atomic| - store_release| - load_acquire| - store_mb| - (?:$barriers) - }x; - my $all_barriers = qr{ - (?:$barriers)| - smp_(?:$barrier_stems)| - virt_(?:$barrier_stems) - }x; - - if ($line =~ /\b(?:$all_barriers)\s*\(/) { - if (!ctx_has_comment($first_line, $linenr)) { - WARN("MEMORY_BARRIER", - "memory barrier without comment\n" . $herecurr); - } - } - - my $underscore_smp_barriers = qr{__smp_(?:$barrier_stems)}x; - - if ($realfile !~ m@^include/asm-generic/@ && - $realfile !~ m@/barrier\.h$@ && - $line =~ m/\b(?:$underscore_smp_barriers)\s*\(/ && - $line !~ m/^.\s*\#\s*define\s+(?:$underscore_smp_barriers)\s*\(/) { - WARN("MEMORY_BARRIER", - "__smp memory barriers shouldn't be used outside barrier.h and asm-generic\n" . $herecurr); - } - -# check for waitqueue_active without a comment. - if ($line =~ /\bwaitqueue_active\s*\(/) { - if (!ctx_has_comment($first_line, $linenr)) { - WARN("WAITQUEUE_ACTIVE", - "waitqueue_active without comment\n" . $herecurr); - } - } - -# check of hardware specific defines - if ($line =~ m@^.\s*\#\s*if.*\b(__i386__|__powerpc64__|__sun__|__s390x__)\b@ && $realfile !~ m@include/asm-@) { - CHK("ARCH_DEFINES", - "architecture specific defines should be avoided\n" . $herecurr); - } - -# check that the storage class is not after a type - if ($line =~ /\b($Type)\s+($Storage)\b/) { - WARN("STORAGE_CLASS", - "storage class '$2' should be located before type '$1'\n" . $herecurr); - } -# Check that the storage class is at the beginning of a declaration - if ($line =~ /\b$Storage\b/ && - $line !~ /^.\s*$Storage/ && - $line =~ /^.\s*(.+?)\$Storage\s/ && - $1 !~ /[\,\)]\s*$/) { - WARN("STORAGE_CLASS", - "storage class should be at the beginning of the declaration\n" . $herecurr); - } - -# check the location of the inline attribute, that it is between -# storage class and type. - if ($line =~ /\b$Type\s+$Inline\b/ || - $line =~ /\b$Inline\s+$Storage\b/) { - ERROR("INLINE_LOCATION", - "inline keyword should sit between storage class and type\n" . $herecurr); - } - -# Check for __inline__ and __inline, prefer inline - if ($realfile !~ m@\binclude/uapi/@ && - $line =~ /\b(__inline__|__inline)\b/) { - if (WARN("INLINE", - "plain inline is preferred over $1\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\b(__inline__|__inline)\b/inline/; - - } - } - -# Check for __attribute__ packed, prefer __packed - if ($realfile !~ m@\binclude/uapi/@ && - $line =~ /\b__attribute__\s*\(\s*\(.*\bpacked\b/) { - WARN("PREFER_PACKED", - "__packed is preferred over __attribute__((packed))\n" . $herecurr); - } - -# Check for __attribute__ aligned, prefer __aligned - if ($realfile !~ m@\binclude/uapi/@ && - $line =~ /\b__attribute__\s*\(\s*\(.*aligned/) { - WARN("PREFER_ALIGNED", - "__aligned(size) is preferred over __attribute__((aligned(size)))\n" . $herecurr); - } - -# Check for __attribute__ format(printf, prefer __printf - if ($realfile !~ m@\binclude/uapi/@ && - $line =~ /\b__attribute__\s*\(\s*\(\s*format\s*\(\s*printf/) { - if (WARN("PREFER_PRINTF", - "__printf(string-index, first-to-check) is preferred over __attribute__((format(printf, string-index, first-to-check)))\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\b__attribute__\s*\(\s*\(\s*format\s*\(\s*printf\s*,\s*(.*)\)\s*\)\s*\)/"__printf(" . trim($1) . ")"/ex; - - } - } - -# Check for __attribute__ format(scanf, prefer __scanf - if ($realfile !~ m@\binclude/uapi/@ && - $line =~ /\b__attribute__\s*\(\s*\(\s*format\s*\(\s*scanf\b/) { - if (WARN("PREFER_SCANF", - "__scanf(string-index, first-to-check) is preferred over __attribute__((format(scanf, string-index, first-to-check)))\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\b__attribute__\s*\(\s*\(\s*format\s*\(\s*scanf\s*,\s*(.*)\)\s*\)\s*\)/"__scanf(" . trim($1) . ")"/ex; - } - } - -# Check for __attribute__ weak, or __weak declarations (may have link issues) - if ($^V && $^V ge 5.10.0 && - $line =~ /(?:$Declare|$DeclareMisordered)\s*$Ident\s*$balanced_parens\s*(?:$Attribute)?\s*;/ && - ($line =~ /\b__attribute__\s*\(\s*\(.*\bweak\b/ || - $line =~ /\b__weak\b/)) { - ERROR("WEAK_DECLARATION", - "Using weak declarations can have unintended link defects\n" . $herecurr); - } - -# check for c99 types like uint8_t - if ($line =~ /\b($Declare)\s*$Ident\s*[=;,\[]/) { - my $type = $1; - if ($type =~ /\b($typeC99Typedefs)\b/) { - $type = $1; - my $kernel_type = 'u'; - $kernel_type = 's' if ($type =~ /^_*[si]/); - $type =~ /(\d+)/; - $kernel_type .= $1.'_t'; - WARN("PREFER_KERNEL_TYPES", - "Prefer kernel type '$kernel_type' over '$type'\n" . $herecurr) - } - } - -# check for cast of C90 native int or longer types constants - if ($line =~ /(\(\s*$C90_int_types\s*\)\s*)($Constant)\b/) { - my $cast = $1; - my $const = $2; - if (WARN("TYPECAST_INT_CONSTANT", - "Unnecessary typecast of c90 int constant\n" . $herecurr) && - $fix) { - my $suffix = ""; - my $newconst = $const; - $newconst =~ s/${Int_type}$//; - $suffix .= 'U' if ($cast =~ /\bunsigned\b/); - if ($cast =~ /\blong\s+long\b/) { - $suffix .= 'LL'; - } elsif ($cast =~ /\blong\b/) { - $suffix .= 'L'; - } - $fixed[$fixlinenr] =~ s/\Q$cast\E$const\b/$newconst$suffix/; - } - } - -# check for sizeof(&) - if ($line =~ /\bsizeof\s*\(\s*\&/) { - WARN("SIZEOF_ADDRESS", - "sizeof(& should be avoided\n" . $herecurr); - } - -# check for sizeof without parenthesis - if ($line =~ /\bsizeof\s+((?:\*\s*|)$Lval|$Type(?:\s+$Lval|))/) { - if (WARN("SIZEOF_PARENTHESIS", - "sizeof $1 should be sizeof($1)\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\bsizeof\s+((?:\*\s*|)$Lval|$Type(?:\s+$Lval|))/"sizeof(" . trim($1) . ")"/ex; - } - } - -# check for struct spinlock declarations - if ($line =~ /^.\s*\bstruct\s+spinlock\s+\w+\s*;/) { - WARN("USE_SPINLOCK_T", - "struct spinlock should be spinlock_t\n" . $herecurr); - } - -# check for seq_printf uses that could be seq_puts - if ($sline =~ /\bseq_printf\s*\(.*"\s*\)\s*;\s*$/) { - my $fmt = get_quoted_string($line, $rawline); - $fmt =~ s/%%//g; - if ($fmt !~ /%/) { - if (WARN("PREFER_SEQ_PUTS", - "Prefer seq_puts to seq_printf\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\bseq_printf\b/seq_puts/; - } - } - } - - # check for vsprintf extension %p<foo> misuses - if ($^V && $^V ge 5.10.0 && - defined $stat && - $stat =~ /^\+(?![^\{]*\{\s*).*\b(\w+)\s*\(.*$String\s*,/s && - $1 !~ /^_*volatile_*$/) { - my $bad_extension = ""; - my $lc = $stat =~ tr@\n@@; - $lc = $lc + $linenr; - for (my $count = $linenr; $count <= $lc; $count++) { - my $fmt = get_quoted_string($lines[$count - 1], raw_line($count, 0)); - $fmt =~ s/%%//g; - if ($fmt =~ /(\%[\*\d\.]*p(?![\WFfSsBKRraEhMmIiUDdgVCbGNO]).)/) { - $bad_extension = $1; - last; - } - } - if ($bad_extension ne "") { - my $stat_real = raw_line($linenr, 0); - for (my $count = $linenr + 1; $count <= $lc; $count++) { - $stat_real = $stat_real . "\n" . raw_line($count, 0); - } - WARN("VSPRINTF_POINTER_EXTENSION", - "Invalid vsprintf pointer extension '$bad_extension'\n" . "$here\n$stat_real\n"); - } - } - -# Check for misused memsets - if ($^V && $^V ge 5.10.0 && - defined $stat && - $stat =~ /^\+(?:.*?)\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*$FuncArg\s*\)/) { - - my $ms_addr = $2; - my $ms_val = $7; - my $ms_size = $12; - - if ($ms_size =~ /^(0x|)0$/i) { - ERROR("MEMSET", - "memset to 0's uses 0 as the 2nd argument, not the 3rd\n" . "$here\n$stat\n"); - } elsif ($ms_size =~ /^(0x|)1$/i) { - WARN("MEMSET", - "single byte memset is suspicious. Swapped 2nd/3rd argument?\n" . "$here\n$stat\n"); - } - } - -# Check for memcpy(foo, bar, ETH_ALEN) that could be ether_addr_copy(foo, bar) -# if ($^V && $^V ge 5.10.0 && -# defined $stat && -# $stat =~ /^\+(?:.*?)\bmemcpy\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) { -# if (WARN("PREFER_ETHER_ADDR_COPY", -# "Prefer ether_addr_copy() over memcpy() if the Ethernet addresses are __aligned(2)\n" . "$here\n$stat\n") && -# $fix) { -# $fixed[$fixlinenr] =~ s/\bmemcpy\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/ether_addr_copy($2, $7)/; -# } -# } - -# Check for memcmp(foo, bar, ETH_ALEN) that could be ether_addr_equal*(foo, bar) -# if ($^V && $^V ge 5.10.0 && -# defined $stat && -# $stat =~ /^\+(?:.*?)\bmemcmp\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) { -# WARN("PREFER_ETHER_ADDR_EQUAL", -# "Prefer ether_addr_equal() or ether_addr_equal_unaligned() over memcmp()\n" . "$here\n$stat\n") -# } - -# check for memset(foo, 0x0, ETH_ALEN) that could be eth_zero_addr -# check for memset(foo, 0xFF, ETH_ALEN) that could be eth_broadcast_addr -# if ($^V && $^V ge 5.10.0 && -# defined $stat && -# $stat =~ /^\+(?:.*?)\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) { -# -# my $ms_val = $7; -# -# if ($ms_val =~ /^(?:0x|)0+$/i) { -# if (WARN("PREFER_ETH_ZERO_ADDR", -# "Prefer eth_zero_addr over memset()\n" . "$here\n$stat\n") && -# $fix) { -# $fixed[$fixlinenr] =~ s/\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*,\s*ETH_ALEN\s*\)/eth_zero_addr($2)/; -# } -# } elsif ($ms_val =~ /^(?:0xff|255)$/i) { -# if (WARN("PREFER_ETH_BROADCAST_ADDR", -# "Prefer eth_broadcast_addr() over memset()\n" . "$here\n$stat\n") && -# $fix) { -# $fixed[$fixlinenr] =~ s/\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*,\s*ETH_ALEN\s*\)/eth_broadcast_addr($2)/; -# } -# } -# } - -# typecasts on min/max could be min_t/max_t - if ($^V && $^V ge 5.10.0 && - defined $stat && - $stat =~ /^\+(?:.*?)\b(min|max)\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\)/) { - if (defined $2 || defined $7) { - my $call = $1; - my $cast1 = deparenthesize($2); - my $arg1 = $3; - my $cast2 = deparenthesize($7); - my $arg2 = $8; - my $cast; - - if ($cast1 ne "" && $cast2 ne "" && $cast1 ne $cast2) { - $cast = "$cast1 or $cast2"; - } elsif ($cast1 ne "") { - $cast = $cast1; - } else { - $cast = $cast2; - } - WARN("MINMAX", - "$call() should probably be ${call}_t($cast, $arg1, $arg2)\n" . "$here\n$stat\n"); - } - } - -# check usleep_range arguments - if ($^V && $^V ge 5.10.0 && - defined $stat && - $stat =~ /^\+(?:.*?)\busleep_range\s*\(\s*($FuncArg)\s*,\s*($FuncArg)\s*\)/) { - my $min = $1; - my $max = $7; - if ($min eq $max) { - WARN("USLEEP_RANGE", - "usleep_range should not use min == max args; see Documentation/timers/timers-howto.txt\n" . "$here\n$stat\n"); - } elsif ($min =~ /^\d+$/ && $max =~ /^\d+$/ && - $min > $max) { - WARN("USLEEP_RANGE", - "usleep_range args reversed, use min then max; see Documentation/timers/timers-howto.txt\n" . "$here\n$stat\n"); - } - } - -# check for naked sscanf - if ($^V && $^V ge 5.10.0 && - defined $stat && - $line =~ /\bsscanf\b/ && - ($stat !~ /$Ident\s*=\s*sscanf\s*$balanced_parens/ && - $stat !~ /\bsscanf\s*$balanced_parens\s*(?:$Compare)/ && - $stat !~ /(?:$Compare)\s*\bsscanf\s*$balanced_parens/)) { - my $lc = $stat =~ tr@\n@@; - $lc = $lc + $linenr; - my $stat_real = raw_line($linenr, 0); - for (my $count = $linenr + 1; $count <= $lc; $count++) { - $stat_real = $stat_real . "\n" . raw_line($count, 0); - } - WARN("NAKED_SSCANF", - "unchecked sscanf return value\n" . "$here\n$stat_real\n"); - } - -# check for simple sscanf that should be kstrto<foo> - if ($^V && $^V ge 5.10.0 && - defined $stat && - $line =~ /\bsscanf\b/) { - my $lc = $stat =~ tr@\n@@; - $lc = $lc + $linenr; - my $stat_real = raw_line($linenr, 0); - for (my $count = $linenr + 1; $count <= $lc; $count++) { - $stat_real = $stat_real . "\n" . raw_line($count, 0); - } - if ($stat_real =~ /\bsscanf\b\s*\(\s*$FuncArg\s*,\s*("[^"]+")/) { - my $format = $6; - my $count = $format =~ tr@%@%@; - if ($count == 1 && - $format =~ /^"\%(?i:ll[udxi]|[udxi]ll|ll|[hl]h?[udxi]|[udxi][hl]h?|[hl]h?|[udxi])"$/) { - WARN("SSCANF_TO_KSTRTO", - "Prefer kstrto<type> to single variable sscanf\n" . "$here\n$stat_real\n"); - } - } - } - -# check for new externs in .h files. - if ($realfile =~ /\.h$/ && - $line =~ /^\+\s*(extern\s+)$Type\s*$Ident\s*\(/s) { - if (CHK("AVOID_EXTERNS", - "extern prototypes should be avoided in .h files\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/(.*)\bextern\b\s*(.*)/$1$2/; - } - } - -# check for new externs in .c files. - if ($realfile =~ /\.c$/ && defined $stat && - $stat =~ /^.\s*(?:extern\s+)?$Type\s+($Ident)(\s*)\(/s) - { - my $function_name = $1; - my $paren_space = $2; - - my $s = $stat; - if (defined $cond) { - substr($s, 0, length($cond), ''); - } - if ($s =~ /^\s*;/ && - $function_name ne 'uninitialized_var') - { - WARN("AVOID_EXTERNS", - "externs should be avoided in .c files\n" . $herecurr); - } - - if ($paren_space =~ /\n/) { - WARN("FUNCTION_ARGUMENTS", - "arguments for function declarations should follow identifier\n" . $herecurr); - } - - } elsif ($realfile =~ /\.c$/ && defined $stat && - $stat =~ /^.\s*extern\s+/) - { - WARN("AVOID_EXTERNS", - "externs should be avoided in .c files\n" . $herecurr); - } - -# check for function declarations that have arguments without identifier names - if (defined $stat && - $stat =~ /^.\s*(?:extern\s+)?$Type\s*$Ident\s*\(\s*([^{]+)\s*\)\s*;/s && - $1 ne "void") { - my $args = trim($1); - while ($args =~ m/\s*($Type\s*(?:$Ident|\(\s*\*\s*$Ident?\s*\)\s*$balanced_parens)?)/g) { - my $arg = trim($1); - if ($arg =~ /^$Type$/ && $arg !~ /enum\s+$Ident$/) { - WARN("FUNCTION_ARGUMENTS", - "function definition argument '$arg' should also have an identifier name\n" . $herecurr); - } - } - } - -# check for function definitions - if ($^V && $^V ge 5.10.0 && - defined $stat && - $stat =~ /^.\s*(?:$Storage\s+)?$Type\s*($Ident)\s*$balanced_parens\s*{/s) { - $context_function = $1; - -# check for multiline function definition with misplaced open brace - my $ok = 0; - my $cnt = statement_rawlines($stat); - my $herectx = $here . "\n"; - for (my $n = 0; $n < $cnt; $n++) { - my $rl = raw_line($linenr, $n); - $herectx .= $rl . "\n"; - $ok = 1 if ($rl =~ /^[ \+]\{/); - $ok = 1 if ($rl =~ /\{/ && $n == 0); - last if $rl =~ /^[ \+].*\{/; - } - if (!$ok) { - ERROR("OPEN_BRACE", - "open brace '{' following function definitions go on the next line\n" . $herectx); - } - } - -# checks for new __setup's - if ($rawline =~ /\b__setup\("([^"]*)"/) { - my $name = $1; - - if (!grep(/$name/, @setup_docs)) { - CHK("UNDOCUMENTED_SETUP", - "__setup appears un-documented -- check Documentation/admin-guide/kernel-parameters.rst\n" . $herecurr); - } - } - -# check for pointless casting of kmalloc return - if ($line =~ /\*\s*\)\s*[kv][czm]alloc(_node){0,1}\b/) { - WARN("UNNECESSARY_CASTS", - "unnecessary cast may hide bugs, see http://c-faq.com/malloc/mallocnocast.html\n" . $herecurr); - } - -# alloc style -# p = alloc(sizeof(struct foo), ...) should be p = alloc(sizeof(*p), ...) - if ($^V && $^V ge 5.10.0 && - $line =~ /\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*([kv][mz]alloc(?:_node)?)\s*\(\s*(sizeof\s*\(\s*struct\s+$Lval\s*\))/) { - CHK("ALLOC_SIZEOF_STRUCT", - "Prefer $3(sizeof(*$1)...) over $3($4...)\n" . $herecurr); - } - -# check for k[mz]alloc with multiplies that could be kmalloc_array/kcalloc - if ($^V && $^V ge 5.10.0 && - defined $stat && - $stat =~ /^\+\s*($Lval)\s*\=\s*(?:$balanced_parens)?\s*(k[mz]alloc)\s*\(\s*($FuncArg)\s*\*\s*($FuncArg)\s*,/) { - my $oldfunc = $3; - my $a1 = $4; - my $a2 = $10; - my $newfunc = "kmalloc_array"; - $newfunc = "kcalloc" if ($oldfunc eq "kzalloc"); - my $r1 = $a1; - my $r2 = $a2; - if ($a1 =~ /^sizeof\s*\S/) { - $r1 = $a2; - $r2 = $a1; - } - if ($r1 !~ /^sizeof\b/ && $r2 =~ /^sizeof\s*\S/ && - !($r1 =~ /^$Constant$/ || $r1 =~ /^[A-Z_][A-Z0-9_]*$/)) { - my $ctx = ''; - my $herectx = $here . "\n"; - my $cnt = statement_rawlines($stat); - for (my $n = 0; $n < $cnt; $n++) { - $herectx .= raw_line($linenr, $n) . "\n"; - } - if (WARN("ALLOC_WITH_MULTIPLY", - "Prefer $newfunc over $oldfunc with multiply\n" . $herectx) && - $cnt == 1 && - $fix) { - $fixed[$fixlinenr] =~ s/\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*(k[mz]alloc)\s*\(\s*($FuncArg)\s*\*\s*($FuncArg)/$1 . ' = ' . "$newfunc(" . trim($r1) . ', ' . trim($r2)/e; - } - } - } - -# check for krealloc arg reuse - if ($^V && $^V ge 5.10.0 && - $line =~ /\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*krealloc\s*\(\s*\1\s*,/) { - WARN("KREALLOC_ARG_REUSE", - "Reusing the krealloc arg is almost always a bug\n" . $herecurr); - } - -# check for alloc argument mismatch - if ($line =~ /\b(kcalloc|kmalloc_array)\s*\(\s*sizeof\b/) { - WARN("ALLOC_ARRAY_ARGS", - "$1 uses number as first arg, sizeof is generally wrong\n" . $herecurr); - } - -# check for multiple semicolons - if ($line =~ /;\s*;\s*$/) { - if (WARN("ONE_SEMICOLON", - "Statements terminations use 1 semicolon\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/(\s*;\s*){2,}$/;/g; - } - } - -# check for #defines like: 1 << <digit> that could be BIT(digit), it is not exported to uapi - if ($realfile !~ m@^include/uapi/@ && - $line =~ /#\s*define\s+\w+\s+\(?\s*1\s*([ulUL]*)\s*\<\<\s*(?:\d+|$Ident)\s*\)?/) { - my $ull = ""; - $ull = "_ULL" if (defined($1) && $1 =~ /ll/i); - if (CHK("BIT_MACRO", - "Prefer using the BIT$ull macro\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\(?\s*1\s*[ulUL]*\s*<<\s*(\d+|$Ident)\s*\)?/BIT${ull}($1)/; - } - } - -# check for #if defined CONFIG_<FOO> || defined CONFIG_<FOO>_MODULE - if ($line =~ /^\+\s*#\s*if\s+defined(?:\s*\(?\s*|\s+)(CONFIG_[A-Z_]+)\s*\)?\s*\|\|\s*defined(?:\s*\(?\s*|\s+)\1_MODULE\s*\)?\s*$/) { - my $config = $1; - if (WARN("PREFER_IS_ENABLED", - "Prefer IS_ENABLED(<FOO>) to CONFIG_<FOO> || CONFIG_<FOO>_MODULE\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] = "\+#if IS_ENABLED($config)"; - } - } - -# check for case / default statements not preceded by break/fallthrough/switch - if ($line =~ /^.\s*(?:case\s+(?:$Ident|$Constant)\s*|default):/) { - my $has_break = 0; - my $has_statement = 0; - my $count = 0; - my $prevline = $linenr; - while ($prevline > 1 && ($file || $count < 3) && !$has_break) { - $prevline--; - my $rline = $rawlines[$prevline - 1]; - my $fline = $lines[$prevline - 1]; - last if ($fline =~ /^\@\@/); - next if ($fline =~ /^\-/); - next if ($fline =~ /^.(?:\s*(?:case\s+(?:$Ident|$Constant)[\s$;]*|default):[\s$;]*)*$/); - $has_break = 1 if ($rline =~ /fall[\s_-]*(through|thru)/i); - next if ($fline =~ /^.[\s$;]*$/); - $has_statement = 1; - $count++; - $has_break = 1 if ($fline =~ /\bswitch\b|\b(?:break\s*;[\s$;]*$|return\b|goto\b|continue\b)/); - } - if (!$has_break && $has_statement) { - WARN("MISSING_BREAK", - "Possible switch case/default not preceded by break or fallthrough comment\n" . $herecurr); - } - } - -# check for switch/default statements without a break; - if ($^V && $^V ge 5.10.0 && - defined $stat && - $stat =~ /^\+[$;\s]*(?:case[$;\s]+\w+[$;\s]*:[$;\s]*|)*[$;\s]*\bdefault[$;\s]*:[$;\s]*;/g) { - my $ctx = ''; - my $herectx = $here . "\n"; - my $cnt = statement_rawlines($stat); - for (my $n = 0; $n < $cnt; $n++) { - $herectx .= raw_line($linenr, $n) . "\n"; - } - WARN("DEFAULT_NO_BREAK", - "switch default: should use break\n" . $herectx); - } - -# check for gcc specific __FUNCTION__ - if ($line =~ /\b__FUNCTION__\b/) { - if (WARN("USE_FUNC", - "__func__ should be used instead of gcc specific __FUNCTION__\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\b__FUNCTION__\b/__func__/g; - } - } - -# check for uses of __DATE__, __TIME__, __TIMESTAMP__ - while ($line =~ /\b(__(?:DATE|TIME|TIMESTAMP)__)\b/g) { - ERROR("DATE_TIME", - "Use of the '$1' macro makes the build non-deterministic\n" . $herecurr); - } - -# check for use of yield() - if ($line =~ /\byield\s*\(\s*\)/) { - WARN("YIELD", - "Using yield() is generally wrong. See yield() kernel-doc (sched/core.c)\n" . $herecurr); - } - -# check for comparisons against true and false - if ($line =~ /\+\s*(.*?)\b(true|false|$Lval)\s*(==|\!=)\s*(true|false|$Lval)\b(.*)$/i) { - my $lead = $1; - my $arg = $2; - my $test = $3; - my $otype = $4; - my $trail = $5; - my $op = "!"; - - ($arg, $otype) = ($otype, $arg) if ($arg =~ /^(?:true|false)$/i); - - my $type = lc($otype); - if ($type =~ /^(?:true|false)$/) { - if (("$test" eq "==" && "$type" eq "true") || - ("$test" eq "!=" && "$type" eq "false")) { - $op = ""; - } - - CHK("BOOL_COMPARISON", - "Using comparison to $otype is error prone\n" . $herecurr); - -## maybe suggesting a correct construct would better -## "Using comparison to $otype is error prone. Perhaps use '${lead}${op}${arg}${trail}'\n" . $herecurr); - - } - } - -# check for semaphores initialized locked - if ($line =~ /^.\s*sema_init.+,\W?0\W?\)/) { - WARN("CONSIDER_COMPLETION", - "consider using a completion\n" . $herecurr); - } - -# recommend kstrto* over simple_strto* and strict_strto* - if ($line =~ /\b((simple|strict)_(strto(l|ll|ul|ull)))\s*\(/) { - WARN("CONSIDER_KSTRTO", - "$1 is obsolete, use k$3 instead\n" . $herecurr); - } - -# check for __initcall(), use device_initcall() explicitly or more appropriate function please - if ($line =~ /^.\s*__initcall\s*\(/) { - WARN("USE_DEVICE_INITCALL", - "please use device_initcall() or more appropriate function instead of __initcall() (see include/linux/init.h)\n" . $herecurr); - } - -# check for various structs that are normally const (ops, kgdb, device_tree) -# and avoid what seem like struct definitions 'struct foo {' - if ($line !~ /\bconst\b/ && - $line =~ /\bstruct\s+($const_structs)\b(?!\s*\{)/) { - WARN("CONST_STRUCT", - "struct $1 should normally be const\n" . $herecurr); - } - -# use of NR_CPUS is usually wrong -# ignore definitions of NR_CPUS and usage to define arrays as likely right - if ($line =~ /\bNR_CPUS\b/ && - $line !~ /^.\s*\s*#\s*if\b.*\bNR_CPUS\b/ && - $line !~ /^.\s*\s*#\s*define\b.*\bNR_CPUS\b/ && - $line !~ /^.\s*$Declare\s.*\[[^\]]*NR_CPUS[^\]]*\]/ && - $line !~ /\[[^\]]*\.\.\.[^\]]*NR_CPUS[^\]]*\]/ && - $line !~ /\[[^\]]*NR_CPUS[^\]]*\.\.\.[^\]]*\]/) - { - WARN("NR_CPUS", - "usage of NR_CPUS is often wrong - consider using cpu_possible(), num_possible_cpus(), for_each_possible_cpu(), etc\n" . $herecurr); - } - -# Use of __ARCH_HAS_<FOO> or ARCH_HAVE_<BAR> is wrong. - if ($line =~ /\+\s*#\s*define\s+((?:__)?ARCH_(?:HAS|HAVE)\w*)\b/) { - ERROR("DEFINE_ARCH_HAS", - "#define of '$1' is wrong - use Kconfig variables or standard guards instead\n" . $herecurr); - } - -# likely/unlikely comparisons similar to "(likely(foo) > 0)" - if ($^V && $^V ge 5.10.0 && - $line =~ /\b((?:un)?likely)\s*\(\s*$FuncArg\s*\)\s*$Compare/) { - WARN("LIKELY_MISUSE", - "Using $1 should generally have parentheses around the comparison\n" . $herecurr); - } - -# whine mightly about in_atomic - if ($line =~ /\bin_atomic\s*\(/) { - if ($realfile =~ m@^drivers/@) { - ERROR("IN_ATOMIC", - "do not use in_atomic in drivers\n" . $herecurr); - } elsif ($realfile !~ m@^kernel/@) { - WARN("IN_ATOMIC", - "use of in_atomic() is incorrect outside core kernel code\n" . $herecurr); - } - } - -# whine about ACCESS_ONCE - if ($^V && $^V ge 5.10.0 && - $line =~ /\bACCESS_ONCE\s*$balanced_parens\s*(=(?!=))?\s*($FuncArg)?/) { - my $par = $1; - my $eq = $2; - my $fun = $3; - $par =~ s/^\(\s*(.*)\s*\)$/$1/; - if (defined($eq)) { - if (WARN("PREFER_WRITE_ONCE", - "Prefer WRITE_ONCE(<FOO>, <BAR>) over ACCESS_ONCE(<FOO>) = <BAR>\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\bACCESS_ONCE\s*\(\s*\Q$par\E\s*\)\s*$eq\s*\Q$fun\E/WRITE_ONCE($par, $fun)/; - } - } else { - if (WARN("PREFER_READ_ONCE", - "Prefer READ_ONCE(<FOO>) over ACCESS_ONCE(<FOO>)\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\bACCESS_ONCE\s*\(\s*\Q$par\E\s*\)/READ_ONCE($par)/; - } - } - } - -# check for mutex_trylock_recursive usage - if ($line =~ /mutex_trylock_recursive/) { - ERROR("LOCKING", - "recursive locking is bad, do not use this ever.\n" . $herecurr); - } - -# check for lockdep_set_novalidate_class - if ($line =~ /^.\s*lockdep_set_novalidate_class\s*\(/ || - $line =~ /__lockdep_no_validate__\s*\)/ ) { - if ($realfile !~ m@^kernel/lockdep@ && - $realfile !~ m@^include/linux/lockdep@ && - $realfile !~ m@^drivers/base/core@) { - ERROR("LOCKDEP", - "lockdep_no_validate class is reserved for device->mutex.\n" . $herecurr); - } - } - - if ($line =~ /debugfs_create_\w+.*\b$mode_perms_world_writable\b/ || - $line =~ /DEVICE_ATTR.*\b$mode_perms_world_writable\b/) { - WARN("EXPORTED_WORLD_WRITABLE", - "Exporting world writable files is usually an error. Consider more restrictive permissions.\n" . $herecurr); - } - -# Mode permission misuses where it seems decimal should be octal -# This uses a shortcut match to avoid unnecessary uses of a slow foreach loop - if ($^V && $^V ge 5.10.0 && - defined $stat && - $line =~ /$mode_perms_search/) { - foreach my $entry (@mode_permission_funcs) { - my $func = $entry->[0]; - my $arg_pos = $entry->[1]; - - my $lc = $stat =~ tr@\n@@; - $lc = $lc + $linenr; - my $stat_real = raw_line($linenr, 0); - for (my $count = $linenr + 1; $count <= $lc; $count++) { - $stat_real = $stat_real . "\n" . raw_line($count, 0); - } - - my $skip_args = ""; - if ($arg_pos > 1) { - $arg_pos--; - $skip_args = "(?:\\s*$FuncArg\\s*,\\s*){$arg_pos,$arg_pos}"; - } - my $test = "\\b$func\\s*\\(${skip_args}($FuncArg(?:\\|\\s*$FuncArg)*)\\s*[,\\)]"; - if ($stat =~ /$test/) { - my $val = $1; - $val = $6 if ($skip_args ne ""); - if (($val =~ /^$Int$/ && $val !~ /^$Octal$/) || - ($val =~ /^$Octal$/ && length($val) ne 4)) { - ERROR("NON_OCTAL_PERMISSIONS", - "Use 4 digit octal (0777) not decimal permissions\n" . "$here\n" . $stat_real); - } - if ($val =~ /^$Octal$/ && (oct($val) & 02)) { - ERROR("EXPORTED_WORLD_WRITABLE", - "Exporting writable files is usually an error. Consider more restrictive permissions.\n" . "$here\n" . $stat_real); - } - } - } - } - -# check for uses of S_<PERMS> that could be octal for readability - if ($line =~ /\b$mode_perms_string_search\b/) { - my $val = ""; - my $oval = ""; - my $to = 0; - my $curpos = 0; - my $lastpos = 0; - while ($line =~ /\b(($mode_perms_string_search)\b(?:\s*\|\s*)?\s*)/g) { - $curpos = pos($line); - my $match = $2; - my $omatch = $1; - last if ($lastpos > 0 && ($curpos - length($omatch) != $lastpos)); - $lastpos = $curpos; - $to |= $mode_permission_string_types{$match}; - $val .= '\s*\|\s*' if ($val ne ""); - $val .= $match; - $oval .= $omatch; - } - $oval =~ s/^\s*\|\s*//; - $oval =~ s/\s*\|\s*$//; - my $octal = sprintf("%04o", $to); - if (WARN("SYMBOLIC_PERMS", - "Symbolic permissions '$oval' are not preferred. Consider using octal permissions '$octal'.\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/$val/$octal/; - } - } - -# validate content of MODULE_LICENSE against list from include/linux/module.h - if ($line =~ /\bMODULE_LICENSE\s*\(\s*($String)\s*\)/) { - my $extracted_string = get_quoted_string($line, $rawline); - my $valid_licenses = qr{ - GPL| - GPL\ v2| - GPL\ and\ additional\ rights| - Dual\ BSD/GPL| - Dual\ MIT/GPL| - Dual\ MPL/GPL| - Proprietary - }x; - if ($extracted_string !~ /^"(?:$valid_licenses)"$/x) { - WARN("MODULE_LICENSE", - "unknown module license " . $extracted_string . "\n" . $herecurr); - } - } - } - - # If we have no input at all, then there is nothing to report on - # so just keep quiet. - if ($#rawlines == -1) { - exit(0); - } - - # In mailback mode only produce a report in the negative, for - # things that appear to be patches. - if ($mailback && ($clean == 1 || !$is_patch)) { - exit(0); - } - - # This is not a patch, and we are are in 'no-patch' mode so - # just keep quiet. - if (!$chk_patch && !$is_patch) { - exit(0); - } - - if (!$is_patch && $file !~ /cover-letter\.patch$/) { - ERROR("NOT_UNIFIED_DIFF", - "Does not appear to be a unified-diff format patch\n"); - } - if ($is_patch && $has_commit_log && $chk_signoff && $signoff == 0) { - ERROR("MISSING_SIGN_OFF", - "Missing Signed-off-by: line(s)\n"); - } - - print report_dump(); - if ($summary && !($clean == 1 && $quiet == 1)) { - print "$filename " if ($summary_file); - print "total: $cnt_error errors, $cnt_warn warnings, " . - (($check)? "$cnt_chk checks, " : "") . - "$cnt_lines lines checked\n"; - } - - if ($quiet == 0) { - # If there were any defects found and not already fixing them - if (!$clean and !$fix) { - print << "EOM" - -NOTE: For some of the reported defects, checkpatch may be able to - mechanically convert to the typical style using --fix or --fix-inplace. -EOM - } - # If there were whitespace errors which cleanpatch can fix - # then suggest that. - if ($rpt_cleaners) { - $rpt_cleaners = 0; - print << "EOM" - -NOTE: Whitespace errors detected. - You may wish to use scripts/cleanpatch or scripts/cleanfile -EOM - } - } - - if ($clean == 0 && $fix && - ("@rawlines" ne "@fixed" || - $#fixed_inserted >= 0 || $#fixed_deleted >= 0)) { - my $newfile = $filename; - $newfile .= ".EXPERIMENTAL-checkpatch-fixes" if (!$fix_inplace); - my $linecount = 0; - my $f; - - @fixed = fix_inserted_deleted_lines(\@fixed, \@fixed_inserted, \@fixed_deleted); - - open($f, '>', $newfile) - or die "$P: Can't open $newfile for write\n"; - foreach my $fixed_line (@fixed) { - $linecount++; - if ($file) { - if ($linecount > 3) { - $fixed_line =~ s/^\+//; - print $f $fixed_line . "\n"; - } - } else { - print $f $fixed_line . "\n"; - } - } - close($f); - - if (!$quiet) { - print << "EOM"; - -Wrote EXPERIMENTAL --fix correction(s) to '$newfile' - -Do _NOT_ trust the results written to this file. -Do _NOT_ submit these changes without inspecting them for correctness. - -This EXPERIMENTAL file is simply a convenience to help rewrite patches. -No warranties, expressed or implied... -EOM - } - } - - if ($quiet == 0) { - print "\n"; - if ($clean == 1) { - print "$vname has no obvious style problems and is ready for submission.\n"; - } else { - print "$vname has style problems, please review.\n"; - } - } - return $clean; -} +#!/usr/bin/env perl +# (c) 2001, Dave Jones. (the file handling bit) +# (c) 2005, Joel Schopp <jschopp@austin.ibm.com> (the ugly bit) +# (c) 2007,2008, Andy Whitcroft <apw@uk.ibm.com> (new conditions, test suite) +# (c) 2008-2010 Andy Whitcroft <apw@canonical.com> +# Licensed under the terms of the GNU GPL License version 2 + +use strict; +use warnings; +use POSIX; +use File::Basename; +use Cwd 'abs_path'; +use Term::ANSIColor qw(:constants); + +my $P = $0; +my $D = dirname(abs_path($P)); + +my $V = '0.32'; + +use Getopt::Long qw(:config no_auto_abbrev); + +my $quiet = 0; +my $tree = 1; +my $chk_signoff = 1; +my $chk_patch = 1; +my $tst_only; +my $emacs = 0; +my $terse = 0; +my $showfile = 0; +my $file = 0; +my $git = 0; +my %git_commits = (); +my $check = 0; +my $check_orig = 0; +my $summary = 1; +my $mailback = 0; +my $summary_file = 0; +my $show_types = 0; +my $list_types = 0; +my $fix = 0; +my $fix_inplace = 0; +my $root; +my %debug; +my %camelcase = (); +my %use_type = (); +my @use = (); +my %ignore_type = (); +my @ignore = (); +my @exclude = (); +my $help = 0; +my $configuration_file = ".checkpatch.conf"; +my $max_line_length = 80; +my $ignore_perl_version = 0; +my $minimum_perl_version = 5.10.0; +my $min_conf_desc_length = 4; +my $spelling_file = "$D/spelling.txt"; +my $codespell = 0; +my $codespellfile = "/usr/share/codespell/dictionary.txt"; +my $conststructsfile = "$D/const_structs.checkpatch"; +my $typedefsfile = ""; +my $color = "auto"; +my $allow_c99_comments = 0; + +sub help { + my ($exitcode) = @_; + + print << "EOM"; +Usage: $P [OPTION]... [FILE]... +Version: $V + +Options: + -q, --quiet quiet + --no-tree run without a kernel tree + --no-signoff do not check for 'Signed-off-by' line + --patch treat FILE as patchfile (default) + --emacs emacs compile window format + --terse one line per report + --showfile emit diffed file position, not input file position + -g, --git treat FILE as a single commit or git revision range + single git commit with: + <rev> + <rev>^ + <rev>~n + multiple git commits with: + <rev1>..<rev2> + <rev1>...<rev2> + <rev>-<count> + git merges are ignored + -f, --file treat FILE as regular source file + --subjective, --strict enable more subjective tests + --list-types list the possible message types + --types TYPE(,TYPE2...) show only these comma separated message types + --ignore TYPE(,TYPE2...) ignore various comma separated message types + --exclude DIR(,DIR22...) exclude directories + --show-types show the specific message type in the output + --max-line-length=n set the maximum line length, if exceeded, warn + --min-conf-desc-length=n set the min description length, if shorter, warn + --root=PATH PATH to the kernel tree root + --no-summary suppress the per-file summary + --mailback only produce a report in case of warnings/errors + --summary-file include the filename in summary + --debug KEY=[0|1] turn on/off debugging of KEY, where KEY is one of + 'values', 'possible', 'type', and 'attr' (default + is all off) + --test-only=WORD report only warnings/errors containing WORD + literally + --fix EXPERIMENTAL - may create horrible results + If correctable single-line errors exist, create + "<inputfile>.EXPERIMENTAL-checkpatch-fixes" + with potential errors corrected to the preferred + checkpatch style + --fix-inplace EXPERIMENTAL - may create horrible results + Is the same as --fix, but overwrites the input + file. It's your fault if there's no backup or git + --ignore-perl-version override checking of perl version. expect + runtime errors. + --codespell Use the codespell dictionary for spelling/typos + (default:/usr/share/codespell/dictionary.txt) + --codespellfile Use this codespell dictionary + --typedefsfile Read additional types from this file + --color[=WHEN] Use colors 'always', 'never', or only when output + is a terminal ('auto'). Default is 'auto'. + -h, --help, --version display this help and exit + +When FILE is - read standard input. +EOM + + exit($exitcode); +} + +sub uniq { + my %seen; + return grep { !$seen{$_}++ } @_; +} + +sub list_types { + my ($exitcode) = @_; + + my $count = 0; + + local $/ = undef; + + open(my $script, '<', abs_path($P)) or + die "$P: Can't read '$P' $!\n"; + + my $text = <$script>; + close($script); + + my @types = (); + # Also catch when type or level is passed through a variable + for ($text =~ /(?:(?:\bCHK|\bWARN|\bERROR|&\{\$msg_level})\s*\(|\$msg_type\s*=)\s*"([^"]+)"/g) { + push (@types, $_); + } + @types = sort(uniq(@types)); + print("#\tMessage type\n\n"); + foreach my $type (@types) { + print(++$count . "\t" . $type . "\n"); + } + + exit($exitcode); +} + +my $conf = which_conf($configuration_file); +if (-f $conf) { + my @conf_args; + open(my $conffile, '<', "$conf") + or warn "$P: Can't find a readable $configuration_file file $!\n"; + + while (<$conffile>) { + my $line = $_; + + $line =~ s/\s*\n?$//g; + $line =~ s/^\s*//g; + $line =~ s/\s+/ /g; + + next if ($line =~ m/^\s*#/); + next if ($line =~ m/^\s*$/); + + my @words = split(" ", $line); + foreach my $word (@words) { + last if ($word =~ m/^#/); + push (@conf_args, $word); + } + } + close($conffile); + unshift(@ARGV, @conf_args) if @conf_args; +} + +# Perl's Getopt::Long allows options to take optional arguments after a space. +# Prevent --color by itself from consuming other arguments +foreach (@ARGV) { + if ($_ eq "--color" || $_ eq "-color") { + $_ = "--color=$color"; + } +} + +GetOptions( + 'q|quiet+' => \$quiet, + 'tree!' => \$tree, + 'signoff!' => \$chk_signoff, + 'patch!' => \$chk_patch, + 'emacs!' => \$emacs, + 'terse!' => \$terse, + 'showfile!' => \$showfile, + 'f|file!' => \$file, + 'g|git!' => \$git, + 'subjective!' => \$check, + 'strict!' => \$check, + 'ignore=s' => \@ignore, + 'exclude=s' => \@exclude, + 'types=s' => \@use, + 'show-types!' => \$show_types, + 'list-types!' => \$list_types, + 'max-line-length=i' => \$max_line_length, + 'min-conf-desc-length=i' => \$min_conf_desc_length, + 'root=s' => \$root, + 'summary!' => \$summary, + 'mailback!' => \$mailback, + 'summary-file!' => \$summary_file, + 'fix!' => \$fix, + 'fix-inplace!' => \$fix_inplace, + 'ignore-perl-version!' => \$ignore_perl_version, + 'debug=s' => \%debug, + 'test-only=s' => \$tst_only, + 'codespell!' => \$codespell, + 'codespellfile=s' => \$codespellfile, + 'typedefsfile=s' => \$typedefsfile, + 'color=s' => \$color, + 'no-color' => \$color, #keep old behaviors of -nocolor + 'nocolor' => \$color, #keep old behaviors of -nocolor + 'h|help' => \$help, + 'version' => \$help +) or help(1); + +help(0) if ($help); + +list_types(0) if ($list_types); + +$fix = 1 if ($fix_inplace); +$check_orig = $check; + +my $exit = 0; + +if ($^V && $^V lt $minimum_perl_version) { + printf "$P: requires at least perl version %vd\n", $minimum_perl_version; + if (!$ignore_perl_version) { + exit(1); + } +} + +#if no filenames are given, push '-' to read patch from stdin +if ($#ARGV < 0) { + push(@ARGV, '-'); +} + +if ($color =~ /^[01]$/) { + $color = !$color; +} elsif ($color =~ /^always$/i) { + $color = 1; +} elsif ($color =~ /^never$/i) { + $color = 0; +} elsif ($color =~ /^auto$/i) { + $color = (-t STDOUT); +} else { + die "Invalid color mode: $color\n"; +} + +sub hash_save_array_words { + my ($hashRef, $arrayRef) = @_; + + my @array = split(/,/, join(',', @$arrayRef)); + foreach my $word (@array) { + $word =~ s/\s*\n?$//g; + $word =~ s/^\s*//g; + $word =~ s/\s+/ /g; + $word =~ tr/[a-z]/[A-Z]/; + + next if ($word =~ m/^\s*#/); + next if ($word =~ m/^\s*$/); + + $hashRef->{$word}++; + } +} + +sub hash_show_words { + my ($hashRef, $prefix) = @_; + + if (keys %$hashRef) { + print "\nNOTE: $prefix message types:"; + foreach my $word (sort keys %$hashRef) { + print " $word"; + } + print "\n"; + } +} + +hash_save_array_words(\%ignore_type, \@ignore); +hash_save_array_words(\%use_type, \@use); + +my $dbg_values = 0; +my $dbg_possible = 0; +my $dbg_type = 0; +my $dbg_attr = 0; +for my $key (keys %debug) { + ## no critic + eval "\${dbg_$key} = '$debug{$key}';"; + die "$@" if ($@); +} + +my $rpt_cleaners = 0; + +if ($terse) { + $emacs = 1; + $quiet++; +} + +if ($tree) { + if (defined $root) { + if (!top_of_kernel_tree($root)) { + die "$P: $root: --root does not point at a valid tree\n"; + } + } else { + if (top_of_kernel_tree('.')) { + $root = '.'; + } elsif ($0 =~ m@(.*)/scripts/[^/]*$@ && + top_of_kernel_tree($1)) { + $root = $1; + } + } + + if (!defined $root) { + print "Must be run from the top-level dir. of a kernel tree\n"; + exit(2); + } +} + +my $emitted_corrupt = 0; + +our $Ident = qr{ + [A-Za-z_][A-Za-z\d_]* + (?:\s*\#\#\s*[A-Za-z_][A-Za-z\d_]*)* + }x; +our $Storage = qr{extern|static|asmlinkage}; +our $Sparse = qr{ + __user| + __force| + __iomem| + __must_check| + __init_refok| + __kprobes| + __ref| + __rcu| + __private + }x; +our $InitAttributePrefix = qr{__(?:mem|cpu|dev|net_|)}; +our $InitAttributeData = qr{$InitAttributePrefix(?:initdata\b)}; +our $InitAttributeConst = qr{$InitAttributePrefix(?:initconst\b)}; +our $InitAttributeInit = qr{$InitAttributePrefix(?:init\b)}; +our $InitAttribute = qr{$InitAttributeData|$InitAttributeConst|$InitAttributeInit}; + +# Notes to $Attribute: +# We need \b after 'init' otherwise 'initconst' will cause a false positive in a check +our $Attribute = qr{ + const| + __percpu| + __nocast| + __safe| + __bitwise| + __packed__| + __packed2__| + __naked| + __maybe_unused| + __always_unused| + __noreturn| + __used| + __cold| + __pure| + __noclone| + __deprecated| + __read_mostly| + __kprobes| + $InitAttribute| + ____cacheline_aligned| + ____cacheline_aligned_in_smp| + ____cacheline_internodealigned_in_smp| + __weak| + __syscall| + __syscall_inline + }x; +our $Modifier; +our $Inline = qr{inline|__always_inline|noinline|__inline|__inline__}; +our $Member = qr{->$Ident|\.$Ident|\[[^]]*\]}; +our $Lval = qr{$Ident(?:$Member)*}; + +our $Int_type = qr{(?i)llu|ull|ll|lu|ul|l|u}; +our $Binary = qr{(?i)0b[01]+$Int_type?}; +our $Hex = qr{(?i)0x[0-9a-f]+$Int_type?}; +our $Int = qr{[0-9]+$Int_type?}; +our $Octal = qr{0[0-7]+$Int_type?}; +our $String = qr{"[X\t]*"}; +our $Float_hex = qr{(?i)0x[0-9a-f]+p-?[0-9]+[fl]?}; +our $Float_dec = qr{(?i)(?:[0-9]+\.[0-9]*|[0-9]*\.[0-9]+)(?:e-?[0-9]+)?[fl]?}; +our $Float_int = qr{(?i)[0-9]+e-?[0-9]+[fl]?}; +our $Float = qr{$Float_hex|$Float_dec|$Float_int}; +our $Constant = qr{$Float|$Binary|$Octal|$Hex|$Int}; +our $Assignment = qr{\*\=|/=|%=|\+=|-=|<<=|>>=|&=|\^=|\|=|=}; +our $Compare = qr{<=|>=|==|!=|<|(?<!-)>}; +our $Arithmetic = qr{\+|-|\*|\/|%}; +our $Operators = qr{ + <=|>=|==|!=| + =>|->|<<|>>|<|>|!|~| + &&|\|\||,|\^|\+\+|--|&|\||$Arithmetic + }x; + +our $c90_Keywords = qr{do|for|while|if|else|return|goto|continue|switch|default|case|break}x; + +our $BasicType; +our $NonptrType; +our $NonptrTypeMisordered; +our $NonptrTypeWithAttr; +our $Type; +our $TypeMisordered; +our $Declare; +our $DeclareMisordered; + +our $NON_ASCII_UTF8 = qr{ + [\xC2-\xDF][\x80-\xBF] # non-overlong 2-byte + | \xE0[\xA0-\xBF][\x80-\xBF] # excluding overlongs + | [\xE1-\xEC\xEE\xEF][\x80-\xBF]{2} # straight 3-byte + | \xED[\x80-\x9F][\x80-\xBF] # excluding surrogates + | \xF0[\x90-\xBF][\x80-\xBF]{2} # planes 1-3 + | [\xF1-\xF3][\x80-\xBF]{3} # planes 4-15 + | \xF4[\x80-\x8F][\x80-\xBF]{2} # plane 16 +}x; + +our $UTF8 = qr{ + [\x09\x0A\x0D\x20-\x7E] # ASCII + | $NON_ASCII_UTF8 +}x; + +our $typeC99Typedefs = qr{(?:__)?(?:[us]_?)?int_?(?:8|16|32|64)_t}; +our $typeOtherOSTypedefs = qr{(?x: + u_(?:char|short|int|long) | # bsd + u(?:nchar|short|int|long) # sysv +)}; +our $typeKernelTypedefs = qr{(?x: + (?:__)?(?:u|s|be|le)(?:8|16|32|64)_t| + atomic_t +)}; +our $typeTypedefs = qr{(?x: + $typeC99Typedefs\b| + $typeOtherOSTypedefs\b| + $typeKernelTypedefs\b +)}; + +our $zero_initializer = qr{(?:(?:0[xX])?0+$Int_type?|NULL|false)\b}; + +our $logFunctions = qr{(?x: + printk(?:_ratelimited|_once|_deferred_once|_deferred|)| + (?:[a-z0-9]+_){1,2}(?:printk|emerg|alert|crit|err|warning|warn|notice|info|debug|dbg|vdbg|devel|cont|WARN)(?:_ratelimited|_once|)| + WARN(?:_RATELIMIT|_ONCE|)| + panic| + MODULE_[A-Z_]+| + seq_vprintf|seq_printf|seq_puts +)}; + +our $signature_tags = qr{(?xi: + Signed-off-by:| + Acked-by:| + Tested-by:| + Reviewed-by:| + Reported-by:| + Suggested-by:| + To:| + Cc: +)}; + +our @typeListMisordered = ( + qr{char\s+(?:un)?signed}, + qr{int\s+(?:(?:un)?signed\s+)?short\s}, + qr{int\s+short(?:\s+(?:un)?signed)}, + qr{short\s+int(?:\s+(?:un)?signed)}, + qr{(?:un)?signed\s+int\s+short}, + qr{short\s+(?:un)?signed}, + qr{long\s+int\s+(?:un)?signed}, + qr{int\s+long\s+(?:un)?signed}, + qr{long\s+(?:un)?signed\s+int}, + qr{int\s+(?:un)?signed\s+long}, + qr{int\s+(?:un)?signed}, + qr{int\s+long\s+long\s+(?:un)?signed}, + qr{long\s+long\s+int\s+(?:un)?signed}, + qr{long\s+long\s+(?:un)?signed\s+int}, + qr{long\s+long\s+(?:un)?signed}, + qr{long\s+(?:un)?signed}, +); + +our @typeList = ( + qr{void}, + qr{(?:(?:un)?signed\s+)?char}, + qr{(?:(?:un)?signed\s+)?short\s+int}, + qr{(?:(?:un)?signed\s+)?short}, + qr{(?:(?:un)?signed\s+)?int}, + qr{(?:(?:un)?signed\s+)?long\s+int}, + qr{(?:(?:un)?signed\s+)?long\s+long\s+int}, + qr{(?:(?:un)?signed\s+)?long\s+long}, + qr{(?:(?:un)?signed\s+)?long}, + qr{(?:un)?signed}, + qr{float}, + qr{double}, + qr{bool}, + qr{struct\s+$Ident}, + qr{union\s+$Ident}, + qr{enum\s+$Ident}, + qr{${Ident}_t}, + qr{${Ident}_handler}, + qr{${Ident}_handler_fn}, + @typeListMisordered, +); + +our $C90_int_types = qr{(?x: + long\s+long\s+int\s+(?:un)?signed| + long\s+long\s+(?:un)?signed\s+int| + long\s+long\s+(?:un)?signed| + (?:(?:un)?signed\s+)?long\s+long\s+int| + (?:(?:un)?signed\s+)?long\s+long| + int\s+long\s+long\s+(?:un)?signed| + int\s+(?:(?:un)?signed\s+)?long\s+long| + + long\s+int\s+(?:un)?signed| + long\s+(?:un)?signed\s+int| + long\s+(?:un)?signed| + (?:(?:un)?signed\s+)?long\s+int| + (?:(?:un)?signed\s+)?long| + int\s+long\s+(?:un)?signed| + int\s+(?:(?:un)?signed\s+)?long| + + int\s+(?:un)?signed| + (?:(?:un)?signed\s+)?int +)}; + +our @typeListFile = (); +our @typeListWithAttr = ( + @typeList, + qr{struct\s+$InitAttribute\s+$Ident}, + qr{union\s+$InitAttribute\s+$Ident}, +); + +our @modifierList = ( + qr{fastcall}, +); +our @modifierListFile = (); + +our @mode_permission_funcs = ( + ["module_param", 3], + ["module_param_(?:array|named|string)", 4], + ["module_param_array_named", 5], + ["debugfs_create_(?:file|u8|u16|u32|u64|x8|x16|x32|x64|size_t|atomic_t|bool|blob|regset32|u32_array)", 2], + ["proc_create(?:_data|)", 2], + ["(?:CLASS|DEVICE|SENSOR|SENSOR_DEVICE|IIO_DEVICE)_ATTR", 2], + ["IIO_DEV_ATTR_[A-Z_]+", 1], + ["SENSOR_(?:DEVICE_|)ATTR_2", 2], + ["SENSOR_TEMPLATE(?:_2|)", 3], + ["__ATTR", 2], +); + +#Create a search pattern for all these functions to speed up a loop below +our $mode_perms_search = ""; +foreach my $entry (@mode_permission_funcs) { + $mode_perms_search .= '|' if ($mode_perms_search ne ""); + $mode_perms_search .= $entry->[0]; +} + +our $mode_perms_world_writable = qr{ + S_IWUGO | + S_IWOTH | + S_IRWXUGO | + S_IALLUGO | + 0[0-7][0-7][2367] +}x; + +our %mode_permission_string_types = ( + "S_IRWXU" => 0700, + "S_IRUSR" => 0400, + "S_IWUSR" => 0200, + "S_IXUSR" => 0100, + "S_IRWXG" => 0070, + "S_IRGRP" => 0040, + "S_IWGRP" => 0020, + "S_IXGRP" => 0010, + "S_IRWXO" => 0007, + "S_IROTH" => 0004, + "S_IWOTH" => 0002, + "S_IXOTH" => 0001, + "S_IRWXUGO" => 0777, + "S_IRUGO" => 0444, + "S_IWUGO" => 0222, + "S_IXUGO" => 0111, +); + +#Create a search pattern for all these strings to speed up a loop below +our $mode_perms_string_search = ""; +foreach my $entry (keys %mode_permission_string_types) { + $mode_perms_string_search .= '|' if ($mode_perms_string_search ne ""); + $mode_perms_string_search .= $entry; +} + +our $allowed_asm_includes = qr{(?x: + irq| + memory| + time| + reboot +)}; +# memory.h: ARM has a custom one + +# Load common spelling mistakes and build regular expression list. +my $misspellings; +my %spelling_fix; + +if (open(my $spelling, '<', $spelling_file)) { + while (<$spelling>) { + my $line = $_; + + $line =~ s/\s*\n?$//g; + $line =~ s/^\s*//g; + + next if ($line =~ m/^\s*#/); + next if ($line =~ m/^\s*$/); + + my ($suspect, $fix) = split(/\|\|/, $line); + + $spelling_fix{$suspect} = $fix; + } + close($spelling); +} else { + warn "No typos will be found - file '$spelling_file': $!\n"; +} + +if ($codespell) { + if (open(my $spelling, '<', $codespellfile)) { + while (<$spelling>) { + my $line = $_; + + $line =~ s/\s*\n?$//g; + $line =~ s/^\s*//g; + + next if ($line =~ m/^\s*#/); + next if ($line =~ m/^\s*$/); + next if ($line =~ m/, disabled/i); + + $line =~ s/,.*$//; + + my ($suspect, $fix) = split(/->/, $line); + + $spelling_fix{$suspect} = $fix; + } + close($spelling); + } else { + warn "No codespell typos will be found - file '$codespellfile': $!\n"; + } +} + +$misspellings = join("|", sort keys %spelling_fix) if keys %spelling_fix; + +sub read_words { + my ($wordsRef, $file) = @_; + + if (open(my $words, '<', $file)) { + while (<$words>) { + my $line = $_; + + $line =~ s/\s*\n?$//g; + $line =~ s/^\s*//g; + + next if ($line =~ m/^\s*#/); + next if ($line =~ m/^\s*$/); + if ($line =~ /\s/) { + print("$file: '$line' invalid - ignored\n"); + next; + } + + $$wordsRef .= '|' if ($$wordsRef ne ""); + $$wordsRef .= $line; + } + close($file); + return 1; + } + + return 0; +} + +my $const_structs = ""; +#read_words(\$const_structs, $conststructsfile) +# or warn "No structs that should be const will be found - file '$conststructsfile': $!\n"; + +my $typeOtherTypedefs = ""; +if (length($typedefsfile)) { + read_words(\$typeOtherTypedefs, $typedefsfile) + or warn "No additional types will be considered - file '$typedefsfile': $!\n"; +} +$typeTypedefs .= '|' . $typeOtherTypedefs if ($typeOtherTypedefs ne ""); + +sub build_types { + my $mods = "(?x: \n" . join("|\n ", (@modifierList, @modifierListFile)) . "\n)"; + my $all = "(?x: \n" . join("|\n ", (@typeList, @typeListFile)) . "\n)"; + my $Misordered = "(?x: \n" . join("|\n ", @typeListMisordered) . "\n)"; + my $allWithAttr = "(?x: \n" . join("|\n ", @typeListWithAttr) . "\n)"; + $Modifier = qr{(?:$Attribute|$Sparse|$mods)}; + $BasicType = qr{ + (?:$typeTypedefs\b)| + (?:${all}\b) + }x; + $NonptrType = qr{ + (?:$Modifier\s+|const\s+)* + (?: + (?:typeof|__typeof__)\s*\([^\)]*\)| + (?:$typeTypedefs\b)| + (?:${all}\b) + ) + (?:\s+$Modifier|\s+const)* + }x; + $NonptrTypeMisordered = qr{ + (?:$Modifier\s+|const\s+)* + (?: + (?:${Misordered}\b) + ) + (?:\s+$Modifier|\s+const)* + }x; + $NonptrTypeWithAttr = qr{ + (?:$Modifier\s+|const\s+)* + (?: + (?:typeof|__typeof__)\s*\([^\)]*\)| + (?:$typeTypedefs\b)| + (?:${allWithAttr}\b) + ) + (?:\s+$Modifier|\s+const)* + }x; + $Type = qr{ + $NonptrType + (?:(?:\s|\*|\[\])+\s*const|(?:\s|\*\s*(?:const\s*)?|\[\])+|(?:\s*\[\s*\])+)? + (?:\s+$Inline|\s+$Modifier)* + }x; + $TypeMisordered = qr{ + $NonptrTypeMisordered + (?:(?:\s|\*|\[\])+\s*const|(?:\s|\*\s*(?:const\s*)?|\[\])+|(?:\s*\[\s*\])+)? + (?:\s+$Inline|\s+$Modifier)* + }x; + $Declare = qr{(?:$Storage\s+(?:$Inline\s+)?)?$Type}; + $DeclareMisordered = qr{(?:$Storage\s+(?:$Inline\s+)?)?$TypeMisordered}; +} +build_types(); + +our $Typecast = qr{\s*(\(\s*$NonptrType\s*\)){0,1}\s*}; + +# Using $balanced_parens, $LvalOrFunc, or $FuncArg +# requires at least perl version v5.10.0 +# Any use must be runtime checked with $^V + +our $balanced_parens = qr/(\((?:[^\(\)]++|(?-1))*\))/; +our $LvalOrFunc = qr{((?:[\&\*]\s*)?$Lval)\s*($balanced_parens{0,1})\s*}; +our $FuncArg = qr{$Typecast{0,1}($LvalOrFunc|$Constant|$String)}; + +our $declaration_macros = qr{(?x: + (?:$Storage\s+)?(?:[A-Z_][A-Z0-9]*_){0,2}(?:DEFINE|DECLARE)(?:_[A-Z0-9]+){1,6}\s*\(| + (?:$Storage\s+)?[HLP]?LIST_HEAD\s*\(| + (?:$Storage\s+)?${Type}\s+uninitialized_var\s*\( +)}; + +sub deparenthesize { + my ($string) = @_; + return "" if (!defined($string)); + + while ($string =~ /^\s*\(.*\)\s*$/) { + $string =~ s@^\s*\(\s*@@; + $string =~ s@\s*\)\s*$@@; + } + + $string =~ s@\s+@ @g; + + return $string; +} + +sub seed_camelcase_file { + my ($file) = @_; + + return if (!(-f $file)); + + local $/; + + open(my $include_file, '<', "$file") + or warn "$P: Can't read '$file' $!\n"; + my $text = <$include_file>; + close($include_file); + + my @lines = split('\n', $text); + + foreach my $line (@lines) { + next if ($line !~ /(?:[A-Z][a-z]|[a-z][A-Z])/); + if ($line =~ /^[ \t]*(?:#[ \t]*define|typedef\s+$Type)\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)/) { + $camelcase{$1} = 1; + } elsif ($line =~ /^\s*$Declare\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)\s*[\(\[,;]/) { + $camelcase{$1} = 1; + } elsif ($line =~ /^\s*(?:union|struct|enum)\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)\s*[;\{]/) { + $camelcase{$1} = 1; + } + } +} + +sub is_maintained_obsolete { + my ($filename) = @_; + + return 0 if (!$tree || !(-e "$root/scripts/get_maintainer.pl")); + + my $status = `perl $root/scripts/get_maintainer.pl --status --nom --nol --nogit --nogit-fallback -f $filename 2>&1`; + + return $status =~ /obsolete/i; +} + +my $camelcase_seeded = 0; +sub seed_camelcase_includes { + return if ($camelcase_seeded); + + my $files; + my $camelcase_cache = ""; + my @include_files = (); + + $camelcase_seeded = 1; + + if (-e ".git") { + my $git_last_include_commit = `git log --no-merges --pretty=format:"%h%n" -1 -- include`; + chomp $git_last_include_commit; + $camelcase_cache = ".checkpatch-camelcase.git.$git_last_include_commit"; + } else { + my $last_mod_date = 0; + $files = `find $root/include -name "*.h"`; + @include_files = split('\n', $files); + foreach my $file (@include_files) { + my $date = POSIX::strftime("%Y%m%d%H%M", + localtime((stat $file)[9])); + $last_mod_date = $date if ($last_mod_date < $date); + } + $camelcase_cache = ".checkpatch-camelcase.date.$last_mod_date"; + } + + if ($camelcase_cache ne "" && -f $camelcase_cache) { + open(my $camelcase_file, '<', "$camelcase_cache") + or warn "$P: Can't read '$camelcase_cache' $!\n"; + while (<$camelcase_file>) { + chomp; + $camelcase{$_} = 1; + } + close($camelcase_file); + + return; + } + + if (-e ".git") { + $files = `git ls-files "include/*.h"`; + @include_files = split('\n', $files); + } + + foreach my $file (@include_files) { + seed_camelcase_file($file); + } + + if ($camelcase_cache ne "") { + unlink glob ".checkpatch-camelcase.*"; + open(my $camelcase_file, '>', "$camelcase_cache") + or warn "$P: Can't write '$camelcase_cache' $!\n"; + foreach (sort { lc($a) cmp lc($b) } keys(%camelcase)) { + print $camelcase_file ("$_\n"); + } + close($camelcase_file); + } +} + +sub git_commit_info { + my ($commit, $id, $desc) = @_; + + return ($id, $desc) if ((which("git") eq "") || !(-e ".git")); + + my $output = `git log --no-color --format='%H %s' -1 $commit 2>&1`; + $output =~ s/^\s*//gm; + my @lines = split("\n", $output); + + return ($id, $desc) if ($#lines < 0); + + if ($lines[0] =~ /^error: short SHA1 $commit is ambiguous\./) { +# Maybe one day convert this block of bash into something that returns +# all matching commit ids, but it's very slow... +# +# echo "checking commits $1..." +# git rev-list --remotes | grep -i "^$1" | +# while read line ; do +# git log --format='%H %s' -1 $line | +# echo "commit $(cut -c 1-12,41-)" +# done + } elsif ($lines[0] =~ /^fatal: ambiguous argument '$commit': unknown revision or path not in the working tree\./) { + $id = undef; + } else { + $id = substr($lines[0], 0, 12); + $desc = substr($lines[0], 41); + } + + return ($id, $desc); +} + +$chk_signoff = 0 if ($file); + +my @rawlines = (); +my @lines = (); +my @fixed = (); +my @fixed_inserted = (); +my @fixed_deleted = (); +my $fixlinenr = -1; + +# If input is git commits, extract all commits from the commit expressions. +# For example, HEAD-3 means we need check 'HEAD, HEAD~1, HEAD~2'. +die "$P: No git repository found\n" if ($git && !-e ".git"); + +if ($git) { + my @commits = (); + foreach my $commit_expr (@ARGV) { + my $git_range; + if ($commit_expr =~ m/^(.*)-(\d+)$/) { + $git_range = "-$2 $1"; + } elsif ($commit_expr =~ m/\.\./) { + $git_range = "$commit_expr"; + } else { + $git_range = "-1 $commit_expr"; + } + my $lines = `git log --no-color --no-merges --pretty=format:'%H %s' $git_range`; + foreach my $line (split(/\n/, $lines)) { + $line =~ /^([0-9a-fA-F]{40,40}) (.*)$/; + next if (!defined($1) || !defined($2)); + my $sha1 = $1; + my $subject = $2; + unshift(@commits, $sha1); + $git_commits{$sha1} = $subject; + } + } + die "$P: no git commits after extraction!\n" if (@commits == 0); + @ARGV = @commits; +} + +my $vname; +for my $filename (@ARGV) { + my $FILE; + if ($git) { + open($FILE, '-|', "git format-patch -M --stdout -1 $filename") || + die "$P: $filename: git format-patch failed - $!\n"; + } elsif ($file) { + open($FILE, '-|', "diff -u /dev/null $filename") || + die "$P: $filename: diff failed - $!\n"; + } elsif ($filename eq '-') { + open($FILE, '<&STDIN'); + } else { + open($FILE, '<', "$filename") || + die "$P: $filename: open failed - $!\n"; + } + if ($filename eq '-') { + $vname = 'Your patch'; + } elsif ($git) { + $vname = "Commit " . substr($filename, 0, 12) . ' ("' . $git_commits{$filename} . '")'; + } else { + $vname = $filename; + } + while (<$FILE>) { + chomp; + push(@rawlines, $_); + } + close($FILE); + + if ($#ARGV > 0 && $quiet == 0) { + print '-' x length($vname) . "\n"; + print "$vname\n"; + print '-' x length($vname) . "\n"; + } + + if (!process($filename)) { + $exit = 1; + } + @rawlines = (); + @lines = (); + @fixed = (); + @fixed_inserted = (); + @fixed_deleted = (); + $fixlinenr = -1; + @modifierListFile = (); + @typeListFile = (); + build_types(); +} + +if (!$quiet) { + hash_show_words(\%use_type, "Used"); + hash_show_words(\%ignore_type, "Ignored"); + + if ($^V lt 5.10.0) { + print << "EOM" + +NOTE: perl $^V is not modern enough to detect all possible issues. + An upgrade to at least perl v5.10.0 is suggested. +EOM + } + if ($exit) { + print << "EOM" + +NOTE: If any of the errors are false positives, please report + them to the maintainers. +EOM + } +} + +exit($exit); + +sub top_of_kernel_tree { + my ($root) = @_; + + my @tree_check = ( + "LICENSE", "CODEOWNERS", "Kconfig", "Makefile", + "README.rst", "doc", "arch", "include", "drivers", + "boards", "kernel", "lib", "scripts", + ); + + foreach my $check (@tree_check) { + if (! -e $root . '/' . $check) { + return 0; + } + } + return 1; +} + +sub parse_email { + my ($formatted_email) = @_; + + my $name = ""; + my $address = ""; + my $comment = ""; + + if ($formatted_email =~ /^(.*)<(\S+\@\S+)>(.*)$/) { + $name = $1; + $address = $2; + $comment = $3 if defined $3; + } elsif ($formatted_email =~ /^\s*<(\S+\@\S+)>(.*)$/) { + $address = $1; + $comment = $2 if defined $2; + } elsif ($formatted_email =~ /(\S+\@\S+)(.*)$/) { + $address = $1; + $comment = $2 if defined $2; + $formatted_email =~ s/$address.*$//; + $name = $formatted_email; + $name = trim($name); + $name =~ s/^\"|\"$//g; + # If there's a name left after stripping spaces and + # leading quotes, and the address doesn't have both + # leading and trailing angle brackets, the address + # is invalid. ie: + # "joe smith joe@smith.com" bad + # "joe smith <joe@smith.com" bad + if ($name ne "" && $address !~ /^<[^>]+>$/) { + $name = ""; + $address = ""; + $comment = ""; + } + } + + $name = trim($name); + $name =~ s/^\"|\"$//g; + $address = trim($address); + $address =~ s/^\<|\>$//g; + + if ($name =~ /[^\w \-]/i) { ##has "must quote" chars + $name =~ s/(?<!\\)"/\\"/g; ##escape quotes + $name = "\"$name\""; + } + + return ($name, $address, $comment); +} + +sub format_email { + my ($name, $address) = @_; + + my $formatted_email; + + $name = trim($name); + $name =~ s/^\"|\"$//g; + $address = trim($address); + + if ($name =~ /[^\w \-]/i) { ##has "must quote" chars + $name =~ s/(?<!\\)"/\\"/g; ##escape quotes + $name = "\"$name\""; + } + + if ("$name" eq "") { + $formatted_email = "$address"; + } else { + $formatted_email = "$name <$address>"; + } + + return $formatted_email; +} + +sub which { + my ($bin) = @_; + + foreach my $path (split(/:/, $ENV{PATH})) { + if (-e "$path/$bin") { + return "$path/$bin"; + } + } + + return ""; +} + +sub which_conf { + my ($conf) = @_; + + foreach my $path (split(/:/, ".:$ENV{HOME}:.scripts")) { + if (-e "$path/$conf") { + return "$path/$conf"; + } + } + + return ""; +} + +sub expand_tabs { + my ($str) = @_; + + my $res = ''; + my $n = 0; + for my $c (split(//, $str)) { + if ($c eq "\t") { + $res .= ' '; + $n++; + for (; ($n % 8) != 0; $n++) { + $res .= ' '; + } + next; + } + $res .= $c; + $n++; + } + + return $res; +} +sub copy_spacing { + (my $res = shift) =~ tr/\t/ /c; + return $res; +} + +sub line_stats { + my ($line) = @_; + + # Drop the diff line leader and expand tabs + $line =~ s/^.//; + $line = expand_tabs($line); + + # Pick the indent from the front of the line. + my ($white) = ($line =~ /^(\s*)/); + + return (length($line), length($white)); +} + +my $sanitise_quote = ''; + +sub sanitise_line_reset { + my ($in_comment) = @_; + + if ($in_comment) { + $sanitise_quote = '*/'; + } else { + $sanitise_quote = ''; + } +} +sub sanitise_line { + my ($line) = @_; + + my $res = ''; + my $l = ''; + + my $qlen = 0; + my $off = 0; + my $c; + + # Always copy over the diff marker. + $res = substr($line, 0, 1); + + for ($off = 1; $off < length($line); $off++) { + $c = substr($line, $off, 1); + + # Comments we are wacking completly including the begin + # and end, all to $;. + if ($sanitise_quote eq '' && substr($line, $off, 2) eq '/*') { + $sanitise_quote = '*/'; + + substr($res, $off, 2, "$;$;"); + $off++; + next; + } + if ($sanitise_quote eq '*/' && substr($line, $off, 2) eq '*/') { + $sanitise_quote = ''; + substr($res, $off, 2, "$;$;"); + $off++; + next; + } + if ($sanitise_quote eq '' && substr($line, $off, 2) eq '//') { + $sanitise_quote = '//'; + + substr($res, $off, 2, $sanitise_quote); + $off++; + next; + } + + # A \ in a string means ignore the next character. + if (($sanitise_quote eq "'" || $sanitise_quote eq '"') && + $c eq "\\") { + substr($res, $off, 2, 'XX'); + $off++; + next; + } + # Regular quotes. + if ($c eq "'" || $c eq '"') { + if ($sanitise_quote eq '') { + $sanitise_quote = $c; + + substr($res, $off, 1, $c); + next; + } elsif ($sanitise_quote eq $c) { + $sanitise_quote = ''; + } + } + + #print "c<$c> SQ<$sanitise_quote>\n"; + if ($off != 0 && $sanitise_quote eq '*/' && $c ne "\t") { + substr($res, $off, 1, $;); + } elsif ($off != 0 && $sanitise_quote eq '//' && $c ne "\t") { + substr($res, $off, 1, $;); + } elsif ($off != 0 && $sanitise_quote && $c ne "\t") { + substr($res, $off, 1, 'X'); + } else { + substr($res, $off, 1, $c); + } + } + + if ($sanitise_quote eq '//') { + $sanitise_quote = ''; + } + + # The pathname on a #include may be surrounded by '<' and '>'. + if ($res =~ /^.\s*\#\s*include\s+\<(.*)\>/) { + my $clean = 'X' x length($1); + $res =~ s@\<.*\>@<$clean>@; + + # The whole of a #error is a string. + } elsif ($res =~ /^.\s*\#\s*(?:error|warning)\s+(.*)\b/) { + my $clean = 'X' x length($1); + $res =~ s@(\#\s*(?:error|warning)\s+).*@$1$clean@; + } + + if ($allow_c99_comments && $res =~ m@(//.*$)@) { + my $match = $1; + $res =~ s/\Q$match\E/"$;" x length($match)/e; + } + + return $res; +} + +sub get_quoted_string { + my ($line, $rawline) = @_; + + return "" if ($line !~ m/($String)/g); + return substr($rawline, $-[0], $+[0] - $-[0]); +} + +sub ctx_statement_block { + my ($linenr, $remain, $off) = @_; + my $line = $linenr - 1; + my $blk = ''; + my $soff = $off; + my $coff = $off - 1; + my $coff_set = 0; + + my $loff = 0; + + my $type = ''; + my $level = 0; + my @stack = (); + my $p; + my $c; + my $len = 0; + + my $remainder; + while (1) { + @stack = (['', 0]) if ($#stack == -1); + + #warn "CSB: blk<$blk> remain<$remain>\n"; + # If we are about to drop off the end, pull in more + # context. + if ($off >= $len) { + for (; $remain > 0; $line++) { + last if (!defined $lines[$line]); + next if ($lines[$line] =~ /^-/); + $remain--; + $loff = $len; + $blk .= $lines[$line] . "\n"; + $len = length($blk); + $line++; + last; + } + # Bail if there is no further context. + #warn "CSB: blk<$blk> off<$off> len<$len>\n"; + if ($off >= $len) { + last; + } + if ($level == 0 && substr($blk, $off) =~ /^.\s*#\s*define/) { + $level++; + $type = '#'; + } + } + $p = $c; + $c = substr($blk, $off, 1); + $remainder = substr($blk, $off); + + #warn "CSB: c<$c> type<$type> level<$level> remainder<$remainder> coff_set<$coff_set>\n"; + + # Handle nested #if/#else. + if ($remainder =~ /^#\s*(?:ifndef|ifdef|if)\s/) { + push(@stack, [ $type, $level ]); + } elsif ($remainder =~ /^#\s*(?:else|elif)\b/) { + ($type, $level) = @{$stack[$#stack - 1]}; + } elsif ($remainder =~ /^#\s*endif\b/) { + ($type, $level) = @{pop(@stack)}; + } + + # Statement ends at the ';' or a close '}' at the + # outermost level. + if ($level == 0 && $c eq ';') { + last; + } + + # An else is really a conditional as long as its not else if + if ($level == 0 && $coff_set == 0 && + (!defined($p) || $p =~ /(?:\s|\}|\+)/) && + $remainder =~ /^(else)(?:\s|{)/ && + $remainder !~ /^else\s+if\b/) { + $coff = $off + length($1) - 1; + $coff_set = 1; + #warn "CSB: mark coff<$coff> soff<$soff> 1<$1>\n"; + #warn "[" . substr($blk, $soff, $coff - $soff + 1) . "]\n"; + } + + if (($type eq '' || $type eq '(') && $c eq '(') { + $level++; + $type = '('; + } + if ($type eq '(' && $c eq ')') { + $level--; + $type = ($level != 0)? '(' : ''; + + if ($level == 0 && $coff < $soff) { + $coff = $off; + $coff_set = 1; + #warn "CSB: mark coff<$coff>\n"; + } + } + if (($type eq '' || $type eq '{') && $c eq '{') { + $level++; + $type = '{'; + } + if ($type eq '{' && $c eq '}') { + $level--; + $type = ($level != 0)? '{' : ''; + + if ($level == 0) { + if (substr($blk, $off + 1, 1) eq ';') { + $off++; + } + last; + } + } + # Preprocessor commands end at the newline unless escaped. + if ($type eq '#' && $c eq "\n" && $p ne "\\") { + $level--; + $type = ''; + $off++; + last; + } + $off++; + } + # We are truly at the end, so shuffle to the next line. + if ($off == $len) { + $loff = $len + 1; + $line++; + $remain--; + } + + my $statement = substr($blk, $soff, $off - $soff + 1); + my $condition = substr($blk, $soff, $coff - $soff + 1); + + #warn "STATEMENT<$statement>\n"; + #warn "CONDITION<$condition>\n"; + + #print "coff<$coff> soff<$off> loff<$loff>\n"; + + return ($statement, $condition, + $line, $remain + 1, $off - $loff + 1, $level); +} + +sub statement_lines { + my ($stmt) = @_; + + # Strip the diff line prefixes and rip blank lines at start and end. + $stmt =~ s/(^|\n)./$1/g; + $stmt =~ s/^\s*//; + $stmt =~ s/\s*$//; + + my @stmt_lines = ($stmt =~ /\n/g); + + return $#stmt_lines + 2; +} + +sub statement_rawlines { + my ($stmt) = @_; + + my @stmt_lines = ($stmt =~ /\n/g); + + return $#stmt_lines + 2; +} + +sub statement_block_size { + my ($stmt) = @_; + + $stmt =~ s/(^|\n)./$1/g; + $stmt =~ s/^\s*{//; + $stmt =~ s/}\s*$//; + $stmt =~ s/^\s*//; + $stmt =~ s/\s*$//; + + my @stmt_lines = ($stmt =~ /\n/g); + my @stmt_statements = ($stmt =~ /;/g); + + my $stmt_lines = $#stmt_lines + 2; + my $stmt_statements = $#stmt_statements + 1; + + if ($stmt_lines > $stmt_statements) { + return $stmt_lines; + } else { + return $stmt_statements; + } +} + +sub ctx_statement_full { + my ($linenr, $remain, $off) = @_; + my ($statement, $condition, $level); + + my (@chunks); + + # Grab the first conditional/block pair. + ($statement, $condition, $linenr, $remain, $off, $level) = + ctx_statement_block($linenr, $remain, $off); + #print "F: c<$condition> s<$statement> remain<$remain>\n"; + push(@chunks, [ $condition, $statement ]); + if (!($remain > 0 && $condition =~ /^\s*(?:\n[+-])?\s*(?:if|else|do)\b/s)) { + return ($level, $linenr, @chunks); + } + + # Pull in the following conditional/block pairs and see if they + # could continue the statement. + for (;;) { + ($statement, $condition, $linenr, $remain, $off, $level) = + ctx_statement_block($linenr, $remain, $off); + #print "C: c<$condition> s<$statement> remain<$remain>\n"; + last if (!($remain > 0 && $condition =~ /^(?:\s*\n[+-])*\s*(?:else|do)\b/s)); + #print "C: push\n"; + push(@chunks, [ $condition, $statement ]); + } + + return ($level, $linenr, @chunks); +} + +sub ctx_block_get { + my ($linenr, $remain, $outer, $open, $close, $off) = @_; + my $line; + my $start = $linenr - 1; + my $blk = ''; + my @o; + my @c; + my @res = (); + + my $level = 0; + my @stack = ($level); + for ($line = $start; $remain > 0; $line++) { + next if ($rawlines[$line] =~ /^-/); + $remain--; + + $blk .= $rawlines[$line]; + + # Handle nested #if/#else. + if ($lines[$line] =~ /^.\s*#\s*(?:ifndef|ifdef|if)\s/) { + push(@stack, $level); + } elsif ($lines[$line] =~ /^.\s*#\s*(?:else|elif)\b/) { + $level = $stack[$#stack - 1]; + } elsif ($lines[$line] =~ /^.\s*#\s*endif\b/) { + $level = pop(@stack); + } + + foreach my $c (split(//, $lines[$line])) { + ##print "C<$c>L<$level><$open$close>O<$off>\n"; + if ($off > 0) { + $off--; + next; + } + + if ($c eq $close && $level > 0) { + $level--; + last if ($level == 0); + } elsif ($c eq $open) { + $level++; + } + } + + if (!$outer || $level <= 1) { + push(@res, $rawlines[$line]); + } + + last if ($level == 0); + } + + return ($level, @res); +} +sub ctx_block_outer { + my ($linenr, $remain) = @_; + + my ($level, @r) = ctx_block_get($linenr, $remain, 1, '{', '}', 0); + return @r; +} +sub ctx_block { + my ($linenr, $remain) = @_; + + my ($level, @r) = ctx_block_get($linenr, $remain, 0, '{', '}', 0); + return @r; +} +sub ctx_statement { + my ($linenr, $remain, $off) = @_; + + my ($level, @r) = ctx_block_get($linenr, $remain, 0, '(', ')', $off); + return @r; +} +sub ctx_block_level { + my ($linenr, $remain) = @_; + + return ctx_block_get($linenr, $remain, 0, '{', '}', 0); +} +sub ctx_statement_level { + my ($linenr, $remain, $off) = @_; + + return ctx_block_get($linenr, $remain, 0, '(', ')', $off); +} + +sub ctx_locate_comment { + my ($first_line, $end_line) = @_; + + # Catch a comment on the end of the line itself. + my ($current_comment) = ($rawlines[$end_line - 1] =~ m@.*(/\*.*\*/)\s*(?:\\\s*)?$@); + return $current_comment if (defined $current_comment); + + # Look through the context and try and figure out if there is a + # comment. + my $in_comment = 0; + $current_comment = ''; + for (my $linenr = $first_line; $linenr < $end_line; $linenr++) { + my $line = $rawlines[$linenr - 1]; + #warn " $line\n"; + if ($linenr == $first_line and $line =~ m@^.\s*\*@) { + $in_comment = 1; + } + if ($line =~ m@/\*@) { + $in_comment = 1; + } + if (!$in_comment && $current_comment ne '') { + $current_comment = ''; + } + $current_comment .= $line . "\n" if ($in_comment); + if ($line =~ m@\*/@) { + $in_comment = 0; + } + } + + chomp($current_comment); + return($current_comment); +} +sub ctx_has_comment { + my ($first_line, $end_line) = @_; + my $cmt = ctx_locate_comment($first_line, $end_line); + + ##print "LINE: $rawlines[$end_line - 1 ]\n"; + ##print "CMMT: $cmt\n"; + + return ($cmt ne ''); +} + +sub raw_line { + my ($linenr, $cnt) = @_; + + my $offset = $linenr - 1; + $cnt++; + + my $line; + while ($cnt) { + $line = $rawlines[$offset++]; + next if (defined($line) && $line =~ /^-/); + $cnt--; + } + + return $line; +} + +sub cat_vet { + my ($vet) = @_; + my ($res, $coded); + + $res = ''; + while ($vet =~ /([^[:cntrl:]]*)([[:cntrl:]]|$)/g) { + $res .= $1; + if ($2 ne '') { + $coded = sprintf("^%c", unpack('C', $2) + 64); + $res .= $coded; + } + } + $res =~ s/$/\$/; + + return $res; +} + +my $av_preprocessor = 0; +my $av_pending; +my @av_paren_type; +my $av_pend_colon; + +sub annotate_reset { + $av_preprocessor = 0; + $av_pending = '_'; + @av_paren_type = ('E'); + $av_pend_colon = 'O'; +} + +sub annotate_values { + my ($stream, $type) = @_; + + my $res; + my $var = '_' x length($stream); + my $cur = $stream; + + print "$stream\n" if ($dbg_values > 1); + + while (length($cur)) { + @av_paren_type = ('E') if ($#av_paren_type < 0); + print " <" . join('', @av_paren_type) . + "> <$type> <$av_pending>" if ($dbg_values > 1); + if ($cur =~ /^(\s+)/o) { + print "WS($1)\n" if ($dbg_values > 1); + if ($1 =~ /\n/ && $av_preprocessor) { + $type = pop(@av_paren_type); + $av_preprocessor = 0; + } + + } elsif ($cur =~ /^(\(\s*$Type\s*)\)/ && $av_pending eq '_') { + print "CAST($1)\n" if ($dbg_values > 1); + push(@av_paren_type, $type); + $type = 'c'; + + } elsif ($cur =~ /^($Type)\s*(?:$Ident|,|\)|\(|\s*$)/) { + print "DECLARE($1)\n" if ($dbg_values > 1); + $type = 'T'; + + } elsif ($cur =~ /^($Modifier)\s*/) { + print "MODIFIER($1)\n" if ($dbg_values > 1); + $type = 'T'; + + } elsif ($cur =~ /^(\#\s*define\s*$Ident)(\(?)/o) { + print "DEFINE($1,$2)\n" if ($dbg_values > 1); + $av_preprocessor = 1; + push(@av_paren_type, $type); + if ($2 ne '') { + $av_pending = 'N'; + } + $type = 'E'; + + } elsif ($cur =~ /^(\#\s*(?:undef\s*$Ident|include\b))/o) { + print "UNDEF($1)\n" if ($dbg_values > 1); + $av_preprocessor = 1; + push(@av_paren_type, $type); + + } elsif ($cur =~ /^(\#\s*(?:ifdef|ifndef|if))/o) { + print "PRE_START($1)\n" if ($dbg_values > 1); + $av_preprocessor = 1; + + push(@av_paren_type, $type); + push(@av_paren_type, $type); + $type = 'E'; + + } elsif ($cur =~ /^(\#\s*(?:else|elif))/o) { + print "PRE_RESTART($1)\n" if ($dbg_values > 1); + $av_preprocessor = 1; + + push(@av_paren_type, $av_paren_type[$#av_paren_type]); + + $type = 'E'; + + } elsif ($cur =~ /^(\#\s*(?:endif))/o) { + print "PRE_END($1)\n" if ($dbg_values > 1); + + $av_preprocessor = 1; + + # Assume all arms of the conditional end as this + # one does, and continue as if the #endif was not here. + pop(@av_paren_type); + push(@av_paren_type, $type); + $type = 'E'; + + } elsif ($cur =~ /^(\\\n)/o) { + print "PRECONT($1)\n" if ($dbg_values > 1); + + } elsif ($cur =~ /^(__attribute__)\s*\(?/o) { + print "ATTR($1)\n" if ($dbg_values > 1); + $av_pending = $type; + $type = 'N'; + + } elsif ($cur =~ /^(sizeof)\s*(\()?/o) { + print "SIZEOF($1)\n" if ($dbg_values > 1); + if (defined $2) { + $av_pending = 'V'; + } + $type = 'N'; + + } elsif ($cur =~ /^(if|while|for)\b/o) { + print "COND($1)\n" if ($dbg_values > 1); + $av_pending = 'E'; + $type = 'N'; + + } elsif ($cur =~/^(case)/o) { + print "CASE($1)\n" if ($dbg_values > 1); + $av_pend_colon = 'C'; + $type = 'N'; + + } elsif ($cur =~/^(return|else|goto|typeof|__typeof__)\b/o) { + print "KEYWORD($1)\n" if ($dbg_values > 1); + $type = 'N'; + + } elsif ($cur =~ /^(\()/o) { + print "PAREN('$1')\n" if ($dbg_values > 1); + push(@av_paren_type, $av_pending); + $av_pending = '_'; + $type = 'N'; + + } elsif ($cur =~ /^(\))/o) { + my $new_type = pop(@av_paren_type); + if ($new_type ne '_') { + $type = $new_type; + print "PAREN('$1') -> $type\n" + if ($dbg_values > 1); + } else { + print "PAREN('$1')\n" if ($dbg_values > 1); + } + + } elsif ($cur =~ /^($Ident)\s*\(/o) { + print "FUNC($1)\n" if ($dbg_values > 1); + $type = 'V'; + $av_pending = 'V'; + + } elsif ($cur =~ /^($Ident\s*):(?:\s*\d+\s*(,|=|;))?/) { + if (defined $2 && $type eq 'C' || $type eq 'T') { + $av_pend_colon = 'B'; + } elsif ($type eq 'E') { + $av_pend_colon = 'L'; + } + print "IDENT_COLON($1,$type>$av_pend_colon)\n" if ($dbg_values > 1); + $type = 'V'; + + } elsif ($cur =~ /^($Ident|$Constant)/o) { + print "IDENT($1)\n" if ($dbg_values > 1); + $type = 'V'; + + } elsif ($cur =~ /^($Assignment)/o) { + print "ASSIGN($1)\n" if ($dbg_values > 1); + $type = 'N'; + + } elsif ($cur =~/^(;|{|})/) { + print "END($1)\n" if ($dbg_values > 1); + $type = 'E'; + $av_pend_colon = 'O'; + + } elsif ($cur =~/^(,)/) { + print "COMMA($1)\n" if ($dbg_values > 1); + $type = 'C'; + + } elsif ($cur =~ /^(\?)/o) { + print "QUESTION($1)\n" if ($dbg_values > 1); + $type = 'N'; + + } elsif ($cur =~ /^(:)/o) { + print "COLON($1,$av_pend_colon)\n" if ($dbg_values > 1); + + substr($var, length($res), 1, $av_pend_colon); + if ($av_pend_colon eq 'C' || $av_pend_colon eq 'L') { + $type = 'E'; + } else { + $type = 'N'; + } + $av_pend_colon = 'O'; + + } elsif ($cur =~ /^(\[)/o) { + print "CLOSE($1)\n" if ($dbg_values > 1); + $type = 'N'; + + } elsif ($cur =~ /^(-(?![->])|\+(?!\+)|\*|\&\&|\&)/o) { + my $variant; + + print "OPV($1)\n" if ($dbg_values > 1); + if ($type eq 'V') { + $variant = 'B'; + } else { + $variant = 'U'; + } + + substr($var, length($res), 1, $variant); + $type = 'N'; + + } elsif ($cur =~ /^($Operators)/o) { + print "OP($1)\n" if ($dbg_values > 1); + if ($1 ne '++' && $1 ne '--') { + $type = 'N'; + } + + } elsif ($cur =~ /(^.)/o) { + print "C($1)\n" if ($dbg_values > 1); + } + if (defined $1) { + $cur = substr($cur, length($1)); + $res .= $type x length($1); + } + } + + return ($res, $var); +} + +sub possible { + my ($possible, $line) = @_; + my $notPermitted = qr{(?: + ^(?: + $Modifier| + $Storage| + $Type| + DEFINE_\S+ + )$| + ^(?: + goto| + return| + case| + else| + asm|__asm__| + do| + \#| + \#\#| + )(?:\s|$)| + ^(?:typedef|struct|enum)\b + )}x; + warn "CHECK<$possible> ($line)\n" if ($dbg_possible > 2); + if ($possible !~ $notPermitted) { + # Check for modifiers. + $possible =~ s/\s*$Storage\s*//g; + $possible =~ s/\s*$Sparse\s*//g; + if ($possible =~ /^\s*$/) { + + } elsif ($possible =~ /\s/) { + $possible =~ s/\s*$Type\s*//g; + for my $modifier (split(' ', $possible)) { + if ($modifier !~ $notPermitted) { + warn "MODIFIER: $modifier ($possible) ($line)\n" if ($dbg_possible); + push(@modifierListFile, $modifier); + } + } + + } else { + warn "POSSIBLE: $possible ($line)\n" if ($dbg_possible); + push(@typeListFile, $possible); + } + build_types(); + } else { + warn "NOTPOSS: $possible ($line)\n" if ($dbg_possible > 1); + } +} + +my $prefix = ''; + +sub show_type { + my ($type) = @_; + + $type =~ tr/[a-z]/[A-Z]/; + + return defined $use_type{$type} if (scalar keys %use_type > 0); + + return !defined $ignore_type{$type}; +} + +sub report { + my ($level, $type, $msg) = @_; + + if (!show_type($type) || + (defined $tst_only && $msg !~ /\Q$tst_only\E/)) { + return 0; + } + my $output = ''; + if ($color) { + if ($level eq 'ERROR') { + $output .= RED; + } elsif ($level eq 'WARNING') { + $output .= YELLOW; + } else { + $output .= GREEN; + } + } + $output .= $prefix . $level . ':'; + if ($show_types) { + $output .= BLUE if ($color); + $output .= "$type:"; + } + $output .= RESET if ($color); + $output .= ' ' . $msg . "\n"; + + if ($showfile) { + my @lines = split("\n", $output, -1); + splice(@lines, 1, 1); + $output = join("\n", @lines); + } + $output = (split('\n', $output))[0] . "\n" if ($terse); + + push(our @report, $output); + + return 1; +} + +sub report_dump { + our @report; +} + +sub fixup_current_range { + my ($lineRef, $offset, $length) = @_; + + if ($$lineRef =~ /^\@\@ -\d+,\d+ \+(\d+),(\d+) \@\@/) { + my $o = $1; + my $l = $2; + my $no = $o + $offset; + my $nl = $l + $length; + $$lineRef =~ s/\+$o,$l \@\@/\+$no,$nl \@\@/; + } +} + +sub fix_inserted_deleted_lines { + my ($linesRef, $insertedRef, $deletedRef) = @_; + + my $range_last_linenr = 0; + my $delta_offset = 0; + + my $old_linenr = 0; + my $new_linenr = 0; + + my $next_insert = 0; + my $next_delete = 0; + + my @lines = (); + + my $inserted = @{$insertedRef}[$next_insert++]; + my $deleted = @{$deletedRef}[$next_delete++]; + + foreach my $old_line (@{$linesRef}) { + my $save_line = 1; + my $line = $old_line; #don't modify the array + if ($line =~ /^(?:\+\+\+|\-\-\-)\s+\S+/) { #new filename + $delta_offset = 0; + } elsif ($line =~ /^\@\@ -\d+,\d+ \+\d+,\d+ \@\@/) { #new hunk + $range_last_linenr = $new_linenr; + fixup_current_range(\$line, $delta_offset, 0); + } + + while (defined($deleted) && ${$deleted}{'LINENR'} == $old_linenr) { + $deleted = @{$deletedRef}[$next_delete++]; + $save_line = 0; + fixup_current_range(\$lines[$range_last_linenr], $delta_offset--, -1); + } + + while (defined($inserted) && ${$inserted}{'LINENR'} == $old_linenr) { + push(@lines, ${$inserted}{'LINE'}); + $inserted = @{$insertedRef}[$next_insert++]; + $new_linenr++; + fixup_current_range(\$lines[$range_last_linenr], $delta_offset++, 1); + } + + if ($save_line) { + push(@lines, $line); + $new_linenr++; + } + + $old_linenr++; + } + + return @lines; +} + +sub fix_insert_line { + my ($linenr, $line) = @_; + + my $inserted = { + LINENR => $linenr, + LINE => $line, + }; + push(@fixed_inserted, $inserted); +} + +sub fix_delete_line { + my ($linenr, $line) = @_; + + my $deleted = { + LINENR => $linenr, + LINE => $line, + }; + + push(@fixed_deleted, $deleted); +} + +sub ERROR { + my ($type, $msg) = @_; + + if (report("ERROR", $type, $msg)) { + our $clean = 0; + our $cnt_error++; + return 1; + } + return 0; +} +sub WARN { + my ($type, $msg) = @_; + + if (report("WARNING", $type, $msg)) { + our $clean = 0; + our $cnt_warn++; + return 1; + } + return 0; +} +sub CHK { + my ($type, $msg) = @_; + + if ($check && report("CHECK", $type, $msg)) { + our $clean = 0; + our $cnt_chk++; + return 1; + } + return 0; +} + +sub check_absolute_file { + my ($absolute, $herecurr) = @_; + my $file = $absolute; + + ##print "absolute<$absolute>\n"; + + # See if any suffix of this path is a path within the tree. + while ($file =~ s@^[^/]*/@@) { + if (-f "$root/$file") { + ##print "file<$file>\n"; + last; + } + } + if (! -f _) { + return 0; + } + + # It is, so see if the prefix is acceptable. + my $prefix = $absolute; + substr($prefix, -length($file)) = ''; + + ##print "prefix<$prefix>\n"; + if ($prefix ne ".../") { + WARN("USE_RELATIVE_PATH", + "use relative pathname instead of absolute in changelog text\n" . $herecurr); + } +} + +sub trim { + my ($string) = @_; + + $string =~ s/^\s+|\s+$//g; + + return $string; +} + +sub ltrim { + my ($string) = @_; + + $string =~ s/^\s+//; + + return $string; +} + +sub rtrim { + my ($string) = @_; + + $string =~ s/\s+$//; + + return $string; +} + +sub string_find_replace { + my ($string, $find, $replace) = @_; + + $string =~ s/$find/$replace/g; + + return $string; +} + +sub tabify { + my ($leading) = @_; + + my $source_indent = 8; + my $max_spaces_before_tab = $source_indent - 1; + my $spaces_to_tab = " " x $source_indent; + + #convert leading spaces to tabs + 1 while $leading =~ s@^([\t]*)$spaces_to_tab@$1\t@g; + #Remove spaces before a tab + 1 while $leading =~ s@^([\t]*)( {1,$max_spaces_before_tab})\t@$1\t@g; + + return "$leading"; +} + +sub pos_last_openparen { + my ($line) = @_; + + my $pos = 0; + + my $opens = $line =~ tr/\(/\(/; + my $closes = $line =~ tr/\)/\)/; + + my $last_openparen = 0; + + if (($opens == 0) || ($closes >= $opens)) { + return -1; + } + + my $len = length($line); + + for ($pos = 0; $pos < $len; $pos++) { + my $string = substr($line, $pos); + if ($string =~ /^($FuncArg|$balanced_parens)/) { + $pos += length($1) - 1; + } elsif (substr($line, $pos, 1) eq '(') { + $last_openparen = $pos; + } elsif (index($string, '(') == -1) { + last; + } + } + + return length(expand_tabs(substr($line, 0, $last_openparen))) + 1; +} + +sub process { + my $filename = shift; + + my $linenr=0; + my $prevline=""; + my $prevrawline=""; + my $stashline=""; + my $stashrawline=""; + + my $length; + my $indent; + my $previndent=0; + my $stashindent=0; + + our $clean = 1; + my $signoff = 0; + my $is_patch = 0; + my $in_header_lines = $file ? 0 : 1; + my $in_commit_log = 0; #Scanning lines before patch + my $has_commit_log = 0; #Encountered lines before patch + my $commit_log_possible_stack_dump = 0; + my $commit_log_long_line = 0; + my $commit_log_has_diff = 0; + my $reported_maintainer_file = 0; + my $non_utf8_charset = 0; + + my $last_blank_line = 0; + my $last_coalesced_string_linenr = -1; + + our @report = (); + our $cnt_lines = 0; + our $cnt_error = 0; + our $cnt_warn = 0; + our $cnt_chk = 0; + + # Trace the real file/line as we go. + my $realfile = ''; + my $realline = 0; + my $realcnt = 0; + my $here = ''; + my $context_function; #undef'd unless there's a known function + my $in_comment = 0; + my $comment_edge = 0; + my $first_line = 0; + my $p1_prefix = ''; + + my $prev_values = 'E'; + + # suppression flags + my %suppress_ifbraces; + my %suppress_whiletrailers; + my %suppress_export; + my $suppress_statement = 0; + + my %signatures = (); + + # Pre-scan the patch sanitizing the lines. + # Pre-scan the patch looking for any __setup documentation. + # + my @setup_docs = (); + my $setup_docs = 0; + + my $camelcase_file_seeded = 0; + + sanitise_line_reset(); + my $line; + foreach my $rawline (@rawlines) { + $linenr++; + $line = $rawline; + + push(@fixed, $rawline) if ($fix); + + if ($rawline=~/^\+\+\+\s+(\S+)/) { + $setup_docs = 0; + if ($1 =~ m@Documentation/admin-guide/kernel-parameters.rst$@) { + $setup_docs = 1; + } + #next; + } + if ($rawline =~ /^\@\@ -\d+(?:,\d+)? \+(\d+)(,(\d+))? \@\@/) { + $realline=$1-1; + if (defined $2) { + $realcnt=$3+1; + } else { + $realcnt=1+1; + } + $in_comment = 0; + + # Guestimate if this is a continuing comment. Run + # the context looking for a comment "edge". If this + # edge is a close comment then we must be in a comment + # at context start. + my $edge; + my $cnt = $realcnt; + for (my $ln = $linenr + 1; $cnt > 0; $ln++) { + next if (defined $rawlines[$ln - 1] && + $rawlines[$ln - 1] =~ /^-/); + $cnt--; + #print "RAW<$rawlines[$ln - 1]>\n"; + last if (!defined $rawlines[$ln - 1]); + if ($rawlines[$ln - 1] =~ m@(/\*|\*/)@ && + $rawlines[$ln - 1] !~ m@"[^"]*(?:/\*|\*/)[^"]*"@) { + ($edge) = $1; + last; + } + } + if (defined $edge && $edge eq '*/') { + $in_comment = 1; + } + + # Guestimate if this is a continuing comment. If this + # is the start of a diff block and this line starts + # ' *' then it is very likely a comment. + if (!defined $edge && + $rawlines[$linenr] =~ m@^.\s*(?:\*\*+| \*)(?:\s|$)@) + { + $in_comment = 1; + } + + ##print "COMMENT:$in_comment edge<$edge> $rawline\n"; + sanitise_line_reset($in_comment); + + } elsif ($realcnt && $rawline =~ /^(?:\+| |$)/) { + # Standardise the strings and chars within the input to + # simplify matching -- only bother with positive lines. + $line = sanitise_line($rawline); + } + push(@lines, $line); + + if ($realcnt > 1) { + $realcnt-- if ($line =~ /^(?:\+| |$)/); + } else { + $realcnt = 0; + } + + #print "==>$rawline\n"; + #print "-->$line\n"; + + if ($setup_docs && $line =~ /^\+/) { + push(@setup_docs, $line); + } + } + + $prefix = ''; + + $realcnt = 0; + $linenr = 0; + $fixlinenr = -1; + foreach my $line (@lines) { + $linenr++; + $fixlinenr++; + my $sline = $line; #copy of $line + $sline =~ s/$;/ /g; #with comments as spaces + + my $rawline = $rawlines[$linenr - 1]; + +#extract the line range in the file after the patch is applied + if (!$in_commit_log && + $line =~ /^\@\@ -\d+(?:,\d+)? \+(\d+)(,(\d+))? \@\@(.*)/) { + my $context = $4; + $is_patch = 1; + $first_line = $linenr + 1; + $realline=$1-1; + if (defined $2) { + $realcnt=$3+1; + } else { + $realcnt=1+1; + } + annotate_reset(); + $prev_values = 'E'; + + %suppress_ifbraces = (); + %suppress_whiletrailers = (); + %suppress_export = (); + $suppress_statement = 0; + if ($context =~ /\b(\w+)\s*\(/) { + $context_function = $1; + } else { + undef $context_function; + } + next; + +# track the line number as we move through the hunk, note that +# new versions of GNU diff omit the leading space on completely +# blank context lines so we need to count that too. + } elsif ($line =~ /^( |\+|$)/) { + $realline++; + $realcnt-- if ($realcnt != 0); + + # Measure the line length and indent. + ($length, $indent) = line_stats($rawline); + + # Track the previous line. + ($prevline, $stashline) = ($stashline, $line); + ($previndent, $stashindent) = ($stashindent, $indent); + ($prevrawline, $stashrawline) = ($stashrawline, $rawline); + + #warn "line<$line>\n"; + + } elsif ($realcnt == 1) { + $realcnt--; + } + + my $hunk_line = ($realcnt != 0); + + $here = "#$linenr: " if (!$file); + $here = "#$realline: " if ($file); + + my $found_file = 0; + # extract the filename as it passes + if ($line =~ /^diff --git.*?(\S+)$/) { + $realfile = $1; + $realfile =~ s@^([^/]*)/@@ if (!$file); + $in_commit_log = 0; + $found_file = 1; + } elsif ($line =~ /^\+\+\+\s+(\S+)/) { + $realfile = $1; + $realfile =~ s@^([^/]*)/@@ if (!$file); + $in_commit_log = 0; + + $p1_prefix = $1; + if (!$file && $tree && $p1_prefix ne '' && + -e "$root/$p1_prefix") { + WARN("PATCH_PREFIX", + "patch prefix '$p1_prefix' exists, appears to be a -p0 patch\n"); + } + + if ($realfile =~ m@^include/asm/@) { + ERROR("MODIFIED_INCLUDE_ASM", + "do not modify files in include/asm, change architecture specific files in include/asm-<architecture>\n" . "$here$rawline\n"); + } + $found_file = 1; + } + my $skipme = 0; + foreach (@exclude) { + if ($realfile =~ m@^(?:$_/)@) { + $skipme = 1; + } + } + if ($skipme) { + next; + } + +#make up the handle for any error we report on this line + if ($showfile) { + $prefix = "$realfile:$realline: " + } elsif ($emacs) { + if ($file) { + $prefix = "$filename:$realline: "; + } else { + $prefix = "$filename:$linenr: "; + } + } + + if ($found_file) { + if (is_maintained_obsolete($realfile)) { + WARN("OBSOLETE", + "$realfile is marked as 'obsolete' in the MAINTAINERS hierarchy. No unnecessary modifications please.\n"); + } + if ($realfile =~ m@^(?:drivers/net/|net/|drivers/staging/)@) { + $check = 1; + } else { + $check = $check_orig; + } + next; + } + + $here .= "FILE: $realfile:$realline:" if ($realcnt != 0); + + my $hereline = "$here\n$rawline\n"; + my $herecurr = "$here\n$rawline\n"; + my $hereprev = "$here\n$prevrawline\n$rawline\n"; + + $cnt_lines++ if ($realcnt != 0); + +# Check if the commit log has what seems like a diff which can confuse patch + if ($in_commit_log && !$commit_log_has_diff && + (($line =~ m@^\s+diff\b.*a/[\w/]+@ && + $line =~ m@^\s+diff\b.*a/([\w/]+)\s+b/$1\b@) || + $line =~ m@^\s*(?:\-\-\-\s+a/|\+\+\+\s+b/)@ || + $line =~ m/^\s*\@\@ \-\d+,\d+ \+\d+,\d+ \@\@/)) { + ERROR("DIFF_IN_COMMIT_MSG", + "Avoid using diff content in the commit message - patch(1) might not work\n" . $herecurr); + $commit_log_has_diff = 1; + } + +# Check for incorrect file permissions + if ($line =~ /^new (file )?mode.*[7531]\d{0,2}$/) { + my $permhere = $here . "FILE: $realfile\n"; + if ($realfile !~ m@scripts/@ && + $realfile !~ /\.(py|pl|awk|sh)$/) { + ERROR("EXECUTE_PERMISSIONS", + "do not set execute permissions for source files\n" . $permhere); + } + } + +# Check the patch for a signoff: + if ($line =~ /^\s*signed-off-by:/i) { + $signoff++; + $in_commit_log = 0; + } + +# Check if CODEOWNERS is being updated. If so, there's probably no need to +# emit the "does CODEOWNERS need updating?" message on file add/move/delete + if ($line =~ /^\s*CODEOWNERS\s*\|/) { + $reported_maintainer_file = 1; + } + +# Check signature styles + if (!$in_header_lines && + $line =~ /^(\s*)([a-z0-9_-]+by:|$signature_tags)(\s*)(.*)/i) { + my $space_before = $1; + my $sign_off = $2; + my $space_after = $3; + my $email = $4; + my $ucfirst_sign_off = ucfirst(lc($sign_off)); + + if ($sign_off !~ /$signature_tags/) { + WARN("BAD_SIGN_OFF", + "Non-standard signature: $sign_off\n" . $herecurr); + } + if (defined $space_before && $space_before ne "") { + if (WARN("BAD_SIGN_OFF", + "Do not use whitespace before $ucfirst_sign_off\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] = + "$ucfirst_sign_off $email"; + } + } + if ($sign_off =~ /-by:$/i && $sign_off ne $ucfirst_sign_off) { + if (WARN("BAD_SIGN_OFF", + "'$ucfirst_sign_off' is the preferred signature form\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] = + "$ucfirst_sign_off $email"; + } + + } + if (!defined $space_after || $space_after ne " ") { + if (WARN("BAD_SIGN_OFF", + "Use a single space after $ucfirst_sign_off\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] = + "$ucfirst_sign_off $email"; + } + } + + my ($email_name, $email_address, $comment) = parse_email($email); + my $suggested_email = format_email(($email_name, $email_address)); + if ($suggested_email eq "") { + ERROR("BAD_SIGN_OFF", + "Unrecognized email address: '$email'\n" . $herecurr); + } else { + my $dequoted = $suggested_email; + $dequoted =~ s/^"//; + $dequoted =~ s/" </ </; + # Don't force email to have quotes + # Allow just an angle bracketed address + if ("$dequoted$comment" ne $email && + "<$email_address>$comment" ne $email && + "$suggested_email$comment" ne $email) { + WARN("BAD_SIGN_OFF", + "email address '$email' might be better as '$suggested_email$comment'\n" . $herecurr); + } + } + +# Check for duplicate signatures + my $sig_nospace = $line; + $sig_nospace =~ s/\s//g; + $sig_nospace = lc($sig_nospace); + if (defined $signatures{$sig_nospace}) { + WARN("BAD_SIGN_OFF", + "Duplicate signature\n" . $herecurr); + } else { + $signatures{$sig_nospace} = 1; + } + } + +# Check email subject for common tools that don't need to be mentioned + if ($in_header_lines && + $line =~ /^Subject:.*\b(?:checkpatch|sparse|smatch)\b[^:]/i) { + WARN("EMAIL_SUBJECT", + "A patch subject line should describe the change not the tool that found it\n" . $herecurr); + } + +# Check for old stable address + if ($line =~ /^\s*cc:\s*.*<?\bstable\@kernel\.org\b>?.*$/i) { + ERROR("STABLE_ADDRESS", + "The 'stable' address should be 'stable\@vger.kernel.org'\n" . $herecurr); + } + +# Check for unwanted Gerrit info + if ($in_commit_log && $line =~ /^\s*change-id:/i) { + ERROR("GERRIT_CHANGE_ID", + "Remove Gerrit Change-Id's before submitting upstream.\n" . $herecurr); + } + +# Check if the commit log is in a possible stack dump + if ($in_commit_log && !$commit_log_possible_stack_dump && + ($line =~ /^\s*(?:WARNING:|BUG:)/ || + $line =~ /^\s*\[\s*\d+\.\d{6,6}\s*\]/ || + # timestamp + $line =~ /^\s*\[\<[0-9a-fA-F]{8,}\>\]/)) { + # stack dump address + $commit_log_possible_stack_dump = 1; + } + +# Check for line lengths > 75 in commit log, warn once + if ($in_commit_log && !$commit_log_long_line && + length($line) > 75 && + !($line =~ /^\s*[a-zA-Z0-9_\/\.]+\s+\|\s+\d+/ || + # file delta changes + $line =~ /^\s*(?:[\w\.\-]+\/)++[\w\.\-]+:/ || + # filename then : + $line =~ /^\s*(?:Fixes:|Link:)/i || + # A Fixes: or Link: line + $commit_log_possible_stack_dump)) { + WARN("COMMIT_LOG_LONG_LINE", + "Possible unwrapped commit description (prefer a maximum 75 chars per line)\n" . $herecurr); + $commit_log_long_line = 1; + } + +# Reset possible stack dump if a blank line is found + if ($in_commit_log && $commit_log_possible_stack_dump && + $line =~ /^\s*$/) { + $commit_log_possible_stack_dump = 0; + } + +# Check for git id commit length and improperly formed commit descriptions + if ($in_commit_log && !$commit_log_possible_stack_dump && + $line !~ /^\s*(?:Link|Patchwork|http|https|BugLink):/i && + $line !~ /^This reverts commit [0-9a-f]{7,40}/ && + ($line =~ /\bcommit\s+[0-9a-f]{5,}\b/i || + ($line =~ /(?:\s|^)[0-9a-f]{12,40}(?:[\s"'\(\[]|$)/i && + $line !~ /[\<\[][0-9a-f]{12,40}[\>\]]/i && + $line !~ /\bfixes:\s*[0-9a-f]{12,40}/i))) { + my $init_char = "c"; + my $orig_commit = ""; + my $short = 1; + my $long = 0; + my $case = 1; + my $space = 1; + my $hasdesc = 0; + my $hasparens = 0; + my $id = '0123456789ab'; + my $orig_desc = "commit description"; + my $description = ""; + + if ($line =~ /\b(c)ommit\s+([0-9a-f]{5,})\b/i) { + $init_char = $1; + $orig_commit = lc($2); + } elsif ($line =~ /\b([0-9a-f]{12,40})\b/i) { + $orig_commit = lc($1); + } + + $short = 0 if ($line =~ /\bcommit\s+[0-9a-f]{12,40}/i); + $long = 1 if ($line =~ /\bcommit\s+[0-9a-f]{41,}/i); + $space = 0 if ($line =~ /\bcommit [0-9a-f]/i); + $case = 0 if ($line =~ /\b[Cc]ommit\s+[0-9a-f]{5,40}[^A-F]/); + if ($line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("([^"]+)"\)/i) { + $orig_desc = $1; + $hasparens = 1; + } elsif ($line =~ /\bcommit\s+[0-9a-f]{5,}\s*$/i && + defined $rawlines[$linenr] && + $rawlines[$linenr] =~ /^\s*\("([^"]+)"\)/) { + $orig_desc = $1; + $hasparens = 1; + } elsif ($line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("[^"]+$/i && + defined $rawlines[$linenr] && + $rawlines[$linenr] =~ /^\s*[^"]+"\)/) { + $line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("([^"]+)$/i; + $orig_desc = $1; + $rawlines[$linenr] =~ /^\s*([^"]+)"\)/; + $orig_desc .= " " . $1; + $hasparens = 1; + } + + ($id, $description) = git_commit_info($orig_commit, + $id, $orig_desc); + + if (defined($id) && + ($short || $long || $space || $case || ($orig_desc ne $description) || !$hasparens)) { + ERROR("GIT_COMMIT_ID", + "Please use git commit description style 'commit <12+ chars of sha1> (\"<title line>\")' - ie: '${init_char}ommit $id (\"$description\")'\n" . $herecurr); + } + } + +# Check for added, moved or deleted files + if (!$reported_maintainer_file && !$in_commit_log && + ($line =~ /^(?:new|deleted) file mode\s*\d+\s*$/ || + $line =~ /^rename (?:from|to) [\w\/\.\-]+\s*$/ || + ($line =~ /\{\s*([\w\/\.\-]*)\s*\=\>\s*([\w\/\.\-]*)\s*\}/ && + (defined($1) || defined($2))))) { + $is_patch = 1; + $reported_maintainer_file = 1; + WARN("FILE_PATH_CHANGES", + "added, moved or deleted file(s), does CODEOWNERS need updating?\n" . $herecurr); + } + +# Check for wrappage within a valid hunk of the file + if ($realcnt != 0 && $line !~ m{^(?:\+|-| |\\ No newline|$)}) { + ERROR("CORRUPTED_PATCH", + "patch seems to be corrupt (line wrapped?)\n" . + $herecurr) if (!$emitted_corrupt++); + } + +# UTF-8 regex found at http://www.w3.org/International/questions/qa-forms-utf-8.en.php + if (($realfile =~ /^$/ || $line =~ /^\+/) && + $rawline !~ m/^$UTF8*$/) { + my ($utf8_prefix) = ($rawline =~ /^($UTF8*)/); + + my $blank = copy_spacing($rawline); + my $ptr = substr($blank, 0, length($utf8_prefix)) . "^"; + my $hereptr = "$hereline$ptr\n"; + + CHK("INVALID_UTF8", + "Invalid UTF-8, patch and commit message should be encoded in UTF-8\n" . $hereptr); + } + +# Check if it's the start of a commit log +# (not a header line and we haven't seen the patch filename) + if ($in_header_lines && $realfile =~ /^$/ && + !($rawline =~ /^\s+(?:\S|$)/ || + $rawline =~ /^(?:commit\b|from\b|[\w-]+:)/i)) { + $in_header_lines = 0; + $in_commit_log = 1; + $has_commit_log = 1; + } + +# Check if there is UTF-8 in a commit log when a mail header has explicitly +# declined it, i.e defined some charset where it is missing. + if ($in_header_lines && + $rawline =~ /^Content-Type:.+charset="(.+)".*$/ && + $1 !~ /utf-8/i) { + $non_utf8_charset = 1; + } + + if ($in_commit_log && $non_utf8_charset && $realfile =~ /^$/ && + $rawline =~ /$NON_ASCII_UTF8/) { + WARN("UTF8_BEFORE_PATCH", + "8-bit UTF-8 used in possible commit log\n" . $herecurr); + } + +# Check for absolute kernel paths in commit message + if ($tree && $in_commit_log) { + while ($line =~ m{(?:^|\s)(/\S*)}g) { + my $file = $1; + + if ($file =~ m{^(.*?)(?::\d+)+:?$} && + check_absolute_file($1, $herecurr)) { + # + } else { + check_absolute_file($file, $herecurr); + } + } + } + +# Check for various typo / spelling mistakes + if (defined($misspellings) && + ($in_commit_log || $line =~ /^(?:\+|Subject:)/i)) { + while ($rawline =~ /(?:^|[^a-z@])($misspellings)(?:\b|$|[^a-z@])/gi) { + my $typo = $1; + my $typo_fix = $spelling_fix{lc($typo)}; + $typo_fix = ucfirst($typo_fix) if ($typo =~ /^[A-Z]/); + $typo_fix = uc($typo_fix) if ($typo =~ /^[A-Z]+$/); + my $msg_level = \&WARN; + $msg_level = \&CHK if ($file); + if (&{$msg_level}("TYPO_SPELLING", + "'$typo' may be misspelled - perhaps '$typo_fix'?\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/(^|[^A-Za-z@])($typo)($|[^A-Za-z@])/$1$typo_fix$3/; + } + } + } + +# ignore non-hunk lines and lines being removed + next if (!$hunk_line || $line =~ /^-/); + +#trailing whitespace + if ($line =~ /^\+.*\015/) { + my $herevet = "$here\n" . cat_vet($rawline) . "\n"; + if (ERROR("DOS_LINE_ENDINGS", + "DOS line endings\n" . $herevet) && + $fix) { + $fixed[$fixlinenr] =~ s/[\s\015]+$//; + } + } elsif ($rawline =~ /^\+.*\S\s+$/ || $rawline =~ /^\+\s+$/) { + my $herevet = "$here\n" . cat_vet($rawline) . "\n"; + if (ERROR("TRAILING_WHITESPACE", + "trailing whitespace\n" . $herevet) && + $fix) { + $fixed[$fixlinenr] =~ s/\s+$//; + } + + $rpt_cleaners = 1; + } + +# Check for FSF mailing addresses. + if ($rawline =~ /\bwrite to the Free/i || + $rawline =~ /\b675\s+Mass\s+Ave/i || + $rawline =~ /\b59\s+Temple\s+Pl/i || + $rawline =~ /\b51\s+Franklin\s+St/i) { + my $herevet = "$here\n" . cat_vet($rawline) . "\n"; + my $msg_level = \&ERROR; + $msg_level = \&CHK if ($file); + &{$msg_level}("FSF_MAILING_ADDRESS", + "Do not include the paragraph about writing to the Free Software Foundation's mailing address from the sample GPL notice. The FSF has changed addresses in the past, and may do so again. Linux already includes a copy of the GPL.\n" . $herevet) + } + +# check for Kconfig help text having a real description +# Only applies when adding the entry originally, after that we do not have +# sufficient context to determine whether it is indeed long enough. + if ($realfile =~ /Kconfig/ && + $line =~ /^\+\s*config\s+/) { + my $length = 0; + my $cnt = $realcnt; + my $ln = $linenr + 1; + my $f; + my $is_start = 0; + my $is_end = 0; + for (; $cnt > 0 && defined $lines[$ln - 1]; $ln++) { + $f = $lines[$ln - 1]; + $cnt-- if ($lines[$ln - 1] !~ /^-/); + $is_end = $lines[$ln - 1] =~ /^\+/; + + next if ($f =~ /^-/); + last if (!$file && $f =~ /^\@\@/); + + if ($lines[$ln - 1] =~ /^\+\s*(?:bool|tristate)\s*\"/) { + $is_start = 1; + } elsif ($lines[$ln - 1] =~ /^\+\s*(?:---)?help(?:---)?$/) { + $length = -1; + } + + $f =~ s/^.//; + $f =~ s/#.*//; + $f =~ s/^\s+//; + next if ($f =~ /^$/); + if ($f =~ /^\s*config\s/) { + $is_end = 1; + last; + } + $length++; + } + if ($is_start && $is_end && $length < $min_conf_desc_length) { + WARN("CONFIG_DESCRIPTION", + "please write a paragraph that describes the config symbol fully\n" . $herecurr); + } + #print "is_start<$is_start> is_end<$is_end> length<$length>\n"; + } + +# check for MAINTAINERS entries that don't have the right form + if ($realfile =~ /^MAINTAINERS$/ && + $rawline =~ /^\+[A-Z]:/ && + $rawline !~ /^\+[A-Z]:\t\S/) { + if (WARN("MAINTAINERS_STYLE", + "MAINTAINERS entries use one tab after TYPE:\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/^(\+[A-Z]):\s*/$1:\t/; + } + } + +# discourage the use of boolean for type definition attributes of Kconfig options + if ($realfile =~ /Kconfig/ && + $line =~ /^\+\s*\bboolean\b/) { + WARN("CONFIG_TYPE_BOOLEAN", + "Use of boolean is deprecated, please use bool instead.\n" . $herecurr); + } + + if (($realfile =~ /Makefile.*/ || $realfile =~ /Kbuild.*/) && + ($line =~ /\+(EXTRA_[A-Z]+FLAGS).*/)) { + my $flag = $1; + my $replacement = { + 'EXTRA_AFLAGS' => 'asflags-y', + 'EXTRA_CFLAGS' => 'ccflags-y', + 'EXTRA_CPPFLAGS' => 'cppflags-y', + 'EXTRA_LDFLAGS' => 'ldflags-y', + }; + + WARN("DEPRECATED_VARIABLE", + "Use of $flag is deprecated, please use \`$replacement->{$flag} instead.\n" . $herecurr) if ($replacement->{$flag}); + } +# Kconfig use tabs and no spaces in line + if ($realfile =~ /Kconfig/ && $rawline =~ /^\+ /) { + my $herevet = "$here\n" . cat_vet($rawline) . "\n"; + WARN("LEADING_SPACE", + "please, no spaces at the start of a line\n" . $herevet); + } + +# check for DT compatible documentation + if (defined $root && + (($realfile =~ /\.dtsi?$/ && $line =~ /^\+\s*compatible\s*=\s*\"/) || + ($realfile =~ /\.[ch]$/ && $line =~ /^\+.*\.compatible\s*=\s*\"/))) { + + my @compats = $rawline =~ /\"([a-zA-Z0-9\-\,\.\+_]+)\"/g; + + my $dt_path = $root . "/dts/bindings/"; + my $vp_file = $dt_path . "vendor-prefixes.txt"; + + foreach my $compat (@compats) { + my $compat2 = $compat; + $compat2 =~ s/\,[a-zA-Z0-9]*\-/\,<\.\*>\-/; + my $compat3 = $compat; + $compat3 =~ s/\,([a-z]*)[0-9]*\-/\,$1<\.\*>\-/; + `grep -Erq "$compat|$compat2|$compat3" $dt_path`; + if ( $? >> 8 ) { + WARN("UNDOCUMENTED_DT_STRING", + "DT compatible string \"$compat\" appears un-documented -- check $dt_path\n" . $herecurr); + } + + next if $compat !~ /^([a-zA-Z0-9\-]+)\,/; + my $vendor = $1; + `grep -Eq "^$vendor\\b" $vp_file`; + if ( $? >> 8 ) { + WARN("UNDOCUMENTED_DT_STRING", + "DT compatible string vendor \"$vendor\" appears un-documented -- check $vp_file\n" . $herecurr); + } + } + } + +# check we are in a valid source file if not then ignore this hunk + next if ($realfile !~ /\.(h|c|s|S|sh|dtsi|dts)$/); + +# line length limit (with some exclusions) +# +# There are a few types of lines that may extend beyond $max_line_length: +# logging functions like pr_info that end in a string +# lines with a single string +# #defines that are a single string +# +# There are 3 different line length message types: +# LONG_LINE_COMMENT a comment starts before but extends beyond $max_line_length +# LONG_LINE_STRING a string starts before but extends beyond $max_line_length +# LONG_LINE all other lines longer than $max_line_length +# +# if LONG_LINE is ignored, the other 2 types are also ignored +# + + if ($line =~ /^\+/ && $length > $max_line_length) { + my $msg_type = "LONG_LINE"; + + # Check the allowed long line types first + + # logging functions that end in a string that starts + # before $max_line_length + if ($line =~ /^\+\s*$logFunctions\s*\(\s*(?:(?:KERN_\S+\s*|[^"]*))?($String\s*(?:|,|\)\s*;)\s*)$/ && + length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) { + $msg_type = ""; + + # lines with only strings (w/ possible termination) + # #defines with only strings + } elsif ($line =~ /^\+\s*$String\s*(?:\s*|,|\)\s*;)\s*$/ || + $line =~ /^\+\s*#\s*define\s+\w+\s+$String$/) { + $msg_type = ""; + + # EFI_GUID is another special case + } elsif ($line =~ /^\+.*\bEFI_GUID\s*\(/) { + $msg_type = ""; + + # Otherwise set the alternate message types + + # a comment starts before $max_line_length + } elsif ($line =~ /($;[\s$;]*)$/ && + length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) { + $msg_type = "LONG_LINE_COMMENT" + + # a quoted string starts before $max_line_length + } elsif ($sline =~ /\s*($String(?:\s*(?:\\|,\s*|\)\s*;\s*))?)$/ && + length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) { + $msg_type = "LONG_LINE_STRING" + } + + if ($msg_type ne "" && + (show_type("LONG_LINE") || show_type($msg_type))) { + WARN($msg_type, + "line over $max_line_length characters\n" . $herecurr); + } + } + +# check for adding lines without a newline. + if ($line =~ /^\+/ && defined $lines[$linenr] && $lines[$linenr] =~ /^\\ No newline at end of file/) { + WARN("MISSING_EOF_NEWLINE", + "adding a line without newline at end of file\n" . $herecurr); + } + +# Blackfin: use hi/lo macros + if ($realfile =~ m@arch/blackfin/.*\.S$@) { + if ($line =~ /\.[lL][[:space:]]*=.*&[[:space:]]*0x[fF][fF][fF][fF]/) { + my $herevet = "$here\n" . cat_vet($line) . "\n"; + ERROR("LO_MACRO", + "use the LO() macro, not (... & 0xFFFF)\n" . $herevet); + } + if ($line =~ /\.[hH][[:space:]]*=.*>>[[:space:]]*16/) { + my $herevet = "$here\n" . cat_vet($line) . "\n"; + ERROR("HI_MACRO", + "use the HI() macro, not (... >> 16)\n" . $herevet); + } + } + +# check we are in a valid source file C or perl if not then ignore this hunk + next if ($realfile !~ /\.(h|c|pl|dtsi|dts)$/); + +# at the beginning of a line any tabs must come first and anything +# more than 8 must use tabs. + if ($rawline =~ /^\+\s* \t\s*\S/ || + $rawline =~ /^\+\s* \s*/) { + my $herevet = "$here\n" . cat_vet($rawline) . "\n"; + $rpt_cleaners = 1; + if (ERROR("CODE_INDENT", + "code indent should use tabs where possible\n" . $herevet) && + $fix) { + $fixed[$fixlinenr] =~ s/^\+([ \t]+)/"\+" . tabify($1)/e; + } + } + +# check for space before tabs. + if ($rawline =~ /^\+/ && $rawline =~ / \t/) { + my $herevet = "$here\n" . cat_vet($rawline) . "\n"; + if (WARN("SPACE_BEFORE_TAB", + "please, no space before tabs\n" . $herevet) && + $fix) { + while ($fixed[$fixlinenr] =~ + s/(^\+.*) {8,8}\t/$1\t\t/) {} + while ($fixed[$fixlinenr] =~ + s/(^\+.*) +\t/$1\t/) {} + } + } + +# check for && or || at the start of a line + if ($rawline =~ /^\+\s*(&&|\|\|)/) { + CHK("LOGICAL_CONTINUATIONS", + "Logical continuations should be on the previous line\n" . $hereprev); + } + +# check indentation starts on a tab stop + if ($^V && $^V ge 5.10.0 && + $sline =~ /^\+\t+( +)(?:$c90_Keywords\b|\{\s*$|\}\s*(?:else\b|while\b|\s*$))/) { + my $indent = length($1); + if ($indent % 8) { + if (WARN("TABSTOP", + "Statements should start on a tabstop\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s@(^\+\t+) +@$1 . "\t" x ($indent/8)@e; + } + } + } + +# check multi-line statement indentation matches previous line + if ($^V && $^V ge 5.10.0 && + $prevline =~ /^\+([ \t]*)((?:$c90_Keywords(?:\s+if)\s*)|(?:$Declare\s*)?(?:$Ident|\(\s*\*\s*$Ident\s*\))\s*|(?:\*\s*)*$Lval\s*=\s*$Ident\s*)\(.*(\&\&|\|\||,)\s*$/) { + $prevline =~ /^\+(\t*)(.*)$/; + my $oldindent = $1; + my $rest = $2; + + my $pos = pos_last_openparen($rest); + if ($pos >= 0) { + $line =~ /^(\+| )([ \t]*)/; + my $newindent = $2; + + my $goodtabindent = $oldindent . + "\t" x ($pos / 8) . + " " x ($pos % 8); + my $goodspaceindent = $oldindent . " " x $pos; + + if ($newindent ne $goodtabindent && + $newindent ne $goodspaceindent) { + + if (CHK("PARENTHESIS_ALIGNMENT", + "Alignment should match open parenthesis\n" . $hereprev) && + $fix && $line =~ /^\+/) { + $fixed[$fixlinenr] =~ + s/^\+[ \t]*/\+$goodtabindent/; + } + } + } + } + +# check for space after cast like "(int) foo" or "(struct foo) bar" +# avoid checking a few false positives: +# "sizeof(<type>)" or "__alignof__(<type>)" +# function pointer declarations like "(*foo)(int) = bar;" +# structure definitions like "(struct foo) { 0 };" +# multiline macros that define functions +# known attributes or the __attribute__ keyword + if ($line =~ /^\+(.*)\(\s*$Type\s*\)([ \t]++)((?![={]|\\$|$Attribute|__attribute__))/ && + (!defined($1) || $1 !~ /\b(?:sizeof|__alignof__)\s*$/)) { + if (CHK("SPACING", + "No space is necessary after a cast\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/(\(\s*$Type\s*\))[ \t]+/$1/; + } + } + +# Block comment styles +# Networking with an initial /* + if ($realfile =~ m@^(drivers/net/|net/)@ && + $prevrawline =~ /^\+[ \t]*\/\*[ \t]*$/ && + $rawline =~ /^\+[ \t]*\*/ && + $realline > 2) { + WARN("NETWORKING_BLOCK_COMMENT_STYLE", + "networking block comments don't use an empty /* line, use /* Comment...\n" . $hereprev); + } + +# Block comments use * on subsequent lines + if ($prevline =~ /$;[ \t]*$/ && #ends in comment + $prevrawline =~ /^\+.*?\/\*/ && #starting /* + $prevrawline !~ /\*\/[ \t]*$/ && #no trailing */ + $rawline =~ /^\+/ && #line is new + $rawline !~ /^\+[ \t]*\*/) { #no leading * + WARN("BLOCK_COMMENT_STYLE", + "Block comments use * on subsequent lines\n" . $hereprev); + } + +# Block comments use */ on trailing lines + if ($rawline !~ m@^\+[ \t]*\*/[ \t]*$@ && #trailing */ + $rawline !~ m@^\+.*/\*.*\*/[ \t]*$@ && #inline /*...*/ + $rawline !~ m@^\+.*\*{2,}/[ \t]*$@ && #trailing **/ + $rawline =~ m@^\+[ \t]*.+\*\/[ \t]*$@) { #non blank */ + WARN("BLOCK_COMMENT_STYLE", + "Block comments use a trailing */ on a separate line\n" . $herecurr); + } + +# Block comment * alignment + if ($prevline =~ /$;[ \t]*$/ && #ends in comment + $line =~ /^\+[ \t]*$;/ && #leading comment + $rawline =~ /^\+[ \t]*\*/ && #leading * + (($prevrawline =~ /^\+.*?\/\*/ && #leading /* + $prevrawline !~ /\*\/[ \t]*$/) || #no trailing */ + $prevrawline =~ /^\+[ \t]*\*/)) { #leading * + my $oldindent; + $prevrawline =~ m@^\+([ \t]*/?)\*@; + if (defined($1)) { + $oldindent = expand_tabs($1); + } else { + $prevrawline =~ m@^\+(.*/?)\*@; + $oldindent = expand_tabs($1); + } + $rawline =~ m@^\+([ \t]*)\*@; + my $newindent = $1; + $newindent = expand_tabs($newindent); + if (length($oldindent) ne length($newindent)) { + WARN("BLOCK_COMMENT_STYLE", + "Block comments should align the * on each line\n" . $hereprev); + } + } + +# check for missing blank lines after struct/union declarations +# with exceptions for various attributes and macros + if ($prevline =~ /^[\+ ]};?\s*$/ && + $line =~ /^\+/ && + !($line =~ /^\+\s*$/ || + $line =~ /^\+\s*EXPORT_SYMBOL/ || + $line =~ /^\+\s*MODULE_/i || + $line =~ /^\+\s*\#\s*(?:end|elif|else)/ || + $line =~ /^\+[a-z_]*init/ || + $line =~ /^\+\s*(?:static\s+)?[A-Z_]*ATTR/ || + $line =~ /^\+\s*DECLARE/ || + $line =~ /^\+\s*__setup/)) { + if (CHK("LINE_SPACING", + "Please use a blank line after function/struct/union/enum declarations\n" . $hereprev) && + $fix) { + fix_insert_line($fixlinenr, "\+"); + } + } + +# check for multiple consecutive blank lines + if ($prevline =~ /^[\+ ]\s*$/ && + $line =~ /^\+\s*$/ && + $last_blank_line != ($linenr - 1)) { + if (CHK("LINE_SPACING", + "Please don't use multiple blank lines\n" . $hereprev) && + $fix) { + fix_delete_line($fixlinenr, $rawline); + } + + $last_blank_line = $linenr; + } + +# check for missing blank lines after declarations + if ($sline =~ /^\+\s+\S/ && #Not at char 1 + # actual declarations + ($prevline =~ /^\+\s+$Declare\s*$Ident\s*[=,;:\[]/ || + # function pointer declarations + $prevline =~ /^\+\s+$Declare\s*\(\s*\*\s*$Ident\s*\)\s*[=,;:\[\(]/ || + # foo bar; where foo is some local typedef or #define + $prevline =~ /^\+\s+$Ident(?:\s+|\s*\*\s*)$Ident\s*[=,;\[]/ || + # known declaration macros + $prevline =~ /^\+\s+$declaration_macros/) && + # for "else if" which can look like "$Ident $Ident" + !($prevline =~ /^\+\s+$c90_Keywords\b/ || + # other possible extensions of declaration lines + $prevline =~ /(?:$Compare|$Assignment|$Operators)\s*$/ || + # not starting a section or a macro "\" extended line + $prevline =~ /(?:\{\s*|\\)$/) && + # looks like a declaration + !($sline =~ /^\+\s+$Declare\s*$Ident\s*[=,;:\[]/ || + # function pointer declarations + $sline =~ /^\+\s+$Declare\s*\(\s*\*\s*$Ident\s*\)\s*[=,;:\[\(]/ || + # foo bar; where foo is some local typedef or #define + $sline =~ /^\+\s+(?:volatile\s+)?$Ident(?:\s+|\s*\*\s*)$Ident\s*[=,;\[]/ || + # known declaration macros + $sline =~ /^\+\s+$declaration_macros/ || + # start of struct or union or enum + $sline =~ /^\+\s+(?:union|struct|enum|typedef)\b/ || + # start or end of block or continuation of declaration + $sline =~ /^\+\s+(?:$|[\{\}\.\#\"\?\:\(\[])/ || + # bitfield continuation + $sline =~ /^\+\s+$Ident\s*:\s*\d+\s*[,;]/ || + # other possible extensions of declaration lines + $sline =~ /^\+\s+\(?\s*(?:$Compare|$Assignment|$Operators)/) && + # indentation of previous and current line are the same + (($prevline =~ /\+(\s+)\S/) && $sline =~ /^\+$1\S/)) { + if (WARN("LINE_SPACING", + "Missing a blank line after declarations\n" . $hereprev) && + $fix) { + fix_insert_line($fixlinenr, "\+"); + } + } + +# check for spaces at the beginning of a line. +# Exceptions: +# 1) within comments +# 2) indented preprocessor commands +# 3) hanging labels + if ($rawline =~ /^\+ / && $line !~ /^\+ *(?:$;|#|$Ident:)/) { + my $herevet = "$here\n" . cat_vet($rawline) . "\n"; + if (WARN("LEADING_SPACE", + "please, no spaces at the start of a line\n" . $herevet) && + $fix) { + $fixed[$fixlinenr] =~ s/^\+([ \t]+)/"\+" . tabify($1)/e; + } + } + +# check we are in a valid C source file if not then ignore this hunk + next if ($realfile !~ /\.(h|c)$/); + +# check if this appears to be the start function declaration, save the name + if ($sline =~ /^\+\{\s*$/ && + $prevline =~ /^\+(?:(?:(?:$Storage|$Inline)\s*)*\s*$Type\s*)?($Ident)\(/) { + $context_function = $1; + } + +# check if this appears to be the end of function declaration + if ($sline =~ /^\+\}\s*$/) { + undef $context_function; + } + +# check indentation of any line with a bare else +# (but not if it is a multiple line "if (foo) return bar; else return baz;") +# if the previous line is a break or return and is indented 1 tab more... + if ($sline =~ /^\+([\t]+)(?:}[ \t]*)?else(?:[ \t]*{)?\s*$/) { + my $tabs = length($1) + 1; + if ($prevline =~ /^\+\t{$tabs,$tabs}break\b/ || + ($prevline =~ /^\+\t{$tabs,$tabs}return\b/ && + defined $lines[$linenr] && + $lines[$linenr] !~ /^[ \+]\t{$tabs,$tabs}return/)) { + WARN("UNNECESSARY_ELSE", + "else is not generally useful after a break or return\n" . $hereprev); + } + } + +# check indentation of a line with a break; +# if the previous line is a goto or return and is indented the same # of tabs + if ($sline =~ /^\+([\t]+)break\s*;\s*$/) { + my $tabs = $1; + if ($prevline =~ /^\+$tabs(?:goto|return)\b/) { + WARN("UNNECESSARY_BREAK", + "break is not useful after a goto or return\n" . $hereprev); + } + } + +# check for RCS/CVS revision markers + if ($rawline =~ /^\+.*\$(Revision|Log|Id)(?:\$|)/) { + WARN("CVS_KEYWORD", + "CVS style keyword markers, these will _not_ be updated\n". $herecurr); + } + +# Blackfin: don't use __builtin_bfin_[cs]sync + if ($line =~ /__builtin_bfin_csync/) { + my $herevet = "$here\n" . cat_vet($line) . "\n"; + ERROR("CSYNC", + "use the CSYNC() macro in asm/blackfin.h\n" . $herevet); + } + if ($line =~ /__builtin_bfin_ssync/) { + my $herevet = "$here\n" . cat_vet($line) . "\n"; + ERROR("SSYNC", + "use the SSYNC() macro in asm/blackfin.h\n" . $herevet); + } + +# check for old HOTPLUG __dev<foo> section markings + if ($line =~ /\b(__dev(init|exit)(data|const|))\b/) { + WARN("HOTPLUG_SECTION", + "Using $1 is unnecessary\n" . $herecurr); + } + +# Check for potential 'bare' types + my ($stat, $cond, $line_nr_next, $remain_next, $off_next, + $realline_next); +#print "LINE<$line>\n"; + if ($linenr > $suppress_statement && + $realcnt && $sline =~ /.\s*\S/) { + ($stat, $cond, $line_nr_next, $remain_next, $off_next) = + ctx_statement_block($linenr, $realcnt, 0); + $stat =~ s/\n./\n /g; + $cond =~ s/\n./\n /g; + +#print "linenr<$linenr> <$stat>\n"; + # If this statement has no statement boundaries within + # it there is no point in retrying a statement scan + # until we hit end of it. + my $frag = $stat; $frag =~ s/;+\s*$//; + if ($frag !~ /(?:{|;)/) { +#print "skip<$line_nr_next>\n"; + $suppress_statement = $line_nr_next; + } + + # Find the real next line. + $realline_next = $line_nr_next; + if (defined $realline_next && + (!defined $lines[$realline_next - 1] || + substr($lines[$realline_next - 1], $off_next) =~ /^\s*$/)) { + $realline_next++; + } + + my $s = $stat; + $s =~ s/{.*$//s; + + # Ignore goto labels. + if ($s =~ /$Ident:\*$/s) { + + # Ignore functions being called + } elsif ($s =~ /^.\s*$Ident\s*\(/s) { + + } elsif ($s =~ /^.\s*else\b/s) { + + # declarations always start with types + } elsif ($prev_values eq 'E' && $s =~ /^.\s*(?:$Storage\s+)?(?:$Inline\s+)?(?:const\s+)?((?:\s*$Ident)+?)\b(?:\s+$Sparse)?\s*\**\s*(?:$Ident|\(\*[^\)]*\))(?:\s*$Modifier)?\s*(?:;|=|,|\()/s) { + my $type = $1; + $type =~ s/\s+/ /g; + possible($type, "A:" . $s); + + # definitions in global scope can only start with types + } elsif ($s =~ /^.(?:$Storage\s+)?(?:$Inline\s+)?(?:const\s+)?($Ident)\b\s*(?!:)/s) { + possible($1, "B:" . $s); + } + + # any (foo ... *) is a pointer cast, and foo is a type + while ($s =~ /\(($Ident)(?:\s+$Sparse)*[\s\*]+\s*\)/sg) { + possible($1, "C:" . $s); + } + + # Check for any sort of function declaration. + # int foo(something bar, other baz); + # void (*store_gdt)(x86_descr_ptr *); + if ($prev_values eq 'E' && $s =~ /^(.(?:typedef\s*)?(?:(?:$Storage|$Inline)\s*)*\s*$Type\s*(?:\b$Ident|\(\*\s*$Ident\))\s*)\(/s) { + my ($name_len) = length($1); + + my $ctx = $s; + substr($ctx, 0, $name_len + 1, ''); + $ctx =~ s/\)[^\)]*$//; + + for my $arg (split(/\s*,\s*/, $ctx)) { + if ($arg =~ /^(?:const\s+)?($Ident)(?:\s+$Sparse)*\s*\**\s*(:?\b$Ident)?$/s || $arg =~ /^($Ident)$/s) { + + possible($1, "D:" . $s); + } + } + } + + } + +# +# Checks which may be anchored in the context. +# + +# Check for switch () and associated case and default +# statements should be at the same indent. + if ($line=~/\bswitch\s*\(.*\)/) { + my $err = ''; + my $sep = ''; + my @ctx = ctx_block_outer($linenr, $realcnt); + shift(@ctx); + for my $ctx (@ctx) { + my ($clen, $cindent) = line_stats($ctx); + if ($ctx =~ /^\+\s*(case\s+|default:)/ && + $indent != $cindent) { + $err .= "$sep$ctx\n"; + $sep = ''; + } else { + $sep = "[...]\n"; + } + } + if ($err ne '') { + ERROR("SWITCH_CASE_INDENT_LEVEL", + "switch and case should be at the same indent\n$hereline$err"); + } + } + +# if/while/etc brace do not go on next line, unless defining a do while loop, +# or if that brace on the next line is for something else + if ($line =~ /(.*)\b((?:if|while|for|switch|(?:[a-z_]+|)for_each[a-z_]+)\s*\(|do\b|else\b)/ && $line !~ /^.\s*\#/) { + my $pre_ctx = "$1$2"; + + my ($level, @ctx) = ctx_statement_level($linenr, $realcnt, 0); + + if ($line =~ /^\+\t{6,}/) { + WARN("DEEP_INDENTATION", + "Too many leading tabs - consider code refactoring\n" . $herecurr); + } + + my $ctx_cnt = $realcnt - $#ctx - 1; + my $ctx = join("\n", @ctx); + + my $ctx_ln = $linenr; + my $ctx_skip = $realcnt; + + while ($ctx_skip > $ctx_cnt || ($ctx_skip == $ctx_cnt && + defined $lines[$ctx_ln - 1] && + $lines[$ctx_ln - 1] =~ /^-/)) { + ##print "SKIP<$ctx_skip> CNT<$ctx_cnt>\n"; + $ctx_skip-- if (!defined $lines[$ctx_ln - 1] || $lines[$ctx_ln - 1] !~ /^-/); + $ctx_ln++; + } + + #print "realcnt<$realcnt> ctx_cnt<$ctx_cnt>\n"; + #print "pre<$pre_ctx>\nline<$line>\nctx<$ctx>\nnext<$lines[$ctx_ln - 1]>\n"; + + if ($ctx !~ /{\s*/ && defined($lines[$ctx_ln - 1]) && $lines[$ctx_ln - 1] =~ /^\+\s*{/) { + ERROR("OPEN_BRACE", + "that open brace { should be on the previous line\n" . + "$here\n$ctx\n$rawlines[$ctx_ln - 1]\n"); + } + if ($level == 0 && $pre_ctx !~ /}\s*while\s*\($/ && + $ctx =~ /\)\s*\;\s*$/ && + defined $lines[$ctx_ln - 1]) + { + my ($nlength, $nindent) = line_stats($lines[$ctx_ln - 1]); + if ($nindent > $indent) { + WARN("TRAILING_SEMICOLON", + "trailing semicolon indicates no statements, indent implies otherwise\n" . + "$here\n$ctx\n$rawlines[$ctx_ln - 1]\n"); + } + } + } + +# Check relative indent for conditionals and blocks. + if ($line =~ /\b(?:(?:if|while|for|(?:[a-z_]+|)for_each[a-z_]+)\s*\(|(?:do|else)\b)/ && $line !~ /^.\s*#/ && $line !~ /\}\s*while\s*/) { + ($stat, $cond, $line_nr_next, $remain_next, $off_next) = + ctx_statement_block($linenr, $realcnt, 0) + if (!defined $stat); + my ($s, $c) = ($stat, $cond); + + substr($s, 0, length($c), ''); + + # remove inline comments + $s =~ s/$;/ /g; + $c =~ s/$;/ /g; + + # Find out how long the conditional actually is. + my @newlines = ($c =~ /\n/gs); + my $cond_lines = 1 + $#newlines; + + # Make sure we remove the line prefixes as we have + # none on the first line, and are going to readd them + # where necessary. + $s =~ s/\n./\n/gs; + while ($s =~ /\n\s+\\\n/) { + $cond_lines += $s =~ s/\n\s+\\\n/\n/g; + } + + # We want to check the first line inside the block + # starting at the end of the conditional, so remove: + # 1) any blank line termination + # 2) any opening brace { on end of the line + # 3) any do (...) { + my $continuation = 0; + my $check = 0; + $s =~ s/^.*\bdo\b//; + $s =~ s/^\s*{//; + if ($s =~ s/^\s*\\//) { + $continuation = 1; + } + if ($s =~ s/^\s*?\n//) { + $check = 1; + $cond_lines++; + } + + # Also ignore a loop construct at the end of a + # preprocessor statement. + if (($prevline =~ /^.\s*#\s*define\s/ || + $prevline =~ /\\\s*$/) && $continuation == 0) { + $check = 0; + } + + my $cond_ptr = -1; + $continuation = 0; + while ($cond_ptr != $cond_lines) { + $cond_ptr = $cond_lines; + + # If we see an #else/#elif then the code + # is not linear. + if ($s =~ /^\s*\#\s*(?:else|elif)/) { + $check = 0; + } + + # Ignore: + # 1) blank lines, they should be at 0, + # 2) preprocessor lines, and + # 3) labels. + if ($continuation || + $s =~ /^\s*?\n/ || + $s =~ /^\s*#\s*?/ || + $s =~ /^\s*$Ident\s*:/) { + $continuation = ($s =~ /^.*?\\\n/) ? 1 : 0; + if ($s =~ s/^.*?\n//) { + $cond_lines++; + } + } + } + + my (undef, $sindent) = line_stats("+" . $s); + my $stat_real = raw_line($linenr, $cond_lines); + + # Check if either of these lines are modified, else + # this is not this patch's fault. + if (!defined($stat_real) || + $stat !~ /^\+/ && $stat_real !~ /^\+/) { + $check = 0; + } + if (defined($stat_real) && $cond_lines > 1) { + $stat_real = "[...]\n$stat_real"; + } + + #print "line<$line> prevline<$prevline> indent<$indent> sindent<$sindent> check<$check> continuation<$continuation> s<$s> cond_lines<$cond_lines> stat_real<$stat_real> stat<$stat>\n"; + + if ($check && $s ne '' && + (($sindent % 8) != 0 || + ($sindent < $indent) || + ($sindent == $indent && + ($s !~ /^\s*(?:\}|\{|else\b)/)) || + ($sindent > $indent + 8))) { + WARN("SUSPECT_CODE_INDENT", + "suspect code indent for conditional statements ($indent, $sindent)\n" . $herecurr . "$stat_real\n"); + } + } + + # Track the 'values' across context and added lines. + my $opline = $line; $opline =~ s/^./ /; + my ($curr_values, $curr_vars) = + annotate_values($opline . "\n", $prev_values); + $curr_values = $prev_values . $curr_values; + if ($dbg_values) { + my $outline = $opline; $outline =~ s/\t/ /g; + print "$linenr > .$outline\n"; + print "$linenr > $curr_values\n"; + print "$linenr > $curr_vars\n"; + } + $prev_values = substr($curr_values, -1); + +#ignore lines not being added + next if ($line =~ /^[^\+]/); + +# check for dereferences that span multiple lines + if ($prevline =~ /^\+.*$Lval\s*(?:\.|->)\s*$/ && + $line =~ /^\+\s*(?!\#\s*(?!define\s+|if))\s*$Lval/) { + $prevline =~ /($Lval\s*(?:\.|->))\s*$/; + my $ref = $1; + $line =~ /^.\s*($Lval)/; + $ref .= $1; + $ref =~ s/\s//g; + WARN("MULTILINE_DEREFERENCE", + "Avoid multiple line dereference - prefer '$ref'\n" . $hereprev); + } + +# check for declarations of signed or unsigned without int + while ($line =~ m{\b($Declare)\s*(?!char\b|short\b|int\b|long\b)\s*($Ident)?\s*[=,;\[\)\(]}g) { + my $type = $1; + my $var = $2; + $var = "" if (!defined $var); + if ($type =~ /^(?:(?:$Storage|$Inline|$Attribute)\s+)*((?:un)?signed)((?:\s*\*)*)\s*$/) { + my $sign = $1; + my $pointer = $2; + + $pointer = "" if (!defined $pointer); + + if (WARN("UNSPECIFIED_INT", + "Prefer '" . trim($sign) . " int" . rtrim($pointer) . "' to bare use of '$sign" . rtrim($pointer) . "'\n" . $herecurr) && + $fix) { + my $decl = trim($sign) . " int "; + my $comp_pointer = $pointer; + $comp_pointer =~ s/\s//g; + $decl .= $comp_pointer; + $decl = rtrim($decl) if ($var eq ""); + $fixed[$fixlinenr] =~ s@\b$sign\s*\Q$pointer\E\s*$var\b@$decl$var@; + } + } + } + +# TEST: allow direct testing of the type matcher. + if ($dbg_type) { + if ($line =~ /^.\s*$Declare\s*$/) { + ERROR("TEST_TYPE", + "TEST: is type\n" . $herecurr); + } elsif ($dbg_type > 1 && $line =~ /^.+($Declare)/) { + ERROR("TEST_NOT_TYPE", + "TEST: is not type ($1 is)\n". $herecurr); + } + next; + } +# TEST: allow direct testing of the attribute matcher. + if ($dbg_attr) { + if ($line =~ /^.\s*$Modifier\s*$/) { + ERROR("TEST_ATTR", + "TEST: is attr\n" . $herecurr); + } elsif ($dbg_attr > 1 && $line =~ /^.+($Modifier)/) { + ERROR("TEST_NOT_ATTR", + "TEST: is not attr ($1 is)\n". $herecurr); + } + next; + } + +# check for initialisation to aggregates open brace on the next line + if ($line =~ /^.\s*{/ && + $prevline =~ /(?:^|[^=])=\s*$/) { + if (ERROR("OPEN_BRACE", + "that open brace { should be on the previous line\n" . $hereprev) && + $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { + fix_delete_line($fixlinenr - 1, $prevrawline); + fix_delete_line($fixlinenr, $rawline); + my $fixedline = $prevrawline; + $fixedline =~ s/\s*=\s*$/ = {/; + fix_insert_line($fixlinenr, $fixedline); + $fixedline = $line; + $fixedline =~ s/^(.\s*)\{\s*/$1/; + fix_insert_line($fixlinenr, $fixedline); + } + } + +# +# Checks which are anchored on the added line. +# + +# check for malformed paths in #include statements (uses RAW line) + if ($rawline =~ m{^.\s*\#\s*include\s+[<"](.*)[">]}) { + my $path = $1; + if ($path =~ m{//}) { + ERROR("MALFORMED_INCLUDE", + "malformed #include filename\n" . $herecurr); + } + if ($path =~ "^uapi/" && $realfile =~ m@\binclude/uapi/@) { + ERROR("UAPI_INCLUDE", + "No #include in ...include/uapi/... should use a uapi/ path prefix\n" . $herecurr); + } + } + +# no C99 // comments + if ($line =~ m{//}) { + if (ERROR("C99_COMMENTS", + "do not use C99 // comments\n" . $herecurr) && + $fix) { + my $line = $fixed[$fixlinenr]; + if ($line =~ /\/\/(.*)$/) { + my $comment = trim($1); + $fixed[$fixlinenr] =~ s@\/\/(.*)$@/\* $comment \*/@; + } + } + } + # Remove C99 comments. + $line =~ s@//.*@@; + $opline =~ s@//.*@@; + +# EXPORT_SYMBOL should immediately follow the thing it is exporting, consider +# the whole statement. +#print "APW <$lines[$realline_next - 1]>\n"; + if (defined $realline_next && + exists $lines[$realline_next - 1] && + !defined $suppress_export{$realline_next} && + ($lines[$realline_next - 1] =~ /EXPORT_SYMBOL.*\((.*)\)/ || + $lines[$realline_next - 1] =~ /EXPORT_UNUSED_SYMBOL.*\((.*)\)/)) { + # Handle definitions which produce identifiers with + # a prefix: + # XXX(foo); + # EXPORT_SYMBOL(something_foo); + my $name = $1; + if ($stat =~ /^(?:.\s*}\s*\n)?.([A-Z_]+)\s*\(\s*($Ident)/ && + $name =~ /^${Ident}_$2/) { +#print "FOO C name<$name>\n"; + $suppress_export{$realline_next} = 1; + + } elsif ($stat !~ /(?: + \n.}\s*$| + ^.DEFINE_$Ident\(\Q$name\E\)| + ^.DECLARE_$Ident\(\Q$name\E\)| + ^.LIST_HEAD\(\Q$name\E\)| + ^.(?:$Storage\s+)?$Type\s*\(\s*\*\s*\Q$name\E\s*\)\s*\(| + \b\Q$name\E(?:\s+$Attribute)*\s*(?:;|=|\[|\() + )/x) { +#print "FOO A<$lines[$realline_next - 1]> stat<$stat> name<$name>\n"; + $suppress_export{$realline_next} = 2; + } else { + $suppress_export{$realline_next} = 1; + } + } + if (!defined $suppress_export{$linenr} && + $prevline =~ /^.\s*$/ && + ($line =~ /EXPORT_SYMBOL.*\((.*)\)/ || + $line =~ /EXPORT_UNUSED_SYMBOL.*\((.*)\)/)) { +#print "FOO B <$lines[$linenr - 1]>\n"; + $suppress_export{$linenr} = 2; + } + if (defined $suppress_export{$linenr} && + $suppress_export{$linenr} == 2) { + WARN("EXPORT_SYMBOL", + "EXPORT_SYMBOL(foo); should immediately follow its function/variable\n" . $herecurr); + } + +# check for global initialisers. + if ($line =~ /^\+$Type\s*$Ident(?:\s+$Modifier)*\s*=\s*($zero_initializer)\s*;/) { + if (ERROR("GLOBAL_INITIALISERS", + "do not initialise globals to $1\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/(^.$Type\s*$Ident(?:\s+$Modifier)*)\s*=\s*$zero_initializer\s*;/$1;/; + } + } +# check for static initialisers. + if ($line =~ /^\+.*\bstatic\s.*=\s*($zero_initializer)\s*;/) { + if (ERROR("INITIALISED_STATIC", + "do not initialise statics to $1\n" . + $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/(\bstatic\s.*?)\s*=\s*$zero_initializer\s*;/$1;/; + } + } + +# check for misordered declarations of char/short/int/long with signed/unsigned + while ($sline =~ m{(\b$TypeMisordered\b)}g) { + my $tmp = trim($1); + WARN("MISORDERED_TYPE", + "type '$tmp' should be specified in [[un]signed] [short|int|long|long long] order\n" . $herecurr); + } + +# check for static const char * arrays. + if ($line =~ /\bstatic\s+const\s+char\s*\*\s*(\w+)\s*\[\s*\]\s*=\s*/) { + WARN("STATIC_CONST_CHAR_ARRAY", + "static const char * array should probably be static const char * const\n" . + $herecurr); + } + +# check for static char foo[] = "bar" declarations. + if ($line =~ /\bstatic\s+char\s+(\w+)\s*\[\s*\]\s*=\s*"/) { + WARN("STATIC_CONST_CHAR_ARRAY", + "static char array declaration should probably be static const char\n" . + $herecurr); + } + +# check for const <foo> const where <foo> is not a pointer or array type + if ($sline =~ /\bconst\s+($BasicType)\s+const\b/) { + my $found = $1; + if ($sline =~ /\bconst\s+\Q$found\E\s+const\b\s*\*/) { + WARN("CONST_CONST", + "'const $found const *' should probably be 'const $found * const'\n" . $herecurr); + } elsif ($sline !~ /\bconst\s+\Q$found\E\s+const\s+\w+\s*\[/) { + WARN("CONST_CONST", + "'const $found const' should probably be 'const $found'\n" . $herecurr); + } + } + +# check for non-global char *foo[] = {"bar", ...} declarations. + if ($line =~ /^.\s+(?:static\s+|const\s+)?char\s+\*\s*\w+\s*\[\s*\]\s*=\s*\{/) { + WARN("STATIC_CONST_CHAR_ARRAY", + "char * array declaration might be better as static const\n" . + $herecurr); + } + +# check for sizeof(foo)/sizeof(foo[0]) that could be ARRAY_SIZE(foo) + if ($line =~ m@\bsizeof\s*\(\s*($Lval)\s*\)@) { + my $array = $1; + if ($line =~ m@\b(sizeof\s*\(\s*\Q$array\E\s*\)\s*/\s*sizeof\s*\(\s*\Q$array\E\s*\[\s*0\s*\]\s*\))@) { + my $array_div = $1; + if (WARN("ARRAY_SIZE", + "Prefer ARRAY_SIZE($array)\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\Q$array_div\E/ARRAY_SIZE($array)/; + } + } + } + +# check for function declarations without arguments like "int foo()" + if ($line =~ /(\b$Type\s+$Ident)\s*\(\s*\)/) { + if (ERROR("FUNCTION_WITHOUT_ARGS", + "Bad function definition - $1() should probably be $1(void)\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/(\b($Type)\s+($Ident))\s*\(\s*\)/$2 $3(void)/; + } + } + +# check for new typedefs, only function parameters and sparse annotations +# make sense. + if ($line =~ /\btypedef\s/ && + $line !~ /\btypedef\s+$Type\s*\(\s*\*?$Ident\s*\)\s*\(/ && + $line !~ /\btypedef\s+$Type\s+$Ident\s*\(/ && + $line !~ /\b$typeTypedefs\b/ && + $line !~ /\b__bitwise\b/) { + WARN("NEW_TYPEDEFS", + "do not add new typedefs\n" . $herecurr); + } + +# * goes on variable not on type + # (char*[ const]) + while ($line =~ m{(\($NonptrType(\s*(?:$Modifier\b\s*|\*\s*)+)\))}g) { + #print "AA<$1>\n"; + my ($ident, $from, $to) = ($1, $2, $2); + + # Should start with a space. + $to =~ s/^(\S)/ $1/; + # Should not end with a space. + $to =~ s/\s+$//; + # '*'s should not have spaces between. + while ($to =~ s/\*\s+\*/\*\*/) { + } + +## print "1: from<$from> to<$to> ident<$ident>\n"; + if ($from ne $to) { + if (ERROR("POINTER_LOCATION", + "\"(foo$from)\" should be \"(foo$to)\"\n" . $herecurr) && + $fix) { + my $sub_from = $ident; + my $sub_to = $ident; + $sub_to =~ s/\Q$from\E/$to/; + $fixed[$fixlinenr] =~ + s@\Q$sub_from\E@$sub_to@; + } + } + } + while ($line =~ m{(\b$NonptrType(\s*(?:$Modifier\b\s*|\*\s*)+)($Ident))}g) { + #print "BB<$1>\n"; + my ($match, $from, $to, $ident) = ($1, $2, $2, $3); + + # Should start with a space. + $to =~ s/^(\S)/ $1/; + # Should not end with a space. + $to =~ s/\s+$//; + # '*'s should not have spaces between. + while ($to =~ s/\*\s+\*/\*\*/) { + } + # Modifiers should have spaces. + $to =~ s/(\b$Modifier$)/$1 /; + +## print "2: from<$from> to<$to> ident<$ident>\n"; + if ($from ne $to && $ident !~ /^$Modifier$/) { + if (ERROR("POINTER_LOCATION", + "\"foo${from}bar\" should be \"foo${to}bar\"\n" . $herecurr) && + $fix) { + + my $sub_from = $match; + my $sub_to = $match; + $sub_to =~ s/\Q$from\E/$to/; + $fixed[$fixlinenr] =~ + s@\Q$sub_from\E@$sub_to@; + } + } + } + +# avoid BUG() or BUG_ON() + if ($line =~ /\b(?:BUG|BUG_ON)\b/) { + my $msg_level = \&WARN; + $msg_level = \&CHK if ($file); + &{$msg_level}("AVOID_BUG", + "Avoid crashing the kernel - try using WARN_ON & recovery code rather than BUG() or BUG_ON()\n" . $herecurr); + } + +# avoid LINUX_VERSION_CODE + if ($line =~ /\bLINUX_VERSION_CODE\b/) { + WARN("LINUX_VERSION_CODE", + "LINUX_VERSION_CODE should be avoided, code should be for the version to which it is merged\n" . $herecurr); + } + +# check for uses of printk_ratelimit + if ($line =~ /\bprintk_ratelimit\s*\(/) { + WARN("PRINTK_RATELIMITED", + "Prefer printk_ratelimited or pr_<level>_ratelimited to printk_ratelimit\n" . $herecurr); + } + +# printk should use KERN_* levels. Note that follow on printk's on the +# same line do not need a level, so we use the current block context +# to try and find and validate the current printk. In summary the current +# printk includes all preceding printk's which have no newline on the end. +# we assume the first bad printk is the one to report. + if ($line =~ /\bprintk\((?!KERN_)\s*"/) { + my $ok = 0; + for (my $ln = $linenr - 1; $ln >= $first_line; $ln--) { + #print "CHECK<$lines[$ln - 1]\n"; + # we have a preceding printk if it ends + # with "\n" ignore it, else it is to blame + if ($lines[$ln - 1] =~ m{\bprintk\(}) { + if ($rawlines[$ln - 1] !~ m{\\n"}) { + $ok = 1; + } + last; + } + } + if ($ok == 0) { + WARN("PRINTK_WITHOUT_KERN_LEVEL", + "printk() should include KERN_ facility level\n" . $herecurr); + } + } + + if ($line =~ /\bprintk\s*\(\s*KERN_([A-Z]+)/) { + my $orig = $1; + my $level = lc($orig); + $level = "warn" if ($level eq "warning"); + my $level2 = $level; + $level2 = "dbg" if ($level eq "debug"); + WARN("PREFER_PR_LEVEL", + "Prefer [subsystem eg: netdev]_$level2([subsystem]dev, ... then dev_$level2(dev, ... then pr_$level(... to printk(KERN_$orig ...\n" . $herecurr); + } + + if ($line =~ /\bpr_warning\s*\(/) { + if (WARN("PREFER_PR_LEVEL", + "Prefer pr_warn(... to pr_warning(...\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/\bpr_warning\b/pr_warn/; + } + } + + if ($line =~ /\bdev_printk\s*\(\s*KERN_([A-Z]+)/) { + my $orig = $1; + my $level = lc($orig); + $level = "warn" if ($level eq "warning"); + $level = "dbg" if ($level eq "debug"); + WARN("PREFER_DEV_LEVEL", + "Prefer dev_$level(... to dev_printk(KERN_$orig, ...\n" . $herecurr); + } + +# ENOSYS means "bad syscall nr" and nothing else. This will have a small +# number of false positives, but assembly files are not checked, so at +# least the arch entry code will not trigger this warning. + if ($line =~ /\bENOSYS\b/) { + WARN("ENOSYS", + "ENOSYS means 'invalid syscall nr' and nothing else\n" . $herecurr); + } + +# function brace can't be on same line, except for #defines of do while, +# or if closed on same line + if (($line=~/$Type\s*$Ident\(.*\).*\s*{/) and + !($line=~/\#\s*define.*do\s\{/) and !($line=~/}/)) { + if (ERROR("OPEN_BRACE", + "open brace '{' following function declarations go on the next line\n" . $herecurr) && + $fix) { + fix_delete_line($fixlinenr, $rawline); + my $fixed_line = $rawline; + $fixed_line =~ /(^..*$Type\s*$Ident\(.*\)\s*){(.*)$/; + my $line1 = $1; + my $line2 = $2; + fix_insert_line($fixlinenr, ltrim($line1)); + fix_insert_line($fixlinenr, "\+{"); + if ($line2 !~ /^\s*$/) { + fix_insert_line($fixlinenr, "\+\t" . trim($line2)); + } + } + } + +# open braces for enum, union and struct go on the same line. + if ($line =~ /^.\s*{/ && + $prevline =~ /^.\s*(?:typedef\s+)?(enum|union|struct)(?:\s+$Ident)?\s*$/) { + if (ERROR("OPEN_BRACE", + "open brace '{' following $1 go on the same line\n" . $hereprev) && + $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { + fix_delete_line($fixlinenr - 1, $prevrawline); + fix_delete_line($fixlinenr, $rawline); + my $fixedline = rtrim($prevrawline) . " {"; + fix_insert_line($fixlinenr, $fixedline); + $fixedline = $rawline; + $fixedline =~ s/^(.\s*)\{\s*/$1\t/; + if ($fixedline !~ /^\+\s*$/) { + fix_insert_line($fixlinenr, $fixedline); + } + } + } + +# missing space after union, struct or enum definition + if ($line =~ /^.\s*(?:typedef\s+)?(enum|union|struct)(?:\s+$Ident){1,2}[=\{]/) { + if (WARN("SPACING", + "missing space after $1 definition\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/^(.\s*(?:typedef\s+)?(?:enum|union|struct)(?:\s+$Ident){1,2})([=\{])/$1 $2/; + } + } + +# Function pointer declarations +# check spacing between type, funcptr, and args +# canonical declaration is "type (*funcptr)(args...)" + if ($line =~ /^.\s*($Declare)\((\s*)\*(\s*)($Ident)(\s*)\)(\s*)\(/) { + my $declare = $1; + my $pre_pointer_space = $2; + my $post_pointer_space = $3; + my $funcname = $4; + my $post_funcname_space = $5; + my $pre_args_space = $6; + +# the $Declare variable will capture all spaces after the type +# so check it for a missing trailing missing space but pointer return types +# don't need a space so don't warn for those. + my $post_declare_space = ""; + if ($declare =~ /(\s+)$/) { + $post_declare_space = $1; + $declare = rtrim($declare); + } + if ($declare !~ /\*$/ && $post_declare_space =~ /^$/) { + WARN("SPACING", + "missing space after return type\n" . $herecurr); + $post_declare_space = " "; + } + +# unnecessary space "type (*funcptr)(args...)" +# This test is not currently implemented because these declarations are +# equivalent to +# int foo(int bar, ...) +# and this is form shouldn't/doesn't generate a checkpatch warning. +# +# elsif ($declare =~ /\s{2,}$/) { +# WARN("SPACING", +# "Multiple spaces after return type\n" . $herecurr); +# } + +# unnecessary space "type ( *funcptr)(args...)" + if (defined $pre_pointer_space && + $pre_pointer_space =~ /^\s/) { + WARN("SPACING", + "Unnecessary space after function pointer open parenthesis\n" . $herecurr); + } + +# unnecessary space "type (* funcptr)(args...)" + if (defined $post_pointer_space && + $post_pointer_space =~ /^\s/) { + WARN("SPACING", + "Unnecessary space before function pointer name\n" . $herecurr); + } + +# unnecessary space "type (*funcptr )(args...)" + if (defined $post_funcname_space && + $post_funcname_space =~ /^\s/) { + WARN("SPACING", + "Unnecessary space after function pointer name\n" . $herecurr); + } + +# unnecessary space "type (*funcptr) (args...)" + if (defined $pre_args_space && + $pre_args_space =~ /^\s/) { + WARN("SPACING", + "Unnecessary space before function pointer arguments\n" . $herecurr); + } + + if (show_type("SPACING") && $fix) { + $fixed[$fixlinenr] =~ + s/^(.\s*)$Declare\s*\(\s*\*\s*$Ident\s*\)\s*\(/$1 . $declare . $post_declare_space . '(*' . $funcname . ')('/ex; + } + } + +# check for spacing round square brackets; allowed: +# 1. with a type on the left -- int [] a; +# 2. at the beginning of a line for slice initialisers -- [0...10] = 5, +# 3. inside a curly brace -- = { [0...10] = 5 } + while ($line =~ /(.*?\s)\[/g) { + my ($where, $prefix) = ($-[1], $1); + if ($prefix !~ /$Type\s+$/ && + ($where != 0 || $prefix !~ /^.\s+$/) && + $prefix !~ /[{,]\s+$/ && + $prefix !~ /:\s+$/) { + if (ERROR("BRACKET_SPACE", + "space prohibited before open square bracket '['\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/^(\+.*?)\s+\[/$1\[/; + } + } + } + +# check for spaces between functions and their parentheses. + while ($line =~ /($Ident)\s+\(/g) { + my $name = $1; + my $ctx_before = substr($line, 0, $-[1]); + my $ctx = "$ctx_before$name"; + + # Ignore those directives where spaces _are_ permitted. + if ($name =~ /^(?: + if|for|while|switch|return|case| + volatile|__volatile__| + __attribute__|format|__extension__| + asm|__asm__)$/x) + { + # cpp #define statements have non-optional spaces, ie + # if there is a space between the name and the open + # parenthesis it is simply not a parameter group. + } elsif ($ctx_before =~ /^.\s*\#\s*define\s*$/) { + + # cpp #elif statement condition may start with a ( + } elsif ($ctx =~ /^.\s*\#\s*elif\s*$/) { + + # If this whole things ends with a type its most + # likely a typedef for a function. + } elsif ($ctx =~ /$Type$/) { + + } else { + if (WARN("SPACING", + "space prohibited between function name and open parenthesis '('\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/\b$name\s+\(/$name\(/; + } + } + } + +# Check operator spacing. + if (!($line=~/\#\s*include/)) { + my $fixed_line = ""; + my $line_fixed = 0; + + my $ops = qr{ + <<=|>>=|<=|>=|==|!=| + \+=|-=|\*=|\/=|%=|\^=|\|=|&=| + =>|->|<<|>>|<|>|=|!|~| + &&|\|\||,|\^|\+\+|--|&|\||\+|-|\*|\/|%| + \?:|\?|: + }x; + my @elements = split(/($ops|;)/, $opline); + +## print("element count: <" . $#elements . ">\n"); +## foreach my $el (@elements) { +## print("el: <$el>\n"); +## } + + my @fix_elements = (); + my $off = 0; + + foreach my $el (@elements) { + push(@fix_elements, substr($rawline, $off, length($el))); + $off += length($el); + } + + $off = 0; + + my $blank = copy_spacing($opline); + my $last_after = -1; + + for (my $n = 0; $n < $#elements; $n += 2) { + + my $good = $fix_elements[$n] . $fix_elements[$n + 1]; + +## print("n: <$n> good: <$good>\n"); + + $off += length($elements[$n]); + + # Pick up the preceding and succeeding characters. + my $ca = substr($opline, 0, $off); + my $cc = ''; + if (length($opline) >= ($off + length($elements[$n + 1]))) { + $cc = substr($opline, $off + length($elements[$n + 1])); + } + my $cb = "$ca$;$cc"; + + my $a = ''; + $a = 'V' if ($elements[$n] ne ''); + $a = 'W' if ($elements[$n] =~ /\s$/); + $a = 'C' if ($elements[$n] =~ /$;$/); + $a = 'B' if ($elements[$n] =~ /(\[|\()$/); + $a = 'O' if ($elements[$n] eq ''); + $a = 'E' if ($ca =~ /^\s*$/); + + my $op = $elements[$n + 1]; + + my $c = ''; + if (defined $elements[$n + 2]) { + $c = 'V' if ($elements[$n + 2] ne ''); + $c = 'W' if ($elements[$n + 2] =~ /^\s/); + $c = 'C' if ($elements[$n + 2] =~ /^$;/); + $c = 'B' if ($elements[$n + 2] =~ /^(\)|\]|;)/); + $c = 'O' if ($elements[$n + 2] eq ''); + $c = 'E' if ($elements[$n + 2] =~ /^\s*\\$/); + } else { + $c = 'E'; + } + + my $ctx = "${a}x${c}"; + + my $at = "(ctx:$ctx)"; + + my $ptr = substr($blank, 0, $off) . "^"; + my $hereptr = "$hereline$ptr\n"; + + # Pull out the value of this operator. + my $op_type = substr($curr_values, $off + 1, 1); + + # Get the full operator variant. + my $opv = $op . substr($curr_vars, $off, 1); + + # Ignore operators passed as parameters. + if ($op_type ne 'V' && + $ca =~ /\s$/ && $cc =~ /^\s*[,\)]/) { + +# # Ignore comments +# } elsif ($op =~ /^$;+$/) { + + # ; should have either the end of line or a space or \ after it + } elsif ($op eq ';') { + if ($ctx !~ /.x[WEBC]/ && + $cc !~ /^\\/ && $cc !~ /^;/) { + if (ERROR("SPACING", + "space required after that '$op' $at\n" . $hereptr)) { + $good = $fix_elements[$n] . trim($fix_elements[$n + 1]) . " "; + $line_fixed = 1; + } + } + + # // is a comment + } elsif ($op eq '//') { + + # : when part of a bitfield + } elsif ($opv eq ':B') { + # skip the bitfield test for now + + # No spaces for: + # -> + } elsif ($op eq '->') { + if ($ctx =~ /Wx.|.xW/) { + if (ERROR("SPACING", + "spaces prohibited around that '$op' $at\n" . $hereptr)) { + $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); + if (defined $fix_elements[$n + 2]) { + $fix_elements[$n + 2] =~ s/^\s+//; + } + $line_fixed = 1; + } + } + + # , must not have a space before and must have a space on the right. + } elsif ($op eq ',') { + my $rtrim_before = 0; + my $space_after = 0; + if ($ctx =~ /Wx./) { + if (ERROR("SPACING", + "space prohibited before that '$op' $at\n" . $hereptr)) { + $line_fixed = 1; + $rtrim_before = 1; + } + } + if ($ctx !~ /.x[WEC]/ && $cc !~ /^}/ && $cc !~ /^\)/) { + if (ERROR("SPACING", + "space required after that '$op' $at\n" . $hereptr)) { + $line_fixed = 1; + $last_after = $n; + $space_after = 1; + } + } + if ($rtrim_before || $space_after) { + if ($rtrim_before) { + $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); + } else { + $good = $fix_elements[$n] . trim($fix_elements[$n + 1]); + } + if ($space_after) { + $good .= " "; + } + } + + # '*' as part of a type definition -- reported already. + } elsif ($opv eq '*_') { + #warn "'*' is part of type\n"; + + # unary operators should have a space before and + # none after. May be left adjacent to another + # unary operator, or a cast + } elsif ($op eq '!' || $op eq '~' || + $opv eq '*U' || $opv eq '-U' || + $opv eq '&U' || $opv eq '&&U') { + if ($ctx !~ /[WEBC]x./ && $ca !~ /(?:\)|!|~|\*|-|\&|\||\+\+|\-\-|\{)$/) { + if (ERROR("SPACING", + "space required before that '$op' $at\n" . $hereptr)) { + if ($n != $last_after + 2) { + $good = $fix_elements[$n] . " " . ltrim($fix_elements[$n + 1]); + $line_fixed = 1; + } + } + } + if ($op eq '*' && $cc =~/\s*$Modifier\b/) { + # A unary '*' may be const + + } elsif ($ctx =~ /.xW/) { + if (ERROR("SPACING", + "space prohibited after that '$op' $at\n" . $hereptr)) { + $good = $fix_elements[$n] . rtrim($fix_elements[$n + 1]); + if (defined $fix_elements[$n + 2]) { + $fix_elements[$n + 2] =~ s/^\s+//; + } + $line_fixed = 1; + } + } + + # unary ++ and unary -- are allowed no space on one side. + } elsif ($op eq '++' or $op eq '--') { + if ($ctx !~ /[WEOBC]x[^W]/ && $ctx !~ /[^W]x[WOBEC]/) { + if (ERROR("SPACING", + "space required one side of that '$op' $at\n" . $hereptr)) { + $good = $fix_elements[$n] . trim($fix_elements[$n + 1]) . " "; + $line_fixed = 1; + } + } + if ($ctx =~ /Wx[BE]/ || + ($ctx =~ /Wx./ && $cc =~ /^;/)) { + if (ERROR("SPACING", + "space prohibited before that '$op' $at\n" . $hereptr)) { + $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); + $line_fixed = 1; + } + } + if ($ctx =~ /ExW/) { + if (ERROR("SPACING", + "space prohibited after that '$op' $at\n" . $hereptr)) { + $good = $fix_elements[$n] . trim($fix_elements[$n + 1]); + if (defined $fix_elements[$n + 2]) { + $fix_elements[$n + 2] =~ s/^\s+//; + } + $line_fixed = 1; + } + } + + # << and >> may either have or not have spaces both sides + } elsif ($op eq '<<' or $op eq '>>' or + $op eq '&' or $op eq '^' or $op eq '|' or + $op eq '+' or $op eq '-' or + $op eq '*' or $op eq '/' or + $op eq '%') + { + if ($check) { + if (defined $fix_elements[$n + 2] && $ctx !~ /[EW]x[EW]/) { + if (CHK("SPACING", + "spaces preferred around that '$op' $at\n" . $hereptr)) { + $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " "; + $fix_elements[$n + 2] =~ s/^\s+//; + $line_fixed = 1; + } + } elsif (!defined $fix_elements[$n + 2] && $ctx !~ /Wx[OE]/) { + if (CHK("SPACING", + "space preferred before that '$op' $at\n" . $hereptr)) { + $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]); + $line_fixed = 1; + } + } + } elsif ($ctx =~ /Wx[^WCE]|[^WCE]xW/) { + if (ERROR("SPACING", + "need consistent spacing around '$op' $at\n" . $hereptr)) { + $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " "; + if (defined $fix_elements[$n + 2]) { + $fix_elements[$n + 2] =~ s/^\s+//; + } + $line_fixed = 1; + } + } + + # A colon needs no spaces before when it is + # terminating a case value or a label. + } elsif ($opv eq ':C' || $opv eq ':L') { + if ($ctx =~ /Wx./) { + if (ERROR("SPACING", + "space prohibited before that '$op' $at\n" . $hereptr)) { + $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); + $line_fixed = 1; + } + } + + # All the others need spaces both sides. + } elsif ($ctx !~ /[EWC]x[CWE]/) { + my $ok = 0; + + # Ignore email addresses <foo@bar> + if (($op eq '<' && + $cc =~ /^\S+\@\S+>/) || + ($op eq '>' && + $ca =~ /<\S+\@\S+$/)) + { + $ok = 1; + } + + # for asm volatile statements + # ignore a colon with another + # colon immediately before or after + if (($op eq ':') && + ($ca =~ /:$/ || $cc =~ /^:/)) { + $ok = 1; + } + + # messages are ERROR, but ?: are CHK + if ($ok == 0) { + my $msg_level = \&ERROR; + $msg_level = \&CHK if (($op eq '?:' || $op eq '?' || $op eq ':') && $ctx =~ /VxV/); + + if (&{$msg_level}("SPACING", + "spaces required around that '$op' $at\n" . $hereptr)) { + $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " "; + if (defined $fix_elements[$n + 2]) { + $fix_elements[$n + 2] =~ s/^\s+//; + } + $line_fixed = 1; + } + } + } + $off += length($elements[$n + 1]); + +## print("n: <$n> GOOD: <$good>\n"); + + $fixed_line = $fixed_line . $good; + } + + if (($#elements % 2) == 0) { + $fixed_line = $fixed_line . $fix_elements[$#elements]; + } + + if ($fix && $line_fixed && $fixed_line ne $fixed[$fixlinenr]) { + $fixed[$fixlinenr] = $fixed_line; + } + + + } + +# check for whitespace before a non-naked semicolon + if ($line =~ /^\+.*\S\s+;\s*$/) { + if (WARN("SPACING", + "space prohibited before semicolon\n" . $herecurr) && + $fix) { + 1 while $fixed[$fixlinenr] =~ + s/^(\+.*\S)\s+;/$1;/; + } + } + +# check for multiple assignments + if ($line =~ /^.\s*$Lval\s*=\s*$Lval\s*=(?!=)/) { + CHK("MULTIPLE_ASSIGNMENTS", + "multiple assignments should be avoided\n" . $herecurr); + } + +## # check for multiple declarations, allowing for a function declaration +## # continuation. +## if ($line =~ /^.\s*$Type\s+$Ident(?:\s*=[^,{]*)?\s*,\s*$Ident.*/ && +## $line !~ /^.\s*$Type\s+$Ident(?:\s*=[^,{]*)?\s*,\s*$Type\s*$Ident.*/) { +## +## # Remove any bracketed sections to ensure we do not +## # falsly report the parameters of functions. +## my $ln = $line; +## while ($ln =~ s/\([^\(\)]*\)//g) { +## } +## if ($ln =~ /,/) { +## WARN("MULTIPLE_DECLARATION", +## "declaring multiple variables together should be avoided\n" . $herecurr); +## } +## } + +#need space before brace following if, while, etc + if (($line =~ /\(.*\)\{/ && $line !~ /\($Type\)\{/) || + $line =~ /do\{/) { + if (ERROR("SPACING", + "space required before the open brace '{'\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/^(\+.*(?:do|\)))\{/$1 {/; + } + } + +## # check for blank lines before declarations +## if ($line =~ /^.\t+$Type\s+$Ident(?:\s*=.*)?;/ && +## $prevrawline =~ /^.\s*$/) { +## WARN("SPACING", +## "No blank lines before declarations\n" . $hereprev); +## } +## + +# closing brace should have a space following it when it has anything +# on the line + if ($line =~ /}(?!(?:,|;|\)))\S/) { + if (ERROR("SPACING", + "space required after that close brace '}'\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/}((?!(?:,|;|\)))\S)/} $1/; + } + } + +# check spacing on square brackets + if ($line =~ /\[\s/ && $line !~ /\[\s*$/) { + if (ERROR("SPACING", + "space prohibited after that open square bracket '['\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/\[\s+/\[/; + } + } + if ($line =~ /\s\]/) { + if (ERROR("SPACING", + "space prohibited before that close square bracket ']'\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/\s+\]/\]/; + } + } + +# check spacing on parentheses + if ($line =~ /\(\s/ && $line !~ /\(\s*(?:\\)?$/ && + $line !~ /for\s*\(\s+;/) { + if (ERROR("SPACING", + "space prohibited after that open parenthesis '('\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/\(\s+/\(/; + } + } + if ($line =~ /(\s+)\)/ && $line !~ /^.\s*\)/ && + $line !~ /for\s*\(.*;\s+\)/ && + $line !~ /:\s+\)/) { + if (ERROR("SPACING", + "space prohibited before that close parenthesis ')'\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/\s+\)/\)/; + } + } + +# check unnecessary parentheses around addressof/dereference single $Lvals +# ie: &(foo->bar) should be &foo->bar and *(foo->bar) should be *foo->bar + + while ($line =~ /(?:[^&]&\s*|\*)\(\s*($Ident\s*(?:$Member\s*)+)\s*\)/g) { + my $var = $1; + if (CHK("UNNECESSARY_PARENTHESES", + "Unnecessary parentheses around $var\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\(\s*\Q$var\E\s*\)/$var/; + } + } + +# check for unnecessary parentheses around function pointer uses +# ie: (foo->bar)(); should be foo->bar(); +# but not "if (foo->bar) (" to avoid some false positives + if ($line =~ /(\bif\s*|)(\(\s*$Ident\s*(?:$Member\s*)+\))[ \t]*\(/ && $1 !~ /^if/) { + my $var = $2; + if (CHK("UNNECESSARY_PARENTHESES", + "Unnecessary parentheses around function pointer $var\n" . $herecurr) && + $fix) { + my $var2 = deparenthesize($var); + $var2 =~ s/\s//g; + $fixed[$fixlinenr] =~ s/\Q$var\E/$var2/; + } + } + +# check for unnecessary parentheses around comparisons in if uses + if ($^V && $^V ge 5.10.0 && defined($stat) && + $stat =~ /(^.\s*if\s*($balanced_parens))/) { + my $if_stat = $1; + my $test = substr($2, 1, -1); + my $herectx; + while ($test =~ /(?:^|[^\w\&\!\~])+\s*\(\s*([\&\!\~]?\s*$Lval\s*(?:$Compare\s*$FuncArg)?)\s*\)/g) { + my $match = $1; + # avoid parentheses around potential macro args + next if ($match =~ /^\s*\w+\s*$/); + if (!defined($herectx)) { + $herectx = $here . "\n"; + my $cnt = statement_rawlines($if_stat); + for (my $n = 0; $n < $cnt; $n++) { + my $rl = raw_line($linenr, $n); + $herectx .= $rl . "\n"; + last if $rl =~ /^[ \+].*\{/; + } + } + CHK("UNNECESSARY_PARENTHESES", + "Unnecessary parentheses around '$match'\n" . $herectx); + } + } + +#goto labels aren't indented, allow a single space however + if ($line=~/^.\s+[A-Za-z\d_]+:(?![0-9]+)/ and + !($line=~/^. [A-Za-z\d_]+:/) and !($line=~/^.\s+default:/)) { + if (WARN("INDENTED_LABEL", + "labels should not be indented\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/^(.)\s+/$1/; + } + } + +# return is not a function + if (defined($stat) && $stat =~ /^.\s*return(\s*)\(/s) { + my $spacing = $1; + if ($^V && $^V ge 5.10.0 && + $stat =~ /^.\s*return\s*($balanced_parens)\s*;\s*$/) { + my $value = $1; + $value = deparenthesize($value); + if ($value =~ m/^\s*$FuncArg\s*(?:\?|$)/) { + ERROR("RETURN_PARENTHESES", + "return is not a function, parentheses are not required\n" . $herecurr); + } + } elsif ($spacing !~ /\s+/) { + ERROR("SPACING", + "space required before the open parenthesis '('\n" . $herecurr); + } + } + +# unnecessary return in a void function +# at end-of-function, with the previous line a single leading tab, then return; +# and the line before that not a goto label target like "out:" + if ($sline =~ /^[ \+]}\s*$/ && + $prevline =~ /^\+\treturn\s*;\s*$/ && + $linenr >= 3 && + $lines[$linenr - 3] =~ /^[ +]/ && + $lines[$linenr - 3] !~ /^[ +]\s*$Ident\s*:/) { + WARN("RETURN_VOID", + "void function return statements are not generally useful\n" . $hereprev); + } + +# if statements using unnecessary parentheses - ie: if ((foo == bar)) + if ($^V && $^V ge 5.10.0 && + $line =~ /\bif\s*((?:\(\s*){2,})/) { + my $openparens = $1; + my $count = $openparens =~ tr@\(@\(@; + my $msg = ""; + if ($line =~ /\bif\s*(?:\(\s*){$count,$count}$LvalOrFunc\s*($Compare)\s*$LvalOrFunc(?:\s*\)){$count,$count}/) { + my $comp = $4; #Not $1 because of $LvalOrFunc + $msg = " - maybe == should be = ?" if ($comp eq "=="); + WARN("UNNECESSARY_PARENTHESES", + "Unnecessary parentheses$msg\n" . $herecurr); + } + } + +# comparisons with a constant or upper case identifier on the left +# avoid cases like "foo + BAR < baz" +# only fix matches surrounded by parentheses to avoid incorrect +# conversions like "FOO < baz() + 5" being "misfixed" to "baz() > FOO + 5" + if ($^V && $^V ge 5.10.0 && + $line =~ /^\+(.*)\b($Constant|[A-Z_][A-Z0-9_]*)\s*($Compare)\s*($LvalOrFunc)/) { + my $lead = $1; + my $const = $2; + my $comp = $3; + my $to = $4; + my $newcomp = $comp; + if ($lead !~ /(?:$Operators|\.)\s*$/ && + $to !~ /^(?:Constant|[A-Z_][A-Z0-9_]*)$/ && + WARN("CONSTANT_COMPARISON", + "Comparisons should place the constant on the right side of the test\n" . $herecurr) && + $fix) { + if ($comp eq "<") { + $newcomp = ">"; + } elsif ($comp eq "<=") { + $newcomp = ">="; + } elsif ($comp eq ">") { + $newcomp = "<"; + } elsif ($comp eq ">=") { + $newcomp = "<="; + } + $fixed[$fixlinenr] =~ s/\(\s*\Q$const\E\s*$Compare\s*\Q$to\E\s*\)/($to $newcomp $const)/; + } + } + +# Return of what appears to be an errno should normally be negative + if ($sline =~ /\breturn(?:\s*\(+\s*|\s+)(E[A-Z]+)(?:\s*\)+\s*|\s*)[;:,]/) { + my $name = $1; + if ($name ne 'EOF' && $name ne 'ERROR') { + WARN("USE_NEGATIVE_ERRNO", + "return of an errno should typically be negative (ie: return -$1)\n" . $herecurr); + } + } + +# Need a space before open parenthesis after if, while etc + if ($line =~ /\b(if|while|for|switch)\(/) { + if (ERROR("SPACING", + "space required before the open parenthesis '('\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/\b(if|while|for|switch)\(/$1 \(/; + } + } + +# Check for illegal assignment in if conditional -- and check for trailing +# statements after the conditional. + if ($line =~ /do\s*(?!{)/) { + ($stat, $cond, $line_nr_next, $remain_next, $off_next) = + ctx_statement_block($linenr, $realcnt, 0) + if (!defined $stat); + my ($stat_next) = ctx_statement_block($line_nr_next, + $remain_next, $off_next); + $stat_next =~ s/\n./\n /g; + ##print "stat<$stat> stat_next<$stat_next>\n"; + + if ($stat_next =~ /^\s*while\b/) { + # If the statement carries leading newlines, + # then count those as offsets. + my ($whitespace) = + ($stat_next =~ /^((?:\s*\n[+-])*\s*)/s); + my $offset = + statement_rawlines($whitespace) - 1; + + $suppress_whiletrailers{$line_nr_next + + $offset} = 1; + } + } + if (!defined $suppress_whiletrailers{$linenr} && + defined($stat) && defined($cond) && + $line =~ /\b(?:if|while|for)\s*\(/ && $line !~ /^.\s*#/) { + my ($s, $c) = ($stat, $cond); + + if ($c =~ /\bif\s*\(.*[^<>!=]=[^=].*/s) { + ERROR("ASSIGN_IN_IF", + "do not use assignment in if condition\n" . $herecurr); + } + + # Find out what is on the end of the line after the + # conditional. + substr($s, 0, length($c), ''); + $s =~ s/\n.*//g; + $s =~ s/$;//g; # Remove any comments + if (length($c) && $s !~ /^\s*{?\s*\\*\s*$/ && + $c !~ /}\s*while\s*/) + { + # Find out how long the conditional actually is. + my @newlines = ($c =~ /\n/gs); + my $cond_lines = 1 + $#newlines; + my $stat_real = ''; + + $stat_real = raw_line($linenr, $cond_lines) + . "\n" if ($cond_lines); + if (defined($stat_real) && $cond_lines > 1) { + $stat_real = "[...]\n$stat_real"; + } + + ERROR("TRAILING_STATEMENTS", + "trailing statements should be on next line\n" . $herecurr . $stat_real); + } + } + +# Check for bitwise tests written as boolean + if ($line =~ / + (?: + (?:\[|\(|\&\&|\|\|) + \s*0[xX][0-9]+\s* + (?:\&\&|\|\|) + | + (?:\&\&|\|\|) + \s*0[xX][0-9]+\s* + (?:\&\&|\|\||\)|\]) + )/x) + { + WARN("HEXADECIMAL_BOOLEAN_TEST", + "boolean test with hexadecimal, perhaps just 1 \& or \|?\n" . $herecurr); + } + +# if and else should not have general statements after it + if ($line =~ /^.\s*(?:}\s*)?else\b(.*)/) { + my $s = $1; + $s =~ s/$;//g; # Remove any comments + if ($s !~ /^\s*(?:\sif|(?:{|)\s*\\?\s*$)/) { + ERROR("TRAILING_STATEMENTS", + "trailing statements should be on next line\n" . $herecurr); + } + } +# if should not continue a brace + if ($line =~ /}\s*if\b/) { + ERROR("TRAILING_STATEMENTS", + "trailing statements should be on next line (or did you mean 'else if'?)\n" . + $herecurr); + } +# case and default should not have general statements after them + if ($line =~ /^.\s*(?:case\s*.*|default\s*):/g && + $line !~ /\G(?: + (?:\s*$;*)(?:\s*{)?(?:\s*$;*)(?:\s*\\)?\s*$| + \s*return\s+ + )/xg) + { + ERROR("TRAILING_STATEMENTS", + "trailing statements should be on next line\n" . $herecurr); + } + + # Check for }<nl>else {, these must be at the same + # indent level to be relevant to each other. + if ($prevline=~/}\s*$/ and $line=~/^.\s*else\s*/ && + $previndent == $indent) { + if (ERROR("ELSE_AFTER_BRACE", + "else should follow close brace '}'\n" . $hereprev) && + $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { + fix_delete_line($fixlinenr - 1, $prevrawline); + fix_delete_line($fixlinenr, $rawline); + my $fixedline = $prevrawline; + $fixedline =~ s/}\s*$//; + if ($fixedline !~ /^\+\s*$/) { + fix_insert_line($fixlinenr, $fixedline); + } + $fixedline = $rawline; + $fixedline =~ s/^(.\s*)else/$1} else/; + fix_insert_line($fixlinenr, $fixedline); + } + } + + if ($prevline=~/}\s*$/ and $line=~/^.\s*while\s*/ && + $previndent == $indent) { + my ($s, $c) = ctx_statement_block($linenr, $realcnt, 0); + + # Find out what is on the end of the line after the + # conditional. + substr($s, 0, length($c), ''); + $s =~ s/\n.*//g; + + if ($s =~ /^\s*;/) { + if (ERROR("WHILE_AFTER_BRACE", + "while should follow close brace '}'\n" . $hereprev) && + $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { + fix_delete_line($fixlinenr - 1, $prevrawline); + fix_delete_line($fixlinenr, $rawline); + my $fixedline = $prevrawline; + my $trailing = $rawline; + $trailing =~ s/^\+//; + $trailing = trim($trailing); + $fixedline =~ s/}\s*$/} $trailing/; + fix_insert_line($fixlinenr, $fixedline); + } + } + } + +#Specific variable tests + while ($line =~ m{($Constant|$Lval)}g) { + my $var = $1; + +#gcc binary extension + if ($var =~ /^$Binary$/) { + if (WARN("GCC_BINARY_CONSTANT", + "Avoid gcc v4.3+ binary constant extension: <$var>\n" . $herecurr) && + $fix) { + my $hexval = sprintf("0x%x", oct($var)); + $fixed[$fixlinenr] =~ + s/\b$var\b/$hexval/; + } + } + +#CamelCase + if ($var !~ /^$Constant$/ && + $var =~ /[A-Z][a-z]|[a-z][A-Z]/ && +#Ignore Page<foo> variants + $var !~ /^(?:Clear|Set|TestClear|TestSet|)Page[A-Z]/ && +#Ignore SI style variants like nS, mV and dB (ie: max_uV, regulator_min_uA_show) + $var !~ /^(?:[a-z_]*?)_?[a-z][A-Z](?:_[a-z_]+)?$/ && +#Ignore some three character SI units explicitly, like MiB and KHz + $var !~ /^(?:[a-z_]*?)_?(?:[KMGT]iB|[KMGT]?Hz)(?:_[a-z_]+)?$/) { + while ($var =~ m{($Ident)}g) { + my $word = $1; + next if ($word !~ /[A-Z][a-z]|[a-z][A-Z]/); + if ($check) { + seed_camelcase_includes(); + if (!$file && !$camelcase_file_seeded) { + seed_camelcase_file($realfile); + $camelcase_file_seeded = 1; + } + } + if (!defined $camelcase{$word}) { + $camelcase{$word} = 1; + CHK("CAMELCASE", + "Avoid CamelCase: <$word>\n" . $herecurr); + } + } + } + } + +#no spaces allowed after \ in define + if ($line =~ /\#\s*define.*\\\s+$/) { + if (WARN("WHITESPACE_AFTER_LINE_CONTINUATION", + "Whitespace after \\ makes next lines useless\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\s+$//; + } + } + +# warn if <asm/foo.h> is #included and <linux/foo.h> is available and includes +# itself <asm/foo.h> (uses RAW line) + if ($tree && $rawline =~ m{^.\s*\#\s*include\s*\<asm\/(.*)\.h\>}) { + my $file = "$1.h"; + my $checkfile = "include/linux/$file"; + if (-f "$root/$checkfile" && + $realfile ne $checkfile && + $1 !~ /$allowed_asm_includes/) + { + my $asminclude = `grep -Ec "#include\\s+<asm/$file>" $root/$checkfile`; + if ($asminclude > 0) { + if ($realfile =~ m{^arch/}) { + CHK("ARCH_INCLUDE_LINUX", + "Consider using #include <linux/$file> instead of <asm/$file>\n" . $herecurr); + } else { + WARN("INCLUDE_LINUX", + "Use #include <linux/$file> instead of <asm/$file>\n" . $herecurr); + } + } + } + } + +# multi-statement macros should be enclosed in a do while loop, grab the +# first statement and ensure its the whole macro if its not enclosed +# in a known good container + if ($realfile !~ m@/vmlinux.lds.h$@ && + $line =~ /^.\s*\#\s*define\s*$Ident(\()?/) { + my $ln = $linenr; + my $cnt = $realcnt; + my ($off, $dstat, $dcond, $rest); + my $ctx = ''; + my $has_flow_statement = 0; + my $has_arg_concat = 0; + ($dstat, $dcond, $ln, $cnt, $off) = + ctx_statement_block($linenr, $realcnt, 0); + $ctx = $dstat; + #print "dstat<$dstat> dcond<$dcond> cnt<$cnt> off<$off>\n"; + #print "LINE<$lines[$ln-1]> len<" . length($lines[$ln-1]) . "\n"; + + $has_flow_statement = 1 if ($ctx =~ /\b(goto|return)\b/); + $has_arg_concat = 1 if ($ctx =~ /\#\#/ && $ctx !~ /\#\#\s*(?:__VA_ARGS__|args)\b/); + + $dstat =~ s/^.\s*\#\s*define\s+$Ident(\([^\)]*\))?\s*//; + my $define_args = $1; + my $define_stmt = $dstat; + my @def_args = (); + + if (defined $define_args && $define_args ne "") { + $define_args = substr($define_args, 1, length($define_args) - 2); + $define_args =~ s/\s*//g; + @def_args = split(",", $define_args); + } + + $dstat =~ s/$;//g; + $dstat =~ s/\\\n.//g; + $dstat =~ s/^\s*//s; + $dstat =~ s/\s*$//s; + + # Flatten any parentheses and braces + while ($dstat =~ s/\([^\(\)]*\)/1/ || + $dstat =~ s/\{[^\{\}]*\}/1/ || + $dstat =~ s/.\[[^\[\]]*\]/1/) + { + } + + # Flatten any obvious string concatentation. + while ($dstat =~ s/($String)\s*$Ident/$1/ || + $dstat =~ s/$Ident\s*($String)/$1/) + { + } + + # Make asm volatile uses seem like a generic function + $dstat =~ s/\b_*asm_*\s+_*volatile_*\b/asm_volatile/g; + + my $exceptions = qr{ + $Declare| + module_param_named| + MODULE_PARM_DESC| + DECLARE_PER_CPU| + DEFINE_PER_CPU| + __typeof__\(| + union| + struct| + \.$Ident\s*=\s*| + ^\"|\"$| + ^\[ + }x; + #print "REST<$rest> dstat<$dstat> ctx<$ctx>\n"; + + $ctx =~ s/\n*$//; + my $herectx = $here . "\n"; + my $stmt_cnt = statement_rawlines($ctx); + + for (my $n = 0; $n < $stmt_cnt; $n++) { + $herectx .= raw_line($linenr, $n) . "\n"; + } + + if ($dstat ne '' && + $dstat !~ /^(?:$Ident|-?$Constant),$/ && # 10, // foo(), + $dstat !~ /^(?:$Ident|-?$Constant);$/ && # foo(); + $dstat !~ /^[!~-]?(?:$Lval|$Constant)$/ && # 10 // foo() // !foo // ~foo // -foo // foo->bar // foo.bar->baz + $dstat !~ /^'X'$/ && $dstat !~ /^'XX'$/ && # character constants + $dstat !~ /$exceptions/ && + $dstat !~ /^\.$Ident\s*=/ && # .foo = + $dstat !~ /^(?:\#\s*$Ident|\#\s*$Constant)\s*$/ && # stringification #foo + $dstat !~ /^do\s*$Constant\s*while\s*$Constant;?$/ && # do {...} while (...); // do {...} while (...) + $dstat !~ /^for\s*$Constant$/ && # for (...) + $dstat !~ /^for\s*$Constant\s+(?:$Ident|-?$Constant)$/ && # for (...) bar() + $dstat !~ /^do\s*{/ && # do {... + $dstat !~ /^\(\{/ && # ({... + $ctx !~ /^.\s*#\s*define\s+TRACE_(?:SYSTEM|INCLUDE_FILE|INCLUDE_PATH)\b/) + { + if ($dstat =~ /^\s*if\b/) { + ERROR("MULTISTATEMENT_MACRO_USE_DO_WHILE", + "Macros starting with if should be enclosed by a do - while loop to avoid possible if/else logic defects\n" . "$herectx"); + } elsif ($dstat =~ /;/) { + ERROR("MULTISTATEMENT_MACRO_USE_DO_WHILE", + "Macros with multiple statements should be enclosed in a do - while loop\n" . "$herectx"); + } else { + WARN("COMPLEX_MACRO", + "Macros with complex values should be enclosed in parentheses\n" . "$herectx"); + } + + } + + # Make $define_stmt single line, comment-free, etc + my @stmt_array = split('\n', $define_stmt); + my $first = 1; + $define_stmt = ""; + foreach my $l (@stmt_array) { + $l =~ s/\\$//; + if ($first) { + $define_stmt = $l; + $first = 0; + } elsif ($l =~ /^[\+ ]/) { + $define_stmt .= substr($l, 1); + } + } + $define_stmt =~ s/$;//g; + $define_stmt =~ s/\s+/ /g; + $define_stmt = trim($define_stmt); + +# check if any macro arguments are reused (ignore '...' and 'type') + foreach my $arg (@def_args) { + next if ($arg =~ /\.\.\./); + next if ($arg =~ /^type$/i); + my $tmp_stmt = $define_stmt; + $tmp_stmt =~ s/\b(typeof|__typeof__|__builtin\w+|typecheck\s*\(\s*$Type\s*,|\#+)\s*\(*\s*$arg\s*\)*\b//g; + $tmp_stmt =~ s/\#+\s*$arg\b//g; + $tmp_stmt =~ s/\b$arg\s*\#\#//g; + my $use_cnt = $tmp_stmt =~ s/\b$arg\b//g; + if ($use_cnt > 1) { + CHK("MACRO_ARG_REUSE", + "Macro argument reuse '$arg' - possible side-effects?\n" . "$herectx"); + } +# check if any macro arguments may have other precedence issues + if ($tmp_stmt =~ m/($Operators)?\s*\b$arg\b\s*($Operators)?/m && + ((defined($1) && $1 ne ',') || + (defined($2) && $2 ne ','))) { + CHK("MACRO_ARG_PRECEDENCE", + "Macro argument '$arg' may be better as '($arg)' to avoid precedence issues\n" . "$herectx"); + } + } + +# check for macros with flow control, but without ## concatenation +# ## concatenation is commonly a macro that defines a function so ignore those + if ($has_flow_statement && !$has_arg_concat) { + my $herectx = $here . "\n"; + my $cnt = statement_rawlines($ctx); + + for (my $n = 0; $n < $cnt; $n++) { + $herectx .= raw_line($linenr, $n) . "\n"; + } + WARN("MACRO_WITH_FLOW_CONTROL", + "Macros with flow control statements should be avoided\n" . "$herectx"); + } + +# check for line continuations outside of #defines, preprocessor #, and asm + + } else { + if ($prevline !~ /^..*\\$/ && + $line !~ /^\+\s*\#.*\\$/ && # preprocessor + $line !~ /^\+.*\b(__asm__|asm)\b.*\\$/ && # asm + $line =~ /^\+.*\\$/) { + WARN("LINE_CONTINUATIONS", + "Avoid unnecessary line continuations\n" . $herecurr); + } + } + +# do {} while (0) macro tests: +# single-statement macros do not need to be enclosed in do while (0) loop, +# macro should not end with a semicolon + if ($^V && $^V ge 5.10.0 && + $realfile !~ m@/vmlinux.lds.h$@ && + $line =~ /^.\s*\#\s*define\s+$Ident(\()?/) { + my $ln = $linenr; + my $cnt = $realcnt; + my ($off, $dstat, $dcond, $rest); + my $ctx = ''; + ($dstat, $dcond, $ln, $cnt, $off) = + ctx_statement_block($linenr, $realcnt, 0); + $ctx = $dstat; + + $dstat =~ s/\\\n.//g; + $dstat =~ s/$;/ /g; + + if ($dstat =~ /^\+\s*#\s*define\s+$Ident\s*${balanced_parens}\s*do\s*{(.*)\s*}\s*while\s*\(\s*0\s*\)\s*([;\s]*)\s*$/) { + my $stmts = $2; + my $semis = $3; + + $ctx =~ s/\n*$//; + my $cnt = statement_rawlines($ctx); + my $herectx = $here . "\n"; + + for (my $n = 0; $n < $cnt; $n++) { + $herectx .= raw_line($linenr, $n) . "\n"; + } + + if (($stmts =~ tr/;/;/) == 1 && + $stmts !~ /^\s*(if|while|for|switch)\b/) { + WARN("SINGLE_STATEMENT_DO_WHILE_MACRO", + "Single statement macros should not use a do {} while (0) loop\n" . "$herectx"); + } + if (defined $semis && $semis ne "") { + WARN("DO_WHILE_MACRO_WITH_TRAILING_SEMICOLON", + "do {} while (0) macros should not be semicolon terminated\n" . "$herectx"); + } + } elsif ($dstat =~ /^\+\s*#\s*define\s+$Ident.*;\s*$/) { + $ctx =~ s/\n*$//; + my $cnt = statement_rawlines($ctx); + my $herectx = $here . "\n"; + + for (my $n = 0; $n < $cnt; $n++) { + $herectx .= raw_line($linenr, $n) . "\n"; + } + + WARN("TRAILING_SEMICOLON", + "macros should not use a trailing semicolon\n" . "$herectx"); + } + } + +# make sure symbols are always wrapped with VMLINUX_SYMBOL() ... +# all assignments may have only one of the following with an assignment: +# . +# ALIGN(...) +# VMLINUX_SYMBOL(...) + if ($realfile eq 'vmlinux.lds.h' && $line =~ /(?:(?:^|\s)$Ident\s*=|=\s*$Ident(?:\s|$))/) { + WARN("MISSING_VMLINUX_SYMBOL", + "vmlinux.lds.h needs VMLINUX_SYMBOL() around C-visible symbols\n" . $herecurr); + } + +# check for redundant bracing round if etc + if ($line =~ /(^.*)\bif\b/ && $1 !~ /else\s*$/) { + my ($level, $endln, @chunks) = + ctx_statement_full($linenr, $realcnt, 1); + #print "chunks<$#chunks> linenr<$linenr> endln<$endln> level<$level>\n"; + #print "APW: <<$chunks[1][0]>><<$chunks[1][1]>>\n"; + if ($#chunks > 0 && $level == 0) { + my @allowed = (); + my $allow = 0; + my $seen = 0; + my $herectx = $here . "\n"; + my $ln = $linenr - 1; + for my $chunk (@chunks) { + my ($cond, $block) = @{$chunk}; + + # If the condition carries leading newlines, then count those as offsets. + my ($whitespace) = ($cond =~ /^((?:\s*\n[+-])*\s*)/s); + my $offset = statement_rawlines($whitespace) - 1; + + $allowed[$allow] = 0; + #print "COND<$cond> whitespace<$whitespace> offset<$offset>\n"; + + # We have looked at and allowed this specific line. + $suppress_ifbraces{$ln + $offset} = 1; + + $herectx .= "$rawlines[$ln + $offset]\n[...]\n"; + $ln += statement_rawlines($block) - 1; + + substr($block, 0, length($cond), ''); + + $seen++ if ($block =~ /^\s*{/); + + #print "cond<$cond> block<$block> allowed<$allowed[$allow]>\n"; + if (statement_lines($cond) > 1) { + #print "APW: ALLOWED: cond<$cond>\n"; + $allowed[$allow] = 1; + } + if ($block =~/\b(?:if|for|while)\b/) { + #print "APW: ALLOWED: block<$block>\n"; + $allowed[$allow] = 1; + } + if (statement_block_size($block) > 1) { + #print "APW: ALLOWED: lines block<$block>\n"; + $allowed[$allow] = 1; + } + $allow++; + } + if ($seen) { + my $sum_allowed = 0; + foreach (@allowed) { + $sum_allowed += $_; + } + if ($sum_allowed == 0) { + WARN("BRACES", + "braces {} are not necessary for any arm of this statement\n" . $herectx); + } elsif ($sum_allowed != $allow && + $seen != $allow) { + CHK("BRACES", + "braces {} should be used on all arms of this statement\n" . $herectx); + } + } + } + } + if (!defined $suppress_ifbraces{$linenr - 1} && + $line =~ /\b(if|while|for|else)\b/) { + my $allowed = 0; + + # Check the pre-context. + if (substr($line, 0, $-[0]) =~ /(\}\s*)$/) { + #print "APW: ALLOWED: pre<$1>\n"; + $allowed = 1; + } + + my ($level, $endln, @chunks) = + ctx_statement_full($linenr, $realcnt, $-[0]); + + # Check the condition. + my ($cond, $block) = @{$chunks[0]}; + #print "CHECKING<$linenr> cond<$cond> block<$block>\n"; + if (defined $cond) { + substr($block, 0, length($cond), ''); + } + if (statement_lines($cond) > 1) { + #print "APW: ALLOWED: cond<$cond>\n"; + $allowed = 1; + } + if ($block =~/\b(?:if|for|while)\b/) { + #print "APW: ALLOWED: block<$block>\n"; + $allowed = 1; + } + if (statement_block_size($block) > 1) { + #print "APW: ALLOWED: lines block<$block>\n"; + $allowed = 1; + } + # Check the post-context. + if (defined $chunks[1]) { + my ($cond, $block) = @{$chunks[1]}; + if (defined $cond) { + substr($block, 0, length($cond), ''); + } + if ($block =~ /^\s*\{/) { + #print "APW: ALLOWED: chunk-1 block<$block>\n"; + $allowed = 1; + } + } + if ($level == 0 && $block =~ /^\s*\{/ && !$allowed) { + my $herectx = $here . "\n"; + my $cnt = statement_rawlines($block); + + for (my $n = 0; $n < $cnt; $n++) { + $herectx .= raw_line($linenr, $n) . "\n"; + } + + WARN("BRACES", + "braces {} are not necessary for single statement blocks\n" . $herectx); + } + } + +# check for single line unbalanced braces + if ($sline =~ /^.\s*\}\s*else\s*$/ || + $sline =~ /^.\s*else\s*\{\s*$/) { + CHK("BRACES", "Unbalanced braces around else statement\n" . $herecurr); + } + +# check for unnecessary blank lines around braces + if (($line =~ /^.\s*}\s*$/ && $prevrawline =~ /^.\s*$/)) { + if (CHK("BRACES", + "Blank lines aren't necessary before a close brace '}'\n" . $hereprev) && + $fix && $prevrawline =~ /^\+/) { + fix_delete_line($fixlinenr - 1, $prevrawline); + } + } + if (($rawline =~ /^.\s*$/ && $prevline =~ /^..*{\s*$/)) { + if (CHK("BRACES", + "Blank lines aren't necessary after an open brace '{'\n" . $hereprev) && + $fix) { + fix_delete_line($fixlinenr, $rawline); + } + } + +# no volatiles please + my $asm_volatile = qr{\b(__asm__|asm)\s+(__volatile__|volatile)\b}; + if ($line =~ /\bvolatile\b/ && $line !~ /$asm_volatile/) { + WARN("VOLATILE", + "Use of volatile is usually wrong: see Documentation/process/volatile-considered-harmful.rst\n" . $herecurr); + } + +# Check for user-visible strings broken across lines, which breaks the ability +# to grep for the string. Make exceptions when the previous string ends in a +# newline (multiple lines in one string constant) or '\t', '\r', ';', or '{' +# (common in inline assembly) or is a octal \123 or hexadecimal \xaf value + if ($line =~ /^\+\s*$String/ && + $prevline =~ /"\s*$/ && + $prevrawline !~ /(?:\\(?:[ntr]|[0-7]{1,3}|x[0-9a-fA-F]{1,2})|;\s*|\{\s*)"\s*$/) { + if (WARN("SPLIT_STRING", + "quoted string split across lines\n" . $hereprev) && + $fix && + $prevrawline =~ /^\+.*"\s*$/ && + $last_coalesced_string_linenr != $linenr - 1) { + my $extracted_string = get_quoted_string($line, $rawline); + my $comma_close = ""; + if ($rawline =~ /\Q$extracted_string\E(\s*\)\s*;\s*$|\s*,\s*)/) { + $comma_close = $1; + } + + fix_delete_line($fixlinenr - 1, $prevrawline); + fix_delete_line($fixlinenr, $rawline); + my $fixedline = $prevrawline; + $fixedline =~ s/"\s*$//; + $fixedline .= substr($extracted_string, 1) . trim($comma_close); + fix_insert_line($fixlinenr - 1, $fixedline); + $fixedline = $rawline; + $fixedline =~ s/\Q$extracted_string\E\Q$comma_close\E//; + if ($fixedline !~ /\+\s*$/) { + fix_insert_line($fixlinenr, $fixedline); + } + $last_coalesced_string_linenr = $linenr; + } + } + +# check for missing a space in a string concatenation + if ($prevrawline =~ /[^\\]\w"$/ && $rawline =~ /^\+[\t ]+"\w/) { + WARN('MISSING_SPACE', + "break quoted strings at a space character\n" . $hereprev); + } + +# check for an embedded function name in a string when the function is known +# This does not work very well for -f --file checking as it depends on patch +# context providing the function name or a single line form for in-file +# function declarations + if ($line =~ /^\+.*$String/ && + defined($context_function) && + get_quoted_string($line, $rawline) =~ /\b$context_function\b/ && + length(get_quoted_string($line, $rawline)) != (length($context_function) + 2)) { + WARN("EMBEDDED_FUNCTION_NAME", + "Prefer using '\"%s...\", __func__' to using '$context_function', this function's name, in a string\n" . $herecurr); + } + +# check for spaces before a quoted newline + if ($rawline =~ /^.*\".*\s\\n/) { + if (WARN("QUOTED_WHITESPACE_BEFORE_NEWLINE", + "unnecessary whitespace before a quoted newline\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/^(\+.*\".*)\s+\\n/$1\\n/; + } + + } + +# concatenated string without spaces between elements + if ($line =~ /$String[A-Z_]/ || $line =~ /[A-Za-z0-9_]$String/) { + CHK("CONCATENATED_STRING", + "Concatenated strings should use spaces between elements\n" . $herecurr); + } + +# uncoalesced string fragments + if ($line =~ /$String\s*"/) { + WARN("STRING_FRAGMENTS", + "Consecutive strings are generally better as a single string\n" . $herecurr); + } + +# check for non-standard and hex prefixed decimal printf formats + my $show_L = 1; #don't show the same defect twice + my $show_Z = 1; + while ($line =~ /(?:^|")([X\t]*)(?:"|$)/g) { + my $string = substr($rawline, $-[1], $+[1] - $-[1]); + $string =~ s/%%/__/g; + # check for %L + if ($show_L && $string =~ /%[\*\d\.\$]*L([diouxX])/) { + WARN("PRINTF_L", + "\%L$1 is non-standard C, use %ll$1\n" . $herecurr); + $show_L = 0; + } + # check for %Z + if ($show_Z && $string =~ /%[\*\d\.\$]*Z([diouxX])/) { + WARN("PRINTF_Z", + "%Z$1 is non-standard C, use %z$1\n" . $herecurr); + $show_Z = 0; + } + # check for 0x<decimal> + if ($string =~ /0x%[\*\d\.\$\Llzth]*[diou]/) { + ERROR("PRINTF_0XDECIMAL", + "Prefixing 0x with decimal output is defective\n" . $herecurr); + } + } + +# check for line continuations in quoted strings with odd counts of " + if ($rawline =~ /\\$/ && $rawline =~ tr/"/"/ % 2) { + WARN("LINE_CONTINUATIONS", + "Avoid line continuations in quoted strings\n" . $herecurr); + } + +# warn about #if 0 + if ($line =~ /^.\s*\#\s*if\s+0\b/) { + CHK("REDUNDANT_CODE", + "if this code is redundant consider removing it\n" . + $herecurr); + } + +# check for needless "if (<foo>) fn(<foo>)" uses + if ($prevline =~ /\bif\s*\(\s*($Lval)\s*\)/) { + my $tested = quotemeta($1); + my $expr = '\s*\(\s*' . $tested . '\s*\)\s*;'; + if ($line =~ /\b(kfree|usb_free_urb|debugfs_remove(?:_recursive)?|(?:kmem_cache|mempool|dma_pool)_destroy)$expr/) { + my $func = $1; + if (WARN('NEEDLESS_IF', + "$func(NULL) is safe and this check is probably not required\n" . $hereprev) && + $fix) { + my $do_fix = 1; + my $leading_tabs = ""; + my $new_leading_tabs = ""; + if ($lines[$linenr - 2] =~ /^\+(\t*)if\s*\(\s*$tested\s*\)\s*$/) { + $leading_tabs = $1; + } else { + $do_fix = 0; + } + if ($lines[$linenr - 1] =~ /^\+(\t+)$func\s*\(\s*$tested\s*\)\s*;\s*$/) { + $new_leading_tabs = $1; + if (length($leading_tabs) + 1 ne length($new_leading_tabs)) { + $do_fix = 0; + } + } else { + $do_fix = 0; + } + if ($do_fix) { + fix_delete_line($fixlinenr - 1, $prevrawline); + $fixed[$fixlinenr] =~ s/^\+$new_leading_tabs/\+$leading_tabs/; + } + } + } + } + +# check for unnecessary "Out of Memory" messages + if ($line =~ /^\+.*\b$logFunctions\s*\(/ && + $prevline =~ /^[ \+]\s*if\s*\(\s*(\!\s*|NULL\s*==\s*)?($Lval)(\s*==\s*NULL\s*)?\s*\)/ && + (defined $1 || defined $3) && + $linenr > 3) { + my $testval = $2; + my $testline = $lines[$linenr - 3]; + + my ($s, $c) = ctx_statement_block($linenr - 3, $realcnt, 0); +# print("line: <$line>\nprevline: <$prevline>\ns: <$s>\nc: <$c>\n\n\n"); + + if ($s =~ /(?:^|\n)[ \+]\s*(?:$Type\s*)?\Q$testval\E\s*=\s*(?:\([^\)]*\)\s*)?\s*(?:devm_)?(?:[kv][czm]alloc(?:_node|_array)?\b|kstrdup|kmemdup|(?:dev_)?alloc_skb)/) { + WARN("OOM_MESSAGE", + "Possible unnecessary 'out of memory' message\n" . $hereprev); + } + } + +# check for logging functions with KERN_<LEVEL> + if ($line !~ /printk(?:_ratelimited|_once)?\s*\(/ && + $line =~ /\b$logFunctions\s*\(.*\b(KERN_[A-Z]+)\b/) { + my $level = $1; + if (WARN("UNNECESSARY_KERN_LEVEL", + "Possible unnecessary $level\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\s*$level\s*//; + } + } + +# check for logging continuations + if ($line =~ /\bprintk\s*\(\s*KERN_CONT\b|\bpr_cont\s*\(/) { + WARN("LOGGING_CONTINUATION", + "Avoid logging continuation uses where feasible\n" . $herecurr); + } + +# check for mask then right shift without a parentheses + if ($^V && $^V ge 5.10.0 && + $line =~ /$LvalOrFunc\s*\&\s*($LvalOrFunc)\s*>>/ && + $4 !~ /^\&/) { # $LvalOrFunc may be &foo, ignore if so + WARN("MASK_THEN_SHIFT", + "Possible precedence defect with mask then right shift - may need parentheses\n" . $herecurr); + } + +# check for pointer comparisons to NULL + if ($^V && $^V ge 5.10.0) { + while ($line =~ /\b$LvalOrFunc\s*(==|\!=)\s*NULL\b/g) { + my $val = $1; + my $equal = "!"; + $equal = "" if ($4 eq "!="); + if (CHK("COMPARISON_TO_NULL", + "Comparison to NULL could be written \"${equal}${val}\"\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\b\Q$val\E\s*(?:==|\!=)\s*NULL\b/$equal$val/; + } + } + } + +# check for bad placement of section $InitAttribute (e.g.: __initdata) + if ($line =~ /(\b$InitAttribute\b)/) { + my $attr = $1; + if ($line =~ /^\+\s*static\s+(?:const\s+)?(?:$attr\s+)?($NonptrTypeWithAttr)\s+(?:$attr\s+)?($Ident(?:\[[^]]*\])?)\s*[=;]/) { + my $ptr = $1; + my $var = $2; + if ((($ptr =~ /\b(union|struct)\s+$attr\b/ && + ERROR("MISPLACED_INIT", + "$attr should be placed after $var\n" . $herecurr)) || + ($ptr !~ /\b(union|struct)\s+$attr\b/ && + WARN("MISPLACED_INIT", + "$attr should be placed after $var\n" . $herecurr))) && + $fix) { + $fixed[$fixlinenr] =~ s/(\bstatic\s+(?:const\s+)?)(?:$attr\s+)?($NonptrTypeWithAttr)\s+(?:$attr\s+)?($Ident(?:\[[^]]*\])?)\s*([=;])\s*/"$1" . trim(string_find_replace($2, "\\s*$attr\\s*", " ")) . " " . trim(string_find_replace($3, "\\s*$attr\\s*", "")) . " $attr" . ("$4" eq ";" ? ";" : " = ")/e; + } + } + } + +# check for $InitAttributeData (ie: __initdata) with const + if ($line =~ /\bconst\b/ && $line =~ /($InitAttributeData)/) { + my $attr = $1; + $attr =~ /($InitAttributePrefix)(.*)/; + my $attr_prefix = $1; + my $attr_type = $2; + if (ERROR("INIT_ATTRIBUTE", + "Use of const init definition must use ${attr_prefix}initconst\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/$InitAttributeData/${attr_prefix}initconst/; + } + } + +# check for $InitAttributeConst (ie: __initconst) without const + if ($line !~ /\bconst\b/ && $line =~ /($InitAttributeConst)/) { + my $attr = $1; + if (ERROR("INIT_ATTRIBUTE", + "Use of $attr requires a separate use of const\n" . $herecurr) && + $fix) { + my $lead = $fixed[$fixlinenr] =~ + /(^\+\s*(?:static\s+))/; + $lead = rtrim($1); + $lead = "$lead " if ($lead !~ /^\+$/); + $lead = "${lead}const "; + $fixed[$fixlinenr] =~ s/(^\+\s*(?:static\s+))/$lead/; + } + } + +# check for __read_mostly with const non-pointer (should just be const) + if ($line =~ /\b__read_mostly\b/ && + $line =~ /($Type)\s*$Ident/ && $1 !~ /\*\s*$/ && $1 =~ /\bconst\b/) { + if (ERROR("CONST_READ_MOSTLY", + "Invalid use of __read_mostly with const type\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\s+__read_mostly\b//; + } + } + +# don't use __constant_<foo> functions outside of include/uapi/ + if ($realfile !~ m@^include/uapi/@ && + $line =~ /(__constant_(?:htons|ntohs|[bl]e(?:16|32|64)_to_cpu|cpu_to_[bl]e(?:16|32|64)))\s*\(/) { + my $constant_func = $1; + my $func = $constant_func; + $func =~ s/^__constant_//; + if (WARN("CONSTANT_CONVERSION", + "$constant_func should be $func\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\b$constant_func\b/$func/g; + } + } + +# prefer usleep_range over udelay + if ($line =~ /\budelay\s*\(\s*(\d+)\s*\)/) { + my $delay = $1; + # ignore udelay's < 10, however + if (! ($delay < 10) ) { + CHK("USLEEP_RANGE", + "usleep_range is preferred over udelay; see Documentation/timers/timers-howto.txt\n" . $herecurr); + } + if ($delay > 2000) { + WARN("LONG_UDELAY", + "long udelay - prefer mdelay; see arch/arm/include/asm/delay.h\n" . $herecurr); + } + } + +# warn about unexpectedly long msleep's + if ($line =~ /\bmsleep\s*\((\d+)\);/) { + if ($1 < 20) { + WARN("MSLEEP", + "msleep < 20ms can sleep for up to 20ms; see Documentation/timers/timers-howto.txt\n" . $herecurr); + } + } + +# check for comparisons of jiffies + if ($line =~ /\bjiffies\s*$Compare|$Compare\s*jiffies\b/) { + WARN("JIFFIES_COMPARISON", + "Comparing jiffies is almost always wrong; prefer time_after, time_before and friends\n" . $herecurr); + } + +# check for comparisons of get_jiffies_64() + if ($line =~ /\bget_jiffies_64\s*\(\s*\)\s*$Compare|$Compare\s*get_jiffies_64\s*\(\s*\)/) { + WARN("JIFFIES_COMPARISON", + "Comparing get_jiffies_64() is almost always wrong; prefer time_after64, time_before64 and friends\n" . $herecurr); + } + +# warn about #ifdefs in C files +# if ($line =~ /^.\s*\#\s*if(|n)def/ && ($realfile =~ /\.c$/)) { +# print "#ifdef in C files should be avoided\n"; +# print "$herecurr"; +# $clean = 0; +# } + +# warn about spacing in #ifdefs + if ($line =~ /^.\s*\#\s*(ifdef|ifndef|elif)\s\s+/) { + if (ERROR("SPACING", + "exactly one space required after that #$1\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/^(.\s*\#\s*(ifdef|ifndef|elif))\s{2,}/$1 /; + } + + } + +# check for spinlock_t definitions without a comment. + if ($line =~ /^.\s*(struct\s+mutex|spinlock_t)\s+\S+;/ || + $line =~ /^.\s*(DEFINE_MUTEX)\s*\(/) { + my $which = $1; + if (!ctx_has_comment($first_line, $linenr)) { + CHK("UNCOMMENTED_DEFINITION", + "$1 definition without comment\n" . $herecurr); + } + } +# check for memory barriers without a comment. + + my $barriers = qr{ + mb| + rmb| + wmb| + read_barrier_depends + }x; + my $barrier_stems = qr{ + mb__before_atomic| + mb__after_atomic| + store_release| + load_acquire| + store_mb| + (?:$barriers) + }x; + my $all_barriers = qr{ + (?:$barriers)| + smp_(?:$barrier_stems)| + virt_(?:$barrier_stems) + }x; + + if ($line =~ /\b(?:$all_barriers)\s*\(/) { + if (!ctx_has_comment($first_line, $linenr)) { + WARN("MEMORY_BARRIER", + "memory barrier without comment\n" . $herecurr); + } + } + + my $underscore_smp_barriers = qr{__smp_(?:$barrier_stems)}x; + + if ($realfile !~ m@^include/asm-generic/@ && + $realfile !~ m@/barrier\.h$@ && + $line =~ m/\b(?:$underscore_smp_barriers)\s*\(/ && + $line !~ m/^.\s*\#\s*define\s+(?:$underscore_smp_barriers)\s*\(/) { + WARN("MEMORY_BARRIER", + "__smp memory barriers shouldn't be used outside barrier.h and asm-generic\n" . $herecurr); + } + +# check for waitqueue_active without a comment. + if ($line =~ /\bwaitqueue_active\s*\(/) { + if (!ctx_has_comment($first_line, $linenr)) { + WARN("WAITQUEUE_ACTIVE", + "waitqueue_active without comment\n" . $herecurr); + } + } + +# check of hardware specific defines + if ($line =~ m@^.\s*\#\s*if.*\b(__i386__|__powerpc64__|__sun__|__s390x__)\b@ && $realfile !~ m@include/asm-@) { + CHK("ARCH_DEFINES", + "architecture specific defines should be avoided\n" . $herecurr); + } + +# check that the storage class is not after a type + if ($line =~ /\b($Type)\s+($Storage)\b/) { + WARN("STORAGE_CLASS", + "storage class '$2' should be located before type '$1'\n" . $herecurr); + } +# Check that the storage class is at the beginning of a declaration + if ($line =~ /\b$Storage\b/ && + $line !~ /^.\s*$Storage/ && + $line =~ /^.\s*(.+?)\$Storage\s/ && + $1 !~ /[\,\)]\s*$/) { + WARN("STORAGE_CLASS", + "storage class should be at the beginning of the declaration\n" . $herecurr); + } + +# check the location of the inline attribute, that it is between +# storage class and type. + if ($line =~ /\b$Type\s+$Inline\b/ || + $line =~ /\b$Inline\s+$Storage\b/) { + ERROR("INLINE_LOCATION", + "inline keyword should sit between storage class and type\n" . $herecurr); + } + +# Check for __inline__ and __inline, prefer inline + if ($realfile !~ m@\binclude/uapi/@ && + $line =~ /\b(__inline__|__inline)\b/) { + if (WARN("INLINE", + "plain inline is preferred over $1\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\b(__inline__|__inline)\b/inline/; + + } + } + +# Check for __attribute__ packed, prefer __packed + if ($realfile !~ m@\binclude/uapi/@ && + $line =~ /\b__attribute__\s*\(\s*\(.*\bpacked\b/) { + WARN("PREFER_PACKED", + "__packed is preferred over __attribute__((packed))\n" . $herecurr); + } + +# Check for __attribute__ aligned, prefer __aligned + if ($realfile !~ m@\binclude/uapi/@ && + $line =~ /\b__attribute__\s*\(\s*\(.*aligned/) { + WARN("PREFER_ALIGNED", + "__aligned(size) is preferred over __attribute__((aligned(size)))\n" . $herecurr); + } + +# Check for __attribute__ format(printf, prefer __printf + if ($realfile !~ m@\binclude/uapi/@ && + $line =~ /\b__attribute__\s*\(\s*\(\s*format\s*\(\s*printf/) { + if (WARN("PREFER_PRINTF", + "__printf(string-index, first-to-check) is preferred over __attribute__((format(printf, string-index, first-to-check)))\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\b__attribute__\s*\(\s*\(\s*format\s*\(\s*printf\s*,\s*(.*)\)\s*\)\s*\)/"__printf(" . trim($1) . ")"/ex; + + } + } + +# Check for __attribute__ format(scanf, prefer __scanf + if ($realfile !~ m@\binclude/uapi/@ && + $line =~ /\b__attribute__\s*\(\s*\(\s*format\s*\(\s*scanf\b/) { + if (WARN("PREFER_SCANF", + "__scanf(string-index, first-to-check) is preferred over __attribute__((format(scanf, string-index, first-to-check)))\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\b__attribute__\s*\(\s*\(\s*format\s*\(\s*scanf\s*,\s*(.*)\)\s*\)\s*\)/"__scanf(" . trim($1) . ")"/ex; + } + } + +# Check for __attribute__ weak, or __weak declarations (may have link issues) + if ($^V && $^V ge 5.10.0 && + $line =~ /(?:$Declare|$DeclareMisordered)\s*$Ident\s*$balanced_parens\s*(?:$Attribute)?\s*;/ && + ($line =~ /\b__attribute__\s*\(\s*\(.*\bweak\b/ || + $line =~ /\b__weak\b/)) { + ERROR("WEAK_DECLARATION", + "Using weak declarations can have unintended link defects\n" . $herecurr); + } + +# check for c99 types like uint8_t + if ($line =~ /\b($Declare)\s*$Ident\s*[=;,\[]/) { + my $type = $1; + if ($type =~ /\b($typeC99Typedefs)\b/) { + $type = $1; + my $kernel_type = 'u'; + $kernel_type = 's' if ($type =~ /^_*[si]/); + $type =~ /(\d+)/; + $kernel_type .= $1.'_t'; + WARN("PREFER_KERNEL_TYPES", + "Prefer kernel type '$kernel_type' over '$type'\n" . $herecurr) + } + } + +# check for cast of C90 native int or longer types constants + if ($line =~ /(\(\s*$C90_int_types\s*\)\s*)($Constant)\b/) { + my $cast = $1; + my $const = $2; + if (WARN("TYPECAST_INT_CONSTANT", + "Unnecessary typecast of c90 int constant\n" . $herecurr) && + $fix) { + my $suffix = ""; + my $newconst = $const; + $newconst =~ s/${Int_type}$//; + $suffix .= 'U' if ($cast =~ /\bunsigned\b/); + if ($cast =~ /\blong\s+long\b/) { + $suffix .= 'LL'; + } elsif ($cast =~ /\blong\b/) { + $suffix .= 'L'; + } + $fixed[$fixlinenr] =~ s/\Q$cast\E$const\b/$newconst$suffix/; + } + } + +# check for sizeof(&) + if ($line =~ /\bsizeof\s*\(\s*\&/) { + WARN("SIZEOF_ADDRESS", + "sizeof(& should be avoided\n" . $herecurr); + } + +# check for sizeof without parenthesis + if ($line =~ /\bsizeof\s+((?:\*\s*|)$Lval|$Type(?:\s+$Lval|))/) { + if (WARN("SIZEOF_PARENTHESIS", + "sizeof $1 should be sizeof($1)\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\bsizeof\s+((?:\*\s*|)$Lval|$Type(?:\s+$Lval|))/"sizeof(" . trim($1) . ")"/ex; + } + } + +# check for struct spinlock declarations + if ($line =~ /^.\s*\bstruct\s+spinlock\s+\w+\s*;/) { + WARN("USE_SPINLOCK_T", + "struct spinlock should be spinlock_t\n" . $herecurr); + } + +# check for seq_printf uses that could be seq_puts + if ($sline =~ /\bseq_printf\s*\(.*"\s*\)\s*;\s*$/) { + my $fmt = get_quoted_string($line, $rawline); + $fmt =~ s/%%//g; + if ($fmt !~ /%/) { + if (WARN("PREFER_SEQ_PUTS", + "Prefer seq_puts to seq_printf\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\bseq_printf\b/seq_puts/; + } + } + } + + # check for vsprintf extension %p<foo> misuses + if ($^V && $^V ge 5.10.0 && + defined $stat && + $stat =~ /^\+(?![^\{]*\{\s*).*\b(\w+)\s*\(.*$String\s*,/s && + $1 !~ /^_*volatile_*$/) { + my $bad_extension = ""; + my $lc = $stat =~ tr@\n@@; + $lc = $lc + $linenr; + for (my $count = $linenr; $count <= $lc; $count++) { + my $fmt = get_quoted_string($lines[$count - 1], raw_line($count, 0)); + $fmt =~ s/%%//g; + if ($fmt =~ /(\%[\*\d\.]*p(?![\WFfSsBKRraEhMmIiUDdgVCbGNO]).)/) { + $bad_extension = $1; + last; + } + } + if ($bad_extension ne "") { + my $stat_real = raw_line($linenr, 0); + for (my $count = $linenr + 1; $count <= $lc; $count++) { + $stat_real = $stat_real . "\n" . raw_line($count, 0); + } + WARN("VSPRINTF_POINTER_EXTENSION", + "Invalid vsprintf pointer extension '$bad_extension'\n" . "$here\n$stat_real\n"); + } + } + +# Check for misused memsets + if ($^V && $^V ge 5.10.0 && + defined $stat && + $stat =~ /^\+(?:.*?)\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*$FuncArg\s*\)/) { + + my $ms_addr = $2; + my $ms_val = $7; + my $ms_size = $12; + + if ($ms_size =~ /^(0x|)0$/i) { + ERROR("MEMSET", + "memset to 0's uses 0 as the 2nd argument, not the 3rd\n" . "$here\n$stat\n"); + } elsif ($ms_size =~ /^(0x|)1$/i) { + WARN("MEMSET", + "single byte memset is suspicious. Swapped 2nd/3rd argument?\n" . "$here\n$stat\n"); + } + } + +# Check for memcpy(foo, bar, ETH_ALEN) that could be ether_addr_copy(foo, bar) +# if ($^V && $^V ge 5.10.0 && +# defined $stat && +# $stat =~ /^\+(?:.*?)\bmemcpy\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) { +# if (WARN("PREFER_ETHER_ADDR_COPY", +# "Prefer ether_addr_copy() over memcpy() if the Ethernet addresses are __aligned(2)\n" . "$here\n$stat\n") && +# $fix) { +# $fixed[$fixlinenr] =~ s/\bmemcpy\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/ether_addr_copy($2, $7)/; +# } +# } + +# Check for memcmp(foo, bar, ETH_ALEN) that could be ether_addr_equal*(foo, bar) +# if ($^V && $^V ge 5.10.0 && +# defined $stat && +# $stat =~ /^\+(?:.*?)\bmemcmp\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) { +# WARN("PREFER_ETHER_ADDR_EQUAL", +# "Prefer ether_addr_equal() or ether_addr_equal_unaligned() over memcmp()\n" . "$here\n$stat\n") +# } + +# check for memset(foo, 0x0, ETH_ALEN) that could be eth_zero_addr +# check for memset(foo, 0xFF, ETH_ALEN) that could be eth_broadcast_addr +# if ($^V && $^V ge 5.10.0 && +# defined $stat && +# $stat =~ /^\+(?:.*?)\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) { +# +# my $ms_val = $7; +# +# if ($ms_val =~ /^(?:0x|)0+$/i) { +# if (WARN("PREFER_ETH_ZERO_ADDR", +# "Prefer eth_zero_addr over memset()\n" . "$here\n$stat\n") && +# $fix) { +# $fixed[$fixlinenr] =~ s/\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*,\s*ETH_ALEN\s*\)/eth_zero_addr($2)/; +# } +# } elsif ($ms_val =~ /^(?:0xff|255)$/i) { +# if (WARN("PREFER_ETH_BROADCAST_ADDR", +# "Prefer eth_broadcast_addr() over memset()\n" . "$here\n$stat\n") && +# $fix) { +# $fixed[$fixlinenr] =~ s/\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*,\s*ETH_ALEN\s*\)/eth_broadcast_addr($2)/; +# } +# } +# } + +# typecasts on min/max could be min_t/max_t + if ($^V && $^V ge 5.10.0 && + defined $stat && + $stat =~ /^\+(?:.*?)\b(min|max)\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\)/) { + if (defined $2 || defined $7) { + my $call = $1; + my $cast1 = deparenthesize($2); + my $arg1 = $3; + my $cast2 = deparenthesize($7); + my $arg2 = $8; + my $cast; + + if ($cast1 ne "" && $cast2 ne "" && $cast1 ne $cast2) { + $cast = "$cast1 or $cast2"; + } elsif ($cast1 ne "") { + $cast = $cast1; + } else { + $cast = $cast2; + } + WARN("MINMAX", + "$call() should probably be ${call}_t($cast, $arg1, $arg2)\n" . "$here\n$stat\n"); + } + } + +# check usleep_range arguments + if ($^V && $^V ge 5.10.0 && + defined $stat && + $stat =~ /^\+(?:.*?)\busleep_range\s*\(\s*($FuncArg)\s*,\s*($FuncArg)\s*\)/) { + my $min = $1; + my $max = $7; + if ($min eq $max) { + WARN("USLEEP_RANGE", + "usleep_range should not use min == max args; see Documentation/timers/timers-howto.txt\n" . "$here\n$stat\n"); + } elsif ($min =~ /^\d+$/ && $max =~ /^\d+$/ && + $min > $max) { + WARN("USLEEP_RANGE", + "usleep_range args reversed, use min then max; see Documentation/timers/timers-howto.txt\n" . "$here\n$stat\n"); + } + } + +# check for naked sscanf + if ($^V && $^V ge 5.10.0 && + defined $stat && + $line =~ /\bsscanf\b/ && + ($stat !~ /$Ident\s*=\s*sscanf\s*$balanced_parens/ && + $stat !~ /\bsscanf\s*$balanced_parens\s*(?:$Compare)/ && + $stat !~ /(?:$Compare)\s*\bsscanf\s*$balanced_parens/)) { + my $lc = $stat =~ tr@\n@@; + $lc = $lc + $linenr; + my $stat_real = raw_line($linenr, 0); + for (my $count = $linenr + 1; $count <= $lc; $count++) { + $stat_real = $stat_real . "\n" . raw_line($count, 0); + } + WARN("NAKED_SSCANF", + "unchecked sscanf return value\n" . "$here\n$stat_real\n"); + } + +# check for simple sscanf that should be kstrto<foo> + if ($^V && $^V ge 5.10.0 && + defined $stat && + $line =~ /\bsscanf\b/) { + my $lc = $stat =~ tr@\n@@; + $lc = $lc + $linenr; + my $stat_real = raw_line($linenr, 0); + for (my $count = $linenr + 1; $count <= $lc; $count++) { + $stat_real = $stat_real . "\n" . raw_line($count, 0); + } + if ($stat_real =~ /\bsscanf\b\s*\(\s*$FuncArg\s*,\s*("[^"]+")/) { + my $format = $6; + my $count = $format =~ tr@%@%@; + if ($count == 1 && + $format =~ /^"\%(?i:ll[udxi]|[udxi]ll|ll|[hl]h?[udxi]|[udxi][hl]h?|[hl]h?|[udxi])"$/) { + WARN("SSCANF_TO_KSTRTO", + "Prefer kstrto<type> to single variable sscanf\n" . "$here\n$stat_real\n"); + } + } + } + +# check for new externs in .h files. + if ($realfile =~ /\.h$/ && + $line =~ /^\+\s*(extern\s+)$Type\s*$Ident\s*\(/s) { + if (CHK("AVOID_EXTERNS", + "extern prototypes should be avoided in .h files\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/(.*)\bextern\b\s*(.*)/$1$2/; + } + } + +# check for new externs in .c files. + if ($realfile =~ /\.c$/ && defined $stat && + $stat =~ /^.\s*(?:extern\s+)?$Type\s+($Ident)(\s*)\(/s) + { + my $function_name = $1; + my $paren_space = $2; + + my $s = $stat; + if (defined $cond) { + substr($s, 0, length($cond), ''); + } + if ($s =~ /^\s*;/ && + $function_name ne 'uninitialized_var') + { + WARN("AVOID_EXTERNS", + "externs should be avoided in .c files\n" . $herecurr); + } + + if ($paren_space =~ /\n/) { + WARN("FUNCTION_ARGUMENTS", + "arguments for function declarations should follow identifier\n" . $herecurr); + } + + } elsif ($realfile =~ /\.c$/ && defined $stat && + $stat =~ /^.\s*extern\s+/) + { + WARN("AVOID_EXTERNS", + "externs should be avoided in .c files\n" . $herecurr); + } + +# check for function declarations that have arguments without identifier names + if (defined $stat && + $stat =~ /^.\s*(?:extern\s+)?$Type\s*$Ident\s*\(\s*([^{]+)\s*\)\s*;/s && + $1 ne "void") { + my $args = trim($1); + while ($args =~ m/\s*($Type\s*(?:$Ident|\(\s*\*\s*$Ident?\s*\)\s*$balanced_parens)?)/g) { + my $arg = trim($1); + if ($arg =~ /^$Type$/ && $arg !~ /enum\s+$Ident$/) { + WARN("FUNCTION_ARGUMENTS", + "function definition argument '$arg' should also have an identifier name\n" . $herecurr); + } + } + } + +# check for function definitions + if ($^V && $^V ge 5.10.0 && + defined $stat && + $stat =~ /^.\s*(?:$Storage\s+)?$Type\s*($Ident)\s*$balanced_parens\s*{/s) { + $context_function = $1; + +# check for multiline function definition with misplaced open brace + my $ok = 0; + my $cnt = statement_rawlines($stat); + my $herectx = $here . "\n"; + for (my $n = 0; $n < $cnt; $n++) { + my $rl = raw_line($linenr, $n); + $herectx .= $rl . "\n"; + $ok = 1 if ($rl =~ /^[ \+]\{/); + $ok = 1 if ($rl =~ /\{/ && $n == 0); + last if $rl =~ /^[ \+].*\{/; + } + if (!$ok) { + ERROR("OPEN_BRACE", + "open brace '{' following function definitions go on the next line\n" . $herectx); + } + } + +# checks for new __setup's + if ($rawline =~ /\b__setup\("([^"]*)"/) { + my $name = $1; + + if (!grep(/$name/, @setup_docs)) { + CHK("UNDOCUMENTED_SETUP", + "__setup appears un-documented -- check Documentation/admin-guide/kernel-parameters.rst\n" . $herecurr); + } + } + +# check for pointless casting of kmalloc return + if ($line =~ /\*\s*\)\s*[kv][czm]alloc(_node){0,1}\b/) { + WARN("UNNECESSARY_CASTS", + "unnecessary cast may hide bugs, see http://c-faq.com/malloc/mallocnocast.html\n" . $herecurr); + } + +# alloc style +# p = alloc(sizeof(struct foo), ...) should be p = alloc(sizeof(*p), ...) + if ($^V && $^V ge 5.10.0 && + $line =~ /\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*([kv][mz]alloc(?:_node)?)\s*\(\s*(sizeof\s*\(\s*struct\s+$Lval\s*\))/) { + CHK("ALLOC_SIZEOF_STRUCT", + "Prefer $3(sizeof(*$1)...) over $3($4...)\n" . $herecurr); + } + +# check for k[mz]alloc with multiplies that could be kmalloc_array/kcalloc + if ($^V && $^V ge 5.10.0 && + defined $stat && + $stat =~ /^\+\s*($Lval)\s*\=\s*(?:$balanced_parens)?\s*(k[mz]alloc)\s*\(\s*($FuncArg)\s*\*\s*($FuncArg)\s*,/) { + my $oldfunc = $3; + my $a1 = $4; + my $a2 = $10; + my $newfunc = "kmalloc_array"; + $newfunc = "kcalloc" if ($oldfunc eq "kzalloc"); + my $r1 = $a1; + my $r2 = $a2; + if ($a1 =~ /^sizeof\s*\S/) { + $r1 = $a2; + $r2 = $a1; + } + if ($r1 !~ /^sizeof\b/ && $r2 =~ /^sizeof\s*\S/ && + !($r1 =~ /^$Constant$/ || $r1 =~ /^[A-Z_][A-Z0-9_]*$/)) { + my $ctx = ''; + my $herectx = $here . "\n"; + my $cnt = statement_rawlines($stat); + for (my $n = 0; $n < $cnt; $n++) { + $herectx .= raw_line($linenr, $n) . "\n"; + } + if (WARN("ALLOC_WITH_MULTIPLY", + "Prefer $newfunc over $oldfunc with multiply\n" . $herectx) && + $cnt == 1 && + $fix) { + $fixed[$fixlinenr] =~ s/\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*(k[mz]alloc)\s*\(\s*($FuncArg)\s*\*\s*($FuncArg)/$1 . ' = ' . "$newfunc(" . trim($r1) . ', ' . trim($r2)/e; + } + } + } + +# check for krealloc arg reuse + if ($^V && $^V ge 5.10.0 && + $line =~ /\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*krealloc\s*\(\s*\1\s*,/) { + WARN("KREALLOC_ARG_REUSE", + "Reusing the krealloc arg is almost always a bug\n" . $herecurr); + } + +# check for alloc argument mismatch + if ($line =~ /\b(kcalloc|kmalloc_array)\s*\(\s*sizeof\b/) { + WARN("ALLOC_ARRAY_ARGS", + "$1 uses number as first arg, sizeof is generally wrong\n" . $herecurr); + } + +# check for multiple semicolons + if ($line =~ /;\s*;\s*$/) { + if (WARN("ONE_SEMICOLON", + "Statements terminations use 1 semicolon\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/(\s*;\s*){2,}$/;/g; + } + } + +# check for #defines like: 1 << <digit> that could be BIT(digit), it is not exported to uapi + if ($realfile !~ m@^include/uapi/@ && + $line =~ /#\s*define\s+\w+\s+\(?\s*1\s*([ulUL]*)\s*\<\<\s*(?:\d+|$Ident)\s*\)?/) { + my $ull = ""; + $ull = "_ULL" if (defined($1) && $1 =~ /ll/i); + if (CHK("BIT_MACRO", + "Prefer using the BIT$ull macro\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\(?\s*1\s*[ulUL]*\s*<<\s*(\d+|$Ident)\s*\)?/BIT${ull}($1)/; + } + } + +# check for #if defined CONFIG_<FOO> || defined CONFIG_<FOO>_MODULE + if ($line =~ /^\+\s*#\s*if\s+defined(?:\s*\(?\s*|\s+)(CONFIG_[A-Z_]+)\s*\)?\s*\|\|\s*defined(?:\s*\(?\s*|\s+)\1_MODULE\s*\)?\s*$/) { + my $config = $1; + if (WARN("PREFER_IS_ENABLED", + "Prefer IS_ENABLED(<FOO>) to CONFIG_<FOO> || CONFIG_<FOO>_MODULE\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] = "\+#if IS_ENABLED($config)"; + } + } + +# check for case / default statements not preceded by break/fallthrough/switch + if ($line =~ /^.\s*(?:case\s+(?:$Ident|$Constant)\s*|default):/) { + my $has_break = 0; + my $has_statement = 0; + my $count = 0; + my $prevline = $linenr; + while ($prevline > 1 && ($file || $count < 3) && !$has_break) { + $prevline--; + my $rline = $rawlines[$prevline - 1]; + my $fline = $lines[$prevline - 1]; + last if ($fline =~ /^\@\@/); + next if ($fline =~ /^\-/); + next if ($fline =~ /^.(?:\s*(?:case\s+(?:$Ident|$Constant)[\s$;]*|default):[\s$;]*)*$/); + $has_break = 1 if ($rline =~ /fall[\s_-]*(through|thru)/i); + next if ($fline =~ /^.[\s$;]*$/); + $has_statement = 1; + $count++; + $has_break = 1 if ($fline =~ /\bswitch\b|\b(?:break\s*;[\s$;]*$|return\b|goto\b|continue\b)/); + } + if (!$has_break && $has_statement) { + WARN("MISSING_BREAK", + "Possible switch case/default not preceded by break or fallthrough comment\n" . $herecurr); + } + } + +# check for switch/default statements without a break; + if ($^V && $^V ge 5.10.0 && + defined $stat && + $stat =~ /^\+[$;\s]*(?:case[$;\s]+\w+[$;\s]*:[$;\s]*|)*[$;\s]*\bdefault[$;\s]*:[$;\s]*;/g) { + my $ctx = ''; + my $herectx = $here . "\n"; + my $cnt = statement_rawlines($stat); + for (my $n = 0; $n < $cnt; $n++) { + $herectx .= raw_line($linenr, $n) . "\n"; + } + WARN("DEFAULT_NO_BREAK", + "switch default: should use break\n" . $herectx); + } + +# check for gcc specific __FUNCTION__ + if ($line =~ /\b__FUNCTION__\b/) { + if (WARN("USE_FUNC", + "__func__ should be used instead of gcc specific __FUNCTION__\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\b__FUNCTION__\b/__func__/g; + } + } + +# check for uses of __DATE__, __TIME__, __TIMESTAMP__ + while ($line =~ /\b(__(?:DATE|TIME|TIMESTAMP)__)\b/g) { + ERROR("DATE_TIME", + "Use of the '$1' macro makes the build non-deterministic\n" . $herecurr); + } + +# check for use of yield() + if ($line =~ /\byield\s*\(\s*\)/) { + WARN("YIELD", + "Using yield() is generally wrong. See yield() kernel-doc (sched/core.c)\n" . $herecurr); + } + +# check for comparisons against true and false + if ($line =~ /\+\s*(.*?)\b(true|false|$Lval)\s*(==|\!=)\s*(true|false|$Lval)\b(.*)$/i) { + my $lead = $1; + my $arg = $2; + my $test = $3; + my $otype = $4; + my $trail = $5; + my $op = "!"; + + ($arg, $otype) = ($otype, $arg) if ($arg =~ /^(?:true|false)$/i); + + my $type = lc($otype); + if ($type =~ /^(?:true|false)$/) { + if (("$test" eq "==" && "$type" eq "true") || + ("$test" eq "!=" && "$type" eq "false")) { + $op = ""; + } + + CHK("BOOL_COMPARISON", + "Using comparison to $otype is error prone\n" . $herecurr); + +## maybe suggesting a correct construct would better +## "Using comparison to $otype is error prone. Perhaps use '${lead}${op}${arg}${trail}'\n" . $herecurr); + + } + } + +# check for semaphores initialized locked + if ($line =~ /^.\s*sema_init.+,\W?0\W?\)/) { + WARN("CONSIDER_COMPLETION", + "consider using a completion\n" . $herecurr); + } + +# recommend kstrto* over simple_strto* and strict_strto* + if ($line =~ /\b((simple|strict)_(strto(l|ll|ul|ull)))\s*\(/) { + WARN("CONSIDER_KSTRTO", + "$1 is obsolete, use k$3 instead\n" . $herecurr); + } + +# check for __initcall(), use device_initcall() explicitly or more appropriate function please + if ($line =~ /^.\s*__initcall\s*\(/) { + WARN("USE_DEVICE_INITCALL", + "please use device_initcall() or more appropriate function instead of __initcall() (see include/linux/init.h)\n" . $herecurr); + } + +# check for various structs that are normally const (ops, kgdb, device_tree) +# and avoid what seem like struct definitions 'struct foo {' + if ($line !~ /\bconst\b/ && + $line =~ /\bstruct\s+($const_structs)\b(?!\s*\{)/) { + WARN("CONST_STRUCT", + "struct $1 should normally be const\n" . $herecurr); + } + +# use of NR_CPUS is usually wrong +# ignore definitions of NR_CPUS and usage to define arrays as likely right + if ($line =~ /\bNR_CPUS\b/ && + $line !~ /^.\s*\s*#\s*if\b.*\bNR_CPUS\b/ && + $line !~ /^.\s*\s*#\s*define\b.*\bNR_CPUS\b/ && + $line !~ /^.\s*$Declare\s.*\[[^\]]*NR_CPUS[^\]]*\]/ && + $line !~ /\[[^\]]*\.\.\.[^\]]*NR_CPUS[^\]]*\]/ && + $line !~ /\[[^\]]*NR_CPUS[^\]]*\.\.\.[^\]]*\]/) + { + WARN("NR_CPUS", + "usage of NR_CPUS is often wrong - consider using cpu_possible(), num_possible_cpus(), for_each_possible_cpu(), etc\n" . $herecurr); + } + +# Use of __ARCH_HAS_<FOO> or ARCH_HAVE_<BAR> is wrong. + if ($line =~ /\+\s*#\s*define\s+((?:__)?ARCH_(?:HAS|HAVE)\w*)\b/) { + ERROR("DEFINE_ARCH_HAS", + "#define of '$1' is wrong - use Kconfig variables or standard guards instead\n" . $herecurr); + } + +# likely/unlikely comparisons similar to "(likely(foo) > 0)" + if ($^V && $^V ge 5.10.0 && + $line =~ /\b((?:un)?likely)\s*\(\s*$FuncArg\s*\)\s*$Compare/) { + WARN("LIKELY_MISUSE", + "Using $1 should generally have parentheses around the comparison\n" . $herecurr); + } + +# whine mightly about in_atomic + if ($line =~ /\bin_atomic\s*\(/) { + if ($realfile =~ m@^drivers/@) { + ERROR("IN_ATOMIC", + "do not use in_atomic in drivers\n" . $herecurr); + } elsif ($realfile !~ m@^kernel/@) { + WARN("IN_ATOMIC", + "use of in_atomic() is incorrect outside core kernel code\n" . $herecurr); + } + } + +# whine about ACCESS_ONCE + if ($^V && $^V ge 5.10.0 && + $line =~ /\bACCESS_ONCE\s*$balanced_parens\s*(=(?!=))?\s*($FuncArg)?/) { + my $par = $1; + my $eq = $2; + my $fun = $3; + $par =~ s/^\(\s*(.*)\s*\)$/$1/; + if (defined($eq)) { + if (WARN("PREFER_WRITE_ONCE", + "Prefer WRITE_ONCE(<FOO>, <BAR>) over ACCESS_ONCE(<FOO>) = <BAR>\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\bACCESS_ONCE\s*\(\s*\Q$par\E\s*\)\s*$eq\s*\Q$fun\E/WRITE_ONCE($par, $fun)/; + } + } else { + if (WARN("PREFER_READ_ONCE", + "Prefer READ_ONCE(<FOO>) over ACCESS_ONCE(<FOO>)\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\bACCESS_ONCE\s*\(\s*\Q$par\E\s*\)/READ_ONCE($par)/; + } + } + } + +# check for mutex_trylock_recursive usage + if ($line =~ /mutex_trylock_recursive/) { + ERROR("LOCKING", + "recursive locking is bad, do not use this ever.\n" . $herecurr); + } + +# check for lockdep_set_novalidate_class + if ($line =~ /^.\s*lockdep_set_novalidate_class\s*\(/ || + $line =~ /__lockdep_no_validate__\s*\)/ ) { + if ($realfile !~ m@^kernel/lockdep@ && + $realfile !~ m@^include/linux/lockdep@ && + $realfile !~ m@^drivers/base/core@) { + ERROR("LOCKDEP", + "lockdep_no_validate class is reserved for device->mutex.\n" . $herecurr); + } + } + + if ($line =~ /debugfs_create_\w+.*\b$mode_perms_world_writable\b/ || + $line =~ /DEVICE_ATTR.*\b$mode_perms_world_writable\b/) { + WARN("EXPORTED_WORLD_WRITABLE", + "Exporting world writable files is usually an error. Consider more restrictive permissions.\n" . $herecurr); + } + +# Mode permission misuses where it seems decimal should be octal +# This uses a shortcut match to avoid unnecessary uses of a slow foreach loop + if ($^V && $^V ge 5.10.0 && + defined $stat && + $line =~ /$mode_perms_search/) { + foreach my $entry (@mode_permission_funcs) { + my $func = $entry->[0]; + my $arg_pos = $entry->[1]; + + my $lc = $stat =~ tr@\n@@; + $lc = $lc + $linenr; + my $stat_real = raw_line($linenr, 0); + for (my $count = $linenr + 1; $count <= $lc; $count++) { + $stat_real = $stat_real . "\n" . raw_line($count, 0); + } + + my $skip_args = ""; + if ($arg_pos > 1) { + $arg_pos--; + $skip_args = "(?:\\s*$FuncArg\\s*,\\s*){$arg_pos,$arg_pos}"; + } + my $test = "\\b$func\\s*\\(${skip_args}($FuncArg(?:\\|\\s*$FuncArg)*)\\s*[,\\)]"; + if ($stat =~ /$test/) { + my $val = $1; + $val = $6 if ($skip_args ne ""); + if (($val =~ /^$Int$/ && $val !~ /^$Octal$/) || + ($val =~ /^$Octal$/ && length($val) ne 4)) { + ERROR("NON_OCTAL_PERMISSIONS", + "Use 4 digit octal (0777) not decimal permissions\n" . "$here\n" . $stat_real); + } + if ($val =~ /^$Octal$/ && (oct($val) & 02)) { + ERROR("EXPORTED_WORLD_WRITABLE", + "Exporting writable files is usually an error. Consider more restrictive permissions.\n" . "$here\n" . $stat_real); + } + } + } + } + +# check for uses of S_<PERMS> that could be octal for readability + if ($line =~ /\b$mode_perms_string_search\b/) { + my $val = ""; + my $oval = ""; + my $to = 0; + my $curpos = 0; + my $lastpos = 0; + while ($line =~ /\b(($mode_perms_string_search)\b(?:\s*\|\s*)?\s*)/g) { + $curpos = pos($line); + my $match = $2; + my $omatch = $1; + last if ($lastpos > 0 && ($curpos - length($omatch) != $lastpos)); + $lastpos = $curpos; + $to |= $mode_permission_string_types{$match}; + $val .= '\s*\|\s*' if ($val ne ""); + $val .= $match; + $oval .= $omatch; + } + $oval =~ s/^\s*\|\s*//; + $oval =~ s/\s*\|\s*$//; + my $octal = sprintf("%04o", $to); + if (WARN("SYMBOLIC_PERMS", + "Symbolic permissions '$oval' are not preferred. Consider using octal permissions '$octal'.\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/$val/$octal/; + } + } + +# validate content of MODULE_LICENSE against list from include/linux/module.h + if ($line =~ /\bMODULE_LICENSE\s*\(\s*($String)\s*\)/) { + my $extracted_string = get_quoted_string($line, $rawline); + my $valid_licenses = qr{ + GPL| + GPL\ v2| + GPL\ and\ additional\ rights| + Dual\ BSD/GPL| + Dual\ MIT/GPL| + Dual\ MPL/GPL| + Proprietary + }x; + if ($extracted_string !~ /^"(?:$valid_licenses)"$/x) { + WARN("MODULE_LICENSE", + "unknown module license " . $extracted_string . "\n" . $herecurr); + } + } + } + + # If we have no input at all, then there is nothing to report on + # so just keep quiet. + if ($#rawlines == -1) { + exit(0); + } + + # In mailback mode only produce a report in the negative, for + # things that appear to be patches. + if ($mailback && ($clean == 1 || !$is_patch)) { + exit(0); + } + + # This is not a patch, and we are are in 'no-patch' mode so + # just keep quiet. + if (!$chk_patch && !$is_patch) { + exit(0); + } + + if (!$is_patch && $file !~ /cover-letter\.patch$/) { + ERROR("NOT_UNIFIED_DIFF", + "Does not appear to be a unified-diff format patch\n"); + } + if ($is_patch && $has_commit_log && $chk_signoff && $signoff == 0) { + ERROR("MISSING_SIGN_OFF", + "Missing Signed-off-by: line(s)\n"); + } + + print report_dump(); + if ($summary && !($clean == 1 && $quiet == 1)) { + print "$filename " if ($summary_file); + print "total: $cnt_error errors, $cnt_warn warnings, " . + (($check)? "$cnt_chk checks, " : "") . + "$cnt_lines lines checked\n"; + } + + if ($quiet == 0) { + # If there were any defects found and not already fixing them + if (!$clean and !$fix) { + print << "EOM" + +NOTE: For some of the reported defects, checkpatch may be able to + mechanically convert to the typical style using --fix or --fix-inplace. +EOM + } + # If there were whitespace errors which cleanpatch can fix + # then suggest that. + if ($rpt_cleaners) { + $rpt_cleaners = 0; + print << "EOM" + +NOTE: Whitespace errors detected. + You may wish to use scripts/cleanpatch or scripts/cleanfile +EOM + } + } + + if ($clean == 0 && $fix && + ("@rawlines" ne "@fixed" || + $#fixed_inserted >= 0 || $#fixed_deleted >= 0)) { + my $newfile = $filename; + $newfile .= ".EXPERIMENTAL-checkpatch-fixes" if (!$fix_inplace); + my $linecount = 0; + my $f; + + @fixed = fix_inserted_deleted_lines(\@fixed, \@fixed_inserted, \@fixed_deleted); + + open($f, '>', $newfile) + or die "$P: Can't open $newfile for write\n"; + foreach my $fixed_line (@fixed) { + $linecount++; + if ($file) { + if ($linecount > 3) { + $fixed_line =~ s/^\+//; + print $f $fixed_line . "\n"; + } + } else { + print $f $fixed_line . "\n"; + } + } + close($f); + + if (!$quiet) { + print << "EOM"; + +Wrote EXPERIMENTAL --fix correction(s) to '$newfile' + +Do _NOT_ trust the results written to this file. +Do _NOT_ submit these changes without inspecting them for correctness. + +This EXPERIMENTAL file is simply a convenience to help rewrite patches. +No warranties, expressed or implied... +EOM + } + } + + if ($quiet == 0) { + print "\n"; + if ($clean == 1) { + print "$vname has no obvious style problems and is ready for submission.\n"; + } else { + print "$vname has style problems, please review.\n"; + } + } + return $clean; +} diff --git a/stm32/system/Middlewares/OpenAMP/libmetal/scripts/ci/check_compliance.py b/stm32/system/Middlewares/OpenAMP/libmetal/scripts/ci/check_compliance.py old mode 100755 new mode 100644 diff --git a/stm32/system/Middlewares/OpenAMP/libmetal/scripts/do_checkpatch.sh b/stm32/system/Middlewares/OpenAMP/libmetal/scripts/do_checkpatch.sh old mode 100755 new mode 100644 diff --git a/stm32/system/Middlewares/OpenAMP/open-amp/.checkpatch.conf b/stm32/system/Middlewares/OpenAMP/open-amp/.checkpatch.conf index 0fb11596..0c4fe8d5 100644 --- a/stm32/system/Middlewares/OpenAMP/open-amp/.checkpatch.conf +++ b/stm32/system/Middlewares/OpenAMP/open-amp/.checkpatch.conf @@ -1,22 +1,22 @@ ---emacs ---no-tree ---summary-file ---show-types ---max-line-length=80 ---min-conf-desc-length=1 - ---ignore BRACES ---ignore PRINTK_WITHOUT_KERN_LEVEL ---ignore SPLIT_STRING ---ignore VOLATILE ---ignore CONFIG_EXPERIMENTAL ---ignore AVOID_EXTERNS ---ignore NETWORKING_BLOCK_COMMENT_STYLE ---ignore DATE_TIME ---ignore MINMAX ---ignore CONST_STRUCT ---ignore FILE_PATH_CHANGES ---ignore BIT_MACRO ---ignore PREFER_KERNEL_TYPES ---ignore NEW_TYPEDEFS +--emacs +--no-tree +--summary-file +--show-types +--max-line-length=80 +--min-conf-desc-length=1 + +--ignore BRACES +--ignore PRINTK_WITHOUT_KERN_LEVEL +--ignore SPLIT_STRING +--ignore VOLATILE +--ignore CONFIG_EXPERIMENTAL +--ignore AVOID_EXTERNS +--ignore NETWORKING_BLOCK_COMMENT_STYLE +--ignore DATE_TIME +--ignore MINMAX +--ignore CONST_STRUCT +--ignore FILE_PATH_CHANGES +--ignore BIT_MACRO +--ignore PREFER_KERNEL_TYPES +--ignore NEW_TYPEDEFS --ignore ARRAY_SIZE \ No newline at end of file diff --git a/stm32/system/Middlewares/OpenAMP/open-amp/.gitlint b/stm32/system/Middlewares/OpenAMP/open-amp/.gitlint index 102cb75b..92d9e1b8 100644 --- a/stm32/system/Middlewares/OpenAMP/open-amp/.gitlint +++ b/stm32/system/Middlewares/OpenAMP/open-amp/.gitlint @@ -1,99 +1,99 @@ -# All these sections are optional, edit this file as you like. -[general] -# Ignore certain rules, you can reference them by their id or by their full name -ignore=title-trailing-punctuation, T3, title-max-length, T1, body-hard-tab, B1, B3 - -# verbosity should be a value between 1 and 3, the commandline -v flags take precedence over this -verbosity = 3 - -# By default gitlint will ignore merge commits. Set to 'false' to disable. -ignore-merge-commits=true - -# By default gitlint will ignore fixup commits. Set to 'false' to disable. -# ignore-fixup-commits=false - -# By default gitlint will ignore squash commits. Set to 'false' to disable. -# ignore-squash-commits=true - -# Ignore any data send to gitlint via stdin -# ignore-stdin=true - -# Enable debug mode (prints more output). Disabled by default. -debug=true - -# Enable community contributed rules -# See http://jorisroovers.github.io/gitlint/contrib_rules for details -# contrib=contrib-title-conventional-commits,CC1 - -# Set the extra-path where gitlint will search for user defined rules -# See http://jorisroovers.github.io/gitlint/user_defined_rules for details -extra-path=scripts/gitlint - -[title-max-length] -line-length=75 - -[body-min-line-count] -min-line-count=1 - -[body-max-line-count] -max-line-count=200 - -[title-must-not-contain-word] -# Comma-separated list of words that should not occur in the title. Matching is case -# insensitive. It's fine if the keyword occurs as part of a larger word (so "WIPING" -# will not cause a violation, but "WIP: my title" will. -words=wip - -# [title-match-regex] -# python like regex (https://docs.python.org/2/library/re.html) that the -# commit-msg title must be matched to. -# Note that the regex can contradict with other rules if not used correctly -# (e.g. title-must-not-contain-word). -# regex=^US[0-9]* - -[body-max-line-length] -# B1 = body-max-line-length -line-length=80 - -[body-min-length] -min-length=3 - -[body-is-missing] -# Whether to ignore this rule on merge commits (which typically only have a title) -# default = True -ignore-merge-commits=false - -# [body-changed-file-mention] -# List of files that need to be explicitly mentioned in the body when they are changed -# This is useful for when developers often erroneously edit certain files or git submodules. -# By specifying this rule, developers can only change the file when they explicitly reference -# it in the commit message. -# files=gitlint/rules.py,README.md - -# [author-valid-email] -# python like regex (https://docs.python.org/2/library/re.html) that the -# commit author email address should be matched to -# For example, use the following regex if you only want to allow email addresses from foo.com -# regex=[^@]+@foo.com - -# [ignore-by-title] -# Ignore certain rules for commits of which the title matches a regex -# E.g. Match commit titles that start with "Release" -# regex=^Release(.*) -# -# Ignore certain rules, you can reference them by their id or by their full name -# Use 'all' to ignore all rules -# ignore=T1,body-min-length - -# [ignore-by-body] -# Ignore certain rules for commits of which the body has a line that matches a regex -# E.g. Match bodies that have a line that that contain "release" -# regex=(.*)release(.*) -# -# Ignore certain rules, you can reference them by their id or by their full name -# Use 'all' to ignore all rules -# ignore=T1,body-min-length - -# [contrib-title-conventional-commits] -# Specify allowed commit types. For details see: https://www.conventionalcommits.org/ +# All these sections are optional, edit this file as you like. +[general] +# Ignore certain rules, you can reference them by their id or by their full name +ignore=title-trailing-punctuation, T3, title-max-length, T1, body-hard-tab, B1, B3 + +# verbosity should be a value between 1 and 3, the commandline -v flags take precedence over this +verbosity = 3 + +# By default gitlint will ignore merge commits. Set to 'false' to disable. +ignore-merge-commits=true + +# By default gitlint will ignore fixup commits. Set to 'false' to disable. +# ignore-fixup-commits=false + +# By default gitlint will ignore squash commits. Set to 'false' to disable. +# ignore-squash-commits=true + +# Ignore any data send to gitlint via stdin +# ignore-stdin=true + +# Enable debug mode (prints more output). Disabled by default. +debug=true + +# Enable community contributed rules +# See http://jorisroovers.github.io/gitlint/contrib_rules for details +# contrib=contrib-title-conventional-commits,CC1 + +# Set the extra-path where gitlint will search for user defined rules +# See http://jorisroovers.github.io/gitlint/user_defined_rules for details +extra-path=scripts/gitlint + +[title-max-length] +line-length=75 + +[body-min-line-count] +min-line-count=1 + +[body-max-line-count] +max-line-count=200 + +[title-must-not-contain-word] +# Comma-separated list of words that should not occur in the title. Matching is case +# insensitive. It's fine if the keyword occurs as part of a larger word (so "WIPING" +# will not cause a violation, but "WIP: my title" will. +words=wip + +# [title-match-regex] +# python like regex (https://docs.python.org/2/library/re.html) that the +# commit-msg title must be matched to. +# Note that the regex can contradict with other rules if not used correctly +# (e.g. title-must-not-contain-word). +# regex=^US[0-9]* + +[body-max-line-length] +# B1 = body-max-line-length +line-length=80 + +[body-min-length] +min-length=3 + +[body-is-missing] +# Whether to ignore this rule on merge commits (which typically only have a title) +# default = True +ignore-merge-commits=false + +# [body-changed-file-mention] +# List of files that need to be explicitly mentioned in the body when they are changed +# This is useful for when developers often erroneously edit certain files or git submodules. +# By specifying this rule, developers can only change the file when they explicitly reference +# it in the commit message. +# files=gitlint/rules.py,README.md + +# [author-valid-email] +# python like regex (https://docs.python.org/2/library/re.html) that the +# commit author email address should be matched to +# For example, use the following regex if you only want to allow email addresses from foo.com +# regex=[^@]+@foo.com + +# [ignore-by-title] +# Ignore certain rules for commits of which the title matches a regex +# E.g. Match commit titles that start with "Release" +# regex=^Release(.*) +# +# Ignore certain rules, you can reference them by their id or by their full name +# Use 'all' to ignore all rules +# ignore=T1,body-min-length + +# [ignore-by-body] +# Ignore certain rules for commits of which the body has a line that matches a regex +# E.g. Match bodies that have a line that that contain "release" +# regex=(.*)release(.*) +# +# Ignore certain rules, you can reference them by their id or by their full name +# Use 'all' to ignore all rules +# ignore=T1,body-min-length + +# [contrib-title-conventional-commits] +# Specify allowed commit types. For details see: https://www.conventionalcommits.org/ # types = bugfix,user-story,epic \ No newline at end of file diff --git a/stm32/system/Middlewares/OpenAMP/open-amp/VERSION b/stm32/system/Middlewares/OpenAMP/open-amp/VERSION index d2fe03a5..61df8f44 100644 --- a/stm32/system/Middlewares/OpenAMP/open-amp/VERSION +++ b/stm32/system/Middlewares/OpenAMP/open-amp/VERSION @@ -1,3 +1,3 @@ -VERSION_MAJOR = 1 -VERSION_MINOR = 1 -VERSION_PATCH = 0 +VERSION_MAJOR = 1 +VERSION_MINOR = 1 +VERSION_PATCH = 0 diff --git a/stm32/system/Middlewares/OpenAMP/open-amp/apps/examples/rpc_demo/rpc_demo.c b/stm32/system/Middlewares/OpenAMP/open-amp/apps/examples/rpc_demo/rpc_demo.c old mode 100644 new mode 100755 diff --git a/stm32/system/Middlewares/OpenAMP/open-amp/apps/examples/rpc_demo/rpc_demod.c b/stm32/system/Middlewares/OpenAMP/open-amp/apps/examples/rpc_demo/rpc_demod.c old mode 100644 new mode 100755 diff --git a/stm32/system/Middlewares/OpenAMP/open-amp/scripts/checkpatch.pl b/stm32/system/Middlewares/OpenAMP/open-amp/scripts/checkpatch.pl old mode 100755 new mode 100644 index 9190fa67..954c4626 --- a/stm32/system/Middlewares/OpenAMP/open-amp/scripts/checkpatch.pl +++ b/stm32/system/Middlewares/OpenAMP/open-amp/scripts/checkpatch.pl @@ -1,6494 +1,6494 @@ -#!/usr/bin/env perl -# (c) 2001, Dave Jones. (the file handling bit) -# (c) 2005, Joel Schopp <jschopp@austin.ibm.com> (the ugly bit) -# (c) 2007,2008, Andy Whitcroft <apw@uk.ibm.com> (new conditions, test suite) -# (c) 2008-2010 Andy Whitcroft <apw@canonical.com> -# Licensed under the terms of the GNU GPL License version 2 - -use strict; -use warnings; -use POSIX; -use File::Basename; -use Cwd 'abs_path'; -use Term::ANSIColor qw(:constants); - -my $P = $0; -my $D = dirname(abs_path($P)); - -my $V = '0.32'; - -use Getopt::Long qw(:config no_auto_abbrev); - -my $quiet = 0; -my $tree = 1; -my $chk_signoff = 1; -my $chk_patch = 1; -my $tst_only; -my $emacs = 0; -my $terse = 0; -my $showfile = 0; -my $file = 0; -my $git = 0; -my %git_commits = (); -my $check = 0; -my $check_orig = 0; -my $summary = 1; -my $mailback = 0; -my $summary_file = 0; -my $show_types = 0; -my $list_types = 0; -my $fix = 0; -my $fix_inplace = 0; -my $root; -my %debug; -my %camelcase = (); -my %use_type = (); -my @use = (); -my %ignore_type = (); -my @ignore = (); -my @exclude = (); -my $help = 0; -my $configuration_file = ".checkpatch.conf"; -my $max_line_length = 80; -my $ignore_perl_version = 0; -my $minimum_perl_version = 5.10.0; -my $min_conf_desc_length = 4; -my $spelling_file = "$D/spelling.txt"; -my $codespell = 0; -my $codespellfile = "/usr/share/codespell/dictionary.txt"; -my $conststructsfile = "$D/const_structs.checkpatch"; -my $typedefsfile = ""; -my $color = "auto"; -my $allow_c99_comments = 0; - -sub help { - my ($exitcode) = @_; - - print << "EOM"; -Usage: $P [OPTION]... [FILE]... -Version: $V - -Options: - -q, --quiet quiet - --no-tree run without a kernel tree - --no-signoff do not check for 'Signed-off-by' line - --patch treat FILE as patchfile (default) - --emacs emacs compile window format - --terse one line per report - --showfile emit diffed file position, not input file position - -g, --git treat FILE as a single commit or git revision range - single git commit with: - <rev> - <rev>^ - <rev>~n - multiple git commits with: - <rev1>..<rev2> - <rev1>...<rev2> - <rev>-<count> - git merges are ignored - -f, --file treat FILE as regular source file - --subjective, --strict enable more subjective tests - --list-types list the possible message types - --types TYPE(,TYPE2...) show only these comma separated message types - --ignore TYPE(,TYPE2...) ignore various comma separated message types - --exclude DIR(,DIR22...) exclude directories - --show-types show the specific message type in the output - --max-line-length=n set the maximum line length, if exceeded, warn - --min-conf-desc-length=n set the min description length, if shorter, warn - --root=PATH PATH to the kernel tree root - --no-summary suppress the per-file summary - --mailback only produce a report in case of warnings/errors - --summary-file include the filename in summary - --debug KEY=[0|1] turn on/off debugging of KEY, where KEY is one of - 'values', 'possible', 'type', and 'attr' (default - is all off) - --test-only=WORD report only warnings/errors containing WORD - literally - --fix EXPERIMENTAL - may create horrible results - If correctable single-line errors exist, create - "<inputfile>.EXPERIMENTAL-checkpatch-fixes" - with potential errors corrected to the preferred - checkpatch style - --fix-inplace EXPERIMENTAL - may create horrible results - Is the same as --fix, but overwrites the input - file. It's your fault if there's no backup or git - --ignore-perl-version override checking of perl version. expect - runtime errors. - --codespell Use the codespell dictionary for spelling/typos - (default:/usr/share/codespell/dictionary.txt) - --codespellfile Use this codespell dictionary - --typedefsfile Read additional types from this file - --color[=WHEN] Use colors 'always', 'never', or only when output - is a terminal ('auto'). Default is 'auto'. - -h, --help, --version display this help and exit - -When FILE is - read standard input. -EOM - - exit($exitcode); -} - -sub uniq { - my %seen; - return grep { !$seen{$_}++ } @_; -} - -sub list_types { - my ($exitcode) = @_; - - my $count = 0; - - local $/ = undef; - - open(my $script, '<', abs_path($P)) or - die "$P: Can't read '$P' $!\n"; - - my $text = <$script>; - close($script); - - my @types = (); - # Also catch when type or level is passed through a variable - for ($text =~ /(?:(?:\bCHK|\bWARN|\bERROR|&\{\$msg_level})\s*\(|\$msg_type\s*=)\s*"([^"]+)"/g) { - push (@types, $_); - } - @types = sort(uniq(@types)); - print("#\tMessage type\n\n"); - foreach my $type (@types) { - print(++$count . "\t" . $type . "\n"); - } - - exit($exitcode); -} - -my $conf = which_conf($configuration_file); -if (-f $conf) { - my @conf_args; - open(my $conffile, '<', "$conf") - or warn "$P: Can't find a readable $configuration_file file $!\n"; - - while (<$conffile>) { - my $line = $_; - - $line =~ s/\s*\n?$//g; - $line =~ s/^\s*//g; - $line =~ s/\s+/ /g; - - next if ($line =~ m/^\s*#/); - next if ($line =~ m/^\s*$/); - - my @words = split(" ", $line); - foreach my $word (@words) { - last if ($word =~ m/^#/); - push (@conf_args, $word); - } - } - close($conffile); - unshift(@ARGV, @conf_args) if @conf_args; -} - -# Perl's Getopt::Long allows options to take optional arguments after a space. -# Prevent --color by itself from consuming other arguments -foreach (@ARGV) { - if ($_ eq "--color" || $_ eq "-color") { - $_ = "--color=$color"; - } -} - -GetOptions( - 'q|quiet+' => \$quiet, - 'tree!' => \$tree, - 'signoff!' => \$chk_signoff, - 'patch!' => \$chk_patch, - 'emacs!' => \$emacs, - 'terse!' => \$terse, - 'showfile!' => \$showfile, - 'f|file!' => \$file, - 'g|git!' => \$git, - 'subjective!' => \$check, - 'strict!' => \$check, - 'ignore=s' => \@ignore, - 'exclude=s' => \@exclude, - 'types=s' => \@use, - 'show-types!' => \$show_types, - 'list-types!' => \$list_types, - 'max-line-length=i' => \$max_line_length, - 'min-conf-desc-length=i' => \$min_conf_desc_length, - 'root=s' => \$root, - 'summary!' => \$summary, - 'mailback!' => \$mailback, - 'summary-file!' => \$summary_file, - 'fix!' => \$fix, - 'fix-inplace!' => \$fix_inplace, - 'ignore-perl-version!' => \$ignore_perl_version, - 'debug=s' => \%debug, - 'test-only=s' => \$tst_only, - 'codespell!' => \$codespell, - 'codespellfile=s' => \$codespellfile, - 'typedefsfile=s' => \$typedefsfile, - 'color=s' => \$color, - 'no-color' => \$color, #keep old behaviors of -nocolor - 'nocolor' => \$color, #keep old behaviors of -nocolor - 'h|help' => \$help, - 'version' => \$help -) or help(1); - -help(0) if ($help); - -list_types(0) if ($list_types); - -$fix = 1 if ($fix_inplace); -$check_orig = $check; - -my $exit = 0; - -if ($^V && $^V lt $minimum_perl_version) { - printf "$P: requires at least perl version %vd\n", $minimum_perl_version; - if (!$ignore_perl_version) { - exit(1); - } -} - -#if no filenames are given, push '-' to read patch from stdin -if ($#ARGV < 0) { - push(@ARGV, '-'); -} - -if ($color =~ /^[01]$/) { - $color = !$color; -} elsif ($color =~ /^always$/i) { - $color = 1; -} elsif ($color =~ /^never$/i) { - $color = 0; -} elsif ($color =~ /^auto$/i) { - $color = (-t STDOUT); -} else { - die "Invalid color mode: $color\n"; -} - -sub hash_save_array_words { - my ($hashRef, $arrayRef) = @_; - - my @array = split(/,/, join(',', @$arrayRef)); - foreach my $word (@array) { - $word =~ s/\s*\n?$//g; - $word =~ s/^\s*//g; - $word =~ s/\s+/ /g; - $word =~ tr/[a-z]/[A-Z]/; - - next if ($word =~ m/^\s*#/); - next if ($word =~ m/^\s*$/); - - $hashRef->{$word}++; - } -} - -sub hash_show_words { - my ($hashRef, $prefix) = @_; - - if (keys %$hashRef) { - print "\nNOTE: $prefix message types:"; - foreach my $word (sort keys %$hashRef) { - print " $word"; - } - print "\n"; - } -} - -hash_save_array_words(\%ignore_type, \@ignore); -hash_save_array_words(\%use_type, \@use); - -my $dbg_values = 0; -my $dbg_possible = 0; -my $dbg_type = 0; -my $dbg_attr = 0; -for my $key (keys %debug) { - ## no critic - eval "\${dbg_$key} = '$debug{$key}';"; - die "$@" if ($@); -} - -my $rpt_cleaners = 0; - -if ($terse) { - $emacs = 1; - $quiet++; -} - -if ($tree) { - if (defined $root) { - if (!top_of_kernel_tree($root)) { - die "$P: $root: --root does not point at a valid tree\n"; - } - } else { - if (top_of_kernel_tree('.')) { - $root = '.'; - } elsif ($0 =~ m@(.*)/scripts/[^/]*$@ && - top_of_kernel_tree($1)) { - $root = $1; - } - } - - if (!defined $root) { - print "Must be run from the top-level dir. of a kernel tree\n"; - exit(2); - } -} - -my $emitted_corrupt = 0; - -our $Ident = qr{ - [A-Za-z_][A-Za-z\d_]* - (?:\s*\#\#\s*[A-Za-z_][A-Za-z\d_]*)* - }x; -our $Storage = qr{extern|static|asmlinkage}; -our $Sparse = qr{ - __user| - __force| - __iomem| - __must_check| - __init_refok| - __kprobes| - __ref| - __rcu| - __private - }x; -our $InitAttributePrefix = qr{__(?:mem|cpu|dev|net_|)}; -our $InitAttributeData = qr{$InitAttributePrefix(?:initdata\b)}; -our $InitAttributeConst = qr{$InitAttributePrefix(?:initconst\b)}; -our $InitAttributeInit = qr{$InitAttributePrefix(?:init\b)}; -our $InitAttribute = qr{$InitAttributeData|$InitAttributeConst|$InitAttributeInit}; - -# Notes to $Attribute: -# We need \b after 'init' otherwise 'initconst' will cause a false positive in a check -our $Attribute = qr{ - const| - __percpu| - __nocast| - __safe| - __bitwise| - __packed__| - __packed2__| - __naked| - __maybe_unused| - __always_unused| - __noreturn| - __used| - __cold| - __pure| - __noclone| - __deprecated| - __read_mostly| - __kprobes| - $InitAttribute| - ____cacheline_aligned| - ____cacheline_aligned_in_smp| - ____cacheline_internodealigned_in_smp| - __weak| - __syscall| - __syscall_inline - }x; -our $Modifier; -our $Inline = qr{inline|__always_inline|noinline|__inline|__inline__}; -our $Member = qr{->$Ident|\.$Ident|\[[^]]*\]}; -our $Lval = qr{$Ident(?:$Member)*}; - -our $Int_type = qr{(?i)llu|ull|ll|lu|ul|l|u}; -our $Binary = qr{(?i)0b[01]+$Int_type?}; -our $Hex = qr{(?i)0x[0-9a-f]+$Int_type?}; -our $Int = qr{[0-9]+$Int_type?}; -our $Octal = qr{0[0-7]+$Int_type?}; -our $String = qr{"[X\t]*"}; -our $Float_hex = qr{(?i)0x[0-9a-f]+p-?[0-9]+[fl]?}; -our $Float_dec = qr{(?i)(?:[0-9]+\.[0-9]*|[0-9]*\.[0-9]+)(?:e-?[0-9]+)?[fl]?}; -our $Float_int = qr{(?i)[0-9]+e-?[0-9]+[fl]?}; -our $Float = qr{$Float_hex|$Float_dec|$Float_int}; -our $Constant = qr{$Float|$Binary|$Octal|$Hex|$Int}; -our $Assignment = qr{\*\=|/=|%=|\+=|-=|<<=|>>=|&=|\^=|\|=|=}; -our $Compare = qr{<=|>=|==|!=|<|(?<!-)>}; -our $Arithmetic = qr{\+|-|\*|\/|%}; -our $Operators = qr{ - <=|>=|==|!=| - =>|->|<<|>>|<|>|!|~| - &&|\|\||,|\^|\+\+|--|&|\||$Arithmetic - }x; - -our $c90_Keywords = qr{do|for|while|if|else|return|goto|continue|switch|default|case|break}x; - -our $BasicType; -our $NonptrType; -our $NonptrTypeMisordered; -our $NonptrTypeWithAttr; -our $Type; -our $TypeMisordered; -our $Declare; -our $DeclareMisordered; - -our $NON_ASCII_UTF8 = qr{ - [\xC2-\xDF][\x80-\xBF] # non-overlong 2-byte - | \xE0[\xA0-\xBF][\x80-\xBF] # excluding overlongs - | [\xE1-\xEC\xEE\xEF][\x80-\xBF]{2} # straight 3-byte - | \xED[\x80-\x9F][\x80-\xBF] # excluding surrogates - | \xF0[\x90-\xBF][\x80-\xBF]{2} # planes 1-3 - | [\xF1-\xF3][\x80-\xBF]{3} # planes 4-15 - | \xF4[\x80-\x8F][\x80-\xBF]{2} # plane 16 -}x; - -our $UTF8 = qr{ - [\x09\x0A\x0D\x20-\x7E] # ASCII - | $NON_ASCII_UTF8 -}x; - -our $typeC99Typedefs = qr{(?:__)?(?:[us]_?)?int_?(?:8|16|32|64)_t}; -our $typeOtherOSTypedefs = qr{(?x: - u_(?:char|short|int|long) | # bsd - u(?:nchar|short|int|long) # sysv -)}; -our $typeKernelTypedefs = qr{(?x: - (?:__)?(?:u|s|be|le)(?:8|16|32|64)_t| - atomic_t -)}; -our $typeTypedefs = qr{(?x: - $typeC99Typedefs\b| - $typeOtherOSTypedefs\b| - $typeKernelTypedefs\b -)}; - -our $zero_initializer = qr{(?:(?:0[xX])?0+$Int_type?|NULL|false)\b}; - -our $logFunctions = qr{(?x: - printk(?:_ratelimited|_once|_deferred_once|_deferred|)| - (?:[a-z0-9]+_){1,2}(?:printk|emerg|alert|crit|err|warning|warn|notice|info|debug|dbg|vdbg|devel|cont|WARN)(?:_ratelimited|_once|)| - WARN(?:_RATELIMIT|_ONCE|)| - panic| - MODULE_[A-Z_]+| - seq_vprintf|seq_printf|seq_puts -)}; - -our $signature_tags = qr{(?xi: - Signed-off-by:| - Acked-by:| - Tested-by:| - Reviewed-by:| - Reported-by:| - Suggested-by:| - To:| - Cc: -)}; - -our @typeListMisordered = ( - qr{char\s+(?:un)?signed}, - qr{int\s+(?:(?:un)?signed\s+)?short\s}, - qr{int\s+short(?:\s+(?:un)?signed)}, - qr{short\s+int(?:\s+(?:un)?signed)}, - qr{(?:un)?signed\s+int\s+short}, - qr{short\s+(?:un)?signed}, - qr{long\s+int\s+(?:un)?signed}, - qr{int\s+long\s+(?:un)?signed}, - qr{long\s+(?:un)?signed\s+int}, - qr{int\s+(?:un)?signed\s+long}, - qr{int\s+(?:un)?signed}, - qr{int\s+long\s+long\s+(?:un)?signed}, - qr{long\s+long\s+int\s+(?:un)?signed}, - qr{long\s+long\s+(?:un)?signed\s+int}, - qr{long\s+long\s+(?:un)?signed}, - qr{long\s+(?:un)?signed}, -); - -our @typeList = ( - qr{void}, - qr{(?:(?:un)?signed\s+)?char}, - qr{(?:(?:un)?signed\s+)?short\s+int}, - qr{(?:(?:un)?signed\s+)?short}, - qr{(?:(?:un)?signed\s+)?int}, - qr{(?:(?:un)?signed\s+)?long\s+int}, - qr{(?:(?:un)?signed\s+)?long\s+long\s+int}, - qr{(?:(?:un)?signed\s+)?long\s+long}, - qr{(?:(?:un)?signed\s+)?long}, - qr{(?:un)?signed}, - qr{float}, - qr{double}, - qr{bool}, - qr{struct\s+$Ident}, - qr{union\s+$Ident}, - qr{enum\s+$Ident}, - qr{${Ident}_t}, - qr{${Ident}_handler}, - qr{${Ident}_handler_fn}, - @typeListMisordered, -); - -our $C90_int_types = qr{(?x: - long\s+long\s+int\s+(?:un)?signed| - long\s+long\s+(?:un)?signed\s+int| - long\s+long\s+(?:un)?signed| - (?:(?:un)?signed\s+)?long\s+long\s+int| - (?:(?:un)?signed\s+)?long\s+long| - int\s+long\s+long\s+(?:un)?signed| - int\s+(?:(?:un)?signed\s+)?long\s+long| - - long\s+int\s+(?:un)?signed| - long\s+(?:un)?signed\s+int| - long\s+(?:un)?signed| - (?:(?:un)?signed\s+)?long\s+int| - (?:(?:un)?signed\s+)?long| - int\s+long\s+(?:un)?signed| - int\s+(?:(?:un)?signed\s+)?long| - - int\s+(?:un)?signed| - (?:(?:un)?signed\s+)?int -)}; - -our @typeListFile = (); -our @typeListWithAttr = ( - @typeList, - qr{struct\s+$InitAttribute\s+$Ident}, - qr{union\s+$InitAttribute\s+$Ident}, -); - -our @modifierList = ( - qr{fastcall}, -); -our @modifierListFile = (); - -our @mode_permission_funcs = ( - ["module_param", 3], - ["module_param_(?:array|named|string)", 4], - ["module_param_array_named", 5], - ["debugfs_create_(?:file|u8|u16|u32|u64|x8|x16|x32|x64|size_t|atomic_t|bool|blob|regset32|u32_array)", 2], - ["proc_create(?:_data|)", 2], - ["(?:CLASS|DEVICE|SENSOR|SENSOR_DEVICE|IIO_DEVICE)_ATTR", 2], - ["IIO_DEV_ATTR_[A-Z_]+", 1], - ["SENSOR_(?:DEVICE_|)ATTR_2", 2], - ["SENSOR_TEMPLATE(?:_2|)", 3], - ["__ATTR", 2], -); - -#Create a search pattern for all these functions to speed up a loop below -our $mode_perms_search = ""; -foreach my $entry (@mode_permission_funcs) { - $mode_perms_search .= '|' if ($mode_perms_search ne ""); - $mode_perms_search .= $entry->[0]; -} - -our $mode_perms_world_writable = qr{ - S_IWUGO | - S_IWOTH | - S_IRWXUGO | - S_IALLUGO | - 0[0-7][0-7][2367] -}x; - -our %mode_permission_string_types = ( - "S_IRWXU" => 0700, - "S_IRUSR" => 0400, - "S_IWUSR" => 0200, - "S_IXUSR" => 0100, - "S_IRWXG" => 0070, - "S_IRGRP" => 0040, - "S_IWGRP" => 0020, - "S_IXGRP" => 0010, - "S_IRWXO" => 0007, - "S_IROTH" => 0004, - "S_IWOTH" => 0002, - "S_IXOTH" => 0001, - "S_IRWXUGO" => 0777, - "S_IRUGO" => 0444, - "S_IWUGO" => 0222, - "S_IXUGO" => 0111, -); - -#Create a search pattern for all these strings to speed up a loop below -our $mode_perms_string_search = ""; -foreach my $entry (keys %mode_permission_string_types) { - $mode_perms_string_search .= '|' if ($mode_perms_string_search ne ""); - $mode_perms_string_search .= $entry; -} - -our $allowed_asm_includes = qr{(?x: - irq| - memory| - time| - reboot -)}; -# memory.h: ARM has a custom one - -# Load common spelling mistakes and build regular expression list. -my $misspellings; -my %spelling_fix; - -if (open(my $spelling, '<', $spelling_file)) { - while (<$spelling>) { - my $line = $_; - - $line =~ s/\s*\n?$//g; - $line =~ s/^\s*//g; - - next if ($line =~ m/^\s*#/); - next if ($line =~ m/^\s*$/); - - my ($suspect, $fix) = split(/\|\|/, $line); - - $spelling_fix{$suspect} = $fix; - } - close($spelling); -} else { - warn "No typos will be found - file '$spelling_file': $!\n"; -} - -if ($codespell) { - if (open(my $spelling, '<', $codespellfile)) { - while (<$spelling>) { - my $line = $_; - - $line =~ s/\s*\n?$//g; - $line =~ s/^\s*//g; - - next if ($line =~ m/^\s*#/); - next if ($line =~ m/^\s*$/); - next if ($line =~ m/, disabled/i); - - $line =~ s/,.*$//; - - my ($suspect, $fix) = split(/->/, $line); - - $spelling_fix{$suspect} = $fix; - } - close($spelling); - } else { - warn "No codespell typos will be found - file '$codespellfile': $!\n"; - } -} - -$misspellings = join("|", sort keys %spelling_fix) if keys %spelling_fix; - -sub read_words { - my ($wordsRef, $file) = @_; - - if (open(my $words, '<', $file)) { - while (<$words>) { - my $line = $_; - - $line =~ s/\s*\n?$//g; - $line =~ s/^\s*//g; - - next if ($line =~ m/^\s*#/); - next if ($line =~ m/^\s*$/); - if ($line =~ /\s/) { - print("$file: '$line' invalid - ignored\n"); - next; - } - - $$wordsRef .= '|' if ($$wordsRef ne ""); - $$wordsRef .= $line; - } - close($file); - return 1; - } - - return 0; -} - -my $const_structs = ""; -#read_words(\$const_structs, $conststructsfile) -# or warn "No structs that should be const will be found - file '$conststructsfile': $!\n"; - -my $typeOtherTypedefs = ""; -if (length($typedefsfile)) { - read_words(\$typeOtherTypedefs, $typedefsfile) - or warn "No additional types will be considered - file '$typedefsfile': $!\n"; -} -$typeTypedefs .= '|' . $typeOtherTypedefs if ($typeOtherTypedefs ne ""); - -sub build_types { - my $mods = "(?x: \n" . join("|\n ", (@modifierList, @modifierListFile)) . "\n)"; - my $all = "(?x: \n" . join("|\n ", (@typeList, @typeListFile)) . "\n)"; - my $Misordered = "(?x: \n" . join("|\n ", @typeListMisordered) . "\n)"; - my $allWithAttr = "(?x: \n" . join("|\n ", @typeListWithAttr) . "\n)"; - $Modifier = qr{(?:$Attribute|$Sparse|$mods)}; - $BasicType = qr{ - (?:$typeTypedefs\b)| - (?:${all}\b) - }x; - $NonptrType = qr{ - (?:$Modifier\s+|const\s+)* - (?: - (?:typeof|__typeof__)\s*\([^\)]*\)| - (?:$typeTypedefs\b)| - (?:${all}\b) - ) - (?:\s+$Modifier|\s+const)* - }x; - $NonptrTypeMisordered = qr{ - (?:$Modifier\s+|const\s+)* - (?: - (?:${Misordered}\b) - ) - (?:\s+$Modifier|\s+const)* - }x; - $NonptrTypeWithAttr = qr{ - (?:$Modifier\s+|const\s+)* - (?: - (?:typeof|__typeof__)\s*\([^\)]*\)| - (?:$typeTypedefs\b)| - (?:${allWithAttr}\b) - ) - (?:\s+$Modifier|\s+const)* - }x; - $Type = qr{ - $NonptrType - (?:(?:\s|\*|\[\])+\s*const|(?:\s|\*\s*(?:const\s*)?|\[\])+|(?:\s*\[\s*\])+)? - (?:\s+$Inline|\s+$Modifier)* - }x; - $TypeMisordered = qr{ - $NonptrTypeMisordered - (?:(?:\s|\*|\[\])+\s*const|(?:\s|\*\s*(?:const\s*)?|\[\])+|(?:\s*\[\s*\])+)? - (?:\s+$Inline|\s+$Modifier)* - }x; - $Declare = qr{(?:$Storage\s+(?:$Inline\s+)?)?$Type}; - $DeclareMisordered = qr{(?:$Storage\s+(?:$Inline\s+)?)?$TypeMisordered}; -} -build_types(); - -our $Typecast = qr{\s*(\(\s*$NonptrType\s*\)){0,1}\s*}; - -# Using $balanced_parens, $LvalOrFunc, or $FuncArg -# requires at least perl version v5.10.0 -# Any use must be runtime checked with $^V - -our $balanced_parens = qr/(\((?:[^\(\)]++|(?-1))*\))/; -our $LvalOrFunc = qr{((?:[\&\*]\s*)?$Lval)\s*($balanced_parens{0,1})\s*}; -our $FuncArg = qr{$Typecast{0,1}($LvalOrFunc|$Constant|$String)}; - -our $declaration_macros = qr{(?x: - (?:$Storage\s+)?(?:[A-Z_][A-Z0-9]*_){0,2}(?:DEFINE|DECLARE)(?:_[A-Z0-9]+){1,6}\s*\(| - (?:$Storage\s+)?[HLP]?LIST_HEAD\s*\(| - (?:$Storage\s+)?${Type}\s+uninitialized_var\s*\( -)}; - -sub deparenthesize { - my ($string) = @_; - return "" if (!defined($string)); - - while ($string =~ /^\s*\(.*\)\s*$/) { - $string =~ s@^\s*\(\s*@@; - $string =~ s@\s*\)\s*$@@; - } - - $string =~ s@\s+@ @g; - - return $string; -} - -sub seed_camelcase_file { - my ($file) = @_; - - return if (!(-f $file)); - - local $/; - - open(my $include_file, '<', "$file") - or warn "$P: Can't read '$file' $!\n"; - my $text = <$include_file>; - close($include_file); - - my @lines = split('\n', $text); - - foreach my $line (@lines) { - next if ($line !~ /(?:[A-Z][a-z]|[a-z][A-Z])/); - if ($line =~ /^[ \t]*(?:#[ \t]*define|typedef\s+$Type)\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)/) { - $camelcase{$1} = 1; - } elsif ($line =~ /^\s*$Declare\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)\s*[\(\[,;]/) { - $camelcase{$1} = 1; - } elsif ($line =~ /^\s*(?:union|struct|enum)\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)\s*[;\{]/) { - $camelcase{$1} = 1; - } - } -} - -sub is_maintained_obsolete { - my ($filename) = @_; - - return 0 if (!$tree || !(-e "$root/scripts/get_maintainer.pl")); - - my $status = `perl $root/scripts/get_maintainer.pl --status --nom --nol --nogit --nogit-fallback -f $filename 2>&1`; - - return $status =~ /obsolete/i; -} - -my $camelcase_seeded = 0; -sub seed_camelcase_includes { - return if ($camelcase_seeded); - - my $files; - my $camelcase_cache = ""; - my @include_files = (); - - $camelcase_seeded = 1; - - if (-e ".git") { - my $git_last_include_commit = `git log --no-merges --pretty=format:"%h%n" -1 -- include`; - chomp $git_last_include_commit; - $camelcase_cache = ".checkpatch-camelcase.git.$git_last_include_commit"; - } else { - my $last_mod_date = 0; - $files = `find $root/include -name "*.h"`; - @include_files = split('\n', $files); - foreach my $file (@include_files) { - my $date = POSIX::strftime("%Y%m%d%H%M", - localtime((stat $file)[9])); - $last_mod_date = $date if ($last_mod_date < $date); - } - $camelcase_cache = ".checkpatch-camelcase.date.$last_mod_date"; - } - - if ($camelcase_cache ne "" && -f $camelcase_cache) { - open(my $camelcase_file, '<', "$camelcase_cache") - or warn "$P: Can't read '$camelcase_cache' $!\n"; - while (<$camelcase_file>) { - chomp; - $camelcase{$_} = 1; - } - close($camelcase_file); - - return; - } - - if (-e ".git") { - $files = `git ls-files "include/*.h"`; - @include_files = split('\n', $files); - } - - foreach my $file (@include_files) { - seed_camelcase_file($file); - } - - if ($camelcase_cache ne "") { - unlink glob ".checkpatch-camelcase.*"; - open(my $camelcase_file, '>', "$camelcase_cache") - or warn "$P: Can't write '$camelcase_cache' $!\n"; - foreach (sort { lc($a) cmp lc($b) } keys(%camelcase)) { - print $camelcase_file ("$_\n"); - } - close($camelcase_file); - } -} - -sub git_commit_info { - my ($commit, $id, $desc) = @_; - - return ($id, $desc) if ((which("git") eq "") || !(-e ".git")); - - my $output = `git log --no-color --format='%H %s' -1 $commit 2>&1`; - $output =~ s/^\s*//gm; - my @lines = split("\n", $output); - - return ($id, $desc) if ($#lines < 0); - - if ($lines[0] =~ /^error: short SHA1 $commit is ambiguous\./) { -# Maybe one day convert this block of bash into something that returns -# all matching commit ids, but it's very slow... -# -# echo "checking commits $1..." -# git rev-list --remotes | grep -i "^$1" | -# while read line ; do -# git log --format='%H %s' -1 $line | -# echo "commit $(cut -c 1-12,41-)" -# done - } elsif ($lines[0] =~ /^fatal: ambiguous argument '$commit': unknown revision or path not in the working tree\./) { - $id = undef; - } else { - $id = substr($lines[0], 0, 12); - $desc = substr($lines[0], 41); - } - - return ($id, $desc); -} - -$chk_signoff = 0 if ($file); - -my @rawlines = (); -my @lines = (); -my @fixed = (); -my @fixed_inserted = (); -my @fixed_deleted = (); -my $fixlinenr = -1; - -# If input is git commits, extract all commits from the commit expressions. -# For example, HEAD-3 means we need check 'HEAD, HEAD~1, HEAD~2'. -die "$P: No git repository found\n" if ($git && !-e ".git"); - -if ($git) { - my @commits = (); - foreach my $commit_expr (@ARGV) { - my $git_range; - if ($commit_expr =~ m/^(.*)-(\d+)$/) { - $git_range = "-$2 $1"; - } elsif ($commit_expr =~ m/\.\./) { - $git_range = "$commit_expr"; - } else { - $git_range = "-1 $commit_expr"; - } - my $lines = `git log --no-color --no-merges --pretty=format:'%H %s' $git_range`; - foreach my $line (split(/\n/, $lines)) { - $line =~ /^([0-9a-fA-F]{40,40}) (.*)$/; - next if (!defined($1) || !defined($2)); - my $sha1 = $1; - my $subject = $2; - unshift(@commits, $sha1); - $git_commits{$sha1} = $subject; - } - } - die "$P: no git commits after extraction!\n" if (@commits == 0); - @ARGV = @commits; -} - -my $vname; -for my $filename (@ARGV) { - my $FILE; - if ($git) { - open($FILE, '-|', "git format-patch -M --stdout -1 $filename") || - die "$P: $filename: git format-patch failed - $!\n"; - } elsif ($file) { - open($FILE, '-|', "diff -u /dev/null $filename") || - die "$P: $filename: diff failed - $!\n"; - } elsif ($filename eq '-') { - open($FILE, '<&STDIN'); - } else { - open($FILE, '<', "$filename") || - die "$P: $filename: open failed - $!\n"; - } - if ($filename eq '-') { - $vname = 'Your patch'; - } elsif ($git) { - $vname = "Commit " . substr($filename, 0, 12) . ' ("' . $git_commits{$filename} . '")'; - } else { - $vname = $filename; - } - while (<$FILE>) { - chomp; - push(@rawlines, $_); - } - close($FILE); - - if ($#ARGV > 0 && $quiet == 0) { - print '-' x length($vname) . "\n"; - print "$vname\n"; - print '-' x length($vname) . "\n"; - } - - if (!process($filename)) { - $exit = 1; - } - @rawlines = (); - @lines = (); - @fixed = (); - @fixed_inserted = (); - @fixed_deleted = (); - $fixlinenr = -1; - @modifierListFile = (); - @typeListFile = (); - build_types(); -} - -if (!$quiet) { - hash_show_words(\%use_type, "Used"); - hash_show_words(\%ignore_type, "Ignored"); - - if ($^V lt 5.10.0) { - print << "EOM" - -NOTE: perl $^V is not modern enough to detect all possible issues. - An upgrade to at least perl v5.10.0 is suggested. -EOM - } - if ($exit) { - print << "EOM" - -NOTE: If any of the errors are false positives, please report - them to the maintainers. -EOM - } -} - -exit($exit); - -sub top_of_kernel_tree { - my ($root) = @_; - - my @tree_check = ( - "LICENSE", "CODEOWNERS", "Kconfig", "Makefile", - "README.rst", "doc", "arch", "include", "drivers", - "boards", "kernel", "lib", "scripts", - ); - - foreach my $check (@tree_check) { - if (! -e $root . '/' . $check) { - return 0; - } - } - return 1; -} - -sub parse_email { - my ($formatted_email) = @_; - - my $name = ""; - my $address = ""; - my $comment = ""; - - if ($formatted_email =~ /^(.*)<(\S+\@\S+)>(.*)$/) { - $name = $1; - $address = $2; - $comment = $3 if defined $3; - } elsif ($formatted_email =~ /^\s*<(\S+\@\S+)>(.*)$/) { - $address = $1; - $comment = $2 if defined $2; - } elsif ($formatted_email =~ /(\S+\@\S+)(.*)$/) { - $address = $1; - $comment = $2 if defined $2; - $formatted_email =~ s/$address.*$//; - $name = $formatted_email; - $name = trim($name); - $name =~ s/^\"|\"$//g; - # If there's a name left after stripping spaces and - # leading quotes, and the address doesn't have both - # leading and trailing angle brackets, the address - # is invalid. ie: - # "joe smith joe@smith.com" bad - # "joe smith <joe@smith.com" bad - if ($name ne "" && $address !~ /^<[^>]+>$/) { - $name = ""; - $address = ""; - $comment = ""; - } - } - - $name = trim($name); - $name =~ s/^\"|\"$//g; - $address = trim($address); - $address =~ s/^\<|\>$//g; - - if ($name =~ /[^\w \-]/i) { ##has "must quote" chars - $name =~ s/(?<!\\)"/\\"/g; ##escape quotes - $name = "\"$name\""; - } - - return ($name, $address, $comment); -} - -sub format_email { - my ($name, $address) = @_; - - my $formatted_email; - - $name = trim($name); - $name =~ s/^\"|\"$//g; - $address = trim($address); - - if ($name =~ /[^\w \-]/i) { ##has "must quote" chars - $name =~ s/(?<!\\)"/\\"/g; ##escape quotes - $name = "\"$name\""; - } - - if ("$name" eq "") { - $formatted_email = "$address"; - } else { - $formatted_email = "$name <$address>"; - } - - return $formatted_email; -} - -sub which { - my ($bin) = @_; - - foreach my $path (split(/:/, $ENV{PATH})) { - if (-e "$path/$bin") { - return "$path/$bin"; - } - } - - return ""; -} - -sub which_conf { - my ($conf) = @_; - - foreach my $path (split(/:/, ".:$ENV{HOME}:.scripts")) { - if (-e "$path/$conf") { - return "$path/$conf"; - } - } - - return ""; -} - -sub expand_tabs { - my ($str) = @_; - - my $res = ''; - my $n = 0; - for my $c (split(//, $str)) { - if ($c eq "\t") { - $res .= ' '; - $n++; - for (; ($n % 8) != 0; $n++) { - $res .= ' '; - } - next; - } - $res .= $c; - $n++; - } - - return $res; -} -sub copy_spacing { - (my $res = shift) =~ tr/\t/ /c; - return $res; -} - -sub line_stats { - my ($line) = @_; - - # Drop the diff line leader and expand tabs - $line =~ s/^.//; - $line = expand_tabs($line); - - # Pick the indent from the front of the line. - my ($white) = ($line =~ /^(\s*)/); - - return (length($line), length($white)); -} - -my $sanitise_quote = ''; - -sub sanitise_line_reset { - my ($in_comment) = @_; - - if ($in_comment) { - $sanitise_quote = '*/'; - } else { - $sanitise_quote = ''; - } -} -sub sanitise_line { - my ($line) = @_; - - my $res = ''; - my $l = ''; - - my $qlen = 0; - my $off = 0; - my $c; - - # Always copy over the diff marker. - $res = substr($line, 0, 1); - - for ($off = 1; $off < length($line); $off++) { - $c = substr($line, $off, 1); - - # Comments we are wacking completly including the begin - # and end, all to $;. - if ($sanitise_quote eq '' && substr($line, $off, 2) eq '/*') { - $sanitise_quote = '*/'; - - substr($res, $off, 2, "$;$;"); - $off++; - next; - } - if ($sanitise_quote eq '*/' && substr($line, $off, 2) eq '*/') { - $sanitise_quote = ''; - substr($res, $off, 2, "$;$;"); - $off++; - next; - } - if ($sanitise_quote eq '' && substr($line, $off, 2) eq '//') { - $sanitise_quote = '//'; - - substr($res, $off, 2, $sanitise_quote); - $off++; - next; - } - - # A \ in a string means ignore the next character. - if (($sanitise_quote eq "'" || $sanitise_quote eq '"') && - $c eq "\\") { - substr($res, $off, 2, 'XX'); - $off++; - next; - } - # Regular quotes. - if ($c eq "'" || $c eq '"') { - if ($sanitise_quote eq '') { - $sanitise_quote = $c; - - substr($res, $off, 1, $c); - next; - } elsif ($sanitise_quote eq $c) { - $sanitise_quote = ''; - } - } - - #print "c<$c> SQ<$sanitise_quote>\n"; - if ($off != 0 && $sanitise_quote eq '*/' && $c ne "\t") { - substr($res, $off, 1, $;); - } elsif ($off != 0 && $sanitise_quote eq '//' && $c ne "\t") { - substr($res, $off, 1, $;); - } elsif ($off != 0 && $sanitise_quote && $c ne "\t") { - substr($res, $off, 1, 'X'); - } else { - substr($res, $off, 1, $c); - } - } - - if ($sanitise_quote eq '//') { - $sanitise_quote = ''; - } - - # The pathname on a #include may be surrounded by '<' and '>'. - if ($res =~ /^.\s*\#\s*include\s+\<(.*)\>/) { - my $clean = 'X' x length($1); - $res =~ s@\<.*\>@<$clean>@; - - # The whole of a #error is a string. - } elsif ($res =~ /^.\s*\#\s*(?:error|warning)\s+(.*)\b/) { - my $clean = 'X' x length($1); - $res =~ s@(\#\s*(?:error|warning)\s+).*@$1$clean@; - } - - if ($allow_c99_comments && $res =~ m@(//.*$)@) { - my $match = $1; - $res =~ s/\Q$match\E/"$;" x length($match)/e; - } - - return $res; -} - -sub get_quoted_string { - my ($line, $rawline) = @_; - - return "" if ($line !~ m/($String)/g); - return substr($rawline, $-[0], $+[0] - $-[0]); -} - -sub ctx_statement_block { - my ($linenr, $remain, $off) = @_; - my $line = $linenr - 1; - my $blk = ''; - my $soff = $off; - my $coff = $off - 1; - my $coff_set = 0; - - my $loff = 0; - - my $type = ''; - my $level = 0; - my @stack = (); - my $p; - my $c; - my $len = 0; - - my $remainder; - while (1) { - @stack = (['', 0]) if ($#stack == -1); - - #warn "CSB: blk<$blk> remain<$remain>\n"; - # If we are about to drop off the end, pull in more - # context. - if ($off >= $len) { - for (; $remain > 0; $line++) { - last if (!defined $lines[$line]); - next if ($lines[$line] =~ /^-/); - $remain--; - $loff = $len; - $blk .= $lines[$line] . "\n"; - $len = length($blk); - $line++; - last; - } - # Bail if there is no further context. - #warn "CSB: blk<$blk> off<$off> len<$len>\n"; - if ($off >= $len) { - last; - } - if ($level == 0 && substr($blk, $off) =~ /^.\s*#\s*define/) { - $level++; - $type = '#'; - } - } - $p = $c; - $c = substr($blk, $off, 1); - $remainder = substr($blk, $off); - - #warn "CSB: c<$c> type<$type> level<$level> remainder<$remainder> coff_set<$coff_set>\n"; - - # Handle nested #if/#else. - if ($remainder =~ /^#\s*(?:ifndef|ifdef|if)\s/) { - push(@stack, [ $type, $level ]); - } elsif ($remainder =~ /^#\s*(?:else|elif)\b/) { - ($type, $level) = @{$stack[$#stack - 1]}; - } elsif ($remainder =~ /^#\s*endif\b/) { - ($type, $level) = @{pop(@stack)}; - } - - # Statement ends at the ';' or a close '}' at the - # outermost level. - if ($level == 0 && $c eq ';') { - last; - } - - # An else is really a conditional as long as its not else if - if ($level == 0 && $coff_set == 0 && - (!defined($p) || $p =~ /(?:\s|\}|\+)/) && - $remainder =~ /^(else)(?:\s|{)/ && - $remainder !~ /^else\s+if\b/) { - $coff = $off + length($1) - 1; - $coff_set = 1; - #warn "CSB: mark coff<$coff> soff<$soff> 1<$1>\n"; - #warn "[" . substr($blk, $soff, $coff - $soff + 1) . "]\n"; - } - - if (($type eq '' || $type eq '(') && $c eq '(') { - $level++; - $type = '('; - } - if ($type eq '(' && $c eq ')') { - $level--; - $type = ($level != 0)? '(' : ''; - - if ($level == 0 && $coff < $soff) { - $coff = $off; - $coff_set = 1; - #warn "CSB: mark coff<$coff>\n"; - } - } - if (($type eq '' || $type eq '{') && $c eq '{') { - $level++; - $type = '{'; - } - if ($type eq '{' && $c eq '}') { - $level--; - $type = ($level != 0)? '{' : ''; - - if ($level == 0) { - if (substr($blk, $off + 1, 1) eq ';') { - $off++; - } - last; - } - } - # Preprocessor commands end at the newline unless escaped. - if ($type eq '#' && $c eq "\n" && $p ne "\\") { - $level--; - $type = ''; - $off++; - last; - } - $off++; - } - # We are truly at the end, so shuffle to the next line. - if ($off == $len) { - $loff = $len + 1; - $line++; - $remain--; - } - - my $statement = substr($blk, $soff, $off - $soff + 1); - my $condition = substr($blk, $soff, $coff - $soff + 1); - - #warn "STATEMENT<$statement>\n"; - #warn "CONDITION<$condition>\n"; - - #print "coff<$coff> soff<$off> loff<$loff>\n"; - - return ($statement, $condition, - $line, $remain + 1, $off - $loff + 1, $level); -} - -sub statement_lines { - my ($stmt) = @_; - - # Strip the diff line prefixes and rip blank lines at start and end. - $stmt =~ s/(^|\n)./$1/g; - $stmt =~ s/^\s*//; - $stmt =~ s/\s*$//; - - my @stmt_lines = ($stmt =~ /\n/g); - - return $#stmt_lines + 2; -} - -sub statement_rawlines { - my ($stmt) = @_; - - my @stmt_lines = ($stmt =~ /\n/g); - - return $#stmt_lines + 2; -} - -sub statement_block_size { - my ($stmt) = @_; - - $stmt =~ s/(^|\n)./$1/g; - $stmt =~ s/^\s*{//; - $stmt =~ s/}\s*$//; - $stmt =~ s/^\s*//; - $stmt =~ s/\s*$//; - - my @stmt_lines = ($stmt =~ /\n/g); - my @stmt_statements = ($stmt =~ /;/g); - - my $stmt_lines = $#stmt_lines + 2; - my $stmt_statements = $#stmt_statements + 1; - - if ($stmt_lines > $stmt_statements) { - return $stmt_lines; - } else { - return $stmt_statements; - } -} - -sub ctx_statement_full { - my ($linenr, $remain, $off) = @_; - my ($statement, $condition, $level); - - my (@chunks); - - # Grab the first conditional/block pair. - ($statement, $condition, $linenr, $remain, $off, $level) = - ctx_statement_block($linenr, $remain, $off); - #print "F: c<$condition> s<$statement> remain<$remain>\n"; - push(@chunks, [ $condition, $statement ]); - if (!($remain > 0 && $condition =~ /^\s*(?:\n[+-])?\s*(?:if|else|do)\b/s)) { - return ($level, $linenr, @chunks); - } - - # Pull in the following conditional/block pairs and see if they - # could continue the statement. - for (;;) { - ($statement, $condition, $linenr, $remain, $off, $level) = - ctx_statement_block($linenr, $remain, $off); - #print "C: c<$condition> s<$statement> remain<$remain>\n"; - last if (!($remain > 0 && $condition =~ /^(?:\s*\n[+-])*\s*(?:else|do)\b/s)); - #print "C: push\n"; - push(@chunks, [ $condition, $statement ]); - } - - return ($level, $linenr, @chunks); -} - -sub ctx_block_get { - my ($linenr, $remain, $outer, $open, $close, $off) = @_; - my $line; - my $start = $linenr - 1; - my $blk = ''; - my @o; - my @c; - my @res = (); - - my $level = 0; - my @stack = ($level); - for ($line = $start; $remain > 0; $line++) { - next if ($rawlines[$line] =~ /^-/); - $remain--; - - $blk .= $rawlines[$line]; - - # Handle nested #if/#else. - if ($lines[$line] =~ /^.\s*#\s*(?:ifndef|ifdef|if)\s/) { - push(@stack, $level); - } elsif ($lines[$line] =~ /^.\s*#\s*(?:else|elif)\b/) { - $level = $stack[$#stack - 1]; - } elsif ($lines[$line] =~ /^.\s*#\s*endif\b/) { - $level = pop(@stack); - } - - foreach my $c (split(//, $lines[$line])) { - ##print "C<$c>L<$level><$open$close>O<$off>\n"; - if ($off > 0) { - $off--; - next; - } - - if ($c eq $close && $level > 0) { - $level--; - last if ($level == 0); - } elsif ($c eq $open) { - $level++; - } - } - - if (!$outer || $level <= 1) { - push(@res, $rawlines[$line]); - } - - last if ($level == 0); - } - - return ($level, @res); -} -sub ctx_block_outer { - my ($linenr, $remain) = @_; - - my ($level, @r) = ctx_block_get($linenr, $remain, 1, '{', '}', 0); - return @r; -} -sub ctx_block { - my ($linenr, $remain) = @_; - - my ($level, @r) = ctx_block_get($linenr, $remain, 0, '{', '}', 0); - return @r; -} -sub ctx_statement { - my ($linenr, $remain, $off) = @_; - - my ($level, @r) = ctx_block_get($linenr, $remain, 0, '(', ')', $off); - return @r; -} -sub ctx_block_level { - my ($linenr, $remain) = @_; - - return ctx_block_get($linenr, $remain, 0, '{', '}', 0); -} -sub ctx_statement_level { - my ($linenr, $remain, $off) = @_; - - return ctx_block_get($linenr, $remain, 0, '(', ')', $off); -} - -sub ctx_locate_comment { - my ($first_line, $end_line) = @_; - - # Catch a comment on the end of the line itself. - my ($current_comment) = ($rawlines[$end_line - 1] =~ m@.*(/\*.*\*/)\s*(?:\\\s*)?$@); - return $current_comment if (defined $current_comment); - - # Look through the context and try and figure out if there is a - # comment. - my $in_comment = 0; - $current_comment = ''; - for (my $linenr = $first_line; $linenr < $end_line; $linenr++) { - my $line = $rawlines[$linenr - 1]; - #warn " $line\n"; - if ($linenr == $first_line and $line =~ m@^.\s*\*@) { - $in_comment = 1; - } - if ($line =~ m@/\*@) { - $in_comment = 1; - } - if (!$in_comment && $current_comment ne '') { - $current_comment = ''; - } - $current_comment .= $line . "\n" if ($in_comment); - if ($line =~ m@\*/@) { - $in_comment = 0; - } - } - - chomp($current_comment); - return($current_comment); -} -sub ctx_has_comment { - my ($first_line, $end_line) = @_; - my $cmt = ctx_locate_comment($first_line, $end_line); - - ##print "LINE: $rawlines[$end_line - 1 ]\n"; - ##print "CMMT: $cmt\n"; - - return ($cmt ne ''); -} - -sub raw_line { - my ($linenr, $cnt) = @_; - - my $offset = $linenr - 1; - $cnt++; - - my $line; - while ($cnt) { - $line = $rawlines[$offset++]; - next if (defined($line) && $line =~ /^-/); - $cnt--; - } - - return $line; -} - -sub cat_vet { - my ($vet) = @_; - my ($res, $coded); - - $res = ''; - while ($vet =~ /([^[:cntrl:]]*)([[:cntrl:]]|$)/g) { - $res .= $1; - if ($2 ne '') { - $coded = sprintf("^%c", unpack('C', $2) + 64); - $res .= $coded; - } - } - $res =~ s/$/\$/; - - return $res; -} - -my $av_preprocessor = 0; -my $av_pending; -my @av_paren_type; -my $av_pend_colon; - -sub annotate_reset { - $av_preprocessor = 0; - $av_pending = '_'; - @av_paren_type = ('E'); - $av_pend_colon = 'O'; -} - -sub annotate_values { - my ($stream, $type) = @_; - - my $res; - my $var = '_' x length($stream); - my $cur = $stream; - - print "$stream\n" if ($dbg_values > 1); - - while (length($cur)) { - @av_paren_type = ('E') if ($#av_paren_type < 0); - print " <" . join('', @av_paren_type) . - "> <$type> <$av_pending>" if ($dbg_values > 1); - if ($cur =~ /^(\s+)/o) { - print "WS($1)\n" if ($dbg_values > 1); - if ($1 =~ /\n/ && $av_preprocessor) { - $type = pop(@av_paren_type); - $av_preprocessor = 0; - } - - } elsif ($cur =~ /^(\(\s*$Type\s*)\)/ && $av_pending eq '_') { - print "CAST($1)\n" if ($dbg_values > 1); - push(@av_paren_type, $type); - $type = 'c'; - - } elsif ($cur =~ /^($Type)\s*(?:$Ident|,|\)|\(|\s*$)/) { - print "DECLARE($1)\n" if ($dbg_values > 1); - $type = 'T'; - - } elsif ($cur =~ /^($Modifier)\s*/) { - print "MODIFIER($1)\n" if ($dbg_values > 1); - $type = 'T'; - - } elsif ($cur =~ /^(\#\s*define\s*$Ident)(\(?)/o) { - print "DEFINE($1,$2)\n" if ($dbg_values > 1); - $av_preprocessor = 1; - push(@av_paren_type, $type); - if ($2 ne '') { - $av_pending = 'N'; - } - $type = 'E'; - - } elsif ($cur =~ /^(\#\s*(?:undef\s*$Ident|include\b))/o) { - print "UNDEF($1)\n" if ($dbg_values > 1); - $av_preprocessor = 1; - push(@av_paren_type, $type); - - } elsif ($cur =~ /^(\#\s*(?:ifdef|ifndef|if))/o) { - print "PRE_START($1)\n" if ($dbg_values > 1); - $av_preprocessor = 1; - - push(@av_paren_type, $type); - push(@av_paren_type, $type); - $type = 'E'; - - } elsif ($cur =~ /^(\#\s*(?:else|elif))/o) { - print "PRE_RESTART($1)\n" if ($dbg_values > 1); - $av_preprocessor = 1; - - push(@av_paren_type, $av_paren_type[$#av_paren_type]); - - $type = 'E'; - - } elsif ($cur =~ /^(\#\s*(?:endif))/o) { - print "PRE_END($1)\n" if ($dbg_values > 1); - - $av_preprocessor = 1; - - # Assume all arms of the conditional end as this - # one does, and continue as if the #endif was not here. - pop(@av_paren_type); - push(@av_paren_type, $type); - $type = 'E'; - - } elsif ($cur =~ /^(\\\n)/o) { - print "PRECONT($1)\n" if ($dbg_values > 1); - - } elsif ($cur =~ /^(__attribute__)\s*\(?/o) { - print "ATTR($1)\n" if ($dbg_values > 1); - $av_pending = $type; - $type = 'N'; - - } elsif ($cur =~ /^(sizeof)\s*(\()?/o) { - print "SIZEOF($1)\n" if ($dbg_values > 1); - if (defined $2) { - $av_pending = 'V'; - } - $type = 'N'; - - } elsif ($cur =~ /^(if|while|for)\b/o) { - print "COND($1)\n" if ($dbg_values > 1); - $av_pending = 'E'; - $type = 'N'; - - } elsif ($cur =~/^(case)/o) { - print "CASE($1)\n" if ($dbg_values > 1); - $av_pend_colon = 'C'; - $type = 'N'; - - } elsif ($cur =~/^(return|else|goto|typeof|__typeof__)\b/o) { - print "KEYWORD($1)\n" if ($dbg_values > 1); - $type = 'N'; - - } elsif ($cur =~ /^(\()/o) { - print "PAREN('$1')\n" if ($dbg_values > 1); - push(@av_paren_type, $av_pending); - $av_pending = '_'; - $type = 'N'; - - } elsif ($cur =~ /^(\))/o) { - my $new_type = pop(@av_paren_type); - if ($new_type ne '_') { - $type = $new_type; - print "PAREN('$1') -> $type\n" - if ($dbg_values > 1); - } else { - print "PAREN('$1')\n" if ($dbg_values > 1); - } - - } elsif ($cur =~ /^($Ident)\s*\(/o) { - print "FUNC($1)\n" if ($dbg_values > 1); - $type = 'V'; - $av_pending = 'V'; - - } elsif ($cur =~ /^($Ident\s*):(?:\s*\d+\s*(,|=|;))?/) { - if (defined $2 && $type eq 'C' || $type eq 'T') { - $av_pend_colon = 'B'; - } elsif ($type eq 'E') { - $av_pend_colon = 'L'; - } - print "IDENT_COLON($1,$type>$av_pend_colon)\n" if ($dbg_values > 1); - $type = 'V'; - - } elsif ($cur =~ /^($Ident|$Constant)/o) { - print "IDENT($1)\n" if ($dbg_values > 1); - $type = 'V'; - - } elsif ($cur =~ /^($Assignment)/o) { - print "ASSIGN($1)\n" if ($dbg_values > 1); - $type = 'N'; - - } elsif ($cur =~/^(;|{|})/) { - print "END($1)\n" if ($dbg_values > 1); - $type = 'E'; - $av_pend_colon = 'O'; - - } elsif ($cur =~/^(,)/) { - print "COMMA($1)\n" if ($dbg_values > 1); - $type = 'C'; - - } elsif ($cur =~ /^(\?)/o) { - print "QUESTION($1)\n" if ($dbg_values > 1); - $type = 'N'; - - } elsif ($cur =~ /^(:)/o) { - print "COLON($1,$av_pend_colon)\n" if ($dbg_values > 1); - - substr($var, length($res), 1, $av_pend_colon); - if ($av_pend_colon eq 'C' || $av_pend_colon eq 'L') { - $type = 'E'; - } else { - $type = 'N'; - } - $av_pend_colon = 'O'; - - } elsif ($cur =~ /^(\[)/o) { - print "CLOSE($1)\n" if ($dbg_values > 1); - $type = 'N'; - - } elsif ($cur =~ /^(-(?![->])|\+(?!\+)|\*|\&\&|\&)/o) { - my $variant; - - print "OPV($1)\n" if ($dbg_values > 1); - if ($type eq 'V') { - $variant = 'B'; - } else { - $variant = 'U'; - } - - substr($var, length($res), 1, $variant); - $type = 'N'; - - } elsif ($cur =~ /^($Operators)/o) { - print "OP($1)\n" if ($dbg_values > 1); - if ($1 ne '++' && $1 ne '--') { - $type = 'N'; - } - - } elsif ($cur =~ /(^.)/o) { - print "C($1)\n" if ($dbg_values > 1); - } - if (defined $1) { - $cur = substr($cur, length($1)); - $res .= $type x length($1); - } - } - - return ($res, $var); -} - -sub possible { - my ($possible, $line) = @_; - my $notPermitted = qr{(?: - ^(?: - $Modifier| - $Storage| - $Type| - DEFINE_\S+ - )$| - ^(?: - goto| - return| - case| - else| - asm|__asm__| - do| - \#| - \#\#| - )(?:\s|$)| - ^(?:typedef|struct|enum)\b - )}x; - warn "CHECK<$possible> ($line)\n" if ($dbg_possible > 2); - if ($possible !~ $notPermitted) { - # Check for modifiers. - $possible =~ s/\s*$Storage\s*//g; - $possible =~ s/\s*$Sparse\s*//g; - if ($possible =~ /^\s*$/) { - - } elsif ($possible =~ /\s/) { - $possible =~ s/\s*$Type\s*//g; - for my $modifier (split(' ', $possible)) { - if ($modifier !~ $notPermitted) { - warn "MODIFIER: $modifier ($possible) ($line)\n" if ($dbg_possible); - push(@modifierListFile, $modifier); - } - } - - } else { - warn "POSSIBLE: $possible ($line)\n" if ($dbg_possible); - push(@typeListFile, $possible); - } - build_types(); - } else { - warn "NOTPOSS: $possible ($line)\n" if ($dbg_possible > 1); - } -} - -my $prefix = ''; - -sub show_type { - my ($type) = @_; - - $type =~ tr/[a-z]/[A-Z]/; - - return defined $use_type{$type} if (scalar keys %use_type > 0); - - return !defined $ignore_type{$type}; -} - -sub report { - my ($level, $type, $msg) = @_; - - if (!show_type($type) || - (defined $tst_only && $msg !~ /\Q$tst_only\E/)) { - return 0; - } - my $output = ''; - if ($color) { - if ($level eq 'ERROR') { - $output .= RED; - } elsif ($level eq 'WARNING') { - $output .= YELLOW; - } else { - $output .= GREEN; - } - } - $output .= $prefix . $level . ':'; - if ($show_types) { - $output .= BLUE if ($color); - $output .= "$type:"; - } - $output .= RESET if ($color); - $output .= ' ' . $msg . "\n"; - - if ($showfile) { - my @lines = split("\n", $output, -1); - splice(@lines, 1, 1); - $output = join("\n", @lines); - } - $output = (split('\n', $output))[0] . "\n" if ($terse); - - push(our @report, $output); - - return 1; -} - -sub report_dump { - our @report; -} - -sub fixup_current_range { - my ($lineRef, $offset, $length) = @_; - - if ($$lineRef =~ /^\@\@ -\d+,\d+ \+(\d+),(\d+) \@\@/) { - my $o = $1; - my $l = $2; - my $no = $o + $offset; - my $nl = $l + $length; - $$lineRef =~ s/\+$o,$l \@\@/\+$no,$nl \@\@/; - } -} - -sub fix_inserted_deleted_lines { - my ($linesRef, $insertedRef, $deletedRef) = @_; - - my $range_last_linenr = 0; - my $delta_offset = 0; - - my $old_linenr = 0; - my $new_linenr = 0; - - my $next_insert = 0; - my $next_delete = 0; - - my @lines = (); - - my $inserted = @{$insertedRef}[$next_insert++]; - my $deleted = @{$deletedRef}[$next_delete++]; - - foreach my $old_line (@{$linesRef}) { - my $save_line = 1; - my $line = $old_line; #don't modify the array - if ($line =~ /^(?:\+\+\+|\-\-\-)\s+\S+/) { #new filename - $delta_offset = 0; - } elsif ($line =~ /^\@\@ -\d+,\d+ \+\d+,\d+ \@\@/) { #new hunk - $range_last_linenr = $new_linenr; - fixup_current_range(\$line, $delta_offset, 0); - } - - while (defined($deleted) && ${$deleted}{'LINENR'} == $old_linenr) { - $deleted = @{$deletedRef}[$next_delete++]; - $save_line = 0; - fixup_current_range(\$lines[$range_last_linenr], $delta_offset--, -1); - } - - while (defined($inserted) && ${$inserted}{'LINENR'} == $old_linenr) { - push(@lines, ${$inserted}{'LINE'}); - $inserted = @{$insertedRef}[$next_insert++]; - $new_linenr++; - fixup_current_range(\$lines[$range_last_linenr], $delta_offset++, 1); - } - - if ($save_line) { - push(@lines, $line); - $new_linenr++; - } - - $old_linenr++; - } - - return @lines; -} - -sub fix_insert_line { - my ($linenr, $line) = @_; - - my $inserted = { - LINENR => $linenr, - LINE => $line, - }; - push(@fixed_inserted, $inserted); -} - -sub fix_delete_line { - my ($linenr, $line) = @_; - - my $deleted = { - LINENR => $linenr, - LINE => $line, - }; - - push(@fixed_deleted, $deleted); -} - -sub ERROR { - my ($type, $msg) = @_; - - if (report("ERROR", $type, $msg)) { - our $clean = 0; - our $cnt_error++; - return 1; - } - return 0; -} -sub WARN { - my ($type, $msg) = @_; - - if (report("WARNING", $type, $msg)) { - our $clean = 0; - our $cnt_warn++; - return 1; - } - return 0; -} -sub CHK { - my ($type, $msg) = @_; - - if ($check && report("CHECK", $type, $msg)) { - our $clean = 0; - our $cnt_chk++; - return 1; - } - return 0; -} - -sub check_absolute_file { - my ($absolute, $herecurr) = @_; - my $file = $absolute; - - ##print "absolute<$absolute>\n"; - - # See if any suffix of this path is a path within the tree. - while ($file =~ s@^[^/]*/@@) { - if (-f "$root/$file") { - ##print "file<$file>\n"; - last; - } - } - if (! -f _) { - return 0; - } - - # It is, so see if the prefix is acceptable. - my $prefix = $absolute; - substr($prefix, -length($file)) = ''; - - ##print "prefix<$prefix>\n"; - if ($prefix ne ".../") { - WARN("USE_RELATIVE_PATH", - "use relative pathname instead of absolute in changelog text\n" . $herecurr); - } -} - -sub trim { - my ($string) = @_; - - $string =~ s/^\s+|\s+$//g; - - return $string; -} - -sub ltrim { - my ($string) = @_; - - $string =~ s/^\s+//; - - return $string; -} - -sub rtrim { - my ($string) = @_; - - $string =~ s/\s+$//; - - return $string; -} - -sub string_find_replace { - my ($string, $find, $replace) = @_; - - $string =~ s/$find/$replace/g; - - return $string; -} - -sub tabify { - my ($leading) = @_; - - my $source_indent = 8; - my $max_spaces_before_tab = $source_indent - 1; - my $spaces_to_tab = " " x $source_indent; - - #convert leading spaces to tabs - 1 while $leading =~ s@^([\t]*)$spaces_to_tab@$1\t@g; - #Remove spaces before a tab - 1 while $leading =~ s@^([\t]*)( {1,$max_spaces_before_tab})\t@$1\t@g; - - return "$leading"; -} - -sub pos_last_openparen { - my ($line) = @_; - - my $pos = 0; - - my $opens = $line =~ tr/\(/\(/; - my $closes = $line =~ tr/\)/\)/; - - my $last_openparen = 0; - - if (($opens == 0) || ($closes >= $opens)) { - return -1; - } - - my $len = length($line); - - for ($pos = 0; $pos < $len; $pos++) { - my $string = substr($line, $pos); - if ($string =~ /^($FuncArg|$balanced_parens)/) { - $pos += length($1) - 1; - } elsif (substr($line, $pos, 1) eq '(') { - $last_openparen = $pos; - } elsif (index($string, '(') == -1) { - last; - } - } - - return length(expand_tabs(substr($line, 0, $last_openparen))) + 1; -} - -sub process { - my $filename = shift; - - my $linenr=0; - my $prevline=""; - my $prevrawline=""; - my $stashline=""; - my $stashrawline=""; - - my $length; - my $indent; - my $previndent=0; - my $stashindent=0; - - our $clean = 1; - my $signoff = 0; - my $is_patch = 0; - my $in_header_lines = $file ? 0 : 1; - my $in_commit_log = 0; #Scanning lines before patch - my $has_commit_log = 0; #Encountered lines before patch - my $commit_log_possible_stack_dump = 0; - my $commit_log_long_line = 0; - my $commit_log_has_diff = 0; - my $reported_maintainer_file = 0; - my $non_utf8_charset = 0; - - my $last_blank_line = 0; - my $last_coalesced_string_linenr = -1; - - our @report = (); - our $cnt_lines = 0; - our $cnt_error = 0; - our $cnt_warn = 0; - our $cnt_chk = 0; - - # Trace the real file/line as we go. - my $realfile = ''; - my $realline = 0; - my $realcnt = 0; - my $here = ''; - my $context_function; #undef'd unless there's a known function - my $in_comment = 0; - my $comment_edge = 0; - my $first_line = 0; - my $p1_prefix = ''; - - my $prev_values = 'E'; - - # suppression flags - my %suppress_ifbraces; - my %suppress_whiletrailers; - my %suppress_export; - my $suppress_statement = 0; - - my %signatures = (); - - # Pre-scan the patch sanitizing the lines. - # Pre-scan the patch looking for any __setup documentation. - # - my @setup_docs = (); - my $setup_docs = 0; - - my $camelcase_file_seeded = 0; - - sanitise_line_reset(); - my $line; - foreach my $rawline (@rawlines) { - $linenr++; - $line = $rawline; - - push(@fixed, $rawline) if ($fix); - - if ($rawline=~/^\+\+\+\s+(\S+)/) { - $setup_docs = 0; - if ($1 =~ m@Documentation/admin-guide/kernel-parameters.rst$@) { - $setup_docs = 1; - } - #next; - } - if ($rawline =~ /^\@\@ -\d+(?:,\d+)? \+(\d+)(,(\d+))? \@\@/) { - $realline=$1-1; - if (defined $2) { - $realcnt=$3+1; - } else { - $realcnt=1+1; - } - $in_comment = 0; - - # Guestimate if this is a continuing comment. Run - # the context looking for a comment "edge". If this - # edge is a close comment then we must be in a comment - # at context start. - my $edge; - my $cnt = $realcnt; - for (my $ln = $linenr + 1; $cnt > 0; $ln++) { - next if (defined $rawlines[$ln - 1] && - $rawlines[$ln - 1] =~ /^-/); - $cnt--; - #print "RAW<$rawlines[$ln - 1]>\n"; - last if (!defined $rawlines[$ln - 1]); - if ($rawlines[$ln - 1] =~ m@(/\*|\*/)@ && - $rawlines[$ln - 1] !~ m@"[^"]*(?:/\*|\*/)[^"]*"@) { - ($edge) = $1; - last; - } - } - if (defined $edge && $edge eq '*/') { - $in_comment = 1; - } - - # Guestimate if this is a continuing comment. If this - # is the start of a diff block and this line starts - # ' *' then it is very likely a comment. - if (!defined $edge && - $rawlines[$linenr] =~ m@^.\s*(?:\*\*+| \*)(?:\s|$)@) - { - $in_comment = 1; - } - - ##print "COMMENT:$in_comment edge<$edge> $rawline\n"; - sanitise_line_reset($in_comment); - - } elsif ($realcnt && $rawline =~ /^(?:\+| |$)/) { - # Standardise the strings and chars within the input to - # simplify matching -- only bother with positive lines. - $line = sanitise_line($rawline); - } - push(@lines, $line); - - if ($realcnt > 1) { - $realcnt-- if ($line =~ /^(?:\+| |$)/); - } else { - $realcnt = 0; - } - - #print "==>$rawline\n"; - #print "-->$line\n"; - - if ($setup_docs && $line =~ /^\+/) { - push(@setup_docs, $line); - } - } - - $prefix = ''; - - $realcnt = 0; - $linenr = 0; - $fixlinenr = -1; - foreach my $line (@lines) { - $linenr++; - $fixlinenr++; - my $sline = $line; #copy of $line - $sline =~ s/$;/ /g; #with comments as spaces - - my $rawline = $rawlines[$linenr - 1]; - -#extract the line range in the file after the patch is applied - if (!$in_commit_log && - $line =~ /^\@\@ -\d+(?:,\d+)? \+(\d+)(,(\d+))? \@\@(.*)/) { - my $context = $4; - $is_patch = 1; - $first_line = $linenr + 1; - $realline=$1-1; - if (defined $2) { - $realcnt=$3+1; - } else { - $realcnt=1+1; - } - annotate_reset(); - $prev_values = 'E'; - - %suppress_ifbraces = (); - %suppress_whiletrailers = (); - %suppress_export = (); - $suppress_statement = 0; - if ($context =~ /\b(\w+)\s*\(/) { - $context_function = $1; - } else { - undef $context_function; - } - next; - -# track the line number as we move through the hunk, note that -# new versions of GNU diff omit the leading space on completely -# blank context lines so we need to count that too. - } elsif ($line =~ /^( |\+|$)/) { - $realline++; - $realcnt-- if ($realcnt != 0); - - # Measure the line length and indent. - ($length, $indent) = line_stats($rawline); - - # Track the previous line. - ($prevline, $stashline) = ($stashline, $line); - ($previndent, $stashindent) = ($stashindent, $indent); - ($prevrawline, $stashrawline) = ($stashrawline, $rawline); - - #warn "line<$line>\n"; - - } elsif ($realcnt == 1) { - $realcnt--; - } - - my $hunk_line = ($realcnt != 0); - - $here = "#$linenr: " if (!$file); - $here = "#$realline: " if ($file); - - my $found_file = 0; - # extract the filename as it passes - if ($line =~ /^diff --git.*?(\S+)$/) { - $realfile = $1; - $realfile =~ s@^([^/]*)/@@ if (!$file); - $in_commit_log = 0; - $found_file = 1; - } elsif ($line =~ /^\+\+\+\s+(\S+)/) { - $realfile = $1; - $realfile =~ s@^([^/]*)/@@ if (!$file); - $in_commit_log = 0; - - $p1_prefix = $1; - if (!$file && $tree && $p1_prefix ne '' && - -e "$root/$p1_prefix") { - WARN("PATCH_PREFIX", - "patch prefix '$p1_prefix' exists, appears to be a -p0 patch\n"); - } - - if ($realfile =~ m@^include/asm/@) { - ERROR("MODIFIED_INCLUDE_ASM", - "do not modify files in include/asm, change architecture specific files in include/asm-<architecture>\n" . "$here$rawline\n"); - } - $found_file = 1; - } - my $skipme = 0; - foreach (@exclude) { - if ($realfile =~ m@^(?:$_/)@) { - $skipme = 1; - } - } - if ($skipme) { - next; - } - -#make up the handle for any error we report on this line - if ($showfile) { - $prefix = "$realfile:$realline: " - } elsif ($emacs) { - if ($file) { - $prefix = "$filename:$realline: "; - } else { - $prefix = "$filename:$linenr: "; - } - } - - if ($found_file) { - if (is_maintained_obsolete($realfile)) { - WARN("OBSOLETE", - "$realfile is marked as 'obsolete' in the MAINTAINERS hierarchy. No unnecessary modifications please.\n"); - } - if ($realfile =~ m@^(?:drivers/net/|net/|drivers/staging/)@) { - $check = 1; - } else { - $check = $check_orig; - } - next; - } - - $here .= "FILE: $realfile:$realline:" if ($realcnt != 0); - - my $hereline = "$here\n$rawline\n"; - my $herecurr = "$here\n$rawline\n"; - my $hereprev = "$here\n$prevrawline\n$rawline\n"; - - $cnt_lines++ if ($realcnt != 0); - -# Check if the commit log has what seems like a diff which can confuse patch - if ($in_commit_log && !$commit_log_has_diff && - (($line =~ m@^\s+diff\b.*a/[\w/]+@ && - $line =~ m@^\s+diff\b.*a/([\w/]+)\s+b/$1\b@) || - $line =~ m@^\s*(?:\-\-\-\s+a/|\+\+\+\s+b/)@ || - $line =~ m/^\s*\@\@ \-\d+,\d+ \+\d+,\d+ \@\@/)) { - ERROR("DIFF_IN_COMMIT_MSG", - "Avoid using diff content in the commit message - patch(1) might not work\n" . $herecurr); - $commit_log_has_diff = 1; - } - -# Check for incorrect file permissions - if ($line =~ /^new (file )?mode.*[7531]\d{0,2}$/) { - my $permhere = $here . "FILE: $realfile\n"; - if ($realfile !~ m@scripts/@ && - $realfile !~ /\.(py|pl|awk|sh)$/) { - ERROR("EXECUTE_PERMISSIONS", - "do not set execute permissions for source files\n" . $permhere); - } - } - -# Check the patch for a signoff: - if ($line =~ /^\s*signed-off-by:/i) { - $signoff++; - $in_commit_log = 0; - } - -# Check if CODEOWNERS is being updated. If so, there's probably no need to -# emit the "does CODEOWNERS need updating?" message on file add/move/delete - if ($line =~ /^\s*CODEOWNERS\s*\|/) { - $reported_maintainer_file = 1; - } - -# Check signature styles - if (!$in_header_lines && - $line =~ /^(\s*)([a-z0-9_-]+by:|$signature_tags)(\s*)(.*)/i) { - my $space_before = $1; - my $sign_off = $2; - my $space_after = $3; - my $email = $4; - my $ucfirst_sign_off = ucfirst(lc($sign_off)); - - if ($sign_off !~ /$signature_tags/) { - WARN("BAD_SIGN_OFF", - "Non-standard signature: $sign_off\n" . $herecurr); - } - if (defined $space_before && $space_before ne "") { - if (WARN("BAD_SIGN_OFF", - "Do not use whitespace before $ucfirst_sign_off\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] = - "$ucfirst_sign_off $email"; - } - } - if ($sign_off =~ /-by:$/i && $sign_off ne $ucfirst_sign_off) { - if (WARN("BAD_SIGN_OFF", - "'$ucfirst_sign_off' is the preferred signature form\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] = - "$ucfirst_sign_off $email"; - } - - } - if (!defined $space_after || $space_after ne " ") { - if (WARN("BAD_SIGN_OFF", - "Use a single space after $ucfirst_sign_off\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] = - "$ucfirst_sign_off $email"; - } - } - - my ($email_name, $email_address, $comment) = parse_email($email); - my $suggested_email = format_email(($email_name, $email_address)); - if ($suggested_email eq "") { - ERROR("BAD_SIGN_OFF", - "Unrecognized email address: '$email'\n" . $herecurr); - } else { - my $dequoted = $suggested_email; - $dequoted =~ s/^"//; - $dequoted =~ s/" </ </; - # Don't force email to have quotes - # Allow just an angle bracketed address - if ("$dequoted$comment" ne $email && - "<$email_address>$comment" ne $email && - "$suggested_email$comment" ne $email) { - WARN("BAD_SIGN_OFF", - "email address '$email' might be better as '$suggested_email$comment'\n" . $herecurr); - } - } - -# Check for duplicate signatures - my $sig_nospace = $line; - $sig_nospace =~ s/\s//g; - $sig_nospace = lc($sig_nospace); - if (defined $signatures{$sig_nospace}) { - WARN("BAD_SIGN_OFF", - "Duplicate signature\n" . $herecurr); - } else { - $signatures{$sig_nospace} = 1; - } - } - -# Check email subject for common tools that don't need to be mentioned - if ($in_header_lines && - $line =~ /^Subject:.*\b(?:checkpatch|sparse|smatch)\b[^:]/i) { - WARN("EMAIL_SUBJECT", - "A patch subject line should describe the change not the tool that found it\n" . $herecurr); - } - -# Check for old stable address - if ($line =~ /^\s*cc:\s*.*<?\bstable\@kernel\.org\b>?.*$/i) { - ERROR("STABLE_ADDRESS", - "The 'stable' address should be 'stable\@vger.kernel.org'\n" . $herecurr); - } - -# Check for unwanted Gerrit info - if ($in_commit_log && $line =~ /^\s*change-id:/i) { - ERROR("GERRIT_CHANGE_ID", - "Remove Gerrit Change-Id's before submitting upstream.\n" . $herecurr); - } - -# Check if the commit log is in a possible stack dump - if ($in_commit_log && !$commit_log_possible_stack_dump && - ($line =~ /^\s*(?:WARNING:|BUG:)/ || - $line =~ /^\s*\[\s*\d+\.\d{6,6}\s*\]/ || - # timestamp - $line =~ /^\s*\[\<[0-9a-fA-F]{8,}\>\]/)) { - # stack dump address - $commit_log_possible_stack_dump = 1; - } - -# Check for line lengths > 75 in commit log, warn once - if ($in_commit_log && !$commit_log_long_line && - length($line) > 75 && - !($line =~ /^\s*[a-zA-Z0-9_\/\.]+\s+\|\s+\d+/ || - # file delta changes - $line =~ /^\s*(?:[\w\.\-]+\/)++[\w\.\-]+:/ || - # filename then : - $line =~ /^\s*(?:Fixes:|Link:)/i || - # A Fixes: or Link: line - $commit_log_possible_stack_dump)) { - WARN("COMMIT_LOG_LONG_LINE", - "Possible unwrapped commit description (prefer a maximum 75 chars per line)\n" . $herecurr); - $commit_log_long_line = 1; - } - -# Reset possible stack dump if a blank line is found - if ($in_commit_log && $commit_log_possible_stack_dump && - $line =~ /^\s*$/) { - $commit_log_possible_stack_dump = 0; - } - -# Check for git id commit length and improperly formed commit descriptions - if ($in_commit_log && !$commit_log_possible_stack_dump && - $line !~ /^\s*(?:Link|Patchwork|http|https|BugLink):/i && - $line !~ /^This reverts commit [0-9a-f]{7,40}/ && - ($line =~ /\bcommit\s+[0-9a-f]{5,}\b/i || - ($line =~ /(?:\s|^)[0-9a-f]{12,40}(?:[\s"'\(\[]|$)/i && - $line !~ /[\<\[][0-9a-f]{12,40}[\>\]]/i && - $line !~ /\bfixes:\s*[0-9a-f]{12,40}/i))) { - my $init_char = "c"; - my $orig_commit = ""; - my $short = 1; - my $long = 0; - my $case = 1; - my $space = 1; - my $hasdesc = 0; - my $hasparens = 0; - my $id = '0123456789ab'; - my $orig_desc = "commit description"; - my $description = ""; - - if ($line =~ /\b(c)ommit\s+([0-9a-f]{5,})\b/i) { - $init_char = $1; - $orig_commit = lc($2); - } elsif ($line =~ /\b([0-9a-f]{12,40})\b/i) { - $orig_commit = lc($1); - } - - $short = 0 if ($line =~ /\bcommit\s+[0-9a-f]{12,40}/i); - $long = 1 if ($line =~ /\bcommit\s+[0-9a-f]{41,}/i); - $space = 0 if ($line =~ /\bcommit [0-9a-f]/i); - $case = 0 if ($line =~ /\b[Cc]ommit\s+[0-9a-f]{5,40}[^A-F]/); - if ($line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("([^"]+)"\)/i) { - $orig_desc = $1; - $hasparens = 1; - } elsif ($line =~ /\bcommit\s+[0-9a-f]{5,}\s*$/i && - defined $rawlines[$linenr] && - $rawlines[$linenr] =~ /^\s*\("([^"]+)"\)/) { - $orig_desc = $1; - $hasparens = 1; - } elsif ($line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("[^"]+$/i && - defined $rawlines[$linenr] && - $rawlines[$linenr] =~ /^\s*[^"]+"\)/) { - $line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("([^"]+)$/i; - $orig_desc = $1; - $rawlines[$linenr] =~ /^\s*([^"]+)"\)/; - $orig_desc .= " " . $1; - $hasparens = 1; - } - - ($id, $description) = git_commit_info($orig_commit, - $id, $orig_desc); - - if (defined($id) && - ($short || $long || $space || $case || ($orig_desc ne $description) || !$hasparens)) { - ERROR("GIT_COMMIT_ID", - "Please use git commit description style 'commit <12+ chars of sha1> (\"<title line>\")' - ie: '${init_char}ommit $id (\"$description\")'\n" . $herecurr); - } - } - -# Check for added, moved or deleted files - if (!$reported_maintainer_file && !$in_commit_log && - ($line =~ /^(?:new|deleted) file mode\s*\d+\s*$/ || - $line =~ /^rename (?:from|to) [\w\/\.\-]+\s*$/ || - ($line =~ /\{\s*([\w\/\.\-]*)\s*\=\>\s*([\w\/\.\-]*)\s*\}/ && - (defined($1) || defined($2))))) { - $is_patch = 1; - $reported_maintainer_file = 1; - WARN("FILE_PATH_CHANGES", - "added, moved or deleted file(s), does CODEOWNERS need updating?\n" . $herecurr); - } - -# Check for wrappage within a valid hunk of the file - if ($realcnt != 0 && $line !~ m{^(?:\+|-| |\\ No newline|$)}) { - ERROR("CORRUPTED_PATCH", - "patch seems to be corrupt (line wrapped?)\n" . - $herecurr) if (!$emitted_corrupt++); - } - -# UTF-8 regex found at http://www.w3.org/International/questions/qa-forms-utf-8.en.php - if (($realfile =~ /^$/ || $line =~ /^\+/) && - $rawline !~ m/^$UTF8*$/) { - my ($utf8_prefix) = ($rawline =~ /^($UTF8*)/); - - my $blank = copy_spacing($rawline); - my $ptr = substr($blank, 0, length($utf8_prefix)) . "^"; - my $hereptr = "$hereline$ptr\n"; - - CHK("INVALID_UTF8", - "Invalid UTF-8, patch and commit message should be encoded in UTF-8\n" . $hereptr); - } - -# Check if it's the start of a commit log -# (not a header line and we haven't seen the patch filename) - if ($in_header_lines && $realfile =~ /^$/ && - !($rawline =~ /^\s+(?:\S|$)/ || - $rawline =~ /^(?:commit\b|from\b|[\w-]+:)/i)) { - $in_header_lines = 0; - $in_commit_log = 1; - $has_commit_log = 1; - } - -# Check if there is UTF-8 in a commit log when a mail header has explicitly -# declined it, i.e defined some charset where it is missing. - if ($in_header_lines && - $rawline =~ /^Content-Type:.+charset="(.+)".*$/ && - $1 !~ /utf-8/i) { - $non_utf8_charset = 1; - } - - if ($in_commit_log && $non_utf8_charset && $realfile =~ /^$/ && - $rawline =~ /$NON_ASCII_UTF8/) { - WARN("UTF8_BEFORE_PATCH", - "8-bit UTF-8 used in possible commit log\n" . $herecurr); - } - -# Check for absolute kernel paths in commit message - if ($tree && $in_commit_log) { - while ($line =~ m{(?:^|\s)(/\S*)}g) { - my $file = $1; - - if ($file =~ m{^(.*?)(?::\d+)+:?$} && - check_absolute_file($1, $herecurr)) { - # - } else { - check_absolute_file($file, $herecurr); - } - } - } - -# Check for various typo / spelling mistakes - if (defined($misspellings) && - ($in_commit_log || $line =~ /^(?:\+|Subject:)/i)) { - while ($rawline =~ /(?:^|[^a-z@])($misspellings)(?:\b|$|[^a-z@])/gi) { - my $typo = $1; - my $typo_fix = $spelling_fix{lc($typo)}; - $typo_fix = ucfirst($typo_fix) if ($typo =~ /^[A-Z]/); - $typo_fix = uc($typo_fix) if ($typo =~ /^[A-Z]+$/); - my $msg_level = \&WARN; - $msg_level = \&CHK if ($file); - if (&{$msg_level}("TYPO_SPELLING", - "'$typo' may be misspelled - perhaps '$typo_fix'?\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/(^|[^A-Za-z@])($typo)($|[^A-Za-z@])/$1$typo_fix$3/; - } - } - } - -# ignore non-hunk lines and lines being removed - next if (!$hunk_line || $line =~ /^-/); - -#trailing whitespace - if ($line =~ /^\+.*\015/) { - my $herevet = "$here\n" . cat_vet($rawline) . "\n"; - if (ERROR("DOS_LINE_ENDINGS", - "DOS line endings\n" . $herevet) && - $fix) { - $fixed[$fixlinenr] =~ s/[\s\015]+$//; - } - } elsif ($rawline =~ /^\+.*\S\s+$/ || $rawline =~ /^\+\s+$/) { - my $herevet = "$here\n" . cat_vet($rawline) . "\n"; - if (ERROR("TRAILING_WHITESPACE", - "trailing whitespace\n" . $herevet) && - $fix) { - $fixed[$fixlinenr] =~ s/\s+$//; - } - - $rpt_cleaners = 1; - } - -# Check for FSF mailing addresses. - if ($rawline =~ /\bwrite to the Free/i || - $rawline =~ /\b675\s+Mass\s+Ave/i || - $rawline =~ /\b59\s+Temple\s+Pl/i || - $rawline =~ /\b51\s+Franklin\s+St/i) { - my $herevet = "$here\n" . cat_vet($rawline) . "\n"; - my $msg_level = \&ERROR; - $msg_level = \&CHK if ($file); - &{$msg_level}("FSF_MAILING_ADDRESS", - "Do not include the paragraph about writing to the Free Software Foundation's mailing address from the sample GPL notice. The FSF has changed addresses in the past, and may do so again. Linux already includes a copy of the GPL.\n" . $herevet) - } - -# check for Kconfig help text having a real description -# Only applies when adding the entry originally, after that we do not have -# sufficient context to determine whether it is indeed long enough. - if ($realfile =~ /Kconfig/ && - $line =~ /^\+\s*config\s+/) { - my $length = 0; - my $cnt = $realcnt; - my $ln = $linenr + 1; - my $f; - my $is_start = 0; - my $is_end = 0; - for (; $cnt > 0 && defined $lines[$ln - 1]; $ln++) { - $f = $lines[$ln - 1]; - $cnt-- if ($lines[$ln - 1] !~ /^-/); - $is_end = $lines[$ln - 1] =~ /^\+/; - - next if ($f =~ /^-/); - last if (!$file && $f =~ /^\@\@/); - - if ($lines[$ln - 1] =~ /^\+\s*(?:bool|tristate)\s*\"/) { - $is_start = 1; - } elsif ($lines[$ln - 1] =~ /^\+\s*(?:---)?help(?:---)?$/) { - $length = -1; - } - - $f =~ s/^.//; - $f =~ s/#.*//; - $f =~ s/^\s+//; - next if ($f =~ /^$/); - if ($f =~ /^\s*config\s/) { - $is_end = 1; - last; - } - $length++; - } - if ($is_start && $is_end && $length < $min_conf_desc_length) { - WARN("CONFIG_DESCRIPTION", - "please write a paragraph that describes the config symbol fully\n" . $herecurr); - } - #print "is_start<$is_start> is_end<$is_end> length<$length>\n"; - } - -# check for MAINTAINERS entries that don't have the right form - if ($realfile =~ /^MAINTAINERS$/ && - $rawline =~ /^\+[A-Z]:/ && - $rawline !~ /^\+[A-Z]:\t\S/) { - if (WARN("MAINTAINERS_STYLE", - "MAINTAINERS entries use one tab after TYPE:\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/^(\+[A-Z]):\s*/$1:\t/; - } - } - -# discourage the use of boolean for type definition attributes of Kconfig options - if ($realfile =~ /Kconfig/ && - $line =~ /^\+\s*\bboolean\b/) { - WARN("CONFIG_TYPE_BOOLEAN", - "Use of boolean is deprecated, please use bool instead.\n" . $herecurr); - } - - if (($realfile =~ /Makefile.*/ || $realfile =~ /Kbuild.*/) && - ($line =~ /\+(EXTRA_[A-Z]+FLAGS).*/)) { - my $flag = $1; - my $replacement = { - 'EXTRA_AFLAGS' => 'asflags-y', - 'EXTRA_CFLAGS' => 'ccflags-y', - 'EXTRA_CPPFLAGS' => 'cppflags-y', - 'EXTRA_LDFLAGS' => 'ldflags-y', - }; - - WARN("DEPRECATED_VARIABLE", - "Use of $flag is deprecated, please use \`$replacement->{$flag} instead.\n" . $herecurr) if ($replacement->{$flag}); - } -# Kconfig use tabs and no spaces in line - if ($realfile =~ /Kconfig/ && $rawline =~ /^\+ /) { - my $herevet = "$here\n" . cat_vet($rawline) . "\n"; - WARN("LEADING_SPACE", - "please, no spaces at the start of a line\n" . $herevet); - } - -# check for DT compatible documentation - if (defined $root && - (($realfile =~ /\.dtsi?$/ && $line =~ /^\+\s*compatible\s*=\s*\"/) || - ($realfile =~ /\.[ch]$/ && $line =~ /^\+.*\.compatible\s*=\s*\"/))) { - - my @compats = $rawline =~ /\"([a-zA-Z0-9\-\,\.\+_]+)\"/g; - - my $dt_path = $root . "/dts/bindings/"; - my $vp_file = $dt_path . "vendor-prefixes.txt"; - - foreach my $compat (@compats) { - my $compat2 = $compat; - $compat2 =~ s/\,[a-zA-Z0-9]*\-/\,<\.\*>\-/; - my $compat3 = $compat; - $compat3 =~ s/\,([a-z]*)[0-9]*\-/\,$1<\.\*>\-/; - `grep -Erq "$compat|$compat2|$compat3" $dt_path`; - if ( $? >> 8 ) { - WARN("UNDOCUMENTED_DT_STRING", - "DT compatible string \"$compat\" appears un-documented -- check $dt_path\n" . $herecurr); - } - - next if $compat !~ /^([a-zA-Z0-9\-]+)\,/; - my $vendor = $1; - `grep -Eq "^$vendor\\b" $vp_file`; - if ( $? >> 8 ) { - WARN("UNDOCUMENTED_DT_STRING", - "DT compatible string vendor \"$vendor\" appears un-documented -- check $vp_file\n" . $herecurr); - } - } - } - -# check we are in a valid source file if not then ignore this hunk - next if ($realfile !~ /\.(h|c|s|S|sh|dtsi|dts)$/); - -# line length limit (with some exclusions) -# -# There are a few types of lines that may extend beyond $max_line_length: -# logging functions like pr_info that end in a string -# lines with a single string -# #defines that are a single string -# -# There are 3 different line length message types: -# LONG_LINE_COMMENT a comment starts before but extends beyond $max_line_length -# LONG_LINE_STRING a string starts before but extends beyond $max_line_length -# LONG_LINE all other lines longer than $max_line_length -# -# if LONG_LINE is ignored, the other 2 types are also ignored -# - - if ($line =~ /^\+/ && $length > $max_line_length) { - my $msg_type = "LONG_LINE"; - - # Check the allowed long line types first - - # logging functions that end in a string that starts - # before $max_line_length - if ($line =~ /^\+\s*$logFunctions\s*\(\s*(?:(?:KERN_\S+\s*|[^"]*))?($String\s*(?:|,|\)\s*;)\s*)$/ && - length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) { - $msg_type = ""; - - # lines with only strings (w/ possible termination) - # #defines with only strings - } elsif ($line =~ /^\+\s*$String\s*(?:\s*|,|\)\s*;)\s*$/ || - $line =~ /^\+\s*#\s*define\s+\w+\s+$String$/) { - $msg_type = ""; - - # EFI_GUID is another special case - } elsif ($line =~ /^\+.*\bEFI_GUID\s*\(/) { - $msg_type = ""; - - # Otherwise set the alternate message types - - # a comment starts before $max_line_length - } elsif ($line =~ /($;[\s$;]*)$/ && - length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) { - $msg_type = "LONG_LINE_COMMENT" - - # a quoted string starts before $max_line_length - } elsif ($sline =~ /\s*($String(?:\s*(?:\\|,\s*|\)\s*;\s*))?)$/ && - length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) { - $msg_type = "LONG_LINE_STRING" - } - - if ($msg_type ne "" && - (show_type("LONG_LINE") || show_type($msg_type))) { - WARN($msg_type, - "line over $max_line_length characters\n" . $herecurr); - } - } - -# check for adding lines without a newline. - if ($line =~ /^\+/ && defined $lines[$linenr] && $lines[$linenr] =~ /^\\ No newline at end of file/) { - WARN("MISSING_EOF_NEWLINE", - "adding a line without newline at end of file\n" . $herecurr); - } - -# Blackfin: use hi/lo macros - if ($realfile =~ m@arch/blackfin/.*\.S$@) { - if ($line =~ /\.[lL][[:space:]]*=.*&[[:space:]]*0x[fF][fF][fF][fF]/) { - my $herevet = "$here\n" . cat_vet($line) . "\n"; - ERROR("LO_MACRO", - "use the LO() macro, not (... & 0xFFFF)\n" . $herevet); - } - if ($line =~ /\.[hH][[:space:]]*=.*>>[[:space:]]*16/) { - my $herevet = "$here\n" . cat_vet($line) . "\n"; - ERROR("HI_MACRO", - "use the HI() macro, not (... >> 16)\n" . $herevet); - } - } - -# check we are in a valid source file C or perl if not then ignore this hunk - next if ($realfile !~ /\.(h|c|pl|dtsi|dts)$/); - -# at the beginning of a line any tabs must come first and anything -# more than 8 must use tabs. - if ($rawline =~ /^\+\s* \t\s*\S/ || - $rawline =~ /^\+\s* \s*/) { - my $herevet = "$here\n" . cat_vet($rawline) . "\n"; - $rpt_cleaners = 1; - if (ERROR("CODE_INDENT", - "code indent should use tabs where possible\n" . $herevet) && - $fix) { - $fixed[$fixlinenr] =~ s/^\+([ \t]+)/"\+" . tabify($1)/e; - } - } - -# check for space before tabs. - if ($rawline =~ /^\+/ && $rawline =~ / \t/) { - my $herevet = "$here\n" . cat_vet($rawline) . "\n"; - if (WARN("SPACE_BEFORE_TAB", - "please, no space before tabs\n" . $herevet) && - $fix) { - while ($fixed[$fixlinenr] =~ - s/(^\+.*) {8,8}\t/$1\t\t/) {} - while ($fixed[$fixlinenr] =~ - s/(^\+.*) +\t/$1\t/) {} - } - } - -# check for && or || at the start of a line - if ($rawline =~ /^\+\s*(&&|\|\|)/) { - CHK("LOGICAL_CONTINUATIONS", - "Logical continuations should be on the previous line\n" . $hereprev); - } - -# check indentation starts on a tab stop - if ($^V && $^V ge 5.10.0 && - $sline =~ /^\+\t+( +)(?:$c90_Keywords\b|\{\s*$|\}\s*(?:else\b|while\b|\s*$))/) { - my $indent = length($1); - if ($indent % 8) { - if (WARN("TABSTOP", - "Statements should start on a tabstop\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s@(^\+\t+) +@$1 . "\t" x ($indent/8)@e; - } - } - } - -# check multi-line statement indentation matches previous line - if ($^V && $^V ge 5.10.0 && - $prevline =~ /^\+([ \t]*)((?:$c90_Keywords(?:\s+if)\s*)|(?:$Declare\s*)?(?:$Ident|\(\s*\*\s*$Ident\s*\))\s*|(?:\*\s*)*$Lval\s*=\s*$Ident\s*)\(.*(\&\&|\|\||,)\s*$/) { - $prevline =~ /^\+(\t*)(.*)$/; - my $oldindent = $1; - my $rest = $2; - - my $pos = pos_last_openparen($rest); - if ($pos >= 0) { - $line =~ /^(\+| )([ \t]*)/; - my $newindent = $2; - - my $goodtabindent = $oldindent . - "\t" x ($pos / 8) . - " " x ($pos % 8); - my $goodspaceindent = $oldindent . " " x $pos; - - if ($newindent ne $goodtabindent && - $newindent ne $goodspaceindent) { - - if (CHK("PARENTHESIS_ALIGNMENT", - "Alignment should match open parenthesis\n" . $hereprev) && - $fix && $line =~ /^\+/) { - $fixed[$fixlinenr] =~ - s/^\+[ \t]*/\+$goodtabindent/; - } - } - } - } - -# check for space after cast like "(int) foo" or "(struct foo) bar" -# avoid checking a few false positives: -# "sizeof(<type>)" or "__alignof__(<type>)" -# function pointer declarations like "(*foo)(int) = bar;" -# structure definitions like "(struct foo) { 0 };" -# multiline macros that define functions -# known attributes or the __attribute__ keyword - if ($line =~ /^\+(.*)\(\s*$Type\s*\)([ \t]++)((?![={]|\\$|$Attribute|__attribute__))/ && - (!defined($1) || $1 !~ /\b(?:sizeof|__alignof__)\s*$/)) { - if (CHK("SPACING", - "No space is necessary after a cast\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/(\(\s*$Type\s*\))[ \t]+/$1/; - } - } - -# Block comment styles -# Networking with an initial /* - if ($realfile =~ m@^(drivers/net/|net/)@ && - $prevrawline =~ /^\+[ \t]*\/\*[ \t]*$/ && - $rawline =~ /^\+[ \t]*\*/ && - $realline > 2) { - WARN("NETWORKING_BLOCK_COMMENT_STYLE", - "networking block comments don't use an empty /* line, use /* Comment...\n" . $hereprev); - } - -# Block comments use * on subsequent lines - if ($prevline =~ /$;[ \t]*$/ && #ends in comment - $prevrawline =~ /^\+.*?\/\*/ && #starting /* - $prevrawline !~ /\*\/[ \t]*$/ && #no trailing */ - $rawline =~ /^\+/ && #line is new - $rawline !~ /^\+[ \t]*\*/) { #no leading * - WARN("BLOCK_COMMENT_STYLE", - "Block comments use * on subsequent lines\n" . $hereprev); - } - -# Block comments use */ on trailing lines - if ($rawline !~ m@^\+[ \t]*\*/[ \t]*$@ && #trailing */ - $rawline !~ m@^\+.*/\*.*\*/[ \t]*$@ && #inline /*...*/ - $rawline !~ m@^\+.*\*{2,}/[ \t]*$@ && #trailing **/ - $rawline =~ m@^\+[ \t]*.+\*\/[ \t]*$@) { #non blank */ - WARN("BLOCK_COMMENT_STYLE", - "Block comments use a trailing */ on a separate line\n" . $herecurr); - } - -# Block comment * alignment - if ($prevline =~ /$;[ \t]*$/ && #ends in comment - $line =~ /^\+[ \t]*$;/ && #leading comment - $rawline =~ /^\+[ \t]*\*/ && #leading * - (($prevrawline =~ /^\+.*?\/\*/ && #leading /* - $prevrawline !~ /\*\/[ \t]*$/) || #no trailing */ - $prevrawline =~ /^\+[ \t]*\*/)) { #leading * - my $oldindent; - $prevrawline =~ m@^\+([ \t]*/?)\*@; - if (defined($1)) { - $oldindent = expand_tabs($1); - } else { - $prevrawline =~ m@^\+(.*/?)\*@; - $oldindent = expand_tabs($1); - } - $rawline =~ m@^\+([ \t]*)\*@; - my $newindent = $1; - $newindent = expand_tabs($newindent); - if (length($oldindent) ne length($newindent)) { - WARN("BLOCK_COMMENT_STYLE", - "Block comments should align the * on each line\n" . $hereprev); - } - } - -# check for missing blank lines after struct/union declarations -# with exceptions for various attributes and macros - if ($prevline =~ /^[\+ ]};?\s*$/ && - $line =~ /^\+/ && - !($line =~ /^\+\s*$/ || - $line =~ /^\+\s*EXPORT_SYMBOL/ || - $line =~ /^\+\s*MODULE_/i || - $line =~ /^\+\s*\#\s*(?:end|elif|else)/ || - $line =~ /^\+[a-z_]*init/ || - $line =~ /^\+\s*(?:static\s+)?[A-Z_]*ATTR/ || - $line =~ /^\+\s*DECLARE/ || - $line =~ /^\+\s*__setup/)) { - if (CHK("LINE_SPACING", - "Please use a blank line after function/struct/union/enum declarations\n" . $hereprev) && - $fix) { - fix_insert_line($fixlinenr, "\+"); - } - } - -# check for multiple consecutive blank lines - if ($prevline =~ /^[\+ ]\s*$/ && - $line =~ /^\+\s*$/ && - $last_blank_line != ($linenr - 1)) { - if (CHK("LINE_SPACING", - "Please don't use multiple blank lines\n" . $hereprev) && - $fix) { - fix_delete_line($fixlinenr, $rawline); - } - - $last_blank_line = $linenr; - } - -# check for missing blank lines after declarations - if ($sline =~ /^\+\s+\S/ && #Not at char 1 - # actual declarations - ($prevline =~ /^\+\s+$Declare\s*$Ident\s*[=,;:\[]/ || - # function pointer declarations - $prevline =~ /^\+\s+$Declare\s*\(\s*\*\s*$Ident\s*\)\s*[=,;:\[\(]/ || - # foo bar; where foo is some local typedef or #define - $prevline =~ /^\+\s+$Ident(?:\s+|\s*\*\s*)$Ident\s*[=,;\[]/ || - # known declaration macros - $prevline =~ /^\+\s+$declaration_macros/) && - # for "else if" which can look like "$Ident $Ident" - !($prevline =~ /^\+\s+$c90_Keywords\b/ || - # other possible extensions of declaration lines - $prevline =~ /(?:$Compare|$Assignment|$Operators)\s*$/ || - # not starting a section or a macro "\" extended line - $prevline =~ /(?:\{\s*|\\)$/) && - # looks like a declaration - !($sline =~ /^\+\s+$Declare\s*$Ident\s*[=,;:\[]/ || - # function pointer declarations - $sline =~ /^\+\s+$Declare\s*\(\s*\*\s*$Ident\s*\)\s*[=,;:\[\(]/ || - # foo bar; where foo is some local typedef or #define - $sline =~ /^\+\s+(?:volatile\s+)?$Ident(?:\s+|\s*\*\s*)$Ident\s*[=,;\[]/ || - # known declaration macros - $sline =~ /^\+\s+$declaration_macros/ || - # start of struct or union or enum - $sline =~ /^\+\s+(?:union|struct|enum|typedef)\b/ || - # start or end of block or continuation of declaration - $sline =~ /^\+\s+(?:$|[\{\}\.\#\"\?\:\(\[])/ || - # bitfield continuation - $sline =~ /^\+\s+$Ident\s*:\s*\d+\s*[,;]/ || - # other possible extensions of declaration lines - $sline =~ /^\+\s+\(?\s*(?:$Compare|$Assignment|$Operators)/) && - # indentation of previous and current line are the same - (($prevline =~ /\+(\s+)\S/) && $sline =~ /^\+$1\S/)) { - if (WARN("LINE_SPACING", - "Missing a blank line after declarations\n" . $hereprev) && - $fix) { - fix_insert_line($fixlinenr, "\+"); - } - } - -# check for spaces at the beginning of a line. -# Exceptions: -# 1) within comments -# 2) indented preprocessor commands -# 3) hanging labels - if ($rawline =~ /^\+ / && $line !~ /^\+ *(?:$;|#|$Ident:)/) { - my $herevet = "$here\n" . cat_vet($rawline) . "\n"; - if (WARN("LEADING_SPACE", - "please, no spaces at the start of a line\n" . $herevet) && - $fix) { - $fixed[$fixlinenr] =~ s/^\+([ \t]+)/"\+" . tabify($1)/e; - } - } - -# check we are in a valid C source file if not then ignore this hunk - next if ($realfile !~ /\.(h|c)$/); - -# check if this appears to be the start function declaration, save the name - if ($sline =~ /^\+\{\s*$/ && - $prevline =~ /^\+(?:(?:(?:$Storage|$Inline)\s*)*\s*$Type\s*)?($Ident)\(/) { - $context_function = $1; - } - -# check if this appears to be the end of function declaration - if ($sline =~ /^\+\}\s*$/) { - undef $context_function; - } - -# check indentation of any line with a bare else -# (but not if it is a multiple line "if (foo) return bar; else return baz;") -# if the previous line is a break or return and is indented 1 tab more... - if ($sline =~ /^\+([\t]+)(?:}[ \t]*)?else(?:[ \t]*{)?\s*$/) { - my $tabs = length($1) + 1; - if ($prevline =~ /^\+\t{$tabs,$tabs}break\b/ || - ($prevline =~ /^\+\t{$tabs,$tabs}return\b/ && - defined $lines[$linenr] && - $lines[$linenr] !~ /^[ \+]\t{$tabs,$tabs}return/)) { - WARN("UNNECESSARY_ELSE", - "else is not generally useful after a break or return\n" . $hereprev); - } - } - -# check indentation of a line with a break; -# if the previous line is a goto or return and is indented the same # of tabs - if ($sline =~ /^\+([\t]+)break\s*;\s*$/) { - my $tabs = $1; - if ($prevline =~ /^\+$tabs(?:goto|return)\b/) { - WARN("UNNECESSARY_BREAK", - "break is not useful after a goto or return\n" . $hereprev); - } - } - -# check for RCS/CVS revision markers - if ($rawline =~ /^\+.*\$(Revision|Log|Id)(?:\$|)/) { - WARN("CVS_KEYWORD", - "CVS style keyword markers, these will _not_ be updated\n". $herecurr); - } - -# Blackfin: don't use __builtin_bfin_[cs]sync - if ($line =~ /__builtin_bfin_csync/) { - my $herevet = "$here\n" . cat_vet($line) . "\n"; - ERROR("CSYNC", - "use the CSYNC() macro in asm/blackfin.h\n" . $herevet); - } - if ($line =~ /__builtin_bfin_ssync/) { - my $herevet = "$here\n" . cat_vet($line) . "\n"; - ERROR("SSYNC", - "use the SSYNC() macro in asm/blackfin.h\n" . $herevet); - } - -# check for old HOTPLUG __dev<foo> section markings - if ($line =~ /\b(__dev(init|exit)(data|const|))\b/) { - WARN("HOTPLUG_SECTION", - "Using $1 is unnecessary\n" . $herecurr); - } - -# Check for potential 'bare' types - my ($stat, $cond, $line_nr_next, $remain_next, $off_next, - $realline_next); -#print "LINE<$line>\n"; - if ($linenr > $suppress_statement && - $realcnt && $sline =~ /.\s*\S/) { - ($stat, $cond, $line_nr_next, $remain_next, $off_next) = - ctx_statement_block($linenr, $realcnt, 0); - $stat =~ s/\n./\n /g; - $cond =~ s/\n./\n /g; - -#print "linenr<$linenr> <$stat>\n"; - # If this statement has no statement boundaries within - # it there is no point in retrying a statement scan - # until we hit end of it. - my $frag = $stat; $frag =~ s/;+\s*$//; - if ($frag !~ /(?:{|;)/) { -#print "skip<$line_nr_next>\n"; - $suppress_statement = $line_nr_next; - } - - # Find the real next line. - $realline_next = $line_nr_next; - if (defined $realline_next && - (!defined $lines[$realline_next - 1] || - substr($lines[$realline_next - 1], $off_next) =~ /^\s*$/)) { - $realline_next++; - } - - my $s = $stat; - $s =~ s/{.*$//s; - - # Ignore goto labels. - if ($s =~ /$Ident:\*$/s) { - - # Ignore functions being called - } elsif ($s =~ /^.\s*$Ident\s*\(/s) { - - } elsif ($s =~ /^.\s*else\b/s) { - - # declarations always start with types - } elsif ($prev_values eq 'E' && $s =~ /^.\s*(?:$Storage\s+)?(?:$Inline\s+)?(?:const\s+)?((?:\s*$Ident)+?)\b(?:\s+$Sparse)?\s*\**\s*(?:$Ident|\(\*[^\)]*\))(?:\s*$Modifier)?\s*(?:;|=|,|\()/s) { - my $type = $1; - $type =~ s/\s+/ /g; - possible($type, "A:" . $s); - - # definitions in global scope can only start with types - } elsif ($s =~ /^.(?:$Storage\s+)?(?:$Inline\s+)?(?:const\s+)?($Ident)\b\s*(?!:)/s) { - possible($1, "B:" . $s); - } - - # any (foo ... *) is a pointer cast, and foo is a type - while ($s =~ /\(($Ident)(?:\s+$Sparse)*[\s\*]+\s*\)/sg) { - possible($1, "C:" . $s); - } - - # Check for any sort of function declaration. - # int foo(something bar, other baz); - # void (*store_gdt)(x86_descr_ptr *); - if ($prev_values eq 'E' && $s =~ /^(.(?:typedef\s*)?(?:(?:$Storage|$Inline)\s*)*\s*$Type\s*(?:\b$Ident|\(\*\s*$Ident\))\s*)\(/s) { - my ($name_len) = length($1); - - my $ctx = $s; - substr($ctx, 0, $name_len + 1, ''); - $ctx =~ s/\)[^\)]*$//; - - for my $arg (split(/\s*,\s*/, $ctx)) { - if ($arg =~ /^(?:const\s+)?($Ident)(?:\s+$Sparse)*\s*\**\s*(:?\b$Ident)?$/s || $arg =~ /^($Ident)$/s) { - - possible($1, "D:" . $s); - } - } - } - - } - -# -# Checks which may be anchored in the context. -# - -# Check for switch () and associated case and default -# statements should be at the same indent. - if ($line=~/\bswitch\s*\(.*\)/) { - my $err = ''; - my $sep = ''; - my @ctx = ctx_block_outer($linenr, $realcnt); - shift(@ctx); - for my $ctx (@ctx) { - my ($clen, $cindent) = line_stats($ctx); - if ($ctx =~ /^\+\s*(case\s+|default:)/ && - $indent != $cindent) { - $err .= "$sep$ctx\n"; - $sep = ''; - } else { - $sep = "[...]\n"; - } - } - if ($err ne '') { - ERROR("SWITCH_CASE_INDENT_LEVEL", - "switch and case should be at the same indent\n$hereline$err"); - } - } - -# if/while/etc brace do not go on next line, unless defining a do while loop, -# or if that brace on the next line is for something else - if ($line =~ /(.*)\b((?:if|while|for|switch|(?:[a-z_]+|)for_each[a-z_]+)\s*\(|do\b|else\b)/ && $line !~ /^.\s*\#/) { - my $pre_ctx = "$1$2"; - - my ($level, @ctx) = ctx_statement_level($linenr, $realcnt, 0); - - if ($line =~ /^\+\t{6,}/) { - WARN("DEEP_INDENTATION", - "Too many leading tabs - consider code refactoring\n" . $herecurr); - } - - my $ctx_cnt = $realcnt - $#ctx - 1; - my $ctx = join("\n", @ctx); - - my $ctx_ln = $linenr; - my $ctx_skip = $realcnt; - - while ($ctx_skip > $ctx_cnt || ($ctx_skip == $ctx_cnt && - defined $lines[$ctx_ln - 1] && - $lines[$ctx_ln - 1] =~ /^-/)) { - ##print "SKIP<$ctx_skip> CNT<$ctx_cnt>\n"; - $ctx_skip-- if (!defined $lines[$ctx_ln - 1] || $lines[$ctx_ln - 1] !~ /^-/); - $ctx_ln++; - } - - #print "realcnt<$realcnt> ctx_cnt<$ctx_cnt>\n"; - #print "pre<$pre_ctx>\nline<$line>\nctx<$ctx>\nnext<$lines[$ctx_ln - 1]>\n"; - - if ($ctx !~ /{\s*/ && defined($lines[$ctx_ln - 1]) && $lines[$ctx_ln - 1] =~ /^\+\s*{/) { - ERROR("OPEN_BRACE", - "that open brace { should be on the previous line\n" . - "$here\n$ctx\n$rawlines[$ctx_ln - 1]\n"); - } - if ($level == 0 && $pre_ctx !~ /}\s*while\s*\($/ && - $ctx =~ /\)\s*\;\s*$/ && - defined $lines[$ctx_ln - 1]) - { - my ($nlength, $nindent) = line_stats($lines[$ctx_ln - 1]); - if ($nindent > $indent) { - WARN("TRAILING_SEMICOLON", - "trailing semicolon indicates no statements, indent implies otherwise\n" . - "$here\n$ctx\n$rawlines[$ctx_ln - 1]\n"); - } - } - } - -# Check relative indent for conditionals and blocks. - if ($line =~ /\b(?:(?:if|while|for|(?:[a-z_]+|)for_each[a-z_]+)\s*\(|(?:do|else)\b)/ && $line !~ /^.\s*#/ && $line !~ /\}\s*while\s*/) { - ($stat, $cond, $line_nr_next, $remain_next, $off_next) = - ctx_statement_block($linenr, $realcnt, 0) - if (!defined $stat); - my ($s, $c) = ($stat, $cond); - - substr($s, 0, length($c), ''); - - # remove inline comments - $s =~ s/$;/ /g; - $c =~ s/$;/ /g; - - # Find out how long the conditional actually is. - my @newlines = ($c =~ /\n/gs); - my $cond_lines = 1 + $#newlines; - - # Make sure we remove the line prefixes as we have - # none on the first line, and are going to readd them - # where necessary. - $s =~ s/\n./\n/gs; - while ($s =~ /\n\s+\\\n/) { - $cond_lines += $s =~ s/\n\s+\\\n/\n/g; - } - - # We want to check the first line inside the block - # starting at the end of the conditional, so remove: - # 1) any blank line termination - # 2) any opening brace { on end of the line - # 3) any do (...) { - my $continuation = 0; - my $check = 0; - $s =~ s/^.*\bdo\b//; - $s =~ s/^\s*{//; - if ($s =~ s/^\s*\\//) { - $continuation = 1; - } - if ($s =~ s/^\s*?\n//) { - $check = 1; - $cond_lines++; - } - - # Also ignore a loop construct at the end of a - # preprocessor statement. - if (($prevline =~ /^.\s*#\s*define\s/ || - $prevline =~ /\\\s*$/) && $continuation == 0) { - $check = 0; - } - - my $cond_ptr = -1; - $continuation = 0; - while ($cond_ptr != $cond_lines) { - $cond_ptr = $cond_lines; - - # If we see an #else/#elif then the code - # is not linear. - if ($s =~ /^\s*\#\s*(?:else|elif)/) { - $check = 0; - } - - # Ignore: - # 1) blank lines, they should be at 0, - # 2) preprocessor lines, and - # 3) labels. - if ($continuation || - $s =~ /^\s*?\n/ || - $s =~ /^\s*#\s*?/ || - $s =~ /^\s*$Ident\s*:/) { - $continuation = ($s =~ /^.*?\\\n/) ? 1 : 0; - if ($s =~ s/^.*?\n//) { - $cond_lines++; - } - } - } - - my (undef, $sindent) = line_stats("+" . $s); - my $stat_real = raw_line($linenr, $cond_lines); - - # Check if either of these lines are modified, else - # this is not this patch's fault. - if (!defined($stat_real) || - $stat !~ /^\+/ && $stat_real !~ /^\+/) { - $check = 0; - } - if (defined($stat_real) && $cond_lines > 1) { - $stat_real = "[...]\n$stat_real"; - } - - #print "line<$line> prevline<$prevline> indent<$indent> sindent<$sindent> check<$check> continuation<$continuation> s<$s> cond_lines<$cond_lines> stat_real<$stat_real> stat<$stat>\n"; - - if ($check && $s ne '' && - (($sindent % 8) != 0 || - ($sindent < $indent) || - ($sindent == $indent && - ($s !~ /^\s*(?:\}|\{|else\b)/)) || - ($sindent > $indent + 8))) { - WARN("SUSPECT_CODE_INDENT", - "suspect code indent for conditional statements ($indent, $sindent)\n" . $herecurr . "$stat_real\n"); - } - } - - # Track the 'values' across context and added lines. - my $opline = $line; $opline =~ s/^./ /; - my ($curr_values, $curr_vars) = - annotate_values($opline . "\n", $prev_values); - $curr_values = $prev_values . $curr_values; - if ($dbg_values) { - my $outline = $opline; $outline =~ s/\t/ /g; - print "$linenr > .$outline\n"; - print "$linenr > $curr_values\n"; - print "$linenr > $curr_vars\n"; - } - $prev_values = substr($curr_values, -1); - -#ignore lines not being added - next if ($line =~ /^[^\+]/); - -# check for dereferences that span multiple lines - if ($prevline =~ /^\+.*$Lval\s*(?:\.|->)\s*$/ && - $line =~ /^\+\s*(?!\#\s*(?!define\s+|if))\s*$Lval/) { - $prevline =~ /($Lval\s*(?:\.|->))\s*$/; - my $ref = $1; - $line =~ /^.\s*($Lval)/; - $ref .= $1; - $ref =~ s/\s//g; - WARN("MULTILINE_DEREFERENCE", - "Avoid multiple line dereference - prefer '$ref'\n" . $hereprev); - } - -# check for declarations of signed or unsigned without int - while ($line =~ m{\b($Declare)\s*(?!char\b|short\b|int\b|long\b)\s*($Ident)?\s*[=,;\[\)\(]}g) { - my $type = $1; - my $var = $2; - $var = "" if (!defined $var); - if ($type =~ /^(?:(?:$Storage|$Inline|$Attribute)\s+)*((?:un)?signed)((?:\s*\*)*)\s*$/) { - my $sign = $1; - my $pointer = $2; - - $pointer = "" if (!defined $pointer); - - if (WARN("UNSPECIFIED_INT", - "Prefer '" . trim($sign) . " int" . rtrim($pointer) . "' to bare use of '$sign" . rtrim($pointer) . "'\n" . $herecurr) && - $fix) { - my $decl = trim($sign) . " int "; - my $comp_pointer = $pointer; - $comp_pointer =~ s/\s//g; - $decl .= $comp_pointer; - $decl = rtrim($decl) if ($var eq ""); - $fixed[$fixlinenr] =~ s@\b$sign\s*\Q$pointer\E\s*$var\b@$decl$var@; - } - } - } - -# TEST: allow direct testing of the type matcher. - if ($dbg_type) { - if ($line =~ /^.\s*$Declare\s*$/) { - ERROR("TEST_TYPE", - "TEST: is type\n" . $herecurr); - } elsif ($dbg_type > 1 && $line =~ /^.+($Declare)/) { - ERROR("TEST_NOT_TYPE", - "TEST: is not type ($1 is)\n". $herecurr); - } - next; - } -# TEST: allow direct testing of the attribute matcher. - if ($dbg_attr) { - if ($line =~ /^.\s*$Modifier\s*$/) { - ERROR("TEST_ATTR", - "TEST: is attr\n" . $herecurr); - } elsif ($dbg_attr > 1 && $line =~ /^.+($Modifier)/) { - ERROR("TEST_NOT_ATTR", - "TEST: is not attr ($1 is)\n". $herecurr); - } - next; - } - -# check for initialisation to aggregates open brace on the next line - if ($line =~ /^.\s*{/ && - $prevline =~ /(?:^|[^=])=\s*$/) { - if (ERROR("OPEN_BRACE", - "that open brace { should be on the previous line\n" . $hereprev) && - $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { - fix_delete_line($fixlinenr - 1, $prevrawline); - fix_delete_line($fixlinenr, $rawline); - my $fixedline = $prevrawline; - $fixedline =~ s/\s*=\s*$/ = {/; - fix_insert_line($fixlinenr, $fixedline); - $fixedline = $line; - $fixedline =~ s/^(.\s*)\{\s*/$1/; - fix_insert_line($fixlinenr, $fixedline); - } - } - -# -# Checks which are anchored on the added line. -# - -# check for malformed paths in #include statements (uses RAW line) - if ($rawline =~ m{^.\s*\#\s*include\s+[<"](.*)[">]}) { - my $path = $1; - if ($path =~ m{//}) { - ERROR("MALFORMED_INCLUDE", - "malformed #include filename\n" . $herecurr); - } - if ($path =~ "^uapi/" && $realfile =~ m@\binclude/uapi/@) { - ERROR("UAPI_INCLUDE", - "No #include in ...include/uapi/... should use a uapi/ path prefix\n" . $herecurr); - } - } - -# no C99 // comments - if ($line =~ m{//}) { - if (ERROR("C99_COMMENTS", - "do not use C99 // comments\n" . $herecurr) && - $fix) { - my $line = $fixed[$fixlinenr]; - if ($line =~ /\/\/(.*)$/) { - my $comment = trim($1); - $fixed[$fixlinenr] =~ s@\/\/(.*)$@/\* $comment \*/@; - } - } - } - # Remove C99 comments. - $line =~ s@//.*@@; - $opline =~ s@//.*@@; - -# EXPORT_SYMBOL should immediately follow the thing it is exporting, consider -# the whole statement. -#print "APW <$lines[$realline_next - 1]>\n"; - if (defined $realline_next && - exists $lines[$realline_next - 1] && - !defined $suppress_export{$realline_next} && - ($lines[$realline_next - 1] =~ /EXPORT_SYMBOL.*\((.*)\)/ || - $lines[$realline_next - 1] =~ /EXPORT_UNUSED_SYMBOL.*\((.*)\)/)) { - # Handle definitions which produce identifiers with - # a prefix: - # XXX(foo); - # EXPORT_SYMBOL(something_foo); - my $name = $1; - if ($stat =~ /^(?:.\s*}\s*\n)?.([A-Z_]+)\s*\(\s*($Ident)/ && - $name =~ /^${Ident}_$2/) { -#print "FOO C name<$name>\n"; - $suppress_export{$realline_next} = 1; - - } elsif ($stat !~ /(?: - \n.}\s*$| - ^.DEFINE_$Ident\(\Q$name\E\)| - ^.DECLARE_$Ident\(\Q$name\E\)| - ^.LIST_HEAD\(\Q$name\E\)| - ^.(?:$Storage\s+)?$Type\s*\(\s*\*\s*\Q$name\E\s*\)\s*\(| - \b\Q$name\E(?:\s+$Attribute)*\s*(?:;|=|\[|\() - )/x) { -#print "FOO A<$lines[$realline_next - 1]> stat<$stat> name<$name>\n"; - $suppress_export{$realline_next} = 2; - } else { - $suppress_export{$realline_next} = 1; - } - } - if (!defined $suppress_export{$linenr} && - $prevline =~ /^.\s*$/ && - ($line =~ /EXPORT_SYMBOL.*\((.*)\)/ || - $line =~ /EXPORT_UNUSED_SYMBOL.*\((.*)\)/)) { -#print "FOO B <$lines[$linenr - 1]>\n"; - $suppress_export{$linenr} = 2; - } - if (defined $suppress_export{$linenr} && - $suppress_export{$linenr} == 2) { - WARN("EXPORT_SYMBOL", - "EXPORT_SYMBOL(foo); should immediately follow its function/variable\n" . $herecurr); - } - -# check for global initialisers. - if ($line =~ /^\+$Type\s*$Ident(?:\s+$Modifier)*\s*=\s*($zero_initializer)\s*;/) { - if (ERROR("GLOBAL_INITIALISERS", - "do not initialise globals to $1\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/(^.$Type\s*$Ident(?:\s+$Modifier)*)\s*=\s*$zero_initializer\s*;/$1;/; - } - } -# check for static initialisers. - if ($line =~ /^\+.*\bstatic\s.*=\s*($zero_initializer)\s*;/) { - if (ERROR("INITIALISED_STATIC", - "do not initialise statics to $1\n" . - $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/(\bstatic\s.*?)\s*=\s*$zero_initializer\s*;/$1;/; - } - } - -# check for misordered declarations of char/short/int/long with signed/unsigned - while ($sline =~ m{(\b$TypeMisordered\b)}g) { - my $tmp = trim($1); - WARN("MISORDERED_TYPE", - "type '$tmp' should be specified in [[un]signed] [short|int|long|long long] order\n" . $herecurr); - } - -# check for static const char * arrays. - if ($line =~ /\bstatic\s+const\s+char\s*\*\s*(\w+)\s*\[\s*\]\s*=\s*/) { - WARN("STATIC_CONST_CHAR_ARRAY", - "static const char * array should probably be static const char * const\n" . - $herecurr); - } - -# check for static char foo[] = "bar" declarations. - if ($line =~ /\bstatic\s+char\s+(\w+)\s*\[\s*\]\s*=\s*"/) { - WARN("STATIC_CONST_CHAR_ARRAY", - "static char array declaration should probably be static const char\n" . - $herecurr); - } - -# check for const <foo> const where <foo> is not a pointer or array type - if ($sline =~ /\bconst\s+($BasicType)\s+const\b/) { - my $found = $1; - if ($sline =~ /\bconst\s+\Q$found\E\s+const\b\s*\*/) { - WARN("CONST_CONST", - "'const $found const *' should probably be 'const $found * const'\n" . $herecurr); - } elsif ($sline !~ /\bconst\s+\Q$found\E\s+const\s+\w+\s*\[/) { - WARN("CONST_CONST", - "'const $found const' should probably be 'const $found'\n" . $herecurr); - } - } - -# check for non-global char *foo[] = {"bar", ...} declarations. - if ($line =~ /^.\s+(?:static\s+|const\s+)?char\s+\*\s*\w+\s*\[\s*\]\s*=\s*\{/) { - WARN("STATIC_CONST_CHAR_ARRAY", - "char * array declaration might be better as static const\n" . - $herecurr); - } - -# check for sizeof(foo)/sizeof(foo[0]) that could be ARRAY_SIZE(foo) - if ($line =~ m@\bsizeof\s*\(\s*($Lval)\s*\)@) { - my $array = $1; - if ($line =~ m@\b(sizeof\s*\(\s*\Q$array\E\s*\)\s*/\s*sizeof\s*\(\s*\Q$array\E\s*\[\s*0\s*\]\s*\))@) { - my $array_div = $1; - if (WARN("ARRAY_SIZE", - "Prefer ARRAY_SIZE($array)\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\Q$array_div\E/ARRAY_SIZE($array)/; - } - } - } - -# check for function declarations without arguments like "int foo()" - if ($line =~ /(\b$Type\s+$Ident)\s*\(\s*\)/) { - if (ERROR("FUNCTION_WITHOUT_ARGS", - "Bad function definition - $1() should probably be $1(void)\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/(\b($Type)\s+($Ident))\s*\(\s*\)/$2 $3(void)/; - } - } - -# check for new typedefs, only function parameters and sparse annotations -# make sense. - if ($line =~ /\btypedef\s/ && - $line !~ /\btypedef\s+$Type\s*\(\s*\*?$Ident\s*\)\s*\(/ && - $line !~ /\btypedef\s+$Type\s+$Ident\s*\(/ && - $line !~ /\b$typeTypedefs\b/ && - $line !~ /\b__bitwise\b/) { - WARN("NEW_TYPEDEFS", - "do not add new typedefs\n" . $herecurr); - } - -# * goes on variable not on type - # (char*[ const]) - while ($line =~ m{(\($NonptrType(\s*(?:$Modifier\b\s*|\*\s*)+)\))}g) { - #print "AA<$1>\n"; - my ($ident, $from, $to) = ($1, $2, $2); - - # Should start with a space. - $to =~ s/^(\S)/ $1/; - # Should not end with a space. - $to =~ s/\s+$//; - # '*'s should not have spaces between. - while ($to =~ s/\*\s+\*/\*\*/) { - } - -## print "1: from<$from> to<$to> ident<$ident>\n"; - if ($from ne $to) { - if (ERROR("POINTER_LOCATION", - "\"(foo$from)\" should be \"(foo$to)\"\n" . $herecurr) && - $fix) { - my $sub_from = $ident; - my $sub_to = $ident; - $sub_to =~ s/\Q$from\E/$to/; - $fixed[$fixlinenr] =~ - s@\Q$sub_from\E@$sub_to@; - } - } - } - while ($line =~ m{(\b$NonptrType(\s*(?:$Modifier\b\s*|\*\s*)+)($Ident))}g) { - #print "BB<$1>\n"; - my ($match, $from, $to, $ident) = ($1, $2, $2, $3); - - # Should start with a space. - $to =~ s/^(\S)/ $1/; - # Should not end with a space. - $to =~ s/\s+$//; - # '*'s should not have spaces between. - while ($to =~ s/\*\s+\*/\*\*/) { - } - # Modifiers should have spaces. - $to =~ s/(\b$Modifier$)/$1 /; - -## print "2: from<$from> to<$to> ident<$ident>\n"; - if ($from ne $to && $ident !~ /^$Modifier$/) { - if (ERROR("POINTER_LOCATION", - "\"foo${from}bar\" should be \"foo${to}bar\"\n" . $herecurr) && - $fix) { - - my $sub_from = $match; - my $sub_to = $match; - $sub_to =~ s/\Q$from\E/$to/; - $fixed[$fixlinenr] =~ - s@\Q$sub_from\E@$sub_to@; - } - } - } - -# avoid BUG() or BUG_ON() - if ($line =~ /\b(?:BUG|BUG_ON)\b/) { - my $msg_level = \&WARN; - $msg_level = \&CHK if ($file); - &{$msg_level}("AVOID_BUG", - "Avoid crashing the kernel - try using WARN_ON & recovery code rather than BUG() or BUG_ON()\n" . $herecurr); - } - -# avoid LINUX_VERSION_CODE - if ($line =~ /\bLINUX_VERSION_CODE\b/) { - WARN("LINUX_VERSION_CODE", - "LINUX_VERSION_CODE should be avoided, code should be for the version to which it is merged\n" . $herecurr); - } - -# check for uses of printk_ratelimit - if ($line =~ /\bprintk_ratelimit\s*\(/) { - WARN("PRINTK_RATELIMITED", - "Prefer printk_ratelimited or pr_<level>_ratelimited to printk_ratelimit\n" . $herecurr); - } - -# printk should use KERN_* levels. Note that follow on printk's on the -# same line do not need a level, so we use the current block context -# to try and find and validate the current printk. In summary the current -# printk includes all preceding printk's which have no newline on the end. -# we assume the first bad printk is the one to report. - if ($line =~ /\bprintk\((?!KERN_)\s*"/) { - my $ok = 0; - for (my $ln = $linenr - 1; $ln >= $first_line; $ln--) { - #print "CHECK<$lines[$ln - 1]\n"; - # we have a preceding printk if it ends - # with "\n" ignore it, else it is to blame - if ($lines[$ln - 1] =~ m{\bprintk\(}) { - if ($rawlines[$ln - 1] !~ m{\\n"}) { - $ok = 1; - } - last; - } - } - if ($ok == 0) { - WARN("PRINTK_WITHOUT_KERN_LEVEL", - "printk() should include KERN_ facility level\n" . $herecurr); - } - } - - if ($line =~ /\bprintk\s*\(\s*KERN_([A-Z]+)/) { - my $orig = $1; - my $level = lc($orig); - $level = "warn" if ($level eq "warning"); - my $level2 = $level; - $level2 = "dbg" if ($level eq "debug"); - WARN("PREFER_PR_LEVEL", - "Prefer [subsystem eg: netdev]_$level2([subsystem]dev, ... then dev_$level2(dev, ... then pr_$level(... to printk(KERN_$orig ...\n" . $herecurr); - } - - if ($line =~ /\bpr_warning\s*\(/) { - if (WARN("PREFER_PR_LEVEL", - "Prefer pr_warn(... to pr_warning(...\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/\bpr_warning\b/pr_warn/; - } - } - - if ($line =~ /\bdev_printk\s*\(\s*KERN_([A-Z]+)/) { - my $orig = $1; - my $level = lc($orig); - $level = "warn" if ($level eq "warning"); - $level = "dbg" if ($level eq "debug"); - WARN("PREFER_DEV_LEVEL", - "Prefer dev_$level(... to dev_printk(KERN_$orig, ...\n" . $herecurr); - } - -# ENOSYS means "bad syscall nr" and nothing else. This will have a small -# number of false positives, but assembly files are not checked, so at -# least the arch entry code will not trigger this warning. - if ($line =~ /\bENOSYS\b/) { - WARN("ENOSYS", - "ENOSYS means 'invalid syscall nr' and nothing else\n" . $herecurr); - } - -# function brace can't be on same line, except for #defines of do while, -# or if closed on same line - if (($line=~/$Type\s*$Ident\(.*\).*\s*{/) and - !($line=~/\#\s*define.*do\s\{/) and !($line=~/}/)) { - if (ERROR("OPEN_BRACE", - "open brace '{' following function declarations go on the next line\n" . $herecurr) && - $fix) { - fix_delete_line($fixlinenr, $rawline); - my $fixed_line = $rawline; - $fixed_line =~ /(^..*$Type\s*$Ident\(.*\)\s*){(.*)$/; - my $line1 = $1; - my $line2 = $2; - fix_insert_line($fixlinenr, ltrim($line1)); - fix_insert_line($fixlinenr, "\+{"); - if ($line2 !~ /^\s*$/) { - fix_insert_line($fixlinenr, "\+\t" . trim($line2)); - } - } - } - -# open braces for enum, union and struct go on the same line. - if ($line =~ /^.\s*{/ && - $prevline =~ /^.\s*(?:typedef\s+)?(enum|union|struct)(?:\s+$Ident)?\s*$/) { - if (ERROR("OPEN_BRACE", - "open brace '{' following $1 go on the same line\n" . $hereprev) && - $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { - fix_delete_line($fixlinenr - 1, $prevrawline); - fix_delete_line($fixlinenr, $rawline); - my $fixedline = rtrim($prevrawline) . " {"; - fix_insert_line($fixlinenr, $fixedline); - $fixedline = $rawline; - $fixedline =~ s/^(.\s*)\{\s*/$1\t/; - if ($fixedline !~ /^\+\s*$/) { - fix_insert_line($fixlinenr, $fixedline); - } - } - } - -# missing space after union, struct or enum definition - if ($line =~ /^.\s*(?:typedef\s+)?(enum|union|struct)(?:\s+$Ident){1,2}[=\{]/) { - if (WARN("SPACING", - "missing space after $1 definition\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/^(.\s*(?:typedef\s+)?(?:enum|union|struct)(?:\s+$Ident){1,2})([=\{])/$1 $2/; - } - } - -# Function pointer declarations -# check spacing between type, funcptr, and args -# canonical declaration is "type (*funcptr)(args...)" - if ($line =~ /^.\s*($Declare)\((\s*)\*(\s*)($Ident)(\s*)\)(\s*)\(/) { - my $declare = $1; - my $pre_pointer_space = $2; - my $post_pointer_space = $3; - my $funcname = $4; - my $post_funcname_space = $5; - my $pre_args_space = $6; - -# the $Declare variable will capture all spaces after the type -# so check it for a missing trailing missing space but pointer return types -# don't need a space so don't warn for those. - my $post_declare_space = ""; - if ($declare =~ /(\s+)$/) { - $post_declare_space = $1; - $declare = rtrim($declare); - } - if ($declare !~ /\*$/ && $post_declare_space =~ /^$/) { - WARN("SPACING", - "missing space after return type\n" . $herecurr); - $post_declare_space = " "; - } - -# unnecessary space "type (*funcptr)(args...)" -# This test is not currently implemented because these declarations are -# equivalent to -# int foo(int bar, ...) -# and this is form shouldn't/doesn't generate a checkpatch warning. -# -# elsif ($declare =~ /\s{2,}$/) { -# WARN("SPACING", -# "Multiple spaces after return type\n" . $herecurr); -# } - -# unnecessary space "type ( *funcptr)(args...)" - if (defined $pre_pointer_space && - $pre_pointer_space =~ /^\s/) { - WARN("SPACING", - "Unnecessary space after function pointer open parenthesis\n" . $herecurr); - } - -# unnecessary space "type (* funcptr)(args...)" - if (defined $post_pointer_space && - $post_pointer_space =~ /^\s/) { - WARN("SPACING", - "Unnecessary space before function pointer name\n" . $herecurr); - } - -# unnecessary space "type (*funcptr )(args...)" - if (defined $post_funcname_space && - $post_funcname_space =~ /^\s/) { - WARN("SPACING", - "Unnecessary space after function pointer name\n" . $herecurr); - } - -# unnecessary space "type (*funcptr) (args...)" - if (defined $pre_args_space && - $pre_args_space =~ /^\s/) { - WARN("SPACING", - "Unnecessary space before function pointer arguments\n" . $herecurr); - } - - if (show_type("SPACING") && $fix) { - $fixed[$fixlinenr] =~ - s/^(.\s*)$Declare\s*\(\s*\*\s*$Ident\s*\)\s*\(/$1 . $declare . $post_declare_space . '(*' . $funcname . ')('/ex; - } - } - -# check for spacing round square brackets; allowed: -# 1. with a type on the left -- int [] a; -# 2. at the beginning of a line for slice initialisers -- [0...10] = 5, -# 3. inside a curly brace -- = { [0...10] = 5 } - while ($line =~ /(.*?\s)\[/g) { - my ($where, $prefix) = ($-[1], $1); - if ($prefix !~ /$Type\s+$/ && - ($where != 0 || $prefix !~ /^.\s+$/) && - $prefix !~ /[{,]\s+$/ && - $prefix !~ /:\s+$/) { - if (ERROR("BRACKET_SPACE", - "space prohibited before open square bracket '['\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/^(\+.*?)\s+\[/$1\[/; - } - } - } - -# check for spaces between functions and their parentheses. - while ($line =~ /($Ident)\s+\(/g) { - my $name = $1; - my $ctx_before = substr($line, 0, $-[1]); - my $ctx = "$ctx_before$name"; - - # Ignore those directives where spaces _are_ permitted. - if ($name =~ /^(?: - if|for|while|switch|return|case| - volatile|__volatile__| - __attribute__|format|__extension__| - asm|__asm__)$/x) - { - # cpp #define statements have non-optional spaces, ie - # if there is a space between the name and the open - # parenthesis it is simply not a parameter group. - } elsif ($ctx_before =~ /^.\s*\#\s*define\s*$/) { - - # cpp #elif statement condition may start with a ( - } elsif ($ctx =~ /^.\s*\#\s*elif\s*$/) { - - # If this whole things ends with a type its most - # likely a typedef for a function. - } elsif ($ctx =~ /$Type$/) { - - } else { - if (WARN("SPACING", - "space prohibited between function name and open parenthesis '('\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/\b$name\s+\(/$name\(/; - } - } - } - -# Check operator spacing. - if (!($line=~/\#\s*include/)) { - my $fixed_line = ""; - my $line_fixed = 0; - - my $ops = qr{ - <<=|>>=|<=|>=|==|!=| - \+=|-=|\*=|\/=|%=|\^=|\|=|&=| - =>|->|<<|>>|<|>|=|!|~| - &&|\|\||,|\^|\+\+|--|&|\||\+|-|\*|\/|%| - \?:|\?|: - }x; - my @elements = split(/($ops|;)/, $opline); - -## print("element count: <" . $#elements . ">\n"); -## foreach my $el (@elements) { -## print("el: <$el>\n"); -## } - - my @fix_elements = (); - my $off = 0; - - foreach my $el (@elements) { - push(@fix_elements, substr($rawline, $off, length($el))); - $off += length($el); - } - - $off = 0; - - my $blank = copy_spacing($opline); - my $last_after = -1; - - for (my $n = 0; $n < $#elements; $n += 2) { - - my $good = $fix_elements[$n] . $fix_elements[$n + 1]; - -## print("n: <$n> good: <$good>\n"); - - $off += length($elements[$n]); - - # Pick up the preceding and succeeding characters. - my $ca = substr($opline, 0, $off); - my $cc = ''; - if (length($opline) >= ($off + length($elements[$n + 1]))) { - $cc = substr($opline, $off + length($elements[$n + 1])); - } - my $cb = "$ca$;$cc"; - - my $a = ''; - $a = 'V' if ($elements[$n] ne ''); - $a = 'W' if ($elements[$n] =~ /\s$/); - $a = 'C' if ($elements[$n] =~ /$;$/); - $a = 'B' if ($elements[$n] =~ /(\[|\()$/); - $a = 'O' if ($elements[$n] eq ''); - $a = 'E' if ($ca =~ /^\s*$/); - - my $op = $elements[$n + 1]; - - my $c = ''; - if (defined $elements[$n + 2]) { - $c = 'V' if ($elements[$n + 2] ne ''); - $c = 'W' if ($elements[$n + 2] =~ /^\s/); - $c = 'C' if ($elements[$n + 2] =~ /^$;/); - $c = 'B' if ($elements[$n + 2] =~ /^(\)|\]|;)/); - $c = 'O' if ($elements[$n + 2] eq ''); - $c = 'E' if ($elements[$n + 2] =~ /^\s*\\$/); - } else { - $c = 'E'; - } - - my $ctx = "${a}x${c}"; - - my $at = "(ctx:$ctx)"; - - my $ptr = substr($blank, 0, $off) . "^"; - my $hereptr = "$hereline$ptr\n"; - - # Pull out the value of this operator. - my $op_type = substr($curr_values, $off + 1, 1); - - # Get the full operator variant. - my $opv = $op . substr($curr_vars, $off, 1); - - # Ignore operators passed as parameters. - if ($op_type ne 'V' && - $ca =~ /\s$/ && $cc =~ /^\s*[,\)]/) { - -# # Ignore comments -# } elsif ($op =~ /^$;+$/) { - - # ; should have either the end of line or a space or \ after it - } elsif ($op eq ';') { - if ($ctx !~ /.x[WEBC]/ && - $cc !~ /^\\/ && $cc !~ /^;/) { - if (ERROR("SPACING", - "space required after that '$op' $at\n" . $hereptr)) { - $good = $fix_elements[$n] . trim($fix_elements[$n + 1]) . " "; - $line_fixed = 1; - } - } - - # // is a comment - } elsif ($op eq '//') { - - # : when part of a bitfield - } elsif ($opv eq ':B') { - # skip the bitfield test for now - - # No spaces for: - # -> - } elsif ($op eq '->') { - if ($ctx =~ /Wx.|.xW/) { - if (ERROR("SPACING", - "spaces prohibited around that '$op' $at\n" . $hereptr)) { - $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); - if (defined $fix_elements[$n + 2]) { - $fix_elements[$n + 2] =~ s/^\s+//; - } - $line_fixed = 1; - } - } - - # , must not have a space before and must have a space on the right. - } elsif ($op eq ',') { - my $rtrim_before = 0; - my $space_after = 0; - if ($ctx =~ /Wx./) { - if (ERROR("SPACING", - "space prohibited before that '$op' $at\n" . $hereptr)) { - $line_fixed = 1; - $rtrim_before = 1; - } - } - if ($ctx !~ /.x[WEC]/ && $cc !~ /^}/ && $cc !~ /^\)/) { - if (ERROR("SPACING", - "space required after that '$op' $at\n" . $hereptr)) { - $line_fixed = 1; - $last_after = $n; - $space_after = 1; - } - } - if ($rtrim_before || $space_after) { - if ($rtrim_before) { - $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); - } else { - $good = $fix_elements[$n] . trim($fix_elements[$n + 1]); - } - if ($space_after) { - $good .= " "; - } - } - - # '*' as part of a type definition -- reported already. - } elsif ($opv eq '*_') { - #warn "'*' is part of type\n"; - - # unary operators should have a space before and - # none after. May be left adjacent to another - # unary operator, or a cast - } elsif ($op eq '!' || $op eq '~' || - $opv eq '*U' || $opv eq '-U' || - $opv eq '&U' || $opv eq '&&U') { - if ($ctx !~ /[WEBC]x./ && $ca !~ /(?:\)|!|~|\*|-|\&|\||\+\+|\-\-|\{)$/) { - if (ERROR("SPACING", - "space required before that '$op' $at\n" . $hereptr)) { - if ($n != $last_after + 2) { - $good = $fix_elements[$n] . " " . ltrim($fix_elements[$n + 1]); - $line_fixed = 1; - } - } - } - if ($op eq '*' && $cc =~/\s*$Modifier\b/) { - # A unary '*' may be const - - } elsif ($ctx =~ /.xW/) { - if (ERROR("SPACING", - "space prohibited after that '$op' $at\n" . $hereptr)) { - $good = $fix_elements[$n] . rtrim($fix_elements[$n + 1]); - if (defined $fix_elements[$n + 2]) { - $fix_elements[$n + 2] =~ s/^\s+//; - } - $line_fixed = 1; - } - } - - # unary ++ and unary -- are allowed no space on one side. - } elsif ($op eq '++' or $op eq '--') { - if ($ctx !~ /[WEOBC]x[^W]/ && $ctx !~ /[^W]x[WOBEC]/) { - if (ERROR("SPACING", - "space required one side of that '$op' $at\n" . $hereptr)) { - $good = $fix_elements[$n] . trim($fix_elements[$n + 1]) . " "; - $line_fixed = 1; - } - } - if ($ctx =~ /Wx[BE]/ || - ($ctx =~ /Wx./ && $cc =~ /^;/)) { - if (ERROR("SPACING", - "space prohibited before that '$op' $at\n" . $hereptr)) { - $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); - $line_fixed = 1; - } - } - if ($ctx =~ /ExW/) { - if (ERROR("SPACING", - "space prohibited after that '$op' $at\n" . $hereptr)) { - $good = $fix_elements[$n] . trim($fix_elements[$n + 1]); - if (defined $fix_elements[$n + 2]) { - $fix_elements[$n + 2] =~ s/^\s+//; - } - $line_fixed = 1; - } - } - - # << and >> may either have or not have spaces both sides - } elsif ($op eq '<<' or $op eq '>>' or - $op eq '&' or $op eq '^' or $op eq '|' or - $op eq '+' or $op eq '-' or - $op eq '*' or $op eq '/' or - $op eq '%') - { - if ($check) { - if (defined $fix_elements[$n + 2] && $ctx !~ /[EW]x[EW]/) { - if (CHK("SPACING", - "spaces preferred around that '$op' $at\n" . $hereptr)) { - $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " "; - $fix_elements[$n + 2] =~ s/^\s+//; - $line_fixed = 1; - } - } elsif (!defined $fix_elements[$n + 2] && $ctx !~ /Wx[OE]/) { - if (CHK("SPACING", - "space preferred before that '$op' $at\n" . $hereptr)) { - $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]); - $line_fixed = 1; - } - } - } elsif ($ctx =~ /Wx[^WCE]|[^WCE]xW/) { - if (ERROR("SPACING", - "need consistent spacing around '$op' $at\n" . $hereptr)) { - $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " "; - if (defined $fix_elements[$n + 2]) { - $fix_elements[$n + 2] =~ s/^\s+//; - } - $line_fixed = 1; - } - } - - # A colon needs no spaces before when it is - # terminating a case value or a label. - } elsif ($opv eq ':C' || $opv eq ':L') { - if ($ctx =~ /Wx./) { - if (ERROR("SPACING", - "space prohibited before that '$op' $at\n" . $hereptr)) { - $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); - $line_fixed = 1; - } - } - - # All the others need spaces both sides. - } elsif ($ctx !~ /[EWC]x[CWE]/) { - my $ok = 0; - - # Ignore email addresses <foo@bar> - if (($op eq '<' && - $cc =~ /^\S+\@\S+>/) || - ($op eq '>' && - $ca =~ /<\S+\@\S+$/)) - { - $ok = 1; - } - - # for asm volatile statements - # ignore a colon with another - # colon immediately before or after - if (($op eq ':') && - ($ca =~ /:$/ || $cc =~ /^:/)) { - $ok = 1; - } - - # messages are ERROR, but ?: are CHK - if ($ok == 0) { - my $msg_level = \&ERROR; - $msg_level = \&CHK if (($op eq '?:' || $op eq '?' || $op eq ':') && $ctx =~ /VxV/); - - if (&{$msg_level}("SPACING", - "spaces required around that '$op' $at\n" . $hereptr)) { - $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " "; - if (defined $fix_elements[$n + 2]) { - $fix_elements[$n + 2] =~ s/^\s+//; - } - $line_fixed = 1; - } - } - } - $off += length($elements[$n + 1]); - -## print("n: <$n> GOOD: <$good>\n"); - - $fixed_line = $fixed_line . $good; - } - - if (($#elements % 2) == 0) { - $fixed_line = $fixed_line . $fix_elements[$#elements]; - } - - if ($fix && $line_fixed && $fixed_line ne $fixed[$fixlinenr]) { - $fixed[$fixlinenr] = $fixed_line; - } - - - } - -# check for whitespace before a non-naked semicolon - if ($line =~ /^\+.*\S\s+;\s*$/) { - if (WARN("SPACING", - "space prohibited before semicolon\n" . $herecurr) && - $fix) { - 1 while $fixed[$fixlinenr] =~ - s/^(\+.*\S)\s+;/$1;/; - } - } - -# check for multiple assignments - if ($line =~ /^.\s*$Lval\s*=\s*$Lval\s*=(?!=)/) { - CHK("MULTIPLE_ASSIGNMENTS", - "multiple assignments should be avoided\n" . $herecurr); - } - -## # check for multiple declarations, allowing for a function declaration -## # continuation. -## if ($line =~ /^.\s*$Type\s+$Ident(?:\s*=[^,{]*)?\s*,\s*$Ident.*/ && -## $line !~ /^.\s*$Type\s+$Ident(?:\s*=[^,{]*)?\s*,\s*$Type\s*$Ident.*/) { -## -## # Remove any bracketed sections to ensure we do not -## # falsly report the parameters of functions. -## my $ln = $line; -## while ($ln =~ s/\([^\(\)]*\)//g) { -## } -## if ($ln =~ /,/) { -## WARN("MULTIPLE_DECLARATION", -## "declaring multiple variables together should be avoided\n" . $herecurr); -## } -## } - -#need space before brace following if, while, etc - if (($line =~ /\(.*\)\{/ && $line !~ /\($Type\)\{/) || - $line =~ /do\{/) { - if (ERROR("SPACING", - "space required before the open brace '{'\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/^(\+.*(?:do|\)))\{/$1 {/; - } - } - -## # check for blank lines before declarations -## if ($line =~ /^.\t+$Type\s+$Ident(?:\s*=.*)?;/ && -## $prevrawline =~ /^.\s*$/) { -## WARN("SPACING", -## "No blank lines before declarations\n" . $hereprev); -## } -## - -# closing brace should have a space following it when it has anything -# on the line - if ($line =~ /}(?!(?:,|;|\)))\S/) { - if (ERROR("SPACING", - "space required after that close brace '}'\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/}((?!(?:,|;|\)))\S)/} $1/; - } - } - -# check spacing on square brackets - if ($line =~ /\[\s/ && $line !~ /\[\s*$/) { - if (ERROR("SPACING", - "space prohibited after that open square bracket '['\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/\[\s+/\[/; - } - } - if ($line =~ /\s\]/) { - if (ERROR("SPACING", - "space prohibited before that close square bracket ']'\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/\s+\]/\]/; - } - } - -# check spacing on parentheses - if ($line =~ /\(\s/ && $line !~ /\(\s*(?:\\)?$/ && - $line !~ /for\s*\(\s+;/) { - if (ERROR("SPACING", - "space prohibited after that open parenthesis '('\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/\(\s+/\(/; - } - } - if ($line =~ /(\s+)\)/ && $line !~ /^.\s*\)/ && - $line !~ /for\s*\(.*;\s+\)/ && - $line !~ /:\s+\)/) { - if (ERROR("SPACING", - "space prohibited before that close parenthesis ')'\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/\s+\)/\)/; - } - } - -# check unnecessary parentheses around addressof/dereference single $Lvals -# ie: &(foo->bar) should be &foo->bar and *(foo->bar) should be *foo->bar - - while ($line =~ /(?:[^&]&\s*|\*)\(\s*($Ident\s*(?:$Member\s*)+)\s*\)/g) { - my $var = $1; - if (CHK("UNNECESSARY_PARENTHESES", - "Unnecessary parentheses around $var\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\(\s*\Q$var\E\s*\)/$var/; - } - } - -# check for unnecessary parentheses around function pointer uses -# ie: (foo->bar)(); should be foo->bar(); -# but not "if (foo->bar) (" to avoid some false positives - if ($line =~ /(\bif\s*|)(\(\s*$Ident\s*(?:$Member\s*)+\))[ \t]*\(/ && $1 !~ /^if/) { - my $var = $2; - if (CHK("UNNECESSARY_PARENTHESES", - "Unnecessary parentheses around function pointer $var\n" . $herecurr) && - $fix) { - my $var2 = deparenthesize($var); - $var2 =~ s/\s//g; - $fixed[$fixlinenr] =~ s/\Q$var\E/$var2/; - } - } - -# check for unnecessary parentheses around comparisons in if uses - if ($^V && $^V ge 5.10.0 && defined($stat) && - $stat =~ /(^.\s*if\s*($balanced_parens))/) { - my $if_stat = $1; - my $test = substr($2, 1, -1); - my $herectx; - while ($test =~ /(?:^|[^\w\&\!\~])+\s*\(\s*([\&\!\~]?\s*$Lval\s*(?:$Compare\s*$FuncArg)?)\s*\)/g) { - my $match = $1; - # avoid parentheses around potential macro args - next if ($match =~ /^\s*\w+\s*$/); - if (!defined($herectx)) { - $herectx = $here . "\n"; - my $cnt = statement_rawlines($if_stat); - for (my $n = 0; $n < $cnt; $n++) { - my $rl = raw_line($linenr, $n); - $herectx .= $rl . "\n"; - last if $rl =~ /^[ \+].*\{/; - } - } - CHK("UNNECESSARY_PARENTHESES", - "Unnecessary parentheses around '$match'\n" . $herectx); - } - } - -#goto labels aren't indented, allow a single space however - if ($line=~/^.\s+[A-Za-z\d_]+:(?![0-9]+)/ and - !($line=~/^. [A-Za-z\d_]+:/) and !($line=~/^.\s+default:/)) { - if (WARN("INDENTED_LABEL", - "labels should not be indented\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/^(.)\s+/$1/; - } - } - -# return is not a function - if (defined($stat) && $stat =~ /^.\s*return(\s*)\(/s) { - my $spacing = $1; - if ($^V && $^V ge 5.10.0 && - $stat =~ /^.\s*return\s*($balanced_parens)\s*;\s*$/) { - my $value = $1; - $value = deparenthesize($value); - if ($value =~ m/^\s*$FuncArg\s*(?:\?|$)/) { - ERROR("RETURN_PARENTHESES", - "return is not a function, parentheses are not required\n" . $herecurr); - } - } elsif ($spacing !~ /\s+/) { - ERROR("SPACING", - "space required before the open parenthesis '('\n" . $herecurr); - } - } - -# unnecessary return in a void function -# at end-of-function, with the previous line a single leading tab, then return; -# and the line before that not a goto label target like "out:" - if ($sline =~ /^[ \+]}\s*$/ && - $prevline =~ /^\+\treturn\s*;\s*$/ && - $linenr >= 3 && - $lines[$linenr - 3] =~ /^[ +]/ && - $lines[$linenr - 3] !~ /^[ +]\s*$Ident\s*:/) { - WARN("RETURN_VOID", - "void function return statements are not generally useful\n" . $hereprev); - } - -# if statements using unnecessary parentheses - ie: if ((foo == bar)) - if ($^V && $^V ge 5.10.0 && - $line =~ /\bif\s*((?:\(\s*){2,})/) { - my $openparens = $1; - my $count = $openparens =~ tr@\(@\(@; - my $msg = ""; - if ($line =~ /\bif\s*(?:\(\s*){$count,$count}$LvalOrFunc\s*($Compare)\s*$LvalOrFunc(?:\s*\)){$count,$count}/) { - my $comp = $4; #Not $1 because of $LvalOrFunc - $msg = " - maybe == should be = ?" if ($comp eq "=="); - WARN("UNNECESSARY_PARENTHESES", - "Unnecessary parentheses$msg\n" . $herecurr); - } - } - -# comparisons with a constant or upper case identifier on the left -# avoid cases like "foo + BAR < baz" -# only fix matches surrounded by parentheses to avoid incorrect -# conversions like "FOO < baz() + 5" being "misfixed" to "baz() > FOO + 5" - if ($^V && $^V ge 5.10.0 && - $line =~ /^\+(.*)\b($Constant|[A-Z_][A-Z0-9_]*)\s*($Compare)\s*($LvalOrFunc)/) { - my $lead = $1; - my $const = $2; - my $comp = $3; - my $to = $4; - my $newcomp = $comp; - if ($lead !~ /(?:$Operators|\.)\s*$/ && - $to !~ /^(?:Constant|[A-Z_][A-Z0-9_]*)$/ && - WARN("CONSTANT_COMPARISON", - "Comparisons should place the constant on the right side of the test\n" . $herecurr) && - $fix) { - if ($comp eq "<") { - $newcomp = ">"; - } elsif ($comp eq "<=") { - $newcomp = ">="; - } elsif ($comp eq ">") { - $newcomp = "<"; - } elsif ($comp eq ">=") { - $newcomp = "<="; - } - $fixed[$fixlinenr] =~ s/\(\s*\Q$const\E\s*$Compare\s*\Q$to\E\s*\)/($to $newcomp $const)/; - } - } - -# Return of what appears to be an errno should normally be negative - if ($sline =~ /\breturn(?:\s*\(+\s*|\s+)(E[A-Z]+)(?:\s*\)+\s*|\s*)[;:,]/) { - my $name = $1; - if ($name ne 'EOF' && $name ne 'ERROR') { - WARN("USE_NEGATIVE_ERRNO", - "return of an errno should typically be negative (ie: return -$1)\n" . $herecurr); - } - } - -# Need a space before open parenthesis after if, while etc - if ($line =~ /\b(if|while|for|switch)\(/) { - if (ERROR("SPACING", - "space required before the open parenthesis '('\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/\b(if|while|for|switch)\(/$1 \(/; - } - } - -# Check for illegal assignment in if conditional -- and check for trailing -# statements after the conditional. - if ($line =~ /do\s*(?!{)/) { - ($stat, $cond, $line_nr_next, $remain_next, $off_next) = - ctx_statement_block($linenr, $realcnt, 0) - if (!defined $stat); - my ($stat_next) = ctx_statement_block($line_nr_next, - $remain_next, $off_next); - $stat_next =~ s/\n./\n /g; - ##print "stat<$stat> stat_next<$stat_next>\n"; - - if ($stat_next =~ /^\s*while\b/) { - # If the statement carries leading newlines, - # then count those as offsets. - my ($whitespace) = - ($stat_next =~ /^((?:\s*\n[+-])*\s*)/s); - my $offset = - statement_rawlines($whitespace) - 1; - - $suppress_whiletrailers{$line_nr_next + - $offset} = 1; - } - } - if (!defined $suppress_whiletrailers{$linenr} && - defined($stat) && defined($cond) && - $line =~ /\b(?:if|while|for)\s*\(/ && $line !~ /^.\s*#/) { - my ($s, $c) = ($stat, $cond); - - if ($c =~ /\bif\s*\(.*[^<>!=]=[^=].*/s) { - ERROR("ASSIGN_IN_IF", - "do not use assignment in if condition\n" . $herecurr); - } - - # Find out what is on the end of the line after the - # conditional. - substr($s, 0, length($c), ''); - $s =~ s/\n.*//g; - $s =~ s/$;//g; # Remove any comments - if (length($c) && $s !~ /^\s*{?\s*\\*\s*$/ && - $c !~ /}\s*while\s*/) - { - # Find out how long the conditional actually is. - my @newlines = ($c =~ /\n/gs); - my $cond_lines = 1 + $#newlines; - my $stat_real = ''; - - $stat_real = raw_line($linenr, $cond_lines) - . "\n" if ($cond_lines); - if (defined($stat_real) && $cond_lines > 1) { - $stat_real = "[...]\n$stat_real"; - } - - ERROR("TRAILING_STATEMENTS", - "trailing statements should be on next line\n" . $herecurr . $stat_real); - } - } - -# Check for bitwise tests written as boolean - if ($line =~ / - (?: - (?:\[|\(|\&\&|\|\|) - \s*0[xX][0-9]+\s* - (?:\&\&|\|\|) - | - (?:\&\&|\|\|) - \s*0[xX][0-9]+\s* - (?:\&\&|\|\||\)|\]) - )/x) - { - WARN("HEXADECIMAL_BOOLEAN_TEST", - "boolean test with hexadecimal, perhaps just 1 \& or \|?\n" . $herecurr); - } - -# if and else should not have general statements after it - if ($line =~ /^.\s*(?:}\s*)?else\b(.*)/) { - my $s = $1; - $s =~ s/$;//g; # Remove any comments - if ($s !~ /^\s*(?:\sif|(?:{|)\s*\\?\s*$)/) { - ERROR("TRAILING_STATEMENTS", - "trailing statements should be on next line\n" . $herecurr); - } - } -# if should not continue a brace - if ($line =~ /}\s*if\b/) { - ERROR("TRAILING_STATEMENTS", - "trailing statements should be on next line (or did you mean 'else if'?)\n" . - $herecurr); - } -# case and default should not have general statements after them - if ($line =~ /^.\s*(?:case\s*.*|default\s*):/g && - $line !~ /\G(?: - (?:\s*$;*)(?:\s*{)?(?:\s*$;*)(?:\s*\\)?\s*$| - \s*return\s+ - )/xg) - { - ERROR("TRAILING_STATEMENTS", - "trailing statements should be on next line\n" . $herecurr); - } - - # Check for }<nl>else {, these must be at the same - # indent level to be relevant to each other. - if ($prevline=~/}\s*$/ and $line=~/^.\s*else\s*/ && - $previndent == $indent) { - if (ERROR("ELSE_AFTER_BRACE", - "else should follow close brace '}'\n" . $hereprev) && - $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { - fix_delete_line($fixlinenr - 1, $prevrawline); - fix_delete_line($fixlinenr, $rawline); - my $fixedline = $prevrawline; - $fixedline =~ s/}\s*$//; - if ($fixedline !~ /^\+\s*$/) { - fix_insert_line($fixlinenr, $fixedline); - } - $fixedline = $rawline; - $fixedline =~ s/^(.\s*)else/$1} else/; - fix_insert_line($fixlinenr, $fixedline); - } - } - - if ($prevline=~/}\s*$/ and $line=~/^.\s*while\s*/ && - $previndent == $indent) { - my ($s, $c) = ctx_statement_block($linenr, $realcnt, 0); - - # Find out what is on the end of the line after the - # conditional. - substr($s, 0, length($c), ''); - $s =~ s/\n.*//g; - - if ($s =~ /^\s*;/) { - if (ERROR("WHILE_AFTER_BRACE", - "while should follow close brace '}'\n" . $hereprev) && - $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { - fix_delete_line($fixlinenr - 1, $prevrawline); - fix_delete_line($fixlinenr, $rawline); - my $fixedline = $prevrawline; - my $trailing = $rawline; - $trailing =~ s/^\+//; - $trailing = trim($trailing); - $fixedline =~ s/}\s*$/} $trailing/; - fix_insert_line($fixlinenr, $fixedline); - } - } - } - -#Specific variable tests - while ($line =~ m{($Constant|$Lval)}g) { - my $var = $1; - -#gcc binary extension - if ($var =~ /^$Binary$/) { - if (WARN("GCC_BINARY_CONSTANT", - "Avoid gcc v4.3+ binary constant extension: <$var>\n" . $herecurr) && - $fix) { - my $hexval = sprintf("0x%x", oct($var)); - $fixed[$fixlinenr] =~ - s/\b$var\b/$hexval/; - } - } - -#CamelCase - if ($var !~ /^$Constant$/ && - $var =~ /[A-Z][a-z]|[a-z][A-Z]/ && -#Ignore Page<foo> variants - $var !~ /^(?:Clear|Set|TestClear|TestSet|)Page[A-Z]/ && -#Ignore SI style variants like nS, mV and dB (ie: max_uV, regulator_min_uA_show) - $var !~ /^(?:[a-z_]*?)_?[a-z][A-Z](?:_[a-z_]+)?$/ && -#Ignore some three character SI units explicitly, like MiB and KHz - $var !~ /^(?:[a-z_]*?)_?(?:[KMGT]iB|[KMGT]?Hz)(?:_[a-z_]+)?$/) { - while ($var =~ m{($Ident)}g) { - my $word = $1; - next if ($word !~ /[A-Z][a-z]|[a-z][A-Z]/); - if ($check) { - seed_camelcase_includes(); - if (!$file && !$camelcase_file_seeded) { - seed_camelcase_file($realfile); - $camelcase_file_seeded = 1; - } - } - if (!defined $camelcase{$word}) { - $camelcase{$word} = 1; - CHK("CAMELCASE", - "Avoid CamelCase: <$word>\n" . $herecurr); - } - } - } - } - -#no spaces allowed after \ in define - if ($line =~ /\#\s*define.*\\\s+$/) { - if (WARN("WHITESPACE_AFTER_LINE_CONTINUATION", - "Whitespace after \\ makes next lines useless\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\s+$//; - } - } - -# warn if <asm/foo.h> is #included and <linux/foo.h> is available and includes -# itself <asm/foo.h> (uses RAW line) - if ($tree && $rawline =~ m{^.\s*\#\s*include\s*\<asm\/(.*)\.h\>}) { - my $file = "$1.h"; - my $checkfile = "include/linux/$file"; - if (-f "$root/$checkfile" && - $realfile ne $checkfile && - $1 !~ /$allowed_asm_includes/) - { - my $asminclude = `grep -Ec "#include\\s+<asm/$file>" $root/$checkfile`; - if ($asminclude > 0) { - if ($realfile =~ m{^arch/}) { - CHK("ARCH_INCLUDE_LINUX", - "Consider using #include <linux/$file> instead of <asm/$file>\n" . $herecurr); - } else { - WARN("INCLUDE_LINUX", - "Use #include <linux/$file> instead of <asm/$file>\n" . $herecurr); - } - } - } - } - -# multi-statement macros should be enclosed in a do while loop, grab the -# first statement and ensure its the whole macro if its not enclosed -# in a known good container - if ($realfile !~ m@/vmlinux.lds.h$@ && - $line =~ /^.\s*\#\s*define\s*$Ident(\()?/) { - my $ln = $linenr; - my $cnt = $realcnt; - my ($off, $dstat, $dcond, $rest); - my $ctx = ''; - my $has_flow_statement = 0; - my $has_arg_concat = 0; - ($dstat, $dcond, $ln, $cnt, $off) = - ctx_statement_block($linenr, $realcnt, 0); - $ctx = $dstat; - #print "dstat<$dstat> dcond<$dcond> cnt<$cnt> off<$off>\n"; - #print "LINE<$lines[$ln-1]> len<" . length($lines[$ln-1]) . "\n"; - - $has_flow_statement = 1 if ($ctx =~ /\b(goto|return)\b/); - $has_arg_concat = 1 if ($ctx =~ /\#\#/ && $ctx !~ /\#\#\s*(?:__VA_ARGS__|args)\b/); - - $dstat =~ s/^.\s*\#\s*define\s+$Ident(\([^\)]*\))?\s*//; - my $define_args = $1; - my $define_stmt = $dstat; - my @def_args = (); - - if (defined $define_args && $define_args ne "") { - $define_args = substr($define_args, 1, length($define_args) - 2); - $define_args =~ s/\s*//g; - @def_args = split(",", $define_args); - } - - $dstat =~ s/$;//g; - $dstat =~ s/\\\n.//g; - $dstat =~ s/^\s*//s; - $dstat =~ s/\s*$//s; - - # Flatten any parentheses and braces - while ($dstat =~ s/\([^\(\)]*\)/1/ || - $dstat =~ s/\{[^\{\}]*\}/1/ || - $dstat =~ s/.\[[^\[\]]*\]/1/) - { - } - - # Flatten any obvious string concatentation. - while ($dstat =~ s/($String)\s*$Ident/$1/ || - $dstat =~ s/$Ident\s*($String)/$1/) - { - } - - # Make asm volatile uses seem like a generic function - $dstat =~ s/\b_*asm_*\s+_*volatile_*\b/asm_volatile/g; - - my $exceptions = qr{ - $Declare| - module_param_named| - MODULE_PARM_DESC| - DECLARE_PER_CPU| - DEFINE_PER_CPU| - __typeof__\(| - union| - struct| - \.$Ident\s*=\s*| - ^\"|\"$| - ^\[ - }x; - #print "REST<$rest> dstat<$dstat> ctx<$ctx>\n"; - - $ctx =~ s/\n*$//; - my $herectx = $here . "\n"; - my $stmt_cnt = statement_rawlines($ctx); - - for (my $n = 0; $n < $stmt_cnt; $n++) { - $herectx .= raw_line($linenr, $n) . "\n"; - } - - if ($dstat ne '' && - $dstat !~ /^(?:$Ident|-?$Constant),$/ && # 10, // foo(), - $dstat !~ /^(?:$Ident|-?$Constant);$/ && # foo(); - $dstat !~ /^[!~-]?(?:$Lval|$Constant)$/ && # 10 // foo() // !foo // ~foo // -foo // foo->bar // foo.bar->baz - $dstat !~ /^'X'$/ && $dstat !~ /^'XX'$/ && # character constants - $dstat !~ /$exceptions/ && - $dstat !~ /^\.$Ident\s*=/ && # .foo = - $dstat !~ /^(?:\#\s*$Ident|\#\s*$Constant)\s*$/ && # stringification #foo - $dstat !~ /^do\s*$Constant\s*while\s*$Constant;?$/ && # do {...} while (...); // do {...} while (...) - $dstat !~ /^for\s*$Constant$/ && # for (...) - $dstat !~ /^for\s*$Constant\s+(?:$Ident|-?$Constant)$/ && # for (...) bar() - $dstat !~ /^do\s*{/ && # do {... - $dstat !~ /^\(\{/ && # ({... - $ctx !~ /^.\s*#\s*define\s+TRACE_(?:SYSTEM|INCLUDE_FILE|INCLUDE_PATH)\b/) - { - if ($dstat =~ /^\s*if\b/) { - ERROR("MULTISTATEMENT_MACRO_USE_DO_WHILE", - "Macros starting with if should be enclosed by a do - while loop to avoid possible if/else logic defects\n" . "$herectx"); - } elsif ($dstat =~ /;/) { - ERROR("MULTISTATEMENT_MACRO_USE_DO_WHILE", - "Macros with multiple statements should be enclosed in a do - while loop\n" . "$herectx"); - } else { - WARN("COMPLEX_MACRO", - "Macros with complex values should be enclosed in parentheses\n" . "$herectx"); - } - - } - - # Make $define_stmt single line, comment-free, etc - my @stmt_array = split('\n', $define_stmt); - my $first = 1; - $define_stmt = ""; - foreach my $l (@stmt_array) { - $l =~ s/\\$//; - if ($first) { - $define_stmt = $l; - $first = 0; - } elsif ($l =~ /^[\+ ]/) { - $define_stmt .= substr($l, 1); - } - } - $define_stmt =~ s/$;//g; - $define_stmt =~ s/\s+/ /g; - $define_stmt = trim($define_stmt); - -# check if any macro arguments are reused (ignore '...' and 'type') - foreach my $arg (@def_args) { - next if ($arg =~ /\.\.\./); - next if ($arg =~ /^type$/i); - my $tmp_stmt = $define_stmt; - $tmp_stmt =~ s/\b(typeof|__typeof__|__builtin\w+|typecheck\s*\(\s*$Type\s*,|\#+)\s*\(*\s*$arg\s*\)*\b//g; - $tmp_stmt =~ s/\#+\s*$arg\b//g; - $tmp_stmt =~ s/\b$arg\s*\#\#//g; - my $use_cnt = $tmp_stmt =~ s/\b$arg\b//g; - if ($use_cnt > 1) { - CHK("MACRO_ARG_REUSE", - "Macro argument reuse '$arg' - possible side-effects?\n" . "$herectx"); - } -# check if any macro arguments may have other precedence issues - if ($tmp_stmt =~ m/($Operators)?\s*\b$arg\b\s*($Operators)?/m && - ((defined($1) && $1 ne ',') || - (defined($2) && $2 ne ','))) { - CHK("MACRO_ARG_PRECEDENCE", - "Macro argument '$arg' may be better as '($arg)' to avoid precedence issues\n" . "$herectx"); - } - } - -# check for macros with flow control, but without ## concatenation -# ## concatenation is commonly a macro that defines a function so ignore those - if ($has_flow_statement && !$has_arg_concat) { - my $herectx = $here . "\n"; - my $cnt = statement_rawlines($ctx); - - for (my $n = 0; $n < $cnt; $n++) { - $herectx .= raw_line($linenr, $n) . "\n"; - } - WARN("MACRO_WITH_FLOW_CONTROL", - "Macros with flow control statements should be avoided\n" . "$herectx"); - } - -# check for line continuations outside of #defines, preprocessor #, and asm - - } else { - if ($prevline !~ /^..*\\$/ && - $line !~ /^\+\s*\#.*\\$/ && # preprocessor - $line !~ /^\+.*\b(__asm__|asm)\b.*\\$/ && # asm - $line =~ /^\+.*\\$/) { - WARN("LINE_CONTINUATIONS", - "Avoid unnecessary line continuations\n" . $herecurr); - } - } - -# do {} while (0) macro tests: -# single-statement macros do not need to be enclosed in do while (0) loop, -# macro should not end with a semicolon - if ($^V && $^V ge 5.10.0 && - $realfile !~ m@/vmlinux.lds.h$@ && - $line =~ /^.\s*\#\s*define\s+$Ident(\()?/) { - my $ln = $linenr; - my $cnt = $realcnt; - my ($off, $dstat, $dcond, $rest); - my $ctx = ''; - ($dstat, $dcond, $ln, $cnt, $off) = - ctx_statement_block($linenr, $realcnt, 0); - $ctx = $dstat; - - $dstat =~ s/\\\n.//g; - $dstat =~ s/$;/ /g; - - if ($dstat =~ /^\+\s*#\s*define\s+$Ident\s*${balanced_parens}\s*do\s*{(.*)\s*}\s*while\s*\(\s*0\s*\)\s*([;\s]*)\s*$/) { - my $stmts = $2; - my $semis = $3; - - $ctx =~ s/\n*$//; - my $cnt = statement_rawlines($ctx); - my $herectx = $here . "\n"; - - for (my $n = 0; $n < $cnt; $n++) { - $herectx .= raw_line($linenr, $n) . "\n"; - } - - if (($stmts =~ tr/;/;/) == 1 && - $stmts !~ /^\s*(if|while|for|switch)\b/) { - WARN("SINGLE_STATEMENT_DO_WHILE_MACRO", - "Single statement macros should not use a do {} while (0) loop\n" . "$herectx"); - } - if (defined $semis && $semis ne "") { - WARN("DO_WHILE_MACRO_WITH_TRAILING_SEMICOLON", - "do {} while (0) macros should not be semicolon terminated\n" . "$herectx"); - } - } elsif ($dstat =~ /^\+\s*#\s*define\s+$Ident.*;\s*$/) { - $ctx =~ s/\n*$//; - my $cnt = statement_rawlines($ctx); - my $herectx = $here . "\n"; - - for (my $n = 0; $n < $cnt; $n++) { - $herectx .= raw_line($linenr, $n) . "\n"; - } - - WARN("TRAILING_SEMICOLON", - "macros should not use a trailing semicolon\n" . "$herectx"); - } - } - -# make sure symbols are always wrapped with VMLINUX_SYMBOL() ... -# all assignments may have only one of the following with an assignment: -# . -# ALIGN(...) -# VMLINUX_SYMBOL(...) - if ($realfile eq 'vmlinux.lds.h' && $line =~ /(?:(?:^|\s)$Ident\s*=|=\s*$Ident(?:\s|$))/) { - WARN("MISSING_VMLINUX_SYMBOL", - "vmlinux.lds.h needs VMLINUX_SYMBOL() around C-visible symbols\n" . $herecurr); - } - -# check for redundant bracing round if etc - if ($line =~ /(^.*)\bif\b/ && $1 !~ /else\s*$/) { - my ($level, $endln, @chunks) = - ctx_statement_full($linenr, $realcnt, 1); - #print "chunks<$#chunks> linenr<$linenr> endln<$endln> level<$level>\n"; - #print "APW: <<$chunks[1][0]>><<$chunks[1][1]>>\n"; - if ($#chunks > 0 && $level == 0) { - my @allowed = (); - my $allow = 0; - my $seen = 0; - my $herectx = $here . "\n"; - my $ln = $linenr - 1; - for my $chunk (@chunks) { - my ($cond, $block) = @{$chunk}; - - # If the condition carries leading newlines, then count those as offsets. - my ($whitespace) = ($cond =~ /^((?:\s*\n[+-])*\s*)/s); - my $offset = statement_rawlines($whitespace) - 1; - - $allowed[$allow] = 0; - #print "COND<$cond> whitespace<$whitespace> offset<$offset>\n"; - - # We have looked at and allowed this specific line. - $suppress_ifbraces{$ln + $offset} = 1; - - $herectx .= "$rawlines[$ln + $offset]\n[...]\n"; - $ln += statement_rawlines($block) - 1; - - substr($block, 0, length($cond), ''); - - $seen++ if ($block =~ /^\s*{/); - - #print "cond<$cond> block<$block> allowed<$allowed[$allow]>\n"; - if (statement_lines($cond) > 1) { - #print "APW: ALLOWED: cond<$cond>\n"; - $allowed[$allow] = 1; - } - if ($block =~/\b(?:if|for|while)\b/) { - #print "APW: ALLOWED: block<$block>\n"; - $allowed[$allow] = 1; - } - if (statement_block_size($block) > 1) { - #print "APW: ALLOWED: lines block<$block>\n"; - $allowed[$allow] = 1; - } - $allow++; - } - if ($seen) { - my $sum_allowed = 0; - foreach (@allowed) { - $sum_allowed += $_; - } - if ($sum_allowed == 0) { - WARN("BRACES", - "braces {} are not necessary for any arm of this statement\n" . $herectx); - } elsif ($sum_allowed != $allow && - $seen != $allow) { - CHK("BRACES", - "braces {} should be used on all arms of this statement\n" . $herectx); - } - } - } - } - if (!defined $suppress_ifbraces{$linenr - 1} && - $line =~ /\b(if|while|for|else)\b/) { - my $allowed = 0; - - # Check the pre-context. - if (substr($line, 0, $-[0]) =~ /(\}\s*)$/) { - #print "APW: ALLOWED: pre<$1>\n"; - $allowed = 1; - } - - my ($level, $endln, @chunks) = - ctx_statement_full($linenr, $realcnt, $-[0]); - - # Check the condition. - my ($cond, $block) = @{$chunks[0]}; - #print "CHECKING<$linenr> cond<$cond> block<$block>\n"; - if (defined $cond) { - substr($block, 0, length($cond), ''); - } - if (statement_lines($cond) > 1) { - #print "APW: ALLOWED: cond<$cond>\n"; - $allowed = 1; - } - if ($block =~/\b(?:if|for|while)\b/) { - #print "APW: ALLOWED: block<$block>\n"; - $allowed = 1; - } - if (statement_block_size($block) > 1) { - #print "APW: ALLOWED: lines block<$block>\n"; - $allowed = 1; - } - # Check the post-context. - if (defined $chunks[1]) { - my ($cond, $block) = @{$chunks[1]}; - if (defined $cond) { - substr($block, 0, length($cond), ''); - } - if ($block =~ /^\s*\{/) { - #print "APW: ALLOWED: chunk-1 block<$block>\n"; - $allowed = 1; - } - } - if ($level == 0 && $block =~ /^\s*\{/ && !$allowed) { - my $herectx = $here . "\n"; - my $cnt = statement_rawlines($block); - - for (my $n = 0; $n < $cnt; $n++) { - $herectx .= raw_line($linenr, $n) . "\n"; - } - - WARN("BRACES", - "braces {} are not necessary for single statement blocks\n" . $herectx); - } - } - -# check for single line unbalanced braces - if ($sline =~ /^.\s*\}\s*else\s*$/ || - $sline =~ /^.\s*else\s*\{\s*$/) { - CHK("BRACES", "Unbalanced braces around else statement\n" . $herecurr); - } - -# check for unnecessary blank lines around braces - if (($line =~ /^.\s*}\s*$/ && $prevrawline =~ /^.\s*$/)) { - if (CHK("BRACES", - "Blank lines aren't necessary before a close brace '}'\n" . $hereprev) && - $fix && $prevrawline =~ /^\+/) { - fix_delete_line($fixlinenr - 1, $prevrawline); - } - } - if (($rawline =~ /^.\s*$/ && $prevline =~ /^..*{\s*$/)) { - if (CHK("BRACES", - "Blank lines aren't necessary after an open brace '{'\n" . $hereprev) && - $fix) { - fix_delete_line($fixlinenr, $rawline); - } - } - -# no volatiles please - my $asm_volatile = qr{\b(__asm__|asm)\s+(__volatile__|volatile)\b}; - if ($line =~ /\bvolatile\b/ && $line !~ /$asm_volatile/) { - WARN("VOLATILE", - "Use of volatile is usually wrong: see Documentation/process/volatile-considered-harmful.rst\n" . $herecurr); - } - -# Check for user-visible strings broken across lines, which breaks the ability -# to grep for the string. Make exceptions when the previous string ends in a -# newline (multiple lines in one string constant) or '\t', '\r', ';', or '{' -# (common in inline assembly) or is a octal \123 or hexadecimal \xaf value - if ($line =~ /^\+\s*$String/ && - $prevline =~ /"\s*$/ && - $prevrawline !~ /(?:\\(?:[ntr]|[0-7]{1,3}|x[0-9a-fA-F]{1,2})|;\s*|\{\s*)"\s*$/) { - if (WARN("SPLIT_STRING", - "quoted string split across lines\n" . $hereprev) && - $fix && - $prevrawline =~ /^\+.*"\s*$/ && - $last_coalesced_string_linenr != $linenr - 1) { - my $extracted_string = get_quoted_string($line, $rawline); - my $comma_close = ""; - if ($rawline =~ /\Q$extracted_string\E(\s*\)\s*;\s*$|\s*,\s*)/) { - $comma_close = $1; - } - - fix_delete_line($fixlinenr - 1, $prevrawline); - fix_delete_line($fixlinenr, $rawline); - my $fixedline = $prevrawline; - $fixedline =~ s/"\s*$//; - $fixedline .= substr($extracted_string, 1) . trim($comma_close); - fix_insert_line($fixlinenr - 1, $fixedline); - $fixedline = $rawline; - $fixedline =~ s/\Q$extracted_string\E\Q$comma_close\E//; - if ($fixedline !~ /\+\s*$/) { - fix_insert_line($fixlinenr, $fixedline); - } - $last_coalesced_string_linenr = $linenr; - } - } - -# check for missing a space in a string concatenation - if ($prevrawline =~ /[^\\]\w"$/ && $rawline =~ /^\+[\t ]+"\w/) { - WARN('MISSING_SPACE', - "break quoted strings at a space character\n" . $hereprev); - } - -# check for an embedded function name in a string when the function is known -# This does not work very well for -f --file checking as it depends on patch -# context providing the function name or a single line form for in-file -# function declarations - if ($line =~ /^\+.*$String/ && - defined($context_function) && - get_quoted_string($line, $rawline) =~ /\b$context_function\b/ && - length(get_quoted_string($line, $rawline)) != (length($context_function) + 2)) { - WARN("EMBEDDED_FUNCTION_NAME", - "Prefer using '\"%s...\", __func__' to using '$context_function', this function's name, in a string\n" . $herecurr); - } - -# check for spaces before a quoted newline - if ($rawline =~ /^.*\".*\s\\n/) { - if (WARN("QUOTED_WHITESPACE_BEFORE_NEWLINE", - "unnecessary whitespace before a quoted newline\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/^(\+.*\".*)\s+\\n/$1\\n/; - } - - } - -# concatenated string without spaces between elements - if ($line =~ /$String[A-Z_]/ || $line =~ /[A-Za-z0-9_]$String/) { - CHK("CONCATENATED_STRING", - "Concatenated strings should use spaces between elements\n" . $herecurr); - } - -# uncoalesced string fragments - if ($line =~ /$String\s*"/) { - WARN("STRING_FRAGMENTS", - "Consecutive strings are generally better as a single string\n" . $herecurr); - } - -# check for non-standard and hex prefixed decimal printf formats - my $show_L = 1; #don't show the same defect twice - my $show_Z = 1; - while ($line =~ /(?:^|")([X\t]*)(?:"|$)/g) { - my $string = substr($rawline, $-[1], $+[1] - $-[1]); - $string =~ s/%%/__/g; - # check for %L - if ($show_L && $string =~ /%[\*\d\.\$]*L([diouxX])/) { - WARN("PRINTF_L", - "\%L$1 is non-standard C, use %ll$1\n" . $herecurr); - $show_L = 0; - } - # check for %Z - if ($show_Z && $string =~ /%[\*\d\.\$]*Z([diouxX])/) { - WARN("PRINTF_Z", - "%Z$1 is non-standard C, use %z$1\n" . $herecurr); - $show_Z = 0; - } - # check for 0x<decimal> - if ($string =~ /0x%[\*\d\.\$\Llzth]*[diou]/) { - ERROR("PRINTF_0XDECIMAL", - "Prefixing 0x with decimal output is defective\n" . $herecurr); - } - } - -# check for line continuations in quoted strings with odd counts of " - if ($rawline =~ /\\$/ && $rawline =~ tr/"/"/ % 2) { - WARN("LINE_CONTINUATIONS", - "Avoid line continuations in quoted strings\n" . $herecurr); - } - -# warn about #if 0 - if ($line =~ /^.\s*\#\s*if\s+0\b/) { - CHK("REDUNDANT_CODE", - "if this code is redundant consider removing it\n" . - $herecurr); - } - -# check for needless "if (<foo>) fn(<foo>)" uses - if ($prevline =~ /\bif\s*\(\s*($Lval)\s*\)/) { - my $tested = quotemeta($1); - my $expr = '\s*\(\s*' . $tested . '\s*\)\s*;'; - if ($line =~ /\b(kfree|usb_free_urb|debugfs_remove(?:_recursive)?|(?:kmem_cache|mempool|dma_pool)_destroy)$expr/) { - my $func = $1; - if (WARN('NEEDLESS_IF', - "$func(NULL) is safe and this check is probably not required\n" . $hereprev) && - $fix) { - my $do_fix = 1; - my $leading_tabs = ""; - my $new_leading_tabs = ""; - if ($lines[$linenr - 2] =~ /^\+(\t*)if\s*\(\s*$tested\s*\)\s*$/) { - $leading_tabs = $1; - } else { - $do_fix = 0; - } - if ($lines[$linenr - 1] =~ /^\+(\t+)$func\s*\(\s*$tested\s*\)\s*;\s*$/) { - $new_leading_tabs = $1; - if (length($leading_tabs) + 1 ne length($new_leading_tabs)) { - $do_fix = 0; - } - } else { - $do_fix = 0; - } - if ($do_fix) { - fix_delete_line($fixlinenr - 1, $prevrawline); - $fixed[$fixlinenr] =~ s/^\+$new_leading_tabs/\+$leading_tabs/; - } - } - } - } - -# check for unnecessary "Out of Memory" messages - if ($line =~ /^\+.*\b$logFunctions\s*\(/ && - $prevline =~ /^[ \+]\s*if\s*\(\s*(\!\s*|NULL\s*==\s*)?($Lval)(\s*==\s*NULL\s*)?\s*\)/ && - (defined $1 || defined $3) && - $linenr > 3) { - my $testval = $2; - my $testline = $lines[$linenr - 3]; - - my ($s, $c) = ctx_statement_block($linenr - 3, $realcnt, 0); -# print("line: <$line>\nprevline: <$prevline>\ns: <$s>\nc: <$c>\n\n\n"); - - if ($s =~ /(?:^|\n)[ \+]\s*(?:$Type\s*)?\Q$testval\E\s*=\s*(?:\([^\)]*\)\s*)?\s*(?:devm_)?(?:[kv][czm]alloc(?:_node|_array)?\b|kstrdup|kmemdup|(?:dev_)?alloc_skb)/) { - WARN("OOM_MESSAGE", - "Possible unnecessary 'out of memory' message\n" . $hereprev); - } - } - -# check for logging functions with KERN_<LEVEL> - if ($line !~ /printk(?:_ratelimited|_once)?\s*\(/ && - $line =~ /\b$logFunctions\s*\(.*\b(KERN_[A-Z]+)\b/) { - my $level = $1; - if (WARN("UNNECESSARY_KERN_LEVEL", - "Possible unnecessary $level\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\s*$level\s*//; - } - } - -# check for logging continuations - if ($line =~ /\bprintk\s*\(\s*KERN_CONT\b|\bpr_cont\s*\(/) { - WARN("LOGGING_CONTINUATION", - "Avoid logging continuation uses where feasible\n" . $herecurr); - } - -# check for mask then right shift without a parentheses - if ($^V && $^V ge 5.10.0 && - $line =~ /$LvalOrFunc\s*\&\s*($LvalOrFunc)\s*>>/ && - $4 !~ /^\&/) { # $LvalOrFunc may be &foo, ignore if so - WARN("MASK_THEN_SHIFT", - "Possible precedence defect with mask then right shift - may need parentheses\n" . $herecurr); - } - -# check for pointer comparisons to NULL - if ($^V && $^V ge 5.10.0) { - while ($line =~ /\b$LvalOrFunc\s*(==|\!=)\s*NULL\b/g) { - my $val = $1; - my $equal = "!"; - $equal = "" if ($4 eq "!="); - if (CHK("COMPARISON_TO_NULL", - "Comparison to NULL could be written \"${equal}${val}\"\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\b\Q$val\E\s*(?:==|\!=)\s*NULL\b/$equal$val/; - } - } - } - -# check for bad placement of section $InitAttribute (e.g.: __initdata) - if ($line =~ /(\b$InitAttribute\b)/) { - my $attr = $1; - if ($line =~ /^\+\s*static\s+(?:const\s+)?(?:$attr\s+)?($NonptrTypeWithAttr)\s+(?:$attr\s+)?($Ident(?:\[[^]]*\])?)\s*[=;]/) { - my $ptr = $1; - my $var = $2; - if ((($ptr =~ /\b(union|struct)\s+$attr\b/ && - ERROR("MISPLACED_INIT", - "$attr should be placed after $var\n" . $herecurr)) || - ($ptr !~ /\b(union|struct)\s+$attr\b/ && - WARN("MISPLACED_INIT", - "$attr should be placed after $var\n" . $herecurr))) && - $fix) { - $fixed[$fixlinenr] =~ s/(\bstatic\s+(?:const\s+)?)(?:$attr\s+)?($NonptrTypeWithAttr)\s+(?:$attr\s+)?($Ident(?:\[[^]]*\])?)\s*([=;])\s*/"$1" . trim(string_find_replace($2, "\\s*$attr\\s*", " ")) . " " . trim(string_find_replace($3, "\\s*$attr\\s*", "")) . " $attr" . ("$4" eq ";" ? ";" : " = ")/e; - } - } - } - -# check for $InitAttributeData (ie: __initdata) with const - if ($line =~ /\bconst\b/ && $line =~ /($InitAttributeData)/) { - my $attr = $1; - $attr =~ /($InitAttributePrefix)(.*)/; - my $attr_prefix = $1; - my $attr_type = $2; - if (ERROR("INIT_ATTRIBUTE", - "Use of const init definition must use ${attr_prefix}initconst\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/$InitAttributeData/${attr_prefix}initconst/; - } - } - -# check for $InitAttributeConst (ie: __initconst) without const - if ($line !~ /\bconst\b/ && $line =~ /($InitAttributeConst)/) { - my $attr = $1; - if (ERROR("INIT_ATTRIBUTE", - "Use of $attr requires a separate use of const\n" . $herecurr) && - $fix) { - my $lead = $fixed[$fixlinenr] =~ - /(^\+\s*(?:static\s+))/; - $lead = rtrim($1); - $lead = "$lead " if ($lead !~ /^\+$/); - $lead = "${lead}const "; - $fixed[$fixlinenr] =~ s/(^\+\s*(?:static\s+))/$lead/; - } - } - -# check for __read_mostly with const non-pointer (should just be const) - if ($line =~ /\b__read_mostly\b/ && - $line =~ /($Type)\s*$Ident/ && $1 !~ /\*\s*$/ && $1 =~ /\bconst\b/) { - if (ERROR("CONST_READ_MOSTLY", - "Invalid use of __read_mostly with const type\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\s+__read_mostly\b//; - } - } - -# don't use __constant_<foo> functions outside of include/uapi/ - if ($realfile !~ m@^include/uapi/@ && - $line =~ /(__constant_(?:htons|ntohs|[bl]e(?:16|32|64)_to_cpu|cpu_to_[bl]e(?:16|32|64)))\s*\(/) { - my $constant_func = $1; - my $func = $constant_func; - $func =~ s/^__constant_//; - if (WARN("CONSTANT_CONVERSION", - "$constant_func should be $func\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\b$constant_func\b/$func/g; - } - } - -# prefer usleep_range over udelay - if ($line =~ /\budelay\s*\(\s*(\d+)\s*\)/) { - my $delay = $1; - # ignore udelay's < 10, however - if (! ($delay < 10) ) { - CHK("USLEEP_RANGE", - "usleep_range is preferred over udelay; see Documentation/timers/timers-howto.txt\n" . $herecurr); - } - if ($delay > 2000) { - WARN("LONG_UDELAY", - "long udelay - prefer mdelay; see arch/arm/include/asm/delay.h\n" . $herecurr); - } - } - -# warn about unexpectedly long msleep's - if ($line =~ /\bmsleep\s*\((\d+)\);/) { - if ($1 < 20) { - WARN("MSLEEP", - "msleep < 20ms can sleep for up to 20ms; see Documentation/timers/timers-howto.txt\n" . $herecurr); - } - } - -# check for comparisons of jiffies - if ($line =~ /\bjiffies\s*$Compare|$Compare\s*jiffies\b/) { - WARN("JIFFIES_COMPARISON", - "Comparing jiffies is almost always wrong; prefer time_after, time_before and friends\n" . $herecurr); - } - -# check for comparisons of get_jiffies_64() - if ($line =~ /\bget_jiffies_64\s*\(\s*\)\s*$Compare|$Compare\s*get_jiffies_64\s*\(\s*\)/) { - WARN("JIFFIES_COMPARISON", - "Comparing get_jiffies_64() is almost always wrong; prefer time_after64, time_before64 and friends\n" . $herecurr); - } - -# warn about #ifdefs in C files -# if ($line =~ /^.\s*\#\s*if(|n)def/ && ($realfile =~ /\.c$/)) { -# print "#ifdef in C files should be avoided\n"; -# print "$herecurr"; -# $clean = 0; -# } - -# warn about spacing in #ifdefs - if ($line =~ /^.\s*\#\s*(ifdef|ifndef|elif)\s\s+/) { - if (ERROR("SPACING", - "exactly one space required after that #$1\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ - s/^(.\s*\#\s*(ifdef|ifndef|elif))\s{2,}/$1 /; - } - - } - -# check for spinlock_t definitions without a comment. - if ($line =~ /^.\s*(struct\s+mutex|spinlock_t)\s+\S+;/ || - $line =~ /^.\s*(DEFINE_MUTEX)\s*\(/) { - my $which = $1; - if (!ctx_has_comment($first_line, $linenr)) { - CHK("UNCOMMENTED_DEFINITION", - "$1 definition without comment\n" . $herecurr); - } - } -# check for memory barriers without a comment. - - my $barriers = qr{ - mb| - rmb| - wmb| - read_barrier_depends - }x; - my $barrier_stems = qr{ - mb__before_atomic| - mb__after_atomic| - store_release| - load_acquire| - store_mb| - (?:$barriers) - }x; - my $all_barriers = qr{ - (?:$barriers)| - smp_(?:$barrier_stems)| - virt_(?:$barrier_stems) - }x; - - if ($line =~ /\b(?:$all_barriers)\s*\(/) { - if (!ctx_has_comment($first_line, $linenr)) { - WARN("MEMORY_BARRIER", - "memory barrier without comment\n" . $herecurr); - } - } - - my $underscore_smp_barriers = qr{__smp_(?:$barrier_stems)}x; - - if ($realfile !~ m@^include/asm-generic/@ && - $realfile !~ m@/barrier\.h$@ && - $line =~ m/\b(?:$underscore_smp_barriers)\s*\(/ && - $line !~ m/^.\s*\#\s*define\s+(?:$underscore_smp_barriers)\s*\(/) { - WARN("MEMORY_BARRIER", - "__smp memory barriers shouldn't be used outside barrier.h and asm-generic\n" . $herecurr); - } - -# check for waitqueue_active without a comment. - if ($line =~ /\bwaitqueue_active\s*\(/) { - if (!ctx_has_comment($first_line, $linenr)) { - WARN("WAITQUEUE_ACTIVE", - "waitqueue_active without comment\n" . $herecurr); - } - } - -# check of hardware specific defines - if ($line =~ m@^.\s*\#\s*if.*\b(__i386__|__powerpc64__|__sun__|__s390x__)\b@ && $realfile !~ m@include/asm-@) { - CHK("ARCH_DEFINES", - "architecture specific defines should be avoided\n" . $herecurr); - } - -# check that the storage class is not after a type - if ($line =~ /\b($Type)\s+($Storage)\b/) { - WARN("STORAGE_CLASS", - "storage class '$2' should be located before type '$1'\n" . $herecurr); - } -# Check that the storage class is at the beginning of a declaration - if ($line =~ /\b$Storage\b/ && - $line !~ /^.\s*$Storage/ && - $line =~ /^.\s*(.+?)\$Storage\s/ && - $1 !~ /[\,\)]\s*$/) { - WARN("STORAGE_CLASS", - "storage class should be at the beginning of the declaration\n" . $herecurr); - } - -# check the location of the inline attribute, that it is between -# storage class and type. - if ($line =~ /\b$Type\s+$Inline\b/ || - $line =~ /\b$Inline\s+$Storage\b/) { - ERROR("INLINE_LOCATION", - "inline keyword should sit between storage class and type\n" . $herecurr); - } - -# Check for __inline__ and __inline, prefer inline - if ($realfile !~ m@\binclude/uapi/@ && - $line =~ /\b(__inline__|__inline)\b/) { - if (WARN("INLINE", - "plain inline is preferred over $1\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\b(__inline__|__inline)\b/inline/; - - } - } - -# Check for __attribute__ packed, prefer __packed - if ($realfile !~ m@\binclude/uapi/@ && - $line =~ /\b__attribute__\s*\(\s*\(.*\bpacked\b/) { - WARN("PREFER_PACKED", - "__packed is preferred over __attribute__((packed))\n" . $herecurr); - } - -# Check for __attribute__ aligned, prefer __aligned - if ($realfile !~ m@\binclude/uapi/@ && - $line =~ /\b__attribute__\s*\(\s*\(.*aligned/) { - WARN("PREFER_ALIGNED", - "__aligned(size) is preferred over __attribute__((aligned(size)))\n" . $herecurr); - } - -# Check for __attribute__ format(printf, prefer __printf - if ($realfile !~ m@\binclude/uapi/@ && - $line =~ /\b__attribute__\s*\(\s*\(\s*format\s*\(\s*printf/) { - if (WARN("PREFER_PRINTF", - "__printf(string-index, first-to-check) is preferred over __attribute__((format(printf, string-index, first-to-check)))\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\b__attribute__\s*\(\s*\(\s*format\s*\(\s*printf\s*,\s*(.*)\)\s*\)\s*\)/"__printf(" . trim($1) . ")"/ex; - - } - } - -# Check for __attribute__ format(scanf, prefer __scanf - if ($realfile !~ m@\binclude/uapi/@ && - $line =~ /\b__attribute__\s*\(\s*\(\s*format\s*\(\s*scanf\b/) { - if (WARN("PREFER_SCANF", - "__scanf(string-index, first-to-check) is preferred over __attribute__((format(scanf, string-index, first-to-check)))\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\b__attribute__\s*\(\s*\(\s*format\s*\(\s*scanf\s*,\s*(.*)\)\s*\)\s*\)/"__scanf(" . trim($1) . ")"/ex; - } - } - -# Check for __attribute__ weak, or __weak declarations (may have link issues) - if ($^V && $^V ge 5.10.0 && - $line =~ /(?:$Declare|$DeclareMisordered)\s*$Ident\s*$balanced_parens\s*(?:$Attribute)?\s*;/ && - ($line =~ /\b__attribute__\s*\(\s*\(.*\bweak\b/ || - $line =~ /\b__weak\b/)) { - ERROR("WEAK_DECLARATION", - "Using weak declarations can have unintended link defects\n" . $herecurr); - } - -# check for c99 types like uint8_t - if ($line =~ /\b($Declare)\s*$Ident\s*[=;,\[]/) { - my $type = $1; - if ($type =~ /\b($typeC99Typedefs)\b/) { - $type = $1; - my $kernel_type = 'u'; - $kernel_type = 's' if ($type =~ /^_*[si]/); - $type =~ /(\d+)/; - $kernel_type .= $1.'_t'; - WARN("PREFER_KERNEL_TYPES", - "Prefer kernel type '$kernel_type' over '$type'\n" . $herecurr) - } - } - -# check for cast of C90 native int or longer types constants - if ($line =~ /(\(\s*$C90_int_types\s*\)\s*)($Constant)\b/) { - my $cast = $1; - my $const = $2; - if (WARN("TYPECAST_INT_CONSTANT", - "Unnecessary typecast of c90 int constant\n" . $herecurr) && - $fix) { - my $suffix = ""; - my $newconst = $const; - $newconst =~ s/${Int_type}$//; - $suffix .= 'U' if ($cast =~ /\bunsigned\b/); - if ($cast =~ /\blong\s+long\b/) { - $suffix .= 'LL'; - } elsif ($cast =~ /\blong\b/) { - $suffix .= 'L'; - } - $fixed[$fixlinenr] =~ s/\Q$cast\E$const\b/$newconst$suffix/; - } - } - -# check for sizeof(&) - if ($line =~ /\bsizeof\s*\(\s*\&/) { - WARN("SIZEOF_ADDRESS", - "sizeof(& should be avoided\n" . $herecurr); - } - -# check for sizeof without parenthesis - if ($line =~ /\bsizeof\s+((?:\*\s*|)$Lval|$Type(?:\s+$Lval|))/) { - if (WARN("SIZEOF_PARENTHESIS", - "sizeof $1 should be sizeof($1)\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\bsizeof\s+((?:\*\s*|)$Lval|$Type(?:\s+$Lval|))/"sizeof(" . trim($1) . ")"/ex; - } - } - -# check for struct spinlock declarations - if ($line =~ /^.\s*\bstruct\s+spinlock\s+\w+\s*;/) { - WARN("USE_SPINLOCK_T", - "struct spinlock should be spinlock_t\n" . $herecurr); - } - -# check for seq_printf uses that could be seq_puts - if ($sline =~ /\bseq_printf\s*\(.*"\s*\)\s*;\s*$/) { - my $fmt = get_quoted_string($line, $rawline); - $fmt =~ s/%%//g; - if ($fmt !~ /%/) { - if (WARN("PREFER_SEQ_PUTS", - "Prefer seq_puts to seq_printf\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\bseq_printf\b/seq_puts/; - } - } - } - - # check for vsprintf extension %p<foo> misuses - if ($^V && $^V ge 5.10.0 && - defined $stat && - $stat =~ /^\+(?![^\{]*\{\s*).*\b(\w+)\s*\(.*$String\s*,/s && - $1 !~ /^_*volatile_*$/) { - my $bad_extension = ""; - my $lc = $stat =~ tr@\n@@; - $lc = $lc + $linenr; - for (my $count = $linenr; $count <= $lc; $count++) { - my $fmt = get_quoted_string($lines[$count - 1], raw_line($count, 0)); - $fmt =~ s/%%//g; - if ($fmt =~ /(\%[\*\d\.]*p(?![\WFfSsBKRraEhMmIiUDdgVCbGNO]).)/) { - $bad_extension = $1; - last; - } - } - if ($bad_extension ne "") { - my $stat_real = raw_line($linenr, 0); - for (my $count = $linenr + 1; $count <= $lc; $count++) { - $stat_real = $stat_real . "\n" . raw_line($count, 0); - } - WARN("VSPRINTF_POINTER_EXTENSION", - "Invalid vsprintf pointer extension '$bad_extension'\n" . "$here\n$stat_real\n"); - } - } - -# Check for misused memsets - if ($^V && $^V ge 5.10.0 && - defined $stat && - $stat =~ /^\+(?:.*?)\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*$FuncArg\s*\)/) { - - my $ms_addr = $2; - my $ms_val = $7; - my $ms_size = $12; - - if ($ms_size =~ /^(0x|)0$/i) { - ERROR("MEMSET", - "memset to 0's uses 0 as the 2nd argument, not the 3rd\n" . "$here\n$stat\n"); - } elsif ($ms_size =~ /^(0x|)1$/i) { - WARN("MEMSET", - "single byte memset is suspicious. Swapped 2nd/3rd argument?\n" . "$here\n$stat\n"); - } - } - -# Check for memcpy(foo, bar, ETH_ALEN) that could be ether_addr_copy(foo, bar) -# if ($^V && $^V ge 5.10.0 && -# defined $stat && -# $stat =~ /^\+(?:.*?)\bmemcpy\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) { -# if (WARN("PREFER_ETHER_ADDR_COPY", -# "Prefer ether_addr_copy() over memcpy() if the Ethernet addresses are __aligned(2)\n" . "$here\n$stat\n") && -# $fix) { -# $fixed[$fixlinenr] =~ s/\bmemcpy\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/ether_addr_copy($2, $7)/; -# } -# } - -# Check for memcmp(foo, bar, ETH_ALEN) that could be ether_addr_equal*(foo, bar) -# if ($^V && $^V ge 5.10.0 && -# defined $stat && -# $stat =~ /^\+(?:.*?)\bmemcmp\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) { -# WARN("PREFER_ETHER_ADDR_EQUAL", -# "Prefer ether_addr_equal() or ether_addr_equal_unaligned() over memcmp()\n" . "$here\n$stat\n") -# } - -# check for memset(foo, 0x0, ETH_ALEN) that could be eth_zero_addr -# check for memset(foo, 0xFF, ETH_ALEN) that could be eth_broadcast_addr -# if ($^V && $^V ge 5.10.0 && -# defined $stat && -# $stat =~ /^\+(?:.*?)\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) { -# -# my $ms_val = $7; -# -# if ($ms_val =~ /^(?:0x|)0+$/i) { -# if (WARN("PREFER_ETH_ZERO_ADDR", -# "Prefer eth_zero_addr over memset()\n" . "$here\n$stat\n") && -# $fix) { -# $fixed[$fixlinenr] =~ s/\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*,\s*ETH_ALEN\s*\)/eth_zero_addr($2)/; -# } -# } elsif ($ms_val =~ /^(?:0xff|255)$/i) { -# if (WARN("PREFER_ETH_BROADCAST_ADDR", -# "Prefer eth_broadcast_addr() over memset()\n" . "$here\n$stat\n") && -# $fix) { -# $fixed[$fixlinenr] =~ s/\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*,\s*ETH_ALEN\s*\)/eth_broadcast_addr($2)/; -# } -# } -# } - -# typecasts on min/max could be min_t/max_t - if ($^V && $^V ge 5.10.0 && - defined $stat && - $stat =~ /^\+(?:.*?)\b(min|max)\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\)/) { - if (defined $2 || defined $7) { - my $call = $1; - my $cast1 = deparenthesize($2); - my $arg1 = $3; - my $cast2 = deparenthesize($7); - my $arg2 = $8; - my $cast; - - if ($cast1 ne "" && $cast2 ne "" && $cast1 ne $cast2) { - $cast = "$cast1 or $cast2"; - } elsif ($cast1 ne "") { - $cast = $cast1; - } else { - $cast = $cast2; - } - WARN("MINMAX", - "$call() should probably be ${call}_t($cast, $arg1, $arg2)\n" . "$here\n$stat\n"); - } - } - -# check usleep_range arguments - if ($^V && $^V ge 5.10.0 && - defined $stat && - $stat =~ /^\+(?:.*?)\busleep_range\s*\(\s*($FuncArg)\s*,\s*($FuncArg)\s*\)/) { - my $min = $1; - my $max = $7; - if ($min eq $max) { - WARN("USLEEP_RANGE", - "usleep_range should not use min == max args; see Documentation/timers/timers-howto.txt\n" . "$here\n$stat\n"); - } elsif ($min =~ /^\d+$/ && $max =~ /^\d+$/ && - $min > $max) { - WARN("USLEEP_RANGE", - "usleep_range args reversed, use min then max; see Documentation/timers/timers-howto.txt\n" . "$here\n$stat\n"); - } - } - -# check for naked sscanf - if ($^V && $^V ge 5.10.0 && - defined $stat && - $line =~ /\bsscanf\b/ && - ($stat !~ /$Ident\s*=\s*sscanf\s*$balanced_parens/ && - $stat !~ /\bsscanf\s*$balanced_parens\s*(?:$Compare)/ && - $stat !~ /(?:$Compare)\s*\bsscanf\s*$balanced_parens/)) { - my $lc = $stat =~ tr@\n@@; - $lc = $lc + $linenr; - my $stat_real = raw_line($linenr, 0); - for (my $count = $linenr + 1; $count <= $lc; $count++) { - $stat_real = $stat_real . "\n" . raw_line($count, 0); - } - WARN("NAKED_SSCANF", - "unchecked sscanf return value\n" . "$here\n$stat_real\n"); - } - -# check for simple sscanf that should be kstrto<foo> - if ($^V && $^V ge 5.10.0 && - defined $stat && - $line =~ /\bsscanf\b/) { - my $lc = $stat =~ tr@\n@@; - $lc = $lc + $linenr; - my $stat_real = raw_line($linenr, 0); - for (my $count = $linenr + 1; $count <= $lc; $count++) { - $stat_real = $stat_real . "\n" . raw_line($count, 0); - } - if ($stat_real =~ /\bsscanf\b\s*\(\s*$FuncArg\s*,\s*("[^"]+")/) { - my $format = $6; - my $count = $format =~ tr@%@%@; - if ($count == 1 && - $format =~ /^"\%(?i:ll[udxi]|[udxi]ll|ll|[hl]h?[udxi]|[udxi][hl]h?|[hl]h?|[udxi])"$/) { - WARN("SSCANF_TO_KSTRTO", - "Prefer kstrto<type> to single variable sscanf\n" . "$here\n$stat_real\n"); - } - } - } - -# check for new externs in .h files. - if ($realfile =~ /\.h$/ && - $line =~ /^\+\s*(extern\s+)$Type\s*$Ident\s*\(/s) { - if (CHK("AVOID_EXTERNS", - "extern prototypes should be avoided in .h files\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/(.*)\bextern\b\s*(.*)/$1$2/; - } - } - -# check for new externs in .c files. - if ($realfile =~ /\.c$/ && defined $stat && - $stat =~ /^.\s*(?:extern\s+)?$Type\s+($Ident)(\s*)\(/s) - { - my $function_name = $1; - my $paren_space = $2; - - my $s = $stat; - if (defined $cond) { - substr($s, 0, length($cond), ''); - } - if ($s =~ /^\s*;/ && - $function_name ne 'uninitialized_var') - { - WARN("AVOID_EXTERNS", - "externs should be avoided in .c files\n" . $herecurr); - } - - if ($paren_space =~ /\n/) { - WARN("FUNCTION_ARGUMENTS", - "arguments for function declarations should follow identifier\n" . $herecurr); - } - - } elsif ($realfile =~ /\.c$/ && defined $stat && - $stat =~ /^.\s*extern\s+/) - { - WARN("AVOID_EXTERNS", - "externs should be avoided in .c files\n" . $herecurr); - } - -# check for function declarations that have arguments without identifier names - if (defined $stat && - $stat =~ /^.\s*(?:extern\s+)?$Type\s*$Ident\s*\(\s*([^{]+)\s*\)\s*;/s && - $1 ne "void") { - my $args = trim($1); - while ($args =~ m/\s*($Type\s*(?:$Ident|\(\s*\*\s*$Ident?\s*\)\s*$balanced_parens)?)/g) { - my $arg = trim($1); - if ($arg =~ /^$Type$/ && $arg !~ /enum\s+$Ident$/) { - WARN("FUNCTION_ARGUMENTS", - "function definition argument '$arg' should also have an identifier name\n" . $herecurr); - } - } - } - -# check for function definitions - if ($^V && $^V ge 5.10.0 && - defined $stat && - $stat =~ /^.\s*(?:$Storage\s+)?$Type\s*($Ident)\s*$balanced_parens\s*{/s) { - $context_function = $1; - -# check for multiline function definition with misplaced open brace - my $ok = 0; - my $cnt = statement_rawlines($stat); - my $herectx = $here . "\n"; - for (my $n = 0; $n < $cnt; $n++) { - my $rl = raw_line($linenr, $n); - $herectx .= $rl . "\n"; - $ok = 1 if ($rl =~ /^[ \+]\{/); - $ok = 1 if ($rl =~ /\{/ && $n == 0); - last if $rl =~ /^[ \+].*\{/; - } - if (!$ok) { - ERROR("OPEN_BRACE", - "open brace '{' following function definitions go on the next line\n" . $herectx); - } - } - -# checks for new __setup's - if ($rawline =~ /\b__setup\("([^"]*)"/) { - my $name = $1; - - if (!grep(/$name/, @setup_docs)) { - CHK("UNDOCUMENTED_SETUP", - "__setup appears un-documented -- check Documentation/admin-guide/kernel-parameters.rst\n" . $herecurr); - } - } - -# check for pointless casting of kmalloc return - if ($line =~ /\*\s*\)\s*[kv][czm]alloc(_node){0,1}\b/) { - WARN("UNNECESSARY_CASTS", - "unnecessary cast may hide bugs, see http://c-faq.com/malloc/mallocnocast.html\n" . $herecurr); - } - -# alloc style -# p = alloc(sizeof(struct foo), ...) should be p = alloc(sizeof(*p), ...) - if ($^V && $^V ge 5.10.0 && - $line =~ /\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*([kv][mz]alloc(?:_node)?)\s*\(\s*(sizeof\s*\(\s*struct\s+$Lval\s*\))/) { - CHK("ALLOC_SIZEOF_STRUCT", - "Prefer $3(sizeof(*$1)...) over $3($4...)\n" . $herecurr); - } - -# check for k[mz]alloc with multiplies that could be kmalloc_array/kcalloc - if ($^V && $^V ge 5.10.0 && - defined $stat && - $stat =~ /^\+\s*($Lval)\s*\=\s*(?:$balanced_parens)?\s*(k[mz]alloc)\s*\(\s*($FuncArg)\s*\*\s*($FuncArg)\s*,/) { - my $oldfunc = $3; - my $a1 = $4; - my $a2 = $10; - my $newfunc = "kmalloc_array"; - $newfunc = "kcalloc" if ($oldfunc eq "kzalloc"); - my $r1 = $a1; - my $r2 = $a2; - if ($a1 =~ /^sizeof\s*\S/) { - $r1 = $a2; - $r2 = $a1; - } - if ($r1 !~ /^sizeof\b/ && $r2 =~ /^sizeof\s*\S/ && - !($r1 =~ /^$Constant$/ || $r1 =~ /^[A-Z_][A-Z0-9_]*$/)) { - my $ctx = ''; - my $herectx = $here . "\n"; - my $cnt = statement_rawlines($stat); - for (my $n = 0; $n < $cnt; $n++) { - $herectx .= raw_line($linenr, $n) . "\n"; - } - if (WARN("ALLOC_WITH_MULTIPLY", - "Prefer $newfunc over $oldfunc with multiply\n" . $herectx) && - $cnt == 1 && - $fix) { - $fixed[$fixlinenr] =~ s/\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*(k[mz]alloc)\s*\(\s*($FuncArg)\s*\*\s*($FuncArg)/$1 . ' = ' . "$newfunc(" . trim($r1) . ', ' . trim($r2)/e; - } - } - } - -# check for krealloc arg reuse - if ($^V && $^V ge 5.10.0 && - $line =~ /\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*krealloc\s*\(\s*\1\s*,/) { - WARN("KREALLOC_ARG_REUSE", - "Reusing the krealloc arg is almost always a bug\n" . $herecurr); - } - -# check for alloc argument mismatch - if ($line =~ /\b(kcalloc|kmalloc_array)\s*\(\s*sizeof\b/) { - WARN("ALLOC_ARRAY_ARGS", - "$1 uses number as first arg, sizeof is generally wrong\n" . $herecurr); - } - -# check for multiple semicolons - if ($line =~ /;\s*;\s*$/) { - if (WARN("ONE_SEMICOLON", - "Statements terminations use 1 semicolon\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/(\s*;\s*){2,}$/;/g; - } - } - -# check for #defines like: 1 << <digit> that could be BIT(digit), it is not exported to uapi - if ($realfile !~ m@^include/uapi/@ && - $line =~ /#\s*define\s+\w+\s+\(?\s*1\s*([ulUL]*)\s*\<\<\s*(?:\d+|$Ident)\s*\)?/) { - my $ull = ""; - $ull = "_ULL" if (defined($1) && $1 =~ /ll/i); - if (CHK("BIT_MACRO", - "Prefer using the BIT$ull macro\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\(?\s*1\s*[ulUL]*\s*<<\s*(\d+|$Ident)\s*\)?/BIT${ull}($1)/; - } - } - -# check for #if defined CONFIG_<FOO> || defined CONFIG_<FOO>_MODULE - if ($line =~ /^\+\s*#\s*if\s+defined(?:\s*\(?\s*|\s+)(CONFIG_[A-Z_]+)\s*\)?\s*\|\|\s*defined(?:\s*\(?\s*|\s+)\1_MODULE\s*\)?\s*$/) { - my $config = $1; - if (WARN("PREFER_IS_ENABLED", - "Prefer IS_ENABLED(<FOO>) to CONFIG_<FOO> || CONFIG_<FOO>_MODULE\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] = "\+#if IS_ENABLED($config)"; - } - } - -# check for case / default statements not preceded by break/fallthrough/switch - if ($line =~ /^.\s*(?:case\s+(?:$Ident|$Constant)\s*|default):/) { - my $has_break = 0; - my $has_statement = 0; - my $count = 0; - my $prevline = $linenr; - while ($prevline > 1 && ($file || $count < 3) && !$has_break) { - $prevline--; - my $rline = $rawlines[$prevline - 1]; - my $fline = $lines[$prevline - 1]; - last if ($fline =~ /^\@\@/); - next if ($fline =~ /^\-/); - next if ($fline =~ /^.(?:\s*(?:case\s+(?:$Ident|$Constant)[\s$;]*|default):[\s$;]*)*$/); - $has_break = 1 if ($rline =~ /fall[\s_-]*(through|thru)/i); - next if ($fline =~ /^.[\s$;]*$/); - $has_statement = 1; - $count++; - $has_break = 1 if ($fline =~ /\bswitch\b|\b(?:break\s*;[\s$;]*$|return\b|goto\b|continue\b)/); - } - if (!$has_break && $has_statement) { - WARN("MISSING_BREAK", - "Possible switch case/default not preceded by break or fallthrough comment\n" . $herecurr); - } - } - -# check for switch/default statements without a break; - if ($^V && $^V ge 5.10.0 && - defined $stat && - $stat =~ /^\+[$;\s]*(?:case[$;\s]+\w+[$;\s]*:[$;\s]*|)*[$;\s]*\bdefault[$;\s]*:[$;\s]*;/g) { - my $ctx = ''; - my $herectx = $here . "\n"; - my $cnt = statement_rawlines($stat); - for (my $n = 0; $n < $cnt; $n++) { - $herectx .= raw_line($linenr, $n) . "\n"; - } - WARN("DEFAULT_NO_BREAK", - "switch default: should use break\n" . $herectx); - } - -# check for gcc specific __FUNCTION__ - if ($line =~ /\b__FUNCTION__\b/) { - if (WARN("USE_FUNC", - "__func__ should be used instead of gcc specific __FUNCTION__\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\b__FUNCTION__\b/__func__/g; - } - } - -# check for uses of __DATE__, __TIME__, __TIMESTAMP__ - while ($line =~ /\b(__(?:DATE|TIME|TIMESTAMP)__)\b/g) { - ERROR("DATE_TIME", - "Use of the '$1' macro makes the build non-deterministic\n" . $herecurr); - } - -# check for use of yield() - if ($line =~ /\byield\s*\(\s*\)/) { - WARN("YIELD", - "Using yield() is generally wrong. See yield() kernel-doc (sched/core.c)\n" . $herecurr); - } - -# check for comparisons against true and false - if ($line =~ /\+\s*(.*?)\b(true|false|$Lval)\s*(==|\!=)\s*(true|false|$Lval)\b(.*)$/i) { - my $lead = $1; - my $arg = $2; - my $test = $3; - my $otype = $4; - my $trail = $5; - my $op = "!"; - - ($arg, $otype) = ($otype, $arg) if ($arg =~ /^(?:true|false)$/i); - - my $type = lc($otype); - if ($type =~ /^(?:true|false)$/) { - if (("$test" eq "==" && "$type" eq "true") || - ("$test" eq "!=" && "$type" eq "false")) { - $op = ""; - } - - CHK("BOOL_COMPARISON", - "Using comparison to $otype is error prone\n" . $herecurr); - -## maybe suggesting a correct construct would better -## "Using comparison to $otype is error prone. Perhaps use '${lead}${op}${arg}${trail}'\n" . $herecurr); - - } - } - -# check for semaphores initialized locked - if ($line =~ /^.\s*sema_init.+,\W?0\W?\)/) { - WARN("CONSIDER_COMPLETION", - "consider using a completion\n" . $herecurr); - } - -# recommend kstrto* over simple_strto* and strict_strto* - if ($line =~ /\b((simple|strict)_(strto(l|ll|ul|ull)))\s*\(/) { - WARN("CONSIDER_KSTRTO", - "$1 is obsolete, use k$3 instead\n" . $herecurr); - } - -# check for __initcall(), use device_initcall() explicitly or more appropriate function please - if ($line =~ /^.\s*__initcall\s*\(/) { - WARN("USE_DEVICE_INITCALL", - "please use device_initcall() or more appropriate function instead of __initcall() (see include/linux/init.h)\n" . $herecurr); - } - -# check for various structs that are normally const (ops, kgdb, device_tree) -# and avoid what seem like struct definitions 'struct foo {' - if ($line !~ /\bconst\b/ && - $line =~ /\bstruct\s+($const_structs)\b(?!\s*\{)/) { - WARN("CONST_STRUCT", - "struct $1 should normally be const\n" . $herecurr); - } - -# use of NR_CPUS is usually wrong -# ignore definitions of NR_CPUS and usage to define arrays as likely right - if ($line =~ /\bNR_CPUS\b/ && - $line !~ /^.\s*\s*#\s*if\b.*\bNR_CPUS\b/ && - $line !~ /^.\s*\s*#\s*define\b.*\bNR_CPUS\b/ && - $line !~ /^.\s*$Declare\s.*\[[^\]]*NR_CPUS[^\]]*\]/ && - $line !~ /\[[^\]]*\.\.\.[^\]]*NR_CPUS[^\]]*\]/ && - $line !~ /\[[^\]]*NR_CPUS[^\]]*\.\.\.[^\]]*\]/) - { - WARN("NR_CPUS", - "usage of NR_CPUS is often wrong - consider using cpu_possible(), num_possible_cpus(), for_each_possible_cpu(), etc\n" . $herecurr); - } - -# Use of __ARCH_HAS_<FOO> or ARCH_HAVE_<BAR> is wrong. - if ($line =~ /\+\s*#\s*define\s+((?:__)?ARCH_(?:HAS|HAVE)\w*)\b/) { - ERROR("DEFINE_ARCH_HAS", - "#define of '$1' is wrong - use Kconfig variables or standard guards instead\n" . $herecurr); - } - -# likely/unlikely comparisons similar to "(likely(foo) > 0)" - if ($^V && $^V ge 5.10.0 && - $line =~ /\b((?:un)?likely)\s*\(\s*$FuncArg\s*\)\s*$Compare/) { - WARN("LIKELY_MISUSE", - "Using $1 should generally have parentheses around the comparison\n" . $herecurr); - } - -# whine mightly about in_atomic - if ($line =~ /\bin_atomic\s*\(/) { - if ($realfile =~ m@^drivers/@) { - ERROR("IN_ATOMIC", - "do not use in_atomic in drivers\n" . $herecurr); - } elsif ($realfile !~ m@^kernel/@) { - WARN("IN_ATOMIC", - "use of in_atomic() is incorrect outside core kernel code\n" . $herecurr); - } - } - -# whine about ACCESS_ONCE - if ($^V && $^V ge 5.10.0 && - $line =~ /\bACCESS_ONCE\s*$balanced_parens\s*(=(?!=))?\s*($FuncArg)?/) { - my $par = $1; - my $eq = $2; - my $fun = $3; - $par =~ s/^\(\s*(.*)\s*\)$/$1/; - if (defined($eq)) { - if (WARN("PREFER_WRITE_ONCE", - "Prefer WRITE_ONCE(<FOO>, <BAR>) over ACCESS_ONCE(<FOO>) = <BAR>\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\bACCESS_ONCE\s*\(\s*\Q$par\E\s*\)\s*$eq\s*\Q$fun\E/WRITE_ONCE($par, $fun)/; - } - } else { - if (WARN("PREFER_READ_ONCE", - "Prefer READ_ONCE(<FOO>) over ACCESS_ONCE(<FOO>)\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/\bACCESS_ONCE\s*\(\s*\Q$par\E\s*\)/READ_ONCE($par)/; - } - } - } - -# check for mutex_trylock_recursive usage - if ($line =~ /mutex_trylock_recursive/) { - ERROR("LOCKING", - "recursive locking is bad, do not use this ever.\n" . $herecurr); - } - -# check for lockdep_set_novalidate_class - if ($line =~ /^.\s*lockdep_set_novalidate_class\s*\(/ || - $line =~ /__lockdep_no_validate__\s*\)/ ) { - if ($realfile !~ m@^kernel/lockdep@ && - $realfile !~ m@^include/linux/lockdep@ && - $realfile !~ m@^drivers/base/core@) { - ERROR("LOCKDEP", - "lockdep_no_validate class is reserved for device->mutex.\n" . $herecurr); - } - } - - if ($line =~ /debugfs_create_\w+.*\b$mode_perms_world_writable\b/ || - $line =~ /DEVICE_ATTR.*\b$mode_perms_world_writable\b/) { - WARN("EXPORTED_WORLD_WRITABLE", - "Exporting world writable files is usually an error. Consider more restrictive permissions.\n" . $herecurr); - } - -# Mode permission misuses where it seems decimal should be octal -# This uses a shortcut match to avoid unnecessary uses of a slow foreach loop - if ($^V && $^V ge 5.10.0 && - defined $stat && - $line =~ /$mode_perms_search/) { - foreach my $entry (@mode_permission_funcs) { - my $func = $entry->[0]; - my $arg_pos = $entry->[1]; - - my $lc = $stat =~ tr@\n@@; - $lc = $lc + $linenr; - my $stat_real = raw_line($linenr, 0); - for (my $count = $linenr + 1; $count <= $lc; $count++) { - $stat_real = $stat_real . "\n" . raw_line($count, 0); - } - - my $skip_args = ""; - if ($arg_pos > 1) { - $arg_pos--; - $skip_args = "(?:\\s*$FuncArg\\s*,\\s*){$arg_pos,$arg_pos}"; - } - my $test = "\\b$func\\s*\\(${skip_args}($FuncArg(?:\\|\\s*$FuncArg)*)\\s*[,\\)]"; - if ($stat =~ /$test/) { - my $val = $1; - $val = $6 if ($skip_args ne ""); - if (($val =~ /^$Int$/ && $val !~ /^$Octal$/) || - ($val =~ /^$Octal$/ && length($val) ne 4)) { - ERROR("NON_OCTAL_PERMISSIONS", - "Use 4 digit octal (0777) not decimal permissions\n" . "$here\n" . $stat_real); - } - if ($val =~ /^$Octal$/ && (oct($val) & 02)) { - ERROR("EXPORTED_WORLD_WRITABLE", - "Exporting writable files is usually an error. Consider more restrictive permissions.\n" . "$here\n" . $stat_real); - } - } - } - } - -# check for uses of S_<PERMS> that could be octal for readability - if ($line =~ /\b$mode_perms_string_search\b/) { - my $val = ""; - my $oval = ""; - my $to = 0; - my $curpos = 0; - my $lastpos = 0; - while ($line =~ /\b(($mode_perms_string_search)\b(?:\s*\|\s*)?\s*)/g) { - $curpos = pos($line); - my $match = $2; - my $omatch = $1; - last if ($lastpos > 0 && ($curpos - length($omatch) != $lastpos)); - $lastpos = $curpos; - $to |= $mode_permission_string_types{$match}; - $val .= '\s*\|\s*' if ($val ne ""); - $val .= $match; - $oval .= $omatch; - } - $oval =~ s/^\s*\|\s*//; - $oval =~ s/\s*\|\s*$//; - my $octal = sprintf("%04o", $to); - if (WARN("SYMBOLIC_PERMS", - "Symbolic permissions '$oval' are not preferred. Consider using octal permissions '$octal'.\n" . $herecurr) && - $fix) { - $fixed[$fixlinenr] =~ s/$val/$octal/; - } - } - -# validate content of MODULE_LICENSE against list from include/linux/module.h - if ($line =~ /\bMODULE_LICENSE\s*\(\s*($String)\s*\)/) { - my $extracted_string = get_quoted_string($line, $rawline); - my $valid_licenses = qr{ - GPL| - GPL\ v2| - GPL\ and\ additional\ rights| - Dual\ BSD/GPL| - Dual\ MIT/GPL| - Dual\ MPL/GPL| - Proprietary - }x; - if ($extracted_string !~ /^"(?:$valid_licenses)"$/x) { - WARN("MODULE_LICENSE", - "unknown module license " . $extracted_string . "\n" . $herecurr); - } - } - } - - # If we have no input at all, then there is nothing to report on - # so just keep quiet. - if ($#rawlines == -1) { - exit(0); - } - - # In mailback mode only produce a report in the negative, for - # things that appear to be patches. - if ($mailback && ($clean == 1 || !$is_patch)) { - exit(0); - } - - # This is not a patch, and we are are in 'no-patch' mode so - # just keep quiet. - if (!$chk_patch && !$is_patch) { - exit(0); - } - - if (!$is_patch && $file !~ /cover-letter\.patch$/) { - ERROR("NOT_UNIFIED_DIFF", - "Does not appear to be a unified-diff format patch\n"); - } - if ($is_patch && $has_commit_log && $chk_signoff && $signoff == 0) { - ERROR("MISSING_SIGN_OFF", - "Missing Signed-off-by: line(s)\n"); - } - - print report_dump(); - if ($summary && !($clean == 1 && $quiet == 1)) { - print "$filename " if ($summary_file); - print "total: $cnt_error errors, $cnt_warn warnings, " . - (($check)? "$cnt_chk checks, " : "") . - "$cnt_lines lines checked\n"; - } - - if ($quiet == 0) { - # If there were any defects found and not already fixing them - if (!$clean and !$fix) { - print << "EOM" - -NOTE: For some of the reported defects, checkpatch may be able to - mechanically convert to the typical style using --fix or --fix-inplace. -EOM - } - # If there were whitespace errors which cleanpatch can fix - # then suggest that. - if ($rpt_cleaners) { - $rpt_cleaners = 0; - print << "EOM" - -NOTE: Whitespace errors detected. - You may wish to use scripts/cleanpatch or scripts/cleanfile -EOM - } - } - - if ($clean == 0 && $fix && - ("@rawlines" ne "@fixed" || - $#fixed_inserted >= 0 || $#fixed_deleted >= 0)) { - my $newfile = $filename; - $newfile .= ".EXPERIMENTAL-checkpatch-fixes" if (!$fix_inplace); - my $linecount = 0; - my $f; - - @fixed = fix_inserted_deleted_lines(\@fixed, \@fixed_inserted, \@fixed_deleted); - - open($f, '>', $newfile) - or die "$P: Can't open $newfile for write\n"; - foreach my $fixed_line (@fixed) { - $linecount++; - if ($file) { - if ($linecount > 3) { - $fixed_line =~ s/^\+//; - print $f $fixed_line . "\n"; - } - } else { - print $f $fixed_line . "\n"; - } - } - close($f); - - if (!$quiet) { - print << "EOM"; - -Wrote EXPERIMENTAL --fix correction(s) to '$newfile' - -Do _NOT_ trust the results written to this file. -Do _NOT_ submit these changes without inspecting them for correctness. - -This EXPERIMENTAL file is simply a convenience to help rewrite patches. -No warranties, expressed or implied... -EOM - } - } - - if ($quiet == 0) { - print "\n"; - if ($clean == 1) { - print "$vname has no obvious style problems and is ready for submission.\n"; - } else { - print "$vname has style problems, please review.\n"; - } - } - return $clean; -} +#!/usr/bin/env perl +# (c) 2001, Dave Jones. (the file handling bit) +# (c) 2005, Joel Schopp <jschopp@austin.ibm.com> (the ugly bit) +# (c) 2007,2008, Andy Whitcroft <apw@uk.ibm.com> (new conditions, test suite) +# (c) 2008-2010 Andy Whitcroft <apw@canonical.com> +# Licensed under the terms of the GNU GPL License version 2 + +use strict; +use warnings; +use POSIX; +use File::Basename; +use Cwd 'abs_path'; +use Term::ANSIColor qw(:constants); + +my $P = $0; +my $D = dirname(abs_path($P)); + +my $V = '0.32'; + +use Getopt::Long qw(:config no_auto_abbrev); + +my $quiet = 0; +my $tree = 1; +my $chk_signoff = 1; +my $chk_patch = 1; +my $tst_only; +my $emacs = 0; +my $terse = 0; +my $showfile = 0; +my $file = 0; +my $git = 0; +my %git_commits = (); +my $check = 0; +my $check_orig = 0; +my $summary = 1; +my $mailback = 0; +my $summary_file = 0; +my $show_types = 0; +my $list_types = 0; +my $fix = 0; +my $fix_inplace = 0; +my $root; +my %debug; +my %camelcase = (); +my %use_type = (); +my @use = (); +my %ignore_type = (); +my @ignore = (); +my @exclude = (); +my $help = 0; +my $configuration_file = ".checkpatch.conf"; +my $max_line_length = 80; +my $ignore_perl_version = 0; +my $minimum_perl_version = 5.10.0; +my $min_conf_desc_length = 4; +my $spelling_file = "$D/spelling.txt"; +my $codespell = 0; +my $codespellfile = "/usr/share/codespell/dictionary.txt"; +my $conststructsfile = "$D/const_structs.checkpatch"; +my $typedefsfile = ""; +my $color = "auto"; +my $allow_c99_comments = 0; + +sub help { + my ($exitcode) = @_; + + print << "EOM"; +Usage: $P [OPTION]... [FILE]... +Version: $V + +Options: + -q, --quiet quiet + --no-tree run without a kernel tree + --no-signoff do not check for 'Signed-off-by' line + --patch treat FILE as patchfile (default) + --emacs emacs compile window format + --terse one line per report + --showfile emit diffed file position, not input file position + -g, --git treat FILE as a single commit or git revision range + single git commit with: + <rev> + <rev>^ + <rev>~n + multiple git commits with: + <rev1>..<rev2> + <rev1>...<rev2> + <rev>-<count> + git merges are ignored + -f, --file treat FILE as regular source file + --subjective, --strict enable more subjective tests + --list-types list the possible message types + --types TYPE(,TYPE2...) show only these comma separated message types + --ignore TYPE(,TYPE2...) ignore various comma separated message types + --exclude DIR(,DIR22...) exclude directories + --show-types show the specific message type in the output + --max-line-length=n set the maximum line length, if exceeded, warn + --min-conf-desc-length=n set the min description length, if shorter, warn + --root=PATH PATH to the kernel tree root + --no-summary suppress the per-file summary + --mailback only produce a report in case of warnings/errors + --summary-file include the filename in summary + --debug KEY=[0|1] turn on/off debugging of KEY, where KEY is one of + 'values', 'possible', 'type', and 'attr' (default + is all off) + --test-only=WORD report only warnings/errors containing WORD + literally + --fix EXPERIMENTAL - may create horrible results + If correctable single-line errors exist, create + "<inputfile>.EXPERIMENTAL-checkpatch-fixes" + with potential errors corrected to the preferred + checkpatch style + --fix-inplace EXPERIMENTAL - may create horrible results + Is the same as --fix, but overwrites the input + file. It's your fault if there's no backup or git + --ignore-perl-version override checking of perl version. expect + runtime errors. + --codespell Use the codespell dictionary for spelling/typos + (default:/usr/share/codespell/dictionary.txt) + --codespellfile Use this codespell dictionary + --typedefsfile Read additional types from this file + --color[=WHEN] Use colors 'always', 'never', or only when output + is a terminal ('auto'). Default is 'auto'. + -h, --help, --version display this help and exit + +When FILE is - read standard input. +EOM + + exit($exitcode); +} + +sub uniq { + my %seen; + return grep { !$seen{$_}++ } @_; +} + +sub list_types { + my ($exitcode) = @_; + + my $count = 0; + + local $/ = undef; + + open(my $script, '<', abs_path($P)) or + die "$P: Can't read '$P' $!\n"; + + my $text = <$script>; + close($script); + + my @types = (); + # Also catch when type or level is passed through a variable + for ($text =~ /(?:(?:\bCHK|\bWARN|\bERROR|&\{\$msg_level})\s*\(|\$msg_type\s*=)\s*"([^"]+)"/g) { + push (@types, $_); + } + @types = sort(uniq(@types)); + print("#\tMessage type\n\n"); + foreach my $type (@types) { + print(++$count . "\t" . $type . "\n"); + } + + exit($exitcode); +} + +my $conf = which_conf($configuration_file); +if (-f $conf) { + my @conf_args; + open(my $conffile, '<', "$conf") + or warn "$P: Can't find a readable $configuration_file file $!\n"; + + while (<$conffile>) { + my $line = $_; + + $line =~ s/\s*\n?$//g; + $line =~ s/^\s*//g; + $line =~ s/\s+/ /g; + + next if ($line =~ m/^\s*#/); + next if ($line =~ m/^\s*$/); + + my @words = split(" ", $line); + foreach my $word (@words) { + last if ($word =~ m/^#/); + push (@conf_args, $word); + } + } + close($conffile); + unshift(@ARGV, @conf_args) if @conf_args; +} + +# Perl's Getopt::Long allows options to take optional arguments after a space. +# Prevent --color by itself from consuming other arguments +foreach (@ARGV) { + if ($_ eq "--color" || $_ eq "-color") { + $_ = "--color=$color"; + } +} + +GetOptions( + 'q|quiet+' => \$quiet, + 'tree!' => \$tree, + 'signoff!' => \$chk_signoff, + 'patch!' => \$chk_patch, + 'emacs!' => \$emacs, + 'terse!' => \$terse, + 'showfile!' => \$showfile, + 'f|file!' => \$file, + 'g|git!' => \$git, + 'subjective!' => \$check, + 'strict!' => \$check, + 'ignore=s' => \@ignore, + 'exclude=s' => \@exclude, + 'types=s' => \@use, + 'show-types!' => \$show_types, + 'list-types!' => \$list_types, + 'max-line-length=i' => \$max_line_length, + 'min-conf-desc-length=i' => \$min_conf_desc_length, + 'root=s' => \$root, + 'summary!' => \$summary, + 'mailback!' => \$mailback, + 'summary-file!' => \$summary_file, + 'fix!' => \$fix, + 'fix-inplace!' => \$fix_inplace, + 'ignore-perl-version!' => \$ignore_perl_version, + 'debug=s' => \%debug, + 'test-only=s' => \$tst_only, + 'codespell!' => \$codespell, + 'codespellfile=s' => \$codespellfile, + 'typedefsfile=s' => \$typedefsfile, + 'color=s' => \$color, + 'no-color' => \$color, #keep old behaviors of -nocolor + 'nocolor' => \$color, #keep old behaviors of -nocolor + 'h|help' => \$help, + 'version' => \$help +) or help(1); + +help(0) if ($help); + +list_types(0) if ($list_types); + +$fix = 1 if ($fix_inplace); +$check_orig = $check; + +my $exit = 0; + +if ($^V && $^V lt $minimum_perl_version) { + printf "$P: requires at least perl version %vd\n", $minimum_perl_version; + if (!$ignore_perl_version) { + exit(1); + } +} + +#if no filenames are given, push '-' to read patch from stdin +if ($#ARGV < 0) { + push(@ARGV, '-'); +} + +if ($color =~ /^[01]$/) { + $color = !$color; +} elsif ($color =~ /^always$/i) { + $color = 1; +} elsif ($color =~ /^never$/i) { + $color = 0; +} elsif ($color =~ /^auto$/i) { + $color = (-t STDOUT); +} else { + die "Invalid color mode: $color\n"; +} + +sub hash_save_array_words { + my ($hashRef, $arrayRef) = @_; + + my @array = split(/,/, join(',', @$arrayRef)); + foreach my $word (@array) { + $word =~ s/\s*\n?$//g; + $word =~ s/^\s*//g; + $word =~ s/\s+/ /g; + $word =~ tr/[a-z]/[A-Z]/; + + next if ($word =~ m/^\s*#/); + next if ($word =~ m/^\s*$/); + + $hashRef->{$word}++; + } +} + +sub hash_show_words { + my ($hashRef, $prefix) = @_; + + if (keys %$hashRef) { + print "\nNOTE: $prefix message types:"; + foreach my $word (sort keys %$hashRef) { + print " $word"; + } + print "\n"; + } +} + +hash_save_array_words(\%ignore_type, \@ignore); +hash_save_array_words(\%use_type, \@use); + +my $dbg_values = 0; +my $dbg_possible = 0; +my $dbg_type = 0; +my $dbg_attr = 0; +for my $key (keys %debug) { + ## no critic + eval "\${dbg_$key} = '$debug{$key}';"; + die "$@" if ($@); +} + +my $rpt_cleaners = 0; + +if ($terse) { + $emacs = 1; + $quiet++; +} + +if ($tree) { + if (defined $root) { + if (!top_of_kernel_tree($root)) { + die "$P: $root: --root does not point at a valid tree\n"; + } + } else { + if (top_of_kernel_tree('.')) { + $root = '.'; + } elsif ($0 =~ m@(.*)/scripts/[^/]*$@ && + top_of_kernel_tree($1)) { + $root = $1; + } + } + + if (!defined $root) { + print "Must be run from the top-level dir. of a kernel tree\n"; + exit(2); + } +} + +my $emitted_corrupt = 0; + +our $Ident = qr{ + [A-Za-z_][A-Za-z\d_]* + (?:\s*\#\#\s*[A-Za-z_][A-Za-z\d_]*)* + }x; +our $Storage = qr{extern|static|asmlinkage}; +our $Sparse = qr{ + __user| + __force| + __iomem| + __must_check| + __init_refok| + __kprobes| + __ref| + __rcu| + __private + }x; +our $InitAttributePrefix = qr{__(?:mem|cpu|dev|net_|)}; +our $InitAttributeData = qr{$InitAttributePrefix(?:initdata\b)}; +our $InitAttributeConst = qr{$InitAttributePrefix(?:initconst\b)}; +our $InitAttributeInit = qr{$InitAttributePrefix(?:init\b)}; +our $InitAttribute = qr{$InitAttributeData|$InitAttributeConst|$InitAttributeInit}; + +# Notes to $Attribute: +# We need \b after 'init' otherwise 'initconst' will cause a false positive in a check +our $Attribute = qr{ + const| + __percpu| + __nocast| + __safe| + __bitwise| + __packed__| + __packed2__| + __naked| + __maybe_unused| + __always_unused| + __noreturn| + __used| + __cold| + __pure| + __noclone| + __deprecated| + __read_mostly| + __kprobes| + $InitAttribute| + ____cacheline_aligned| + ____cacheline_aligned_in_smp| + ____cacheline_internodealigned_in_smp| + __weak| + __syscall| + __syscall_inline + }x; +our $Modifier; +our $Inline = qr{inline|__always_inline|noinline|__inline|__inline__}; +our $Member = qr{->$Ident|\.$Ident|\[[^]]*\]}; +our $Lval = qr{$Ident(?:$Member)*}; + +our $Int_type = qr{(?i)llu|ull|ll|lu|ul|l|u}; +our $Binary = qr{(?i)0b[01]+$Int_type?}; +our $Hex = qr{(?i)0x[0-9a-f]+$Int_type?}; +our $Int = qr{[0-9]+$Int_type?}; +our $Octal = qr{0[0-7]+$Int_type?}; +our $String = qr{"[X\t]*"}; +our $Float_hex = qr{(?i)0x[0-9a-f]+p-?[0-9]+[fl]?}; +our $Float_dec = qr{(?i)(?:[0-9]+\.[0-9]*|[0-9]*\.[0-9]+)(?:e-?[0-9]+)?[fl]?}; +our $Float_int = qr{(?i)[0-9]+e-?[0-9]+[fl]?}; +our $Float = qr{$Float_hex|$Float_dec|$Float_int}; +our $Constant = qr{$Float|$Binary|$Octal|$Hex|$Int}; +our $Assignment = qr{\*\=|/=|%=|\+=|-=|<<=|>>=|&=|\^=|\|=|=}; +our $Compare = qr{<=|>=|==|!=|<|(?<!-)>}; +our $Arithmetic = qr{\+|-|\*|\/|%}; +our $Operators = qr{ + <=|>=|==|!=| + =>|->|<<|>>|<|>|!|~| + &&|\|\||,|\^|\+\+|--|&|\||$Arithmetic + }x; + +our $c90_Keywords = qr{do|for|while|if|else|return|goto|continue|switch|default|case|break}x; + +our $BasicType; +our $NonptrType; +our $NonptrTypeMisordered; +our $NonptrTypeWithAttr; +our $Type; +our $TypeMisordered; +our $Declare; +our $DeclareMisordered; + +our $NON_ASCII_UTF8 = qr{ + [\xC2-\xDF][\x80-\xBF] # non-overlong 2-byte + | \xE0[\xA0-\xBF][\x80-\xBF] # excluding overlongs + | [\xE1-\xEC\xEE\xEF][\x80-\xBF]{2} # straight 3-byte + | \xED[\x80-\x9F][\x80-\xBF] # excluding surrogates + | \xF0[\x90-\xBF][\x80-\xBF]{2} # planes 1-3 + | [\xF1-\xF3][\x80-\xBF]{3} # planes 4-15 + | \xF4[\x80-\x8F][\x80-\xBF]{2} # plane 16 +}x; + +our $UTF8 = qr{ + [\x09\x0A\x0D\x20-\x7E] # ASCII + | $NON_ASCII_UTF8 +}x; + +our $typeC99Typedefs = qr{(?:__)?(?:[us]_?)?int_?(?:8|16|32|64)_t}; +our $typeOtherOSTypedefs = qr{(?x: + u_(?:char|short|int|long) | # bsd + u(?:nchar|short|int|long) # sysv +)}; +our $typeKernelTypedefs = qr{(?x: + (?:__)?(?:u|s|be|le)(?:8|16|32|64)_t| + atomic_t +)}; +our $typeTypedefs = qr{(?x: + $typeC99Typedefs\b| + $typeOtherOSTypedefs\b| + $typeKernelTypedefs\b +)}; + +our $zero_initializer = qr{(?:(?:0[xX])?0+$Int_type?|NULL|false)\b}; + +our $logFunctions = qr{(?x: + printk(?:_ratelimited|_once|_deferred_once|_deferred|)| + (?:[a-z0-9]+_){1,2}(?:printk|emerg|alert|crit|err|warning|warn|notice|info|debug|dbg|vdbg|devel|cont|WARN)(?:_ratelimited|_once|)| + WARN(?:_RATELIMIT|_ONCE|)| + panic| + MODULE_[A-Z_]+| + seq_vprintf|seq_printf|seq_puts +)}; + +our $signature_tags = qr{(?xi: + Signed-off-by:| + Acked-by:| + Tested-by:| + Reviewed-by:| + Reported-by:| + Suggested-by:| + To:| + Cc: +)}; + +our @typeListMisordered = ( + qr{char\s+(?:un)?signed}, + qr{int\s+(?:(?:un)?signed\s+)?short\s}, + qr{int\s+short(?:\s+(?:un)?signed)}, + qr{short\s+int(?:\s+(?:un)?signed)}, + qr{(?:un)?signed\s+int\s+short}, + qr{short\s+(?:un)?signed}, + qr{long\s+int\s+(?:un)?signed}, + qr{int\s+long\s+(?:un)?signed}, + qr{long\s+(?:un)?signed\s+int}, + qr{int\s+(?:un)?signed\s+long}, + qr{int\s+(?:un)?signed}, + qr{int\s+long\s+long\s+(?:un)?signed}, + qr{long\s+long\s+int\s+(?:un)?signed}, + qr{long\s+long\s+(?:un)?signed\s+int}, + qr{long\s+long\s+(?:un)?signed}, + qr{long\s+(?:un)?signed}, +); + +our @typeList = ( + qr{void}, + qr{(?:(?:un)?signed\s+)?char}, + qr{(?:(?:un)?signed\s+)?short\s+int}, + qr{(?:(?:un)?signed\s+)?short}, + qr{(?:(?:un)?signed\s+)?int}, + qr{(?:(?:un)?signed\s+)?long\s+int}, + qr{(?:(?:un)?signed\s+)?long\s+long\s+int}, + qr{(?:(?:un)?signed\s+)?long\s+long}, + qr{(?:(?:un)?signed\s+)?long}, + qr{(?:un)?signed}, + qr{float}, + qr{double}, + qr{bool}, + qr{struct\s+$Ident}, + qr{union\s+$Ident}, + qr{enum\s+$Ident}, + qr{${Ident}_t}, + qr{${Ident}_handler}, + qr{${Ident}_handler_fn}, + @typeListMisordered, +); + +our $C90_int_types = qr{(?x: + long\s+long\s+int\s+(?:un)?signed| + long\s+long\s+(?:un)?signed\s+int| + long\s+long\s+(?:un)?signed| + (?:(?:un)?signed\s+)?long\s+long\s+int| + (?:(?:un)?signed\s+)?long\s+long| + int\s+long\s+long\s+(?:un)?signed| + int\s+(?:(?:un)?signed\s+)?long\s+long| + + long\s+int\s+(?:un)?signed| + long\s+(?:un)?signed\s+int| + long\s+(?:un)?signed| + (?:(?:un)?signed\s+)?long\s+int| + (?:(?:un)?signed\s+)?long| + int\s+long\s+(?:un)?signed| + int\s+(?:(?:un)?signed\s+)?long| + + int\s+(?:un)?signed| + (?:(?:un)?signed\s+)?int +)}; + +our @typeListFile = (); +our @typeListWithAttr = ( + @typeList, + qr{struct\s+$InitAttribute\s+$Ident}, + qr{union\s+$InitAttribute\s+$Ident}, +); + +our @modifierList = ( + qr{fastcall}, +); +our @modifierListFile = (); + +our @mode_permission_funcs = ( + ["module_param", 3], + ["module_param_(?:array|named|string)", 4], + ["module_param_array_named", 5], + ["debugfs_create_(?:file|u8|u16|u32|u64|x8|x16|x32|x64|size_t|atomic_t|bool|blob|regset32|u32_array)", 2], + ["proc_create(?:_data|)", 2], + ["(?:CLASS|DEVICE|SENSOR|SENSOR_DEVICE|IIO_DEVICE)_ATTR", 2], + ["IIO_DEV_ATTR_[A-Z_]+", 1], + ["SENSOR_(?:DEVICE_|)ATTR_2", 2], + ["SENSOR_TEMPLATE(?:_2|)", 3], + ["__ATTR", 2], +); + +#Create a search pattern for all these functions to speed up a loop below +our $mode_perms_search = ""; +foreach my $entry (@mode_permission_funcs) { + $mode_perms_search .= '|' if ($mode_perms_search ne ""); + $mode_perms_search .= $entry->[0]; +} + +our $mode_perms_world_writable = qr{ + S_IWUGO | + S_IWOTH | + S_IRWXUGO | + S_IALLUGO | + 0[0-7][0-7][2367] +}x; + +our %mode_permission_string_types = ( + "S_IRWXU" => 0700, + "S_IRUSR" => 0400, + "S_IWUSR" => 0200, + "S_IXUSR" => 0100, + "S_IRWXG" => 0070, + "S_IRGRP" => 0040, + "S_IWGRP" => 0020, + "S_IXGRP" => 0010, + "S_IRWXO" => 0007, + "S_IROTH" => 0004, + "S_IWOTH" => 0002, + "S_IXOTH" => 0001, + "S_IRWXUGO" => 0777, + "S_IRUGO" => 0444, + "S_IWUGO" => 0222, + "S_IXUGO" => 0111, +); + +#Create a search pattern for all these strings to speed up a loop below +our $mode_perms_string_search = ""; +foreach my $entry (keys %mode_permission_string_types) { + $mode_perms_string_search .= '|' if ($mode_perms_string_search ne ""); + $mode_perms_string_search .= $entry; +} + +our $allowed_asm_includes = qr{(?x: + irq| + memory| + time| + reboot +)}; +# memory.h: ARM has a custom one + +# Load common spelling mistakes and build regular expression list. +my $misspellings; +my %spelling_fix; + +if (open(my $spelling, '<', $spelling_file)) { + while (<$spelling>) { + my $line = $_; + + $line =~ s/\s*\n?$//g; + $line =~ s/^\s*//g; + + next if ($line =~ m/^\s*#/); + next if ($line =~ m/^\s*$/); + + my ($suspect, $fix) = split(/\|\|/, $line); + + $spelling_fix{$suspect} = $fix; + } + close($spelling); +} else { + warn "No typos will be found - file '$spelling_file': $!\n"; +} + +if ($codespell) { + if (open(my $spelling, '<', $codespellfile)) { + while (<$spelling>) { + my $line = $_; + + $line =~ s/\s*\n?$//g; + $line =~ s/^\s*//g; + + next if ($line =~ m/^\s*#/); + next if ($line =~ m/^\s*$/); + next if ($line =~ m/, disabled/i); + + $line =~ s/,.*$//; + + my ($suspect, $fix) = split(/->/, $line); + + $spelling_fix{$suspect} = $fix; + } + close($spelling); + } else { + warn "No codespell typos will be found - file '$codespellfile': $!\n"; + } +} + +$misspellings = join("|", sort keys %spelling_fix) if keys %spelling_fix; + +sub read_words { + my ($wordsRef, $file) = @_; + + if (open(my $words, '<', $file)) { + while (<$words>) { + my $line = $_; + + $line =~ s/\s*\n?$//g; + $line =~ s/^\s*//g; + + next if ($line =~ m/^\s*#/); + next if ($line =~ m/^\s*$/); + if ($line =~ /\s/) { + print("$file: '$line' invalid - ignored\n"); + next; + } + + $$wordsRef .= '|' if ($$wordsRef ne ""); + $$wordsRef .= $line; + } + close($file); + return 1; + } + + return 0; +} + +my $const_structs = ""; +#read_words(\$const_structs, $conststructsfile) +# or warn "No structs that should be const will be found - file '$conststructsfile': $!\n"; + +my $typeOtherTypedefs = ""; +if (length($typedefsfile)) { + read_words(\$typeOtherTypedefs, $typedefsfile) + or warn "No additional types will be considered - file '$typedefsfile': $!\n"; +} +$typeTypedefs .= '|' . $typeOtherTypedefs if ($typeOtherTypedefs ne ""); + +sub build_types { + my $mods = "(?x: \n" . join("|\n ", (@modifierList, @modifierListFile)) . "\n)"; + my $all = "(?x: \n" . join("|\n ", (@typeList, @typeListFile)) . "\n)"; + my $Misordered = "(?x: \n" . join("|\n ", @typeListMisordered) . "\n)"; + my $allWithAttr = "(?x: \n" . join("|\n ", @typeListWithAttr) . "\n)"; + $Modifier = qr{(?:$Attribute|$Sparse|$mods)}; + $BasicType = qr{ + (?:$typeTypedefs\b)| + (?:${all}\b) + }x; + $NonptrType = qr{ + (?:$Modifier\s+|const\s+)* + (?: + (?:typeof|__typeof__)\s*\([^\)]*\)| + (?:$typeTypedefs\b)| + (?:${all}\b) + ) + (?:\s+$Modifier|\s+const)* + }x; + $NonptrTypeMisordered = qr{ + (?:$Modifier\s+|const\s+)* + (?: + (?:${Misordered}\b) + ) + (?:\s+$Modifier|\s+const)* + }x; + $NonptrTypeWithAttr = qr{ + (?:$Modifier\s+|const\s+)* + (?: + (?:typeof|__typeof__)\s*\([^\)]*\)| + (?:$typeTypedefs\b)| + (?:${allWithAttr}\b) + ) + (?:\s+$Modifier|\s+const)* + }x; + $Type = qr{ + $NonptrType + (?:(?:\s|\*|\[\])+\s*const|(?:\s|\*\s*(?:const\s*)?|\[\])+|(?:\s*\[\s*\])+)? + (?:\s+$Inline|\s+$Modifier)* + }x; + $TypeMisordered = qr{ + $NonptrTypeMisordered + (?:(?:\s|\*|\[\])+\s*const|(?:\s|\*\s*(?:const\s*)?|\[\])+|(?:\s*\[\s*\])+)? + (?:\s+$Inline|\s+$Modifier)* + }x; + $Declare = qr{(?:$Storage\s+(?:$Inline\s+)?)?$Type}; + $DeclareMisordered = qr{(?:$Storage\s+(?:$Inline\s+)?)?$TypeMisordered}; +} +build_types(); + +our $Typecast = qr{\s*(\(\s*$NonptrType\s*\)){0,1}\s*}; + +# Using $balanced_parens, $LvalOrFunc, or $FuncArg +# requires at least perl version v5.10.0 +# Any use must be runtime checked with $^V + +our $balanced_parens = qr/(\((?:[^\(\)]++|(?-1))*\))/; +our $LvalOrFunc = qr{((?:[\&\*]\s*)?$Lval)\s*($balanced_parens{0,1})\s*}; +our $FuncArg = qr{$Typecast{0,1}($LvalOrFunc|$Constant|$String)}; + +our $declaration_macros = qr{(?x: + (?:$Storage\s+)?(?:[A-Z_][A-Z0-9]*_){0,2}(?:DEFINE|DECLARE)(?:_[A-Z0-9]+){1,6}\s*\(| + (?:$Storage\s+)?[HLP]?LIST_HEAD\s*\(| + (?:$Storage\s+)?${Type}\s+uninitialized_var\s*\( +)}; + +sub deparenthesize { + my ($string) = @_; + return "" if (!defined($string)); + + while ($string =~ /^\s*\(.*\)\s*$/) { + $string =~ s@^\s*\(\s*@@; + $string =~ s@\s*\)\s*$@@; + } + + $string =~ s@\s+@ @g; + + return $string; +} + +sub seed_camelcase_file { + my ($file) = @_; + + return if (!(-f $file)); + + local $/; + + open(my $include_file, '<', "$file") + or warn "$P: Can't read '$file' $!\n"; + my $text = <$include_file>; + close($include_file); + + my @lines = split('\n', $text); + + foreach my $line (@lines) { + next if ($line !~ /(?:[A-Z][a-z]|[a-z][A-Z])/); + if ($line =~ /^[ \t]*(?:#[ \t]*define|typedef\s+$Type)\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)/) { + $camelcase{$1} = 1; + } elsif ($line =~ /^\s*$Declare\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)\s*[\(\[,;]/) { + $camelcase{$1} = 1; + } elsif ($line =~ /^\s*(?:union|struct|enum)\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)\s*[;\{]/) { + $camelcase{$1} = 1; + } + } +} + +sub is_maintained_obsolete { + my ($filename) = @_; + + return 0 if (!$tree || !(-e "$root/scripts/get_maintainer.pl")); + + my $status = `perl $root/scripts/get_maintainer.pl --status --nom --nol --nogit --nogit-fallback -f $filename 2>&1`; + + return $status =~ /obsolete/i; +} + +my $camelcase_seeded = 0; +sub seed_camelcase_includes { + return if ($camelcase_seeded); + + my $files; + my $camelcase_cache = ""; + my @include_files = (); + + $camelcase_seeded = 1; + + if (-e ".git") { + my $git_last_include_commit = `git log --no-merges --pretty=format:"%h%n" -1 -- include`; + chomp $git_last_include_commit; + $camelcase_cache = ".checkpatch-camelcase.git.$git_last_include_commit"; + } else { + my $last_mod_date = 0; + $files = `find $root/include -name "*.h"`; + @include_files = split('\n', $files); + foreach my $file (@include_files) { + my $date = POSIX::strftime("%Y%m%d%H%M", + localtime((stat $file)[9])); + $last_mod_date = $date if ($last_mod_date < $date); + } + $camelcase_cache = ".checkpatch-camelcase.date.$last_mod_date"; + } + + if ($camelcase_cache ne "" && -f $camelcase_cache) { + open(my $camelcase_file, '<', "$camelcase_cache") + or warn "$P: Can't read '$camelcase_cache' $!\n"; + while (<$camelcase_file>) { + chomp; + $camelcase{$_} = 1; + } + close($camelcase_file); + + return; + } + + if (-e ".git") { + $files = `git ls-files "include/*.h"`; + @include_files = split('\n', $files); + } + + foreach my $file (@include_files) { + seed_camelcase_file($file); + } + + if ($camelcase_cache ne "") { + unlink glob ".checkpatch-camelcase.*"; + open(my $camelcase_file, '>', "$camelcase_cache") + or warn "$P: Can't write '$camelcase_cache' $!\n"; + foreach (sort { lc($a) cmp lc($b) } keys(%camelcase)) { + print $camelcase_file ("$_\n"); + } + close($camelcase_file); + } +} + +sub git_commit_info { + my ($commit, $id, $desc) = @_; + + return ($id, $desc) if ((which("git") eq "") || !(-e ".git")); + + my $output = `git log --no-color --format='%H %s' -1 $commit 2>&1`; + $output =~ s/^\s*//gm; + my @lines = split("\n", $output); + + return ($id, $desc) if ($#lines < 0); + + if ($lines[0] =~ /^error: short SHA1 $commit is ambiguous\./) { +# Maybe one day convert this block of bash into something that returns +# all matching commit ids, but it's very slow... +# +# echo "checking commits $1..." +# git rev-list --remotes | grep -i "^$1" | +# while read line ; do +# git log --format='%H %s' -1 $line | +# echo "commit $(cut -c 1-12,41-)" +# done + } elsif ($lines[0] =~ /^fatal: ambiguous argument '$commit': unknown revision or path not in the working tree\./) { + $id = undef; + } else { + $id = substr($lines[0], 0, 12); + $desc = substr($lines[0], 41); + } + + return ($id, $desc); +} + +$chk_signoff = 0 if ($file); + +my @rawlines = (); +my @lines = (); +my @fixed = (); +my @fixed_inserted = (); +my @fixed_deleted = (); +my $fixlinenr = -1; + +# If input is git commits, extract all commits from the commit expressions. +# For example, HEAD-3 means we need check 'HEAD, HEAD~1, HEAD~2'. +die "$P: No git repository found\n" if ($git && !-e ".git"); + +if ($git) { + my @commits = (); + foreach my $commit_expr (@ARGV) { + my $git_range; + if ($commit_expr =~ m/^(.*)-(\d+)$/) { + $git_range = "-$2 $1"; + } elsif ($commit_expr =~ m/\.\./) { + $git_range = "$commit_expr"; + } else { + $git_range = "-1 $commit_expr"; + } + my $lines = `git log --no-color --no-merges --pretty=format:'%H %s' $git_range`; + foreach my $line (split(/\n/, $lines)) { + $line =~ /^([0-9a-fA-F]{40,40}) (.*)$/; + next if (!defined($1) || !defined($2)); + my $sha1 = $1; + my $subject = $2; + unshift(@commits, $sha1); + $git_commits{$sha1} = $subject; + } + } + die "$P: no git commits after extraction!\n" if (@commits == 0); + @ARGV = @commits; +} + +my $vname; +for my $filename (@ARGV) { + my $FILE; + if ($git) { + open($FILE, '-|', "git format-patch -M --stdout -1 $filename") || + die "$P: $filename: git format-patch failed - $!\n"; + } elsif ($file) { + open($FILE, '-|', "diff -u /dev/null $filename") || + die "$P: $filename: diff failed - $!\n"; + } elsif ($filename eq '-') { + open($FILE, '<&STDIN'); + } else { + open($FILE, '<', "$filename") || + die "$P: $filename: open failed - $!\n"; + } + if ($filename eq '-') { + $vname = 'Your patch'; + } elsif ($git) { + $vname = "Commit " . substr($filename, 0, 12) . ' ("' . $git_commits{$filename} . '")'; + } else { + $vname = $filename; + } + while (<$FILE>) { + chomp; + push(@rawlines, $_); + } + close($FILE); + + if ($#ARGV > 0 && $quiet == 0) { + print '-' x length($vname) . "\n"; + print "$vname\n"; + print '-' x length($vname) . "\n"; + } + + if (!process($filename)) { + $exit = 1; + } + @rawlines = (); + @lines = (); + @fixed = (); + @fixed_inserted = (); + @fixed_deleted = (); + $fixlinenr = -1; + @modifierListFile = (); + @typeListFile = (); + build_types(); +} + +if (!$quiet) { + hash_show_words(\%use_type, "Used"); + hash_show_words(\%ignore_type, "Ignored"); + + if ($^V lt 5.10.0) { + print << "EOM" + +NOTE: perl $^V is not modern enough to detect all possible issues. + An upgrade to at least perl v5.10.0 is suggested. +EOM + } + if ($exit) { + print << "EOM" + +NOTE: If any of the errors are false positives, please report + them to the maintainers. +EOM + } +} + +exit($exit); + +sub top_of_kernel_tree { + my ($root) = @_; + + my @tree_check = ( + "LICENSE", "CODEOWNERS", "Kconfig", "Makefile", + "README.rst", "doc", "arch", "include", "drivers", + "boards", "kernel", "lib", "scripts", + ); + + foreach my $check (@tree_check) { + if (! -e $root . '/' . $check) { + return 0; + } + } + return 1; +} + +sub parse_email { + my ($formatted_email) = @_; + + my $name = ""; + my $address = ""; + my $comment = ""; + + if ($formatted_email =~ /^(.*)<(\S+\@\S+)>(.*)$/) { + $name = $1; + $address = $2; + $comment = $3 if defined $3; + } elsif ($formatted_email =~ /^\s*<(\S+\@\S+)>(.*)$/) { + $address = $1; + $comment = $2 if defined $2; + } elsif ($formatted_email =~ /(\S+\@\S+)(.*)$/) { + $address = $1; + $comment = $2 if defined $2; + $formatted_email =~ s/$address.*$//; + $name = $formatted_email; + $name = trim($name); + $name =~ s/^\"|\"$//g; + # If there's a name left after stripping spaces and + # leading quotes, and the address doesn't have both + # leading and trailing angle brackets, the address + # is invalid. ie: + # "joe smith joe@smith.com" bad + # "joe smith <joe@smith.com" bad + if ($name ne "" && $address !~ /^<[^>]+>$/) { + $name = ""; + $address = ""; + $comment = ""; + } + } + + $name = trim($name); + $name =~ s/^\"|\"$//g; + $address = trim($address); + $address =~ s/^\<|\>$//g; + + if ($name =~ /[^\w \-]/i) { ##has "must quote" chars + $name =~ s/(?<!\\)"/\\"/g; ##escape quotes + $name = "\"$name\""; + } + + return ($name, $address, $comment); +} + +sub format_email { + my ($name, $address) = @_; + + my $formatted_email; + + $name = trim($name); + $name =~ s/^\"|\"$//g; + $address = trim($address); + + if ($name =~ /[^\w \-]/i) { ##has "must quote" chars + $name =~ s/(?<!\\)"/\\"/g; ##escape quotes + $name = "\"$name\""; + } + + if ("$name" eq "") { + $formatted_email = "$address"; + } else { + $formatted_email = "$name <$address>"; + } + + return $formatted_email; +} + +sub which { + my ($bin) = @_; + + foreach my $path (split(/:/, $ENV{PATH})) { + if (-e "$path/$bin") { + return "$path/$bin"; + } + } + + return ""; +} + +sub which_conf { + my ($conf) = @_; + + foreach my $path (split(/:/, ".:$ENV{HOME}:.scripts")) { + if (-e "$path/$conf") { + return "$path/$conf"; + } + } + + return ""; +} + +sub expand_tabs { + my ($str) = @_; + + my $res = ''; + my $n = 0; + for my $c (split(//, $str)) { + if ($c eq "\t") { + $res .= ' '; + $n++; + for (; ($n % 8) != 0; $n++) { + $res .= ' '; + } + next; + } + $res .= $c; + $n++; + } + + return $res; +} +sub copy_spacing { + (my $res = shift) =~ tr/\t/ /c; + return $res; +} + +sub line_stats { + my ($line) = @_; + + # Drop the diff line leader and expand tabs + $line =~ s/^.//; + $line = expand_tabs($line); + + # Pick the indent from the front of the line. + my ($white) = ($line =~ /^(\s*)/); + + return (length($line), length($white)); +} + +my $sanitise_quote = ''; + +sub sanitise_line_reset { + my ($in_comment) = @_; + + if ($in_comment) { + $sanitise_quote = '*/'; + } else { + $sanitise_quote = ''; + } +} +sub sanitise_line { + my ($line) = @_; + + my $res = ''; + my $l = ''; + + my $qlen = 0; + my $off = 0; + my $c; + + # Always copy over the diff marker. + $res = substr($line, 0, 1); + + for ($off = 1; $off < length($line); $off++) { + $c = substr($line, $off, 1); + + # Comments we are wacking completly including the begin + # and end, all to $;. + if ($sanitise_quote eq '' && substr($line, $off, 2) eq '/*') { + $sanitise_quote = '*/'; + + substr($res, $off, 2, "$;$;"); + $off++; + next; + } + if ($sanitise_quote eq '*/' && substr($line, $off, 2) eq '*/') { + $sanitise_quote = ''; + substr($res, $off, 2, "$;$;"); + $off++; + next; + } + if ($sanitise_quote eq '' && substr($line, $off, 2) eq '//') { + $sanitise_quote = '//'; + + substr($res, $off, 2, $sanitise_quote); + $off++; + next; + } + + # A \ in a string means ignore the next character. + if (($sanitise_quote eq "'" || $sanitise_quote eq '"') && + $c eq "\\") { + substr($res, $off, 2, 'XX'); + $off++; + next; + } + # Regular quotes. + if ($c eq "'" || $c eq '"') { + if ($sanitise_quote eq '') { + $sanitise_quote = $c; + + substr($res, $off, 1, $c); + next; + } elsif ($sanitise_quote eq $c) { + $sanitise_quote = ''; + } + } + + #print "c<$c> SQ<$sanitise_quote>\n"; + if ($off != 0 && $sanitise_quote eq '*/' && $c ne "\t") { + substr($res, $off, 1, $;); + } elsif ($off != 0 && $sanitise_quote eq '//' && $c ne "\t") { + substr($res, $off, 1, $;); + } elsif ($off != 0 && $sanitise_quote && $c ne "\t") { + substr($res, $off, 1, 'X'); + } else { + substr($res, $off, 1, $c); + } + } + + if ($sanitise_quote eq '//') { + $sanitise_quote = ''; + } + + # The pathname on a #include may be surrounded by '<' and '>'. + if ($res =~ /^.\s*\#\s*include\s+\<(.*)\>/) { + my $clean = 'X' x length($1); + $res =~ s@\<.*\>@<$clean>@; + + # The whole of a #error is a string. + } elsif ($res =~ /^.\s*\#\s*(?:error|warning)\s+(.*)\b/) { + my $clean = 'X' x length($1); + $res =~ s@(\#\s*(?:error|warning)\s+).*@$1$clean@; + } + + if ($allow_c99_comments && $res =~ m@(//.*$)@) { + my $match = $1; + $res =~ s/\Q$match\E/"$;" x length($match)/e; + } + + return $res; +} + +sub get_quoted_string { + my ($line, $rawline) = @_; + + return "" if ($line !~ m/($String)/g); + return substr($rawline, $-[0], $+[0] - $-[0]); +} + +sub ctx_statement_block { + my ($linenr, $remain, $off) = @_; + my $line = $linenr - 1; + my $blk = ''; + my $soff = $off; + my $coff = $off - 1; + my $coff_set = 0; + + my $loff = 0; + + my $type = ''; + my $level = 0; + my @stack = (); + my $p; + my $c; + my $len = 0; + + my $remainder; + while (1) { + @stack = (['', 0]) if ($#stack == -1); + + #warn "CSB: blk<$blk> remain<$remain>\n"; + # If we are about to drop off the end, pull in more + # context. + if ($off >= $len) { + for (; $remain > 0; $line++) { + last if (!defined $lines[$line]); + next if ($lines[$line] =~ /^-/); + $remain--; + $loff = $len; + $blk .= $lines[$line] . "\n"; + $len = length($blk); + $line++; + last; + } + # Bail if there is no further context. + #warn "CSB: blk<$blk> off<$off> len<$len>\n"; + if ($off >= $len) { + last; + } + if ($level == 0 && substr($blk, $off) =~ /^.\s*#\s*define/) { + $level++; + $type = '#'; + } + } + $p = $c; + $c = substr($blk, $off, 1); + $remainder = substr($blk, $off); + + #warn "CSB: c<$c> type<$type> level<$level> remainder<$remainder> coff_set<$coff_set>\n"; + + # Handle nested #if/#else. + if ($remainder =~ /^#\s*(?:ifndef|ifdef|if)\s/) { + push(@stack, [ $type, $level ]); + } elsif ($remainder =~ /^#\s*(?:else|elif)\b/) { + ($type, $level) = @{$stack[$#stack - 1]}; + } elsif ($remainder =~ /^#\s*endif\b/) { + ($type, $level) = @{pop(@stack)}; + } + + # Statement ends at the ';' or a close '}' at the + # outermost level. + if ($level == 0 && $c eq ';') { + last; + } + + # An else is really a conditional as long as its not else if + if ($level == 0 && $coff_set == 0 && + (!defined($p) || $p =~ /(?:\s|\}|\+)/) && + $remainder =~ /^(else)(?:\s|{)/ && + $remainder !~ /^else\s+if\b/) { + $coff = $off + length($1) - 1; + $coff_set = 1; + #warn "CSB: mark coff<$coff> soff<$soff> 1<$1>\n"; + #warn "[" . substr($blk, $soff, $coff - $soff + 1) . "]\n"; + } + + if (($type eq '' || $type eq '(') && $c eq '(') { + $level++; + $type = '('; + } + if ($type eq '(' && $c eq ')') { + $level--; + $type = ($level != 0)? '(' : ''; + + if ($level == 0 && $coff < $soff) { + $coff = $off; + $coff_set = 1; + #warn "CSB: mark coff<$coff>\n"; + } + } + if (($type eq '' || $type eq '{') && $c eq '{') { + $level++; + $type = '{'; + } + if ($type eq '{' && $c eq '}') { + $level--; + $type = ($level != 0)? '{' : ''; + + if ($level == 0) { + if (substr($blk, $off + 1, 1) eq ';') { + $off++; + } + last; + } + } + # Preprocessor commands end at the newline unless escaped. + if ($type eq '#' && $c eq "\n" && $p ne "\\") { + $level--; + $type = ''; + $off++; + last; + } + $off++; + } + # We are truly at the end, so shuffle to the next line. + if ($off == $len) { + $loff = $len + 1; + $line++; + $remain--; + } + + my $statement = substr($blk, $soff, $off - $soff + 1); + my $condition = substr($blk, $soff, $coff - $soff + 1); + + #warn "STATEMENT<$statement>\n"; + #warn "CONDITION<$condition>\n"; + + #print "coff<$coff> soff<$off> loff<$loff>\n"; + + return ($statement, $condition, + $line, $remain + 1, $off - $loff + 1, $level); +} + +sub statement_lines { + my ($stmt) = @_; + + # Strip the diff line prefixes and rip blank lines at start and end. + $stmt =~ s/(^|\n)./$1/g; + $stmt =~ s/^\s*//; + $stmt =~ s/\s*$//; + + my @stmt_lines = ($stmt =~ /\n/g); + + return $#stmt_lines + 2; +} + +sub statement_rawlines { + my ($stmt) = @_; + + my @stmt_lines = ($stmt =~ /\n/g); + + return $#stmt_lines + 2; +} + +sub statement_block_size { + my ($stmt) = @_; + + $stmt =~ s/(^|\n)./$1/g; + $stmt =~ s/^\s*{//; + $stmt =~ s/}\s*$//; + $stmt =~ s/^\s*//; + $stmt =~ s/\s*$//; + + my @stmt_lines = ($stmt =~ /\n/g); + my @stmt_statements = ($stmt =~ /;/g); + + my $stmt_lines = $#stmt_lines + 2; + my $stmt_statements = $#stmt_statements + 1; + + if ($stmt_lines > $stmt_statements) { + return $stmt_lines; + } else { + return $stmt_statements; + } +} + +sub ctx_statement_full { + my ($linenr, $remain, $off) = @_; + my ($statement, $condition, $level); + + my (@chunks); + + # Grab the first conditional/block pair. + ($statement, $condition, $linenr, $remain, $off, $level) = + ctx_statement_block($linenr, $remain, $off); + #print "F: c<$condition> s<$statement> remain<$remain>\n"; + push(@chunks, [ $condition, $statement ]); + if (!($remain > 0 && $condition =~ /^\s*(?:\n[+-])?\s*(?:if|else|do)\b/s)) { + return ($level, $linenr, @chunks); + } + + # Pull in the following conditional/block pairs and see if they + # could continue the statement. + for (;;) { + ($statement, $condition, $linenr, $remain, $off, $level) = + ctx_statement_block($linenr, $remain, $off); + #print "C: c<$condition> s<$statement> remain<$remain>\n"; + last if (!($remain > 0 && $condition =~ /^(?:\s*\n[+-])*\s*(?:else|do)\b/s)); + #print "C: push\n"; + push(@chunks, [ $condition, $statement ]); + } + + return ($level, $linenr, @chunks); +} + +sub ctx_block_get { + my ($linenr, $remain, $outer, $open, $close, $off) = @_; + my $line; + my $start = $linenr - 1; + my $blk = ''; + my @o; + my @c; + my @res = (); + + my $level = 0; + my @stack = ($level); + for ($line = $start; $remain > 0; $line++) { + next if ($rawlines[$line] =~ /^-/); + $remain--; + + $blk .= $rawlines[$line]; + + # Handle nested #if/#else. + if ($lines[$line] =~ /^.\s*#\s*(?:ifndef|ifdef|if)\s/) { + push(@stack, $level); + } elsif ($lines[$line] =~ /^.\s*#\s*(?:else|elif)\b/) { + $level = $stack[$#stack - 1]; + } elsif ($lines[$line] =~ /^.\s*#\s*endif\b/) { + $level = pop(@stack); + } + + foreach my $c (split(//, $lines[$line])) { + ##print "C<$c>L<$level><$open$close>O<$off>\n"; + if ($off > 0) { + $off--; + next; + } + + if ($c eq $close && $level > 0) { + $level--; + last if ($level == 0); + } elsif ($c eq $open) { + $level++; + } + } + + if (!$outer || $level <= 1) { + push(@res, $rawlines[$line]); + } + + last if ($level == 0); + } + + return ($level, @res); +} +sub ctx_block_outer { + my ($linenr, $remain) = @_; + + my ($level, @r) = ctx_block_get($linenr, $remain, 1, '{', '}', 0); + return @r; +} +sub ctx_block { + my ($linenr, $remain) = @_; + + my ($level, @r) = ctx_block_get($linenr, $remain, 0, '{', '}', 0); + return @r; +} +sub ctx_statement { + my ($linenr, $remain, $off) = @_; + + my ($level, @r) = ctx_block_get($linenr, $remain, 0, '(', ')', $off); + return @r; +} +sub ctx_block_level { + my ($linenr, $remain) = @_; + + return ctx_block_get($linenr, $remain, 0, '{', '}', 0); +} +sub ctx_statement_level { + my ($linenr, $remain, $off) = @_; + + return ctx_block_get($linenr, $remain, 0, '(', ')', $off); +} + +sub ctx_locate_comment { + my ($first_line, $end_line) = @_; + + # Catch a comment on the end of the line itself. + my ($current_comment) = ($rawlines[$end_line - 1] =~ m@.*(/\*.*\*/)\s*(?:\\\s*)?$@); + return $current_comment if (defined $current_comment); + + # Look through the context and try and figure out if there is a + # comment. + my $in_comment = 0; + $current_comment = ''; + for (my $linenr = $first_line; $linenr < $end_line; $linenr++) { + my $line = $rawlines[$linenr - 1]; + #warn " $line\n"; + if ($linenr == $first_line and $line =~ m@^.\s*\*@) { + $in_comment = 1; + } + if ($line =~ m@/\*@) { + $in_comment = 1; + } + if (!$in_comment && $current_comment ne '') { + $current_comment = ''; + } + $current_comment .= $line . "\n" if ($in_comment); + if ($line =~ m@\*/@) { + $in_comment = 0; + } + } + + chomp($current_comment); + return($current_comment); +} +sub ctx_has_comment { + my ($first_line, $end_line) = @_; + my $cmt = ctx_locate_comment($first_line, $end_line); + + ##print "LINE: $rawlines[$end_line - 1 ]\n"; + ##print "CMMT: $cmt\n"; + + return ($cmt ne ''); +} + +sub raw_line { + my ($linenr, $cnt) = @_; + + my $offset = $linenr - 1; + $cnt++; + + my $line; + while ($cnt) { + $line = $rawlines[$offset++]; + next if (defined($line) && $line =~ /^-/); + $cnt--; + } + + return $line; +} + +sub cat_vet { + my ($vet) = @_; + my ($res, $coded); + + $res = ''; + while ($vet =~ /([^[:cntrl:]]*)([[:cntrl:]]|$)/g) { + $res .= $1; + if ($2 ne '') { + $coded = sprintf("^%c", unpack('C', $2) + 64); + $res .= $coded; + } + } + $res =~ s/$/\$/; + + return $res; +} + +my $av_preprocessor = 0; +my $av_pending; +my @av_paren_type; +my $av_pend_colon; + +sub annotate_reset { + $av_preprocessor = 0; + $av_pending = '_'; + @av_paren_type = ('E'); + $av_pend_colon = 'O'; +} + +sub annotate_values { + my ($stream, $type) = @_; + + my $res; + my $var = '_' x length($stream); + my $cur = $stream; + + print "$stream\n" if ($dbg_values > 1); + + while (length($cur)) { + @av_paren_type = ('E') if ($#av_paren_type < 0); + print " <" . join('', @av_paren_type) . + "> <$type> <$av_pending>" if ($dbg_values > 1); + if ($cur =~ /^(\s+)/o) { + print "WS($1)\n" if ($dbg_values > 1); + if ($1 =~ /\n/ && $av_preprocessor) { + $type = pop(@av_paren_type); + $av_preprocessor = 0; + } + + } elsif ($cur =~ /^(\(\s*$Type\s*)\)/ && $av_pending eq '_') { + print "CAST($1)\n" if ($dbg_values > 1); + push(@av_paren_type, $type); + $type = 'c'; + + } elsif ($cur =~ /^($Type)\s*(?:$Ident|,|\)|\(|\s*$)/) { + print "DECLARE($1)\n" if ($dbg_values > 1); + $type = 'T'; + + } elsif ($cur =~ /^($Modifier)\s*/) { + print "MODIFIER($1)\n" if ($dbg_values > 1); + $type = 'T'; + + } elsif ($cur =~ /^(\#\s*define\s*$Ident)(\(?)/o) { + print "DEFINE($1,$2)\n" if ($dbg_values > 1); + $av_preprocessor = 1; + push(@av_paren_type, $type); + if ($2 ne '') { + $av_pending = 'N'; + } + $type = 'E'; + + } elsif ($cur =~ /^(\#\s*(?:undef\s*$Ident|include\b))/o) { + print "UNDEF($1)\n" if ($dbg_values > 1); + $av_preprocessor = 1; + push(@av_paren_type, $type); + + } elsif ($cur =~ /^(\#\s*(?:ifdef|ifndef|if))/o) { + print "PRE_START($1)\n" if ($dbg_values > 1); + $av_preprocessor = 1; + + push(@av_paren_type, $type); + push(@av_paren_type, $type); + $type = 'E'; + + } elsif ($cur =~ /^(\#\s*(?:else|elif))/o) { + print "PRE_RESTART($1)\n" if ($dbg_values > 1); + $av_preprocessor = 1; + + push(@av_paren_type, $av_paren_type[$#av_paren_type]); + + $type = 'E'; + + } elsif ($cur =~ /^(\#\s*(?:endif))/o) { + print "PRE_END($1)\n" if ($dbg_values > 1); + + $av_preprocessor = 1; + + # Assume all arms of the conditional end as this + # one does, and continue as if the #endif was not here. + pop(@av_paren_type); + push(@av_paren_type, $type); + $type = 'E'; + + } elsif ($cur =~ /^(\\\n)/o) { + print "PRECONT($1)\n" if ($dbg_values > 1); + + } elsif ($cur =~ /^(__attribute__)\s*\(?/o) { + print "ATTR($1)\n" if ($dbg_values > 1); + $av_pending = $type; + $type = 'N'; + + } elsif ($cur =~ /^(sizeof)\s*(\()?/o) { + print "SIZEOF($1)\n" if ($dbg_values > 1); + if (defined $2) { + $av_pending = 'V'; + } + $type = 'N'; + + } elsif ($cur =~ /^(if|while|for)\b/o) { + print "COND($1)\n" if ($dbg_values > 1); + $av_pending = 'E'; + $type = 'N'; + + } elsif ($cur =~/^(case)/o) { + print "CASE($1)\n" if ($dbg_values > 1); + $av_pend_colon = 'C'; + $type = 'N'; + + } elsif ($cur =~/^(return|else|goto|typeof|__typeof__)\b/o) { + print "KEYWORD($1)\n" if ($dbg_values > 1); + $type = 'N'; + + } elsif ($cur =~ /^(\()/o) { + print "PAREN('$1')\n" if ($dbg_values > 1); + push(@av_paren_type, $av_pending); + $av_pending = '_'; + $type = 'N'; + + } elsif ($cur =~ /^(\))/o) { + my $new_type = pop(@av_paren_type); + if ($new_type ne '_') { + $type = $new_type; + print "PAREN('$1') -> $type\n" + if ($dbg_values > 1); + } else { + print "PAREN('$1')\n" if ($dbg_values > 1); + } + + } elsif ($cur =~ /^($Ident)\s*\(/o) { + print "FUNC($1)\n" if ($dbg_values > 1); + $type = 'V'; + $av_pending = 'V'; + + } elsif ($cur =~ /^($Ident\s*):(?:\s*\d+\s*(,|=|;))?/) { + if (defined $2 && $type eq 'C' || $type eq 'T') { + $av_pend_colon = 'B'; + } elsif ($type eq 'E') { + $av_pend_colon = 'L'; + } + print "IDENT_COLON($1,$type>$av_pend_colon)\n" if ($dbg_values > 1); + $type = 'V'; + + } elsif ($cur =~ /^($Ident|$Constant)/o) { + print "IDENT($1)\n" if ($dbg_values > 1); + $type = 'V'; + + } elsif ($cur =~ /^($Assignment)/o) { + print "ASSIGN($1)\n" if ($dbg_values > 1); + $type = 'N'; + + } elsif ($cur =~/^(;|{|})/) { + print "END($1)\n" if ($dbg_values > 1); + $type = 'E'; + $av_pend_colon = 'O'; + + } elsif ($cur =~/^(,)/) { + print "COMMA($1)\n" if ($dbg_values > 1); + $type = 'C'; + + } elsif ($cur =~ /^(\?)/o) { + print "QUESTION($1)\n" if ($dbg_values > 1); + $type = 'N'; + + } elsif ($cur =~ /^(:)/o) { + print "COLON($1,$av_pend_colon)\n" if ($dbg_values > 1); + + substr($var, length($res), 1, $av_pend_colon); + if ($av_pend_colon eq 'C' || $av_pend_colon eq 'L') { + $type = 'E'; + } else { + $type = 'N'; + } + $av_pend_colon = 'O'; + + } elsif ($cur =~ /^(\[)/o) { + print "CLOSE($1)\n" if ($dbg_values > 1); + $type = 'N'; + + } elsif ($cur =~ /^(-(?![->])|\+(?!\+)|\*|\&\&|\&)/o) { + my $variant; + + print "OPV($1)\n" if ($dbg_values > 1); + if ($type eq 'V') { + $variant = 'B'; + } else { + $variant = 'U'; + } + + substr($var, length($res), 1, $variant); + $type = 'N'; + + } elsif ($cur =~ /^($Operators)/o) { + print "OP($1)\n" if ($dbg_values > 1); + if ($1 ne '++' && $1 ne '--') { + $type = 'N'; + } + + } elsif ($cur =~ /(^.)/o) { + print "C($1)\n" if ($dbg_values > 1); + } + if (defined $1) { + $cur = substr($cur, length($1)); + $res .= $type x length($1); + } + } + + return ($res, $var); +} + +sub possible { + my ($possible, $line) = @_; + my $notPermitted = qr{(?: + ^(?: + $Modifier| + $Storage| + $Type| + DEFINE_\S+ + )$| + ^(?: + goto| + return| + case| + else| + asm|__asm__| + do| + \#| + \#\#| + )(?:\s|$)| + ^(?:typedef|struct|enum)\b + )}x; + warn "CHECK<$possible> ($line)\n" if ($dbg_possible > 2); + if ($possible !~ $notPermitted) { + # Check for modifiers. + $possible =~ s/\s*$Storage\s*//g; + $possible =~ s/\s*$Sparse\s*//g; + if ($possible =~ /^\s*$/) { + + } elsif ($possible =~ /\s/) { + $possible =~ s/\s*$Type\s*//g; + for my $modifier (split(' ', $possible)) { + if ($modifier !~ $notPermitted) { + warn "MODIFIER: $modifier ($possible) ($line)\n" if ($dbg_possible); + push(@modifierListFile, $modifier); + } + } + + } else { + warn "POSSIBLE: $possible ($line)\n" if ($dbg_possible); + push(@typeListFile, $possible); + } + build_types(); + } else { + warn "NOTPOSS: $possible ($line)\n" if ($dbg_possible > 1); + } +} + +my $prefix = ''; + +sub show_type { + my ($type) = @_; + + $type =~ tr/[a-z]/[A-Z]/; + + return defined $use_type{$type} if (scalar keys %use_type > 0); + + return !defined $ignore_type{$type}; +} + +sub report { + my ($level, $type, $msg) = @_; + + if (!show_type($type) || + (defined $tst_only && $msg !~ /\Q$tst_only\E/)) { + return 0; + } + my $output = ''; + if ($color) { + if ($level eq 'ERROR') { + $output .= RED; + } elsif ($level eq 'WARNING') { + $output .= YELLOW; + } else { + $output .= GREEN; + } + } + $output .= $prefix . $level . ':'; + if ($show_types) { + $output .= BLUE if ($color); + $output .= "$type:"; + } + $output .= RESET if ($color); + $output .= ' ' . $msg . "\n"; + + if ($showfile) { + my @lines = split("\n", $output, -1); + splice(@lines, 1, 1); + $output = join("\n", @lines); + } + $output = (split('\n', $output))[0] . "\n" if ($terse); + + push(our @report, $output); + + return 1; +} + +sub report_dump { + our @report; +} + +sub fixup_current_range { + my ($lineRef, $offset, $length) = @_; + + if ($$lineRef =~ /^\@\@ -\d+,\d+ \+(\d+),(\d+) \@\@/) { + my $o = $1; + my $l = $2; + my $no = $o + $offset; + my $nl = $l + $length; + $$lineRef =~ s/\+$o,$l \@\@/\+$no,$nl \@\@/; + } +} + +sub fix_inserted_deleted_lines { + my ($linesRef, $insertedRef, $deletedRef) = @_; + + my $range_last_linenr = 0; + my $delta_offset = 0; + + my $old_linenr = 0; + my $new_linenr = 0; + + my $next_insert = 0; + my $next_delete = 0; + + my @lines = (); + + my $inserted = @{$insertedRef}[$next_insert++]; + my $deleted = @{$deletedRef}[$next_delete++]; + + foreach my $old_line (@{$linesRef}) { + my $save_line = 1; + my $line = $old_line; #don't modify the array + if ($line =~ /^(?:\+\+\+|\-\-\-)\s+\S+/) { #new filename + $delta_offset = 0; + } elsif ($line =~ /^\@\@ -\d+,\d+ \+\d+,\d+ \@\@/) { #new hunk + $range_last_linenr = $new_linenr; + fixup_current_range(\$line, $delta_offset, 0); + } + + while (defined($deleted) && ${$deleted}{'LINENR'} == $old_linenr) { + $deleted = @{$deletedRef}[$next_delete++]; + $save_line = 0; + fixup_current_range(\$lines[$range_last_linenr], $delta_offset--, -1); + } + + while (defined($inserted) && ${$inserted}{'LINENR'} == $old_linenr) { + push(@lines, ${$inserted}{'LINE'}); + $inserted = @{$insertedRef}[$next_insert++]; + $new_linenr++; + fixup_current_range(\$lines[$range_last_linenr], $delta_offset++, 1); + } + + if ($save_line) { + push(@lines, $line); + $new_linenr++; + } + + $old_linenr++; + } + + return @lines; +} + +sub fix_insert_line { + my ($linenr, $line) = @_; + + my $inserted = { + LINENR => $linenr, + LINE => $line, + }; + push(@fixed_inserted, $inserted); +} + +sub fix_delete_line { + my ($linenr, $line) = @_; + + my $deleted = { + LINENR => $linenr, + LINE => $line, + }; + + push(@fixed_deleted, $deleted); +} + +sub ERROR { + my ($type, $msg) = @_; + + if (report("ERROR", $type, $msg)) { + our $clean = 0; + our $cnt_error++; + return 1; + } + return 0; +} +sub WARN { + my ($type, $msg) = @_; + + if (report("WARNING", $type, $msg)) { + our $clean = 0; + our $cnt_warn++; + return 1; + } + return 0; +} +sub CHK { + my ($type, $msg) = @_; + + if ($check && report("CHECK", $type, $msg)) { + our $clean = 0; + our $cnt_chk++; + return 1; + } + return 0; +} + +sub check_absolute_file { + my ($absolute, $herecurr) = @_; + my $file = $absolute; + + ##print "absolute<$absolute>\n"; + + # See if any suffix of this path is a path within the tree. + while ($file =~ s@^[^/]*/@@) { + if (-f "$root/$file") { + ##print "file<$file>\n"; + last; + } + } + if (! -f _) { + return 0; + } + + # It is, so see if the prefix is acceptable. + my $prefix = $absolute; + substr($prefix, -length($file)) = ''; + + ##print "prefix<$prefix>\n"; + if ($prefix ne ".../") { + WARN("USE_RELATIVE_PATH", + "use relative pathname instead of absolute in changelog text\n" . $herecurr); + } +} + +sub trim { + my ($string) = @_; + + $string =~ s/^\s+|\s+$//g; + + return $string; +} + +sub ltrim { + my ($string) = @_; + + $string =~ s/^\s+//; + + return $string; +} + +sub rtrim { + my ($string) = @_; + + $string =~ s/\s+$//; + + return $string; +} + +sub string_find_replace { + my ($string, $find, $replace) = @_; + + $string =~ s/$find/$replace/g; + + return $string; +} + +sub tabify { + my ($leading) = @_; + + my $source_indent = 8; + my $max_spaces_before_tab = $source_indent - 1; + my $spaces_to_tab = " " x $source_indent; + + #convert leading spaces to tabs + 1 while $leading =~ s@^([\t]*)$spaces_to_tab@$1\t@g; + #Remove spaces before a tab + 1 while $leading =~ s@^([\t]*)( {1,$max_spaces_before_tab})\t@$1\t@g; + + return "$leading"; +} + +sub pos_last_openparen { + my ($line) = @_; + + my $pos = 0; + + my $opens = $line =~ tr/\(/\(/; + my $closes = $line =~ tr/\)/\)/; + + my $last_openparen = 0; + + if (($opens == 0) || ($closes >= $opens)) { + return -1; + } + + my $len = length($line); + + for ($pos = 0; $pos < $len; $pos++) { + my $string = substr($line, $pos); + if ($string =~ /^($FuncArg|$balanced_parens)/) { + $pos += length($1) - 1; + } elsif (substr($line, $pos, 1) eq '(') { + $last_openparen = $pos; + } elsif (index($string, '(') == -1) { + last; + } + } + + return length(expand_tabs(substr($line, 0, $last_openparen))) + 1; +} + +sub process { + my $filename = shift; + + my $linenr=0; + my $prevline=""; + my $prevrawline=""; + my $stashline=""; + my $stashrawline=""; + + my $length; + my $indent; + my $previndent=0; + my $stashindent=0; + + our $clean = 1; + my $signoff = 0; + my $is_patch = 0; + my $in_header_lines = $file ? 0 : 1; + my $in_commit_log = 0; #Scanning lines before patch + my $has_commit_log = 0; #Encountered lines before patch + my $commit_log_possible_stack_dump = 0; + my $commit_log_long_line = 0; + my $commit_log_has_diff = 0; + my $reported_maintainer_file = 0; + my $non_utf8_charset = 0; + + my $last_blank_line = 0; + my $last_coalesced_string_linenr = -1; + + our @report = (); + our $cnt_lines = 0; + our $cnt_error = 0; + our $cnt_warn = 0; + our $cnt_chk = 0; + + # Trace the real file/line as we go. + my $realfile = ''; + my $realline = 0; + my $realcnt = 0; + my $here = ''; + my $context_function; #undef'd unless there's a known function + my $in_comment = 0; + my $comment_edge = 0; + my $first_line = 0; + my $p1_prefix = ''; + + my $prev_values = 'E'; + + # suppression flags + my %suppress_ifbraces; + my %suppress_whiletrailers; + my %suppress_export; + my $suppress_statement = 0; + + my %signatures = (); + + # Pre-scan the patch sanitizing the lines. + # Pre-scan the patch looking for any __setup documentation. + # + my @setup_docs = (); + my $setup_docs = 0; + + my $camelcase_file_seeded = 0; + + sanitise_line_reset(); + my $line; + foreach my $rawline (@rawlines) { + $linenr++; + $line = $rawline; + + push(@fixed, $rawline) if ($fix); + + if ($rawline=~/^\+\+\+\s+(\S+)/) { + $setup_docs = 0; + if ($1 =~ m@Documentation/admin-guide/kernel-parameters.rst$@) { + $setup_docs = 1; + } + #next; + } + if ($rawline =~ /^\@\@ -\d+(?:,\d+)? \+(\d+)(,(\d+))? \@\@/) { + $realline=$1-1; + if (defined $2) { + $realcnt=$3+1; + } else { + $realcnt=1+1; + } + $in_comment = 0; + + # Guestimate if this is a continuing comment. Run + # the context looking for a comment "edge". If this + # edge is a close comment then we must be in a comment + # at context start. + my $edge; + my $cnt = $realcnt; + for (my $ln = $linenr + 1; $cnt > 0; $ln++) { + next if (defined $rawlines[$ln - 1] && + $rawlines[$ln - 1] =~ /^-/); + $cnt--; + #print "RAW<$rawlines[$ln - 1]>\n"; + last if (!defined $rawlines[$ln - 1]); + if ($rawlines[$ln - 1] =~ m@(/\*|\*/)@ && + $rawlines[$ln - 1] !~ m@"[^"]*(?:/\*|\*/)[^"]*"@) { + ($edge) = $1; + last; + } + } + if (defined $edge && $edge eq '*/') { + $in_comment = 1; + } + + # Guestimate if this is a continuing comment. If this + # is the start of a diff block and this line starts + # ' *' then it is very likely a comment. + if (!defined $edge && + $rawlines[$linenr] =~ m@^.\s*(?:\*\*+| \*)(?:\s|$)@) + { + $in_comment = 1; + } + + ##print "COMMENT:$in_comment edge<$edge> $rawline\n"; + sanitise_line_reset($in_comment); + + } elsif ($realcnt && $rawline =~ /^(?:\+| |$)/) { + # Standardise the strings and chars within the input to + # simplify matching -- only bother with positive lines. + $line = sanitise_line($rawline); + } + push(@lines, $line); + + if ($realcnt > 1) { + $realcnt-- if ($line =~ /^(?:\+| |$)/); + } else { + $realcnt = 0; + } + + #print "==>$rawline\n"; + #print "-->$line\n"; + + if ($setup_docs && $line =~ /^\+/) { + push(@setup_docs, $line); + } + } + + $prefix = ''; + + $realcnt = 0; + $linenr = 0; + $fixlinenr = -1; + foreach my $line (@lines) { + $linenr++; + $fixlinenr++; + my $sline = $line; #copy of $line + $sline =~ s/$;/ /g; #with comments as spaces + + my $rawline = $rawlines[$linenr - 1]; + +#extract the line range in the file after the patch is applied + if (!$in_commit_log && + $line =~ /^\@\@ -\d+(?:,\d+)? \+(\d+)(,(\d+))? \@\@(.*)/) { + my $context = $4; + $is_patch = 1; + $first_line = $linenr + 1; + $realline=$1-1; + if (defined $2) { + $realcnt=$3+1; + } else { + $realcnt=1+1; + } + annotate_reset(); + $prev_values = 'E'; + + %suppress_ifbraces = (); + %suppress_whiletrailers = (); + %suppress_export = (); + $suppress_statement = 0; + if ($context =~ /\b(\w+)\s*\(/) { + $context_function = $1; + } else { + undef $context_function; + } + next; + +# track the line number as we move through the hunk, note that +# new versions of GNU diff omit the leading space on completely +# blank context lines so we need to count that too. + } elsif ($line =~ /^( |\+|$)/) { + $realline++; + $realcnt-- if ($realcnt != 0); + + # Measure the line length and indent. + ($length, $indent) = line_stats($rawline); + + # Track the previous line. + ($prevline, $stashline) = ($stashline, $line); + ($previndent, $stashindent) = ($stashindent, $indent); + ($prevrawline, $stashrawline) = ($stashrawline, $rawline); + + #warn "line<$line>\n"; + + } elsif ($realcnt == 1) { + $realcnt--; + } + + my $hunk_line = ($realcnt != 0); + + $here = "#$linenr: " if (!$file); + $here = "#$realline: " if ($file); + + my $found_file = 0; + # extract the filename as it passes + if ($line =~ /^diff --git.*?(\S+)$/) { + $realfile = $1; + $realfile =~ s@^([^/]*)/@@ if (!$file); + $in_commit_log = 0; + $found_file = 1; + } elsif ($line =~ /^\+\+\+\s+(\S+)/) { + $realfile = $1; + $realfile =~ s@^([^/]*)/@@ if (!$file); + $in_commit_log = 0; + + $p1_prefix = $1; + if (!$file && $tree && $p1_prefix ne '' && + -e "$root/$p1_prefix") { + WARN("PATCH_PREFIX", + "patch prefix '$p1_prefix' exists, appears to be a -p0 patch\n"); + } + + if ($realfile =~ m@^include/asm/@) { + ERROR("MODIFIED_INCLUDE_ASM", + "do not modify files in include/asm, change architecture specific files in include/asm-<architecture>\n" . "$here$rawline\n"); + } + $found_file = 1; + } + my $skipme = 0; + foreach (@exclude) { + if ($realfile =~ m@^(?:$_/)@) { + $skipme = 1; + } + } + if ($skipme) { + next; + } + +#make up the handle for any error we report on this line + if ($showfile) { + $prefix = "$realfile:$realline: " + } elsif ($emacs) { + if ($file) { + $prefix = "$filename:$realline: "; + } else { + $prefix = "$filename:$linenr: "; + } + } + + if ($found_file) { + if (is_maintained_obsolete($realfile)) { + WARN("OBSOLETE", + "$realfile is marked as 'obsolete' in the MAINTAINERS hierarchy. No unnecessary modifications please.\n"); + } + if ($realfile =~ m@^(?:drivers/net/|net/|drivers/staging/)@) { + $check = 1; + } else { + $check = $check_orig; + } + next; + } + + $here .= "FILE: $realfile:$realline:" if ($realcnt != 0); + + my $hereline = "$here\n$rawline\n"; + my $herecurr = "$here\n$rawline\n"; + my $hereprev = "$here\n$prevrawline\n$rawline\n"; + + $cnt_lines++ if ($realcnt != 0); + +# Check if the commit log has what seems like a diff which can confuse patch + if ($in_commit_log && !$commit_log_has_diff && + (($line =~ m@^\s+diff\b.*a/[\w/]+@ && + $line =~ m@^\s+diff\b.*a/([\w/]+)\s+b/$1\b@) || + $line =~ m@^\s*(?:\-\-\-\s+a/|\+\+\+\s+b/)@ || + $line =~ m/^\s*\@\@ \-\d+,\d+ \+\d+,\d+ \@\@/)) { + ERROR("DIFF_IN_COMMIT_MSG", + "Avoid using diff content in the commit message - patch(1) might not work\n" . $herecurr); + $commit_log_has_diff = 1; + } + +# Check for incorrect file permissions + if ($line =~ /^new (file )?mode.*[7531]\d{0,2}$/) { + my $permhere = $here . "FILE: $realfile\n"; + if ($realfile !~ m@scripts/@ && + $realfile !~ /\.(py|pl|awk|sh)$/) { + ERROR("EXECUTE_PERMISSIONS", + "do not set execute permissions for source files\n" . $permhere); + } + } + +# Check the patch for a signoff: + if ($line =~ /^\s*signed-off-by:/i) { + $signoff++; + $in_commit_log = 0; + } + +# Check if CODEOWNERS is being updated. If so, there's probably no need to +# emit the "does CODEOWNERS need updating?" message on file add/move/delete + if ($line =~ /^\s*CODEOWNERS\s*\|/) { + $reported_maintainer_file = 1; + } + +# Check signature styles + if (!$in_header_lines && + $line =~ /^(\s*)([a-z0-9_-]+by:|$signature_tags)(\s*)(.*)/i) { + my $space_before = $1; + my $sign_off = $2; + my $space_after = $3; + my $email = $4; + my $ucfirst_sign_off = ucfirst(lc($sign_off)); + + if ($sign_off !~ /$signature_tags/) { + WARN("BAD_SIGN_OFF", + "Non-standard signature: $sign_off\n" . $herecurr); + } + if (defined $space_before && $space_before ne "") { + if (WARN("BAD_SIGN_OFF", + "Do not use whitespace before $ucfirst_sign_off\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] = + "$ucfirst_sign_off $email"; + } + } + if ($sign_off =~ /-by:$/i && $sign_off ne $ucfirst_sign_off) { + if (WARN("BAD_SIGN_OFF", + "'$ucfirst_sign_off' is the preferred signature form\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] = + "$ucfirst_sign_off $email"; + } + + } + if (!defined $space_after || $space_after ne " ") { + if (WARN("BAD_SIGN_OFF", + "Use a single space after $ucfirst_sign_off\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] = + "$ucfirst_sign_off $email"; + } + } + + my ($email_name, $email_address, $comment) = parse_email($email); + my $suggested_email = format_email(($email_name, $email_address)); + if ($suggested_email eq "") { + ERROR("BAD_SIGN_OFF", + "Unrecognized email address: '$email'\n" . $herecurr); + } else { + my $dequoted = $suggested_email; + $dequoted =~ s/^"//; + $dequoted =~ s/" </ </; + # Don't force email to have quotes + # Allow just an angle bracketed address + if ("$dequoted$comment" ne $email && + "<$email_address>$comment" ne $email && + "$suggested_email$comment" ne $email) { + WARN("BAD_SIGN_OFF", + "email address '$email' might be better as '$suggested_email$comment'\n" . $herecurr); + } + } + +# Check for duplicate signatures + my $sig_nospace = $line; + $sig_nospace =~ s/\s//g; + $sig_nospace = lc($sig_nospace); + if (defined $signatures{$sig_nospace}) { + WARN("BAD_SIGN_OFF", + "Duplicate signature\n" . $herecurr); + } else { + $signatures{$sig_nospace} = 1; + } + } + +# Check email subject for common tools that don't need to be mentioned + if ($in_header_lines && + $line =~ /^Subject:.*\b(?:checkpatch|sparse|smatch)\b[^:]/i) { + WARN("EMAIL_SUBJECT", + "A patch subject line should describe the change not the tool that found it\n" . $herecurr); + } + +# Check for old stable address + if ($line =~ /^\s*cc:\s*.*<?\bstable\@kernel\.org\b>?.*$/i) { + ERROR("STABLE_ADDRESS", + "The 'stable' address should be 'stable\@vger.kernel.org'\n" . $herecurr); + } + +# Check for unwanted Gerrit info + if ($in_commit_log && $line =~ /^\s*change-id:/i) { + ERROR("GERRIT_CHANGE_ID", + "Remove Gerrit Change-Id's before submitting upstream.\n" . $herecurr); + } + +# Check if the commit log is in a possible stack dump + if ($in_commit_log && !$commit_log_possible_stack_dump && + ($line =~ /^\s*(?:WARNING:|BUG:)/ || + $line =~ /^\s*\[\s*\d+\.\d{6,6}\s*\]/ || + # timestamp + $line =~ /^\s*\[\<[0-9a-fA-F]{8,}\>\]/)) { + # stack dump address + $commit_log_possible_stack_dump = 1; + } + +# Check for line lengths > 75 in commit log, warn once + if ($in_commit_log && !$commit_log_long_line && + length($line) > 75 && + !($line =~ /^\s*[a-zA-Z0-9_\/\.]+\s+\|\s+\d+/ || + # file delta changes + $line =~ /^\s*(?:[\w\.\-]+\/)++[\w\.\-]+:/ || + # filename then : + $line =~ /^\s*(?:Fixes:|Link:)/i || + # A Fixes: or Link: line + $commit_log_possible_stack_dump)) { + WARN("COMMIT_LOG_LONG_LINE", + "Possible unwrapped commit description (prefer a maximum 75 chars per line)\n" . $herecurr); + $commit_log_long_line = 1; + } + +# Reset possible stack dump if a blank line is found + if ($in_commit_log && $commit_log_possible_stack_dump && + $line =~ /^\s*$/) { + $commit_log_possible_stack_dump = 0; + } + +# Check for git id commit length and improperly formed commit descriptions + if ($in_commit_log && !$commit_log_possible_stack_dump && + $line !~ /^\s*(?:Link|Patchwork|http|https|BugLink):/i && + $line !~ /^This reverts commit [0-9a-f]{7,40}/ && + ($line =~ /\bcommit\s+[0-9a-f]{5,}\b/i || + ($line =~ /(?:\s|^)[0-9a-f]{12,40}(?:[\s"'\(\[]|$)/i && + $line !~ /[\<\[][0-9a-f]{12,40}[\>\]]/i && + $line !~ /\bfixes:\s*[0-9a-f]{12,40}/i))) { + my $init_char = "c"; + my $orig_commit = ""; + my $short = 1; + my $long = 0; + my $case = 1; + my $space = 1; + my $hasdesc = 0; + my $hasparens = 0; + my $id = '0123456789ab'; + my $orig_desc = "commit description"; + my $description = ""; + + if ($line =~ /\b(c)ommit\s+([0-9a-f]{5,})\b/i) { + $init_char = $1; + $orig_commit = lc($2); + } elsif ($line =~ /\b([0-9a-f]{12,40})\b/i) { + $orig_commit = lc($1); + } + + $short = 0 if ($line =~ /\bcommit\s+[0-9a-f]{12,40}/i); + $long = 1 if ($line =~ /\bcommit\s+[0-9a-f]{41,}/i); + $space = 0 if ($line =~ /\bcommit [0-9a-f]/i); + $case = 0 if ($line =~ /\b[Cc]ommit\s+[0-9a-f]{5,40}[^A-F]/); + if ($line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("([^"]+)"\)/i) { + $orig_desc = $1; + $hasparens = 1; + } elsif ($line =~ /\bcommit\s+[0-9a-f]{5,}\s*$/i && + defined $rawlines[$linenr] && + $rawlines[$linenr] =~ /^\s*\("([^"]+)"\)/) { + $orig_desc = $1; + $hasparens = 1; + } elsif ($line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("[^"]+$/i && + defined $rawlines[$linenr] && + $rawlines[$linenr] =~ /^\s*[^"]+"\)/) { + $line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("([^"]+)$/i; + $orig_desc = $1; + $rawlines[$linenr] =~ /^\s*([^"]+)"\)/; + $orig_desc .= " " . $1; + $hasparens = 1; + } + + ($id, $description) = git_commit_info($orig_commit, + $id, $orig_desc); + + if (defined($id) && + ($short || $long || $space || $case || ($orig_desc ne $description) || !$hasparens)) { + ERROR("GIT_COMMIT_ID", + "Please use git commit description style 'commit <12+ chars of sha1> (\"<title line>\")' - ie: '${init_char}ommit $id (\"$description\")'\n" . $herecurr); + } + } + +# Check for added, moved or deleted files + if (!$reported_maintainer_file && !$in_commit_log && + ($line =~ /^(?:new|deleted) file mode\s*\d+\s*$/ || + $line =~ /^rename (?:from|to) [\w\/\.\-]+\s*$/ || + ($line =~ /\{\s*([\w\/\.\-]*)\s*\=\>\s*([\w\/\.\-]*)\s*\}/ && + (defined($1) || defined($2))))) { + $is_patch = 1; + $reported_maintainer_file = 1; + WARN("FILE_PATH_CHANGES", + "added, moved or deleted file(s), does CODEOWNERS need updating?\n" . $herecurr); + } + +# Check for wrappage within a valid hunk of the file + if ($realcnt != 0 && $line !~ m{^(?:\+|-| |\\ No newline|$)}) { + ERROR("CORRUPTED_PATCH", + "patch seems to be corrupt (line wrapped?)\n" . + $herecurr) if (!$emitted_corrupt++); + } + +# UTF-8 regex found at http://www.w3.org/International/questions/qa-forms-utf-8.en.php + if (($realfile =~ /^$/ || $line =~ /^\+/) && + $rawline !~ m/^$UTF8*$/) { + my ($utf8_prefix) = ($rawline =~ /^($UTF8*)/); + + my $blank = copy_spacing($rawline); + my $ptr = substr($blank, 0, length($utf8_prefix)) . "^"; + my $hereptr = "$hereline$ptr\n"; + + CHK("INVALID_UTF8", + "Invalid UTF-8, patch and commit message should be encoded in UTF-8\n" . $hereptr); + } + +# Check if it's the start of a commit log +# (not a header line and we haven't seen the patch filename) + if ($in_header_lines && $realfile =~ /^$/ && + !($rawline =~ /^\s+(?:\S|$)/ || + $rawline =~ /^(?:commit\b|from\b|[\w-]+:)/i)) { + $in_header_lines = 0; + $in_commit_log = 1; + $has_commit_log = 1; + } + +# Check if there is UTF-8 in a commit log when a mail header has explicitly +# declined it, i.e defined some charset where it is missing. + if ($in_header_lines && + $rawline =~ /^Content-Type:.+charset="(.+)".*$/ && + $1 !~ /utf-8/i) { + $non_utf8_charset = 1; + } + + if ($in_commit_log && $non_utf8_charset && $realfile =~ /^$/ && + $rawline =~ /$NON_ASCII_UTF8/) { + WARN("UTF8_BEFORE_PATCH", + "8-bit UTF-8 used in possible commit log\n" . $herecurr); + } + +# Check for absolute kernel paths in commit message + if ($tree && $in_commit_log) { + while ($line =~ m{(?:^|\s)(/\S*)}g) { + my $file = $1; + + if ($file =~ m{^(.*?)(?::\d+)+:?$} && + check_absolute_file($1, $herecurr)) { + # + } else { + check_absolute_file($file, $herecurr); + } + } + } + +# Check for various typo / spelling mistakes + if (defined($misspellings) && + ($in_commit_log || $line =~ /^(?:\+|Subject:)/i)) { + while ($rawline =~ /(?:^|[^a-z@])($misspellings)(?:\b|$|[^a-z@])/gi) { + my $typo = $1; + my $typo_fix = $spelling_fix{lc($typo)}; + $typo_fix = ucfirst($typo_fix) if ($typo =~ /^[A-Z]/); + $typo_fix = uc($typo_fix) if ($typo =~ /^[A-Z]+$/); + my $msg_level = \&WARN; + $msg_level = \&CHK if ($file); + if (&{$msg_level}("TYPO_SPELLING", + "'$typo' may be misspelled - perhaps '$typo_fix'?\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/(^|[^A-Za-z@])($typo)($|[^A-Za-z@])/$1$typo_fix$3/; + } + } + } + +# ignore non-hunk lines and lines being removed + next if (!$hunk_line || $line =~ /^-/); + +#trailing whitespace + if ($line =~ /^\+.*\015/) { + my $herevet = "$here\n" . cat_vet($rawline) . "\n"; + if (ERROR("DOS_LINE_ENDINGS", + "DOS line endings\n" . $herevet) && + $fix) { + $fixed[$fixlinenr] =~ s/[\s\015]+$//; + } + } elsif ($rawline =~ /^\+.*\S\s+$/ || $rawline =~ /^\+\s+$/) { + my $herevet = "$here\n" . cat_vet($rawline) . "\n"; + if (ERROR("TRAILING_WHITESPACE", + "trailing whitespace\n" . $herevet) && + $fix) { + $fixed[$fixlinenr] =~ s/\s+$//; + } + + $rpt_cleaners = 1; + } + +# Check for FSF mailing addresses. + if ($rawline =~ /\bwrite to the Free/i || + $rawline =~ /\b675\s+Mass\s+Ave/i || + $rawline =~ /\b59\s+Temple\s+Pl/i || + $rawline =~ /\b51\s+Franklin\s+St/i) { + my $herevet = "$here\n" . cat_vet($rawline) . "\n"; + my $msg_level = \&ERROR; + $msg_level = \&CHK if ($file); + &{$msg_level}("FSF_MAILING_ADDRESS", + "Do not include the paragraph about writing to the Free Software Foundation's mailing address from the sample GPL notice. The FSF has changed addresses in the past, and may do so again. Linux already includes a copy of the GPL.\n" . $herevet) + } + +# check for Kconfig help text having a real description +# Only applies when adding the entry originally, after that we do not have +# sufficient context to determine whether it is indeed long enough. + if ($realfile =~ /Kconfig/ && + $line =~ /^\+\s*config\s+/) { + my $length = 0; + my $cnt = $realcnt; + my $ln = $linenr + 1; + my $f; + my $is_start = 0; + my $is_end = 0; + for (; $cnt > 0 && defined $lines[$ln - 1]; $ln++) { + $f = $lines[$ln - 1]; + $cnt-- if ($lines[$ln - 1] !~ /^-/); + $is_end = $lines[$ln - 1] =~ /^\+/; + + next if ($f =~ /^-/); + last if (!$file && $f =~ /^\@\@/); + + if ($lines[$ln - 1] =~ /^\+\s*(?:bool|tristate)\s*\"/) { + $is_start = 1; + } elsif ($lines[$ln - 1] =~ /^\+\s*(?:---)?help(?:---)?$/) { + $length = -1; + } + + $f =~ s/^.//; + $f =~ s/#.*//; + $f =~ s/^\s+//; + next if ($f =~ /^$/); + if ($f =~ /^\s*config\s/) { + $is_end = 1; + last; + } + $length++; + } + if ($is_start && $is_end && $length < $min_conf_desc_length) { + WARN("CONFIG_DESCRIPTION", + "please write a paragraph that describes the config symbol fully\n" . $herecurr); + } + #print "is_start<$is_start> is_end<$is_end> length<$length>\n"; + } + +# check for MAINTAINERS entries that don't have the right form + if ($realfile =~ /^MAINTAINERS$/ && + $rawline =~ /^\+[A-Z]:/ && + $rawline !~ /^\+[A-Z]:\t\S/) { + if (WARN("MAINTAINERS_STYLE", + "MAINTAINERS entries use one tab after TYPE:\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/^(\+[A-Z]):\s*/$1:\t/; + } + } + +# discourage the use of boolean for type definition attributes of Kconfig options + if ($realfile =~ /Kconfig/ && + $line =~ /^\+\s*\bboolean\b/) { + WARN("CONFIG_TYPE_BOOLEAN", + "Use of boolean is deprecated, please use bool instead.\n" . $herecurr); + } + + if (($realfile =~ /Makefile.*/ || $realfile =~ /Kbuild.*/) && + ($line =~ /\+(EXTRA_[A-Z]+FLAGS).*/)) { + my $flag = $1; + my $replacement = { + 'EXTRA_AFLAGS' => 'asflags-y', + 'EXTRA_CFLAGS' => 'ccflags-y', + 'EXTRA_CPPFLAGS' => 'cppflags-y', + 'EXTRA_LDFLAGS' => 'ldflags-y', + }; + + WARN("DEPRECATED_VARIABLE", + "Use of $flag is deprecated, please use \`$replacement->{$flag} instead.\n" . $herecurr) if ($replacement->{$flag}); + } +# Kconfig use tabs and no spaces in line + if ($realfile =~ /Kconfig/ && $rawline =~ /^\+ /) { + my $herevet = "$here\n" . cat_vet($rawline) . "\n"; + WARN("LEADING_SPACE", + "please, no spaces at the start of a line\n" . $herevet); + } + +# check for DT compatible documentation + if (defined $root && + (($realfile =~ /\.dtsi?$/ && $line =~ /^\+\s*compatible\s*=\s*\"/) || + ($realfile =~ /\.[ch]$/ && $line =~ /^\+.*\.compatible\s*=\s*\"/))) { + + my @compats = $rawline =~ /\"([a-zA-Z0-9\-\,\.\+_]+)\"/g; + + my $dt_path = $root . "/dts/bindings/"; + my $vp_file = $dt_path . "vendor-prefixes.txt"; + + foreach my $compat (@compats) { + my $compat2 = $compat; + $compat2 =~ s/\,[a-zA-Z0-9]*\-/\,<\.\*>\-/; + my $compat3 = $compat; + $compat3 =~ s/\,([a-z]*)[0-9]*\-/\,$1<\.\*>\-/; + `grep -Erq "$compat|$compat2|$compat3" $dt_path`; + if ( $? >> 8 ) { + WARN("UNDOCUMENTED_DT_STRING", + "DT compatible string \"$compat\" appears un-documented -- check $dt_path\n" . $herecurr); + } + + next if $compat !~ /^([a-zA-Z0-9\-]+)\,/; + my $vendor = $1; + `grep -Eq "^$vendor\\b" $vp_file`; + if ( $? >> 8 ) { + WARN("UNDOCUMENTED_DT_STRING", + "DT compatible string vendor \"$vendor\" appears un-documented -- check $vp_file\n" . $herecurr); + } + } + } + +# check we are in a valid source file if not then ignore this hunk + next if ($realfile !~ /\.(h|c|s|S|sh|dtsi|dts)$/); + +# line length limit (with some exclusions) +# +# There are a few types of lines that may extend beyond $max_line_length: +# logging functions like pr_info that end in a string +# lines with a single string +# #defines that are a single string +# +# There are 3 different line length message types: +# LONG_LINE_COMMENT a comment starts before but extends beyond $max_line_length +# LONG_LINE_STRING a string starts before but extends beyond $max_line_length +# LONG_LINE all other lines longer than $max_line_length +# +# if LONG_LINE is ignored, the other 2 types are also ignored +# + + if ($line =~ /^\+/ && $length > $max_line_length) { + my $msg_type = "LONG_LINE"; + + # Check the allowed long line types first + + # logging functions that end in a string that starts + # before $max_line_length + if ($line =~ /^\+\s*$logFunctions\s*\(\s*(?:(?:KERN_\S+\s*|[^"]*))?($String\s*(?:|,|\)\s*;)\s*)$/ && + length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) { + $msg_type = ""; + + # lines with only strings (w/ possible termination) + # #defines with only strings + } elsif ($line =~ /^\+\s*$String\s*(?:\s*|,|\)\s*;)\s*$/ || + $line =~ /^\+\s*#\s*define\s+\w+\s+$String$/) { + $msg_type = ""; + + # EFI_GUID is another special case + } elsif ($line =~ /^\+.*\bEFI_GUID\s*\(/) { + $msg_type = ""; + + # Otherwise set the alternate message types + + # a comment starts before $max_line_length + } elsif ($line =~ /($;[\s$;]*)$/ && + length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) { + $msg_type = "LONG_LINE_COMMENT" + + # a quoted string starts before $max_line_length + } elsif ($sline =~ /\s*($String(?:\s*(?:\\|,\s*|\)\s*;\s*))?)$/ && + length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) { + $msg_type = "LONG_LINE_STRING" + } + + if ($msg_type ne "" && + (show_type("LONG_LINE") || show_type($msg_type))) { + WARN($msg_type, + "line over $max_line_length characters\n" . $herecurr); + } + } + +# check for adding lines without a newline. + if ($line =~ /^\+/ && defined $lines[$linenr] && $lines[$linenr] =~ /^\\ No newline at end of file/) { + WARN("MISSING_EOF_NEWLINE", + "adding a line without newline at end of file\n" . $herecurr); + } + +# Blackfin: use hi/lo macros + if ($realfile =~ m@arch/blackfin/.*\.S$@) { + if ($line =~ /\.[lL][[:space:]]*=.*&[[:space:]]*0x[fF][fF][fF][fF]/) { + my $herevet = "$here\n" . cat_vet($line) . "\n"; + ERROR("LO_MACRO", + "use the LO() macro, not (... & 0xFFFF)\n" . $herevet); + } + if ($line =~ /\.[hH][[:space:]]*=.*>>[[:space:]]*16/) { + my $herevet = "$here\n" . cat_vet($line) . "\n"; + ERROR("HI_MACRO", + "use the HI() macro, not (... >> 16)\n" . $herevet); + } + } + +# check we are in a valid source file C or perl if not then ignore this hunk + next if ($realfile !~ /\.(h|c|pl|dtsi|dts)$/); + +# at the beginning of a line any tabs must come first and anything +# more than 8 must use tabs. + if ($rawline =~ /^\+\s* \t\s*\S/ || + $rawline =~ /^\+\s* \s*/) { + my $herevet = "$here\n" . cat_vet($rawline) . "\n"; + $rpt_cleaners = 1; + if (ERROR("CODE_INDENT", + "code indent should use tabs where possible\n" . $herevet) && + $fix) { + $fixed[$fixlinenr] =~ s/^\+([ \t]+)/"\+" . tabify($1)/e; + } + } + +# check for space before tabs. + if ($rawline =~ /^\+/ && $rawline =~ / \t/) { + my $herevet = "$here\n" . cat_vet($rawline) . "\n"; + if (WARN("SPACE_BEFORE_TAB", + "please, no space before tabs\n" . $herevet) && + $fix) { + while ($fixed[$fixlinenr] =~ + s/(^\+.*) {8,8}\t/$1\t\t/) {} + while ($fixed[$fixlinenr] =~ + s/(^\+.*) +\t/$1\t/) {} + } + } + +# check for && or || at the start of a line + if ($rawline =~ /^\+\s*(&&|\|\|)/) { + CHK("LOGICAL_CONTINUATIONS", + "Logical continuations should be on the previous line\n" . $hereprev); + } + +# check indentation starts on a tab stop + if ($^V && $^V ge 5.10.0 && + $sline =~ /^\+\t+( +)(?:$c90_Keywords\b|\{\s*$|\}\s*(?:else\b|while\b|\s*$))/) { + my $indent = length($1); + if ($indent % 8) { + if (WARN("TABSTOP", + "Statements should start on a tabstop\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s@(^\+\t+) +@$1 . "\t" x ($indent/8)@e; + } + } + } + +# check multi-line statement indentation matches previous line + if ($^V && $^V ge 5.10.0 && + $prevline =~ /^\+([ \t]*)((?:$c90_Keywords(?:\s+if)\s*)|(?:$Declare\s*)?(?:$Ident|\(\s*\*\s*$Ident\s*\))\s*|(?:\*\s*)*$Lval\s*=\s*$Ident\s*)\(.*(\&\&|\|\||,)\s*$/) { + $prevline =~ /^\+(\t*)(.*)$/; + my $oldindent = $1; + my $rest = $2; + + my $pos = pos_last_openparen($rest); + if ($pos >= 0) { + $line =~ /^(\+| )([ \t]*)/; + my $newindent = $2; + + my $goodtabindent = $oldindent . + "\t" x ($pos / 8) . + " " x ($pos % 8); + my $goodspaceindent = $oldindent . " " x $pos; + + if ($newindent ne $goodtabindent && + $newindent ne $goodspaceindent) { + + if (CHK("PARENTHESIS_ALIGNMENT", + "Alignment should match open parenthesis\n" . $hereprev) && + $fix && $line =~ /^\+/) { + $fixed[$fixlinenr] =~ + s/^\+[ \t]*/\+$goodtabindent/; + } + } + } + } + +# check for space after cast like "(int) foo" or "(struct foo) bar" +# avoid checking a few false positives: +# "sizeof(<type>)" or "__alignof__(<type>)" +# function pointer declarations like "(*foo)(int) = bar;" +# structure definitions like "(struct foo) { 0 };" +# multiline macros that define functions +# known attributes or the __attribute__ keyword + if ($line =~ /^\+(.*)\(\s*$Type\s*\)([ \t]++)((?![={]|\\$|$Attribute|__attribute__))/ && + (!defined($1) || $1 !~ /\b(?:sizeof|__alignof__)\s*$/)) { + if (CHK("SPACING", + "No space is necessary after a cast\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/(\(\s*$Type\s*\))[ \t]+/$1/; + } + } + +# Block comment styles +# Networking with an initial /* + if ($realfile =~ m@^(drivers/net/|net/)@ && + $prevrawline =~ /^\+[ \t]*\/\*[ \t]*$/ && + $rawline =~ /^\+[ \t]*\*/ && + $realline > 2) { + WARN("NETWORKING_BLOCK_COMMENT_STYLE", + "networking block comments don't use an empty /* line, use /* Comment...\n" . $hereprev); + } + +# Block comments use * on subsequent lines + if ($prevline =~ /$;[ \t]*$/ && #ends in comment + $prevrawline =~ /^\+.*?\/\*/ && #starting /* + $prevrawline !~ /\*\/[ \t]*$/ && #no trailing */ + $rawline =~ /^\+/ && #line is new + $rawline !~ /^\+[ \t]*\*/) { #no leading * + WARN("BLOCK_COMMENT_STYLE", + "Block comments use * on subsequent lines\n" . $hereprev); + } + +# Block comments use */ on trailing lines + if ($rawline !~ m@^\+[ \t]*\*/[ \t]*$@ && #trailing */ + $rawline !~ m@^\+.*/\*.*\*/[ \t]*$@ && #inline /*...*/ + $rawline !~ m@^\+.*\*{2,}/[ \t]*$@ && #trailing **/ + $rawline =~ m@^\+[ \t]*.+\*\/[ \t]*$@) { #non blank */ + WARN("BLOCK_COMMENT_STYLE", + "Block comments use a trailing */ on a separate line\n" . $herecurr); + } + +# Block comment * alignment + if ($prevline =~ /$;[ \t]*$/ && #ends in comment + $line =~ /^\+[ \t]*$;/ && #leading comment + $rawline =~ /^\+[ \t]*\*/ && #leading * + (($prevrawline =~ /^\+.*?\/\*/ && #leading /* + $prevrawline !~ /\*\/[ \t]*$/) || #no trailing */ + $prevrawline =~ /^\+[ \t]*\*/)) { #leading * + my $oldindent; + $prevrawline =~ m@^\+([ \t]*/?)\*@; + if (defined($1)) { + $oldindent = expand_tabs($1); + } else { + $prevrawline =~ m@^\+(.*/?)\*@; + $oldindent = expand_tabs($1); + } + $rawline =~ m@^\+([ \t]*)\*@; + my $newindent = $1; + $newindent = expand_tabs($newindent); + if (length($oldindent) ne length($newindent)) { + WARN("BLOCK_COMMENT_STYLE", + "Block comments should align the * on each line\n" . $hereprev); + } + } + +# check for missing blank lines after struct/union declarations +# with exceptions for various attributes and macros + if ($prevline =~ /^[\+ ]};?\s*$/ && + $line =~ /^\+/ && + !($line =~ /^\+\s*$/ || + $line =~ /^\+\s*EXPORT_SYMBOL/ || + $line =~ /^\+\s*MODULE_/i || + $line =~ /^\+\s*\#\s*(?:end|elif|else)/ || + $line =~ /^\+[a-z_]*init/ || + $line =~ /^\+\s*(?:static\s+)?[A-Z_]*ATTR/ || + $line =~ /^\+\s*DECLARE/ || + $line =~ /^\+\s*__setup/)) { + if (CHK("LINE_SPACING", + "Please use a blank line after function/struct/union/enum declarations\n" . $hereprev) && + $fix) { + fix_insert_line($fixlinenr, "\+"); + } + } + +# check for multiple consecutive blank lines + if ($prevline =~ /^[\+ ]\s*$/ && + $line =~ /^\+\s*$/ && + $last_blank_line != ($linenr - 1)) { + if (CHK("LINE_SPACING", + "Please don't use multiple blank lines\n" . $hereprev) && + $fix) { + fix_delete_line($fixlinenr, $rawline); + } + + $last_blank_line = $linenr; + } + +# check for missing blank lines after declarations + if ($sline =~ /^\+\s+\S/ && #Not at char 1 + # actual declarations + ($prevline =~ /^\+\s+$Declare\s*$Ident\s*[=,;:\[]/ || + # function pointer declarations + $prevline =~ /^\+\s+$Declare\s*\(\s*\*\s*$Ident\s*\)\s*[=,;:\[\(]/ || + # foo bar; where foo is some local typedef or #define + $prevline =~ /^\+\s+$Ident(?:\s+|\s*\*\s*)$Ident\s*[=,;\[]/ || + # known declaration macros + $prevline =~ /^\+\s+$declaration_macros/) && + # for "else if" which can look like "$Ident $Ident" + !($prevline =~ /^\+\s+$c90_Keywords\b/ || + # other possible extensions of declaration lines + $prevline =~ /(?:$Compare|$Assignment|$Operators)\s*$/ || + # not starting a section or a macro "\" extended line + $prevline =~ /(?:\{\s*|\\)$/) && + # looks like a declaration + !($sline =~ /^\+\s+$Declare\s*$Ident\s*[=,;:\[]/ || + # function pointer declarations + $sline =~ /^\+\s+$Declare\s*\(\s*\*\s*$Ident\s*\)\s*[=,;:\[\(]/ || + # foo bar; where foo is some local typedef or #define + $sline =~ /^\+\s+(?:volatile\s+)?$Ident(?:\s+|\s*\*\s*)$Ident\s*[=,;\[]/ || + # known declaration macros + $sline =~ /^\+\s+$declaration_macros/ || + # start of struct or union or enum + $sline =~ /^\+\s+(?:union|struct|enum|typedef)\b/ || + # start or end of block or continuation of declaration + $sline =~ /^\+\s+(?:$|[\{\}\.\#\"\?\:\(\[])/ || + # bitfield continuation + $sline =~ /^\+\s+$Ident\s*:\s*\d+\s*[,;]/ || + # other possible extensions of declaration lines + $sline =~ /^\+\s+\(?\s*(?:$Compare|$Assignment|$Operators)/) && + # indentation of previous and current line are the same + (($prevline =~ /\+(\s+)\S/) && $sline =~ /^\+$1\S/)) { + if (WARN("LINE_SPACING", + "Missing a blank line after declarations\n" . $hereprev) && + $fix) { + fix_insert_line($fixlinenr, "\+"); + } + } + +# check for spaces at the beginning of a line. +# Exceptions: +# 1) within comments +# 2) indented preprocessor commands +# 3) hanging labels + if ($rawline =~ /^\+ / && $line !~ /^\+ *(?:$;|#|$Ident:)/) { + my $herevet = "$here\n" . cat_vet($rawline) . "\n"; + if (WARN("LEADING_SPACE", + "please, no spaces at the start of a line\n" . $herevet) && + $fix) { + $fixed[$fixlinenr] =~ s/^\+([ \t]+)/"\+" . tabify($1)/e; + } + } + +# check we are in a valid C source file if not then ignore this hunk + next if ($realfile !~ /\.(h|c)$/); + +# check if this appears to be the start function declaration, save the name + if ($sline =~ /^\+\{\s*$/ && + $prevline =~ /^\+(?:(?:(?:$Storage|$Inline)\s*)*\s*$Type\s*)?($Ident)\(/) { + $context_function = $1; + } + +# check if this appears to be the end of function declaration + if ($sline =~ /^\+\}\s*$/) { + undef $context_function; + } + +# check indentation of any line with a bare else +# (but not if it is a multiple line "if (foo) return bar; else return baz;") +# if the previous line is a break or return and is indented 1 tab more... + if ($sline =~ /^\+([\t]+)(?:}[ \t]*)?else(?:[ \t]*{)?\s*$/) { + my $tabs = length($1) + 1; + if ($prevline =~ /^\+\t{$tabs,$tabs}break\b/ || + ($prevline =~ /^\+\t{$tabs,$tabs}return\b/ && + defined $lines[$linenr] && + $lines[$linenr] !~ /^[ \+]\t{$tabs,$tabs}return/)) { + WARN("UNNECESSARY_ELSE", + "else is not generally useful after a break or return\n" . $hereprev); + } + } + +# check indentation of a line with a break; +# if the previous line is a goto or return and is indented the same # of tabs + if ($sline =~ /^\+([\t]+)break\s*;\s*$/) { + my $tabs = $1; + if ($prevline =~ /^\+$tabs(?:goto|return)\b/) { + WARN("UNNECESSARY_BREAK", + "break is not useful after a goto or return\n" . $hereprev); + } + } + +# check for RCS/CVS revision markers + if ($rawline =~ /^\+.*\$(Revision|Log|Id)(?:\$|)/) { + WARN("CVS_KEYWORD", + "CVS style keyword markers, these will _not_ be updated\n". $herecurr); + } + +# Blackfin: don't use __builtin_bfin_[cs]sync + if ($line =~ /__builtin_bfin_csync/) { + my $herevet = "$here\n" . cat_vet($line) . "\n"; + ERROR("CSYNC", + "use the CSYNC() macro in asm/blackfin.h\n" . $herevet); + } + if ($line =~ /__builtin_bfin_ssync/) { + my $herevet = "$here\n" . cat_vet($line) . "\n"; + ERROR("SSYNC", + "use the SSYNC() macro in asm/blackfin.h\n" . $herevet); + } + +# check for old HOTPLUG __dev<foo> section markings + if ($line =~ /\b(__dev(init|exit)(data|const|))\b/) { + WARN("HOTPLUG_SECTION", + "Using $1 is unnecessary\n" . $herecurr); + } + +# Check for potential 'bare' types + my ($stat, $cond, $line_nr_next, $remain_next, $off_next, + $realline_next); +#print "LINE<$line>\n"; + if ($linenr > $suppress_statement && + $realcnt && $sline =~ /.\s*\S/) { + ($stat, $cond, $line_nr_next, $remain_next, $off_next) = + ctx_statement_block($linenr, $realcnt, 0); + $stat =~ s/\n./\n /g; + $cond =~ s/\n./\n /g; + +#print "linenr<$linenr> <$stat>\n"; + # If this statement has no statement boundaries within + # it there is no point in retrying a statement scan + # until we hit end of it. + my $frag = $stat; $frag =~ s/;+\s*$//; + if ($frag !~ /(?:{|;)/) { +#print "skip<$line_nr_next>\n"; + $suppress_statement = $line_nr_next; + } + + # Find the real next line. + $realline_next = $line_nr_next; + if (defined $realline_next && + (!defined $lines[$realline_next - 1] || + substr($lines[$realline_next - 1], $off_next) =~ /^\s*$/)) { + $realline_next++; + } + + my $s = $stat; + $s =~ s/{.*$//s; + + # Ignore goto labels. + if ($s =~ /$Ident:\*$/s) { + + # Ignore functions being called + } elsif ($s =~ /^.\s*$Ident\s*\(/s) { + + } elsif ($s =~ /^.\s*else\b/s) { + + # declarations always start with types + } elsif ($prev_values eq 'E' && $s =~ /^.\s*(?:$Storage\s+)?(?:$Inline\s+)?(?:const\s+)?((?:\s*$Ident)+?)\b(?:\s+$Sparse)?\s*\**\s*(?:$Ident|\(\*[^\)]*\))(?:\s*$Modifier)?\s*(?:;|=|,|\()/s) { + my $type = $1; + $type =~ s/\s+/ /g; + possible($type, "A:" . $s); + + # definitions in global scope can only start with types + } elsif ($s =~ /^.(?:$Storage\s+)?(?:$Inline\s+)?(?:const\s+)?($Ident)\b\s*(?!:)/s) { + possible($1, "B:" . $s); + } + + # any (foo ... *) is a pointer cast, and foo is a type + while ($s =~ /\(($Ident)(?:\s+$Sparse)*[\s\*]+\s*\)/sg) { + possible($1, "C:" . $s); + } + + # Check for any sort of function declaration. + # int foo(something bar, other baz); + # void (*store_gdt)(x86_descr_ptr *); + if ($prev_values eq 'E' && $s =~ /^(.(?:typedef\s*)?(?:(?:$Storage|$Inline)\s*)*\s*$Type\s*(?:\b$Ident|\(\*\s*$Ident\))\s*)\(/s) { + my ($name_len) = length($1); + + my $ctx = $s; + substr($ctx, 0, $name_len + 1, ''); + $ctx =~ s/\)[^\)]*$//; + + for my $arg (split(/\s*,\s*/, $ctx)) { + if ($arg =~ /^(?:const\s+)?($Ident)(?:\s+$Sparse)*\s*\**\s*(:?\b$Ident)?$/s || $arg =~ /^($Ident)$/s) { + + possible($1, "D:" . $s); + } + } + } + + } + +# +# Checks which may be anchored in the context. +# + +# Check for switch () and associated case and default +# statements should be at the same indent. + if ($line=~/\bswitch\s*\(.*\)/) { + my $err = ''; + my $sep = ''; + my @ctx = ctx_block_outer($linenr, $realcnt); + shift(@ctx); + for my $ctx (@ctx) { + my ($clen, $cindent) = line_stats($ctx); + if ($ctx =~ /^\+\s*(case\s+|default:)/ && + $indent != $cindent) { + $err .= "$sep$ctx\n"; + $sep = ''; + } else { + $sep = "[...]\n"; + } + } + if ($err ne '') { + ERROR("SWITCH_CASE_INDENT_LEVEL", + "switch and case should be at the same indent\n$hereline$err"); + } + } + +# if/while/etc brace do not go on next line, unless defining a do while loop, +# or if that brace on the next line is for something else + if ($line =~ /(.*)\b((?:if|while|for|switch|(?:[a-z_]+|)for_each[a-z_]+)\s*\(|do\b|else\b)/ && $line !~ /^.\s*\#/) { + my $pre_ctx = "$1$2"; + + my ($level, @ctx) = ctx_statement_level($linenr, $realcnt, 0); + + if ($line =~ /^\+\t{6,}/) { + WARN("DEEP_INDENTATION", + "Too many leading tabs - consider code refactoring\n" . $herecurr); + } + + my $ctx_cnt = $realcnt - $#ctx - 1; + my $ctx = join("\n", @ctx); + + my $ctx_ln = $linenr; + my $ctx_skip = $realcnt; + + while ($ctx_skip > $ctx_cnt || ($ctx_skip == $ctx_cnt && + defined $lines[$ctx_ln - 1] && + $lines[$ctx_ln - 1] =~ /^-/)) { + ##print "SKIP<$ctx_skip> CNT<$ctx_cnt>\n"; + $ctx_skip-- if (!defined $lines[$ctx_ln - 1] || $lines[$ctx_ln - 1] !~ /^-/); + $ctx_ln++; + } + + #print "realcnt<$realcnt> ctx_cnt<$ctx_cnt>\n"; + #print "pre<$pre_ctx>\nline<$line>\nctx<$ctx>\nnext<$lines[$ctx_ln - 1]>\n"; + + if ($ctx !~ /{\s*/ && defined($lines[$ctx_ln - 1]) && $lines[$ctx_ln - 1] =~ /^\+\s*{/) { + ERROR("OPEN_BRACE", + "that open brace { should be on the previous line\n" . + "$here\n$ctx\n$rawlines[$ctx_ln - 1]\n"); + } + if ($level == 0 && $pre_ctx !~ /}\s*while\s*\($/ && + $ctx =~ /\)\s*\;\s*$/ && + defined $lines[$ctx_ln - 1]) + { + my ($nlength, $nindent) = line_stats($lines[$ctx_ln - 1]); + if ($nindent > $indent) { + WARN("TRAILING_SEMICOLON", + "trailing semicolon indicates no statements, indent implies otherwise\n" . + "$here\n$ctx\n$rawlines[$ctx_ln - 1]\n"); + } + } + } + +# Check relative indent for conditionals and blocks. + if ($line =~ /\b(?:(?:if|while|for|(?:[a-z_]+|)for_each[a-z_]+)\s*\(|(?:do|else)\b)/ && $line !~ /^.\s*#/ && $line !~ /\}\s*while\s*/) { + ($stat, $cond, $line_nr_next, $remain_next, $off_next) = + ctx_statement_block($linenr, $realcnt, 0) + if (!defined $stat); + my ($s, $c) = ($stat, $cond); + + substr($s, 0, length($c), ''); + + # remove inline comments + $s =~ s/$;/ /g; + $c =~ s/$;/ /g; + + # Find out how long the conditional actually is. + my @newlines = ($c =~ /\n/gs); + my $cond_lines = 1 + $#newlines; + + # Make sure we remove the line prefixes as we have + # none on the first line, and are going to readd them + # where necessary. + $s =~ s/\n./\n/gs; + while ($s =~ /\n\s+\\\n/) { + $cond_lines += $s =~ s/\n\s+\\\n/\n/g; + } + + # We want to check the first line inside the block + # starting at the end of the conditional, so remove: + # 1) any blank line termination + # 2) any opening brace { on end of the line + # 3) any do (...) { + my $continuation = 0; + my $check = 0; + $s =~ s/^.*\bdo\b//; + $s =~ s/^\s*{//; + if ($s =~ s/^\s*\\//) { + $continuation = 1; + } + if ($s =~ s/^\s*?\n//) { + $check = 1; + $cond_lines++; + } + + # Also ignore a loop construct at the end of a + # preprocessor statement. + if (($prevline =~ /^.\s*#\s*define\s/ || + $prevline =~ /\\\s*$/) && $continuation == 0) { + $check = 0; + } + + my $cond_ptr = -1; + $continuation = 0; + while ($cond_ptr != $cond_lines) { + $cond_ptr = $cond_lines; + + # If we see an #else/#elif then the code + # is not linear. + if ($s =~ /^\s*\#\s*(?:else|elif)/) { + $check = 0; + } + + # Ignore: + # 1) blank lines, they should be at 0, + # 2) preprocessor lines, and + # 3) labels. + if ($continuation || + $s =~ /^\s*?\n/ || + $s =~ /^\s*#\s*?/ || + $s =~ /^\s*$Ident\s*:/) { + $continuation = ($s =~ /^.*?\\\n/) ? 1 : 0; + if ($s =~ s/^.*?\n//) { + $cond_lines++; + } + } + } + + my (undef, $sindent) = line_stats("+" . $s); + my $stat_real = raw_line($linenr, $cond_lines); + + # Check if either of these lines are modified, else + # this is not this patch's fault. + if (!defined($stat_real) || + $stat !~ /^\+/ && $stat_real !~ /^\+/) { + $check = 0; + } + if (defined($stat_real) && $cond_lines > 1) { + $stat_real = "[...]\n$stat_real"; + } + + #print "line<$line> prevline<$prevline> indent<$indent> sindent<$sindent> check<$check> continuation<$continuation> s<$s> cond_lines<$cond_lines> stat_real<$stat_real> stat<$stat>\n"; + + if ($check && $s ne '' && + (($sindent % 8) != 0 || + ($sindent < $indent) || + ($sindent == $indent && + ($s !~ /^\s*(?:\}|\{|else\b)/)) || + ($sindent > $indent + 8))) { + WARN("SUSPECT_CODE_INDENT", + "suspect code indent for conditional statements ($indent, $sindent)\n" . $herecurr . "$stat_real\n"); + } + } + + # Track the 'values' across context and added lines. + my $opline = $line; $opline =~ s/^./ /; + my ($curr_values, $curr_vars) = + annotate_values($opline . "\n", $prev_values); + $curr_values = $prev_values . $curr_values; + if ($dbg_values) { + my $outline = $opline; $outline =~ s/\t/ /g; + print "$linenr > .$outline\n"; + print "$linenr > $curr_values\n"; + print "$linenr > $curr_vars\n"; + } + $prev_values = substr($curr_values, -1); + +#ignore lines not being added + next if ($line =~ /^[^\+]/); + +# check for dereferences that span multiple lines + if ($prevline =~ /^\+.*$Lval\s*(?:\.|->)\s*$/ && + $line =~ /^\+\s*(?!\#\s*(?!define\s+|if))\s*$Lval/) { + $prevline =~ /($Lval\s*(?:\.|->))\s*$/; + my $ref = $1; + $line =~ /^.\s*($Lval)/; + $ref .= $1; + $ref =~ s/\s//g; + WARN("MULTILINE_DEREFERENCE", + "Avoid multiple line dereference - prefer '$ref'\n" . $hereprev); + } + +# check for declarations of signed or unsigned without int + while ($line =~ m{\b($Declare)\s*(?!char\b|short\b|int\b|long\b)\s*($Ident)?\s*[=,;\[\)\(]}g) { + my $type = $1; + my $var = $2; + $var = "" if (!defined $var); + if ($type =~ /^(?:(?:$Storage|$Inline|$Attribute)\s+)*((?:un)?signed)((?:\s*\*)*)\s*$/) { + my $sign = $1; + my $pointer = $2; + + $pointer = "" if (!defined $pointer); + + if (WARN("UNSPECIFIED_INT", + "Prefer '" . trim($sign) . " int" . rtrim($pointer) . "' to bare use of '$sign" . rtrim($pointer) . "'\n" . $herecurr) && + $fix) { + my $decl = trim($sign) . " int "; + my $comp_pointer = $pointer; + $comp_pointer =~ s/\s//g; + $decl .= $comp_pointer; + $decl = rtrim($decl) if ($var eq ""); + $fixed[$fixlinenr] =~ s@\b$sign\s*\Q$pointer\E\s*$var\b@$decl$var@; + } + } + } + +# TEST: allow direct testing of the type matcher. + if ($dbg_type) { + if ($line =~ /^.\s*$Declare\s*$/) { + ERROR("TEST_TYPE", + "TEST: is type\n" . $herecurr); + } elsif ($dbg_type > 1 && $line =~ /^.+($Declare)/) { + ERROR("TEST_NOT_TYPE", + "TEST: is not type ($1 is)\n". $herecurr); + } + next; + } +# TEST: allow direct testing of the attribute matcher. + if ($dbg_attr) { + if ($line =~ /^.\s*$Modifier\s*$/) { + ERROR("TEST_ATTR", + "TEST: is attr\n" . $herecurr); + } elsif ($dbg_attr > 1 && $line =~ /^.+($Modifier)/) { + ERROR("TEST_NOT_ATTR", + "TEST: is not attr ($1 is)\n". $herecurr); + } + next; + } + +# check for initialisation to aggregates open brace on the next line + if ($line =~ /^.\s*{/ && + $prevline =~ /(?:^|[^=])=\s*$/) { + if (ERROR("OPEN_BRACE", + "that open brace { should be on the previous line\n" . $hereprev) && + $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { + fix_delete_line($fixlinenr - 1, $prevrawline); + fix_delete_line($fixlinenr, $rawline); + my $fixedline = $prevrawline; + $fixedline =~ s/\s*=\s*$/ = {/; + fix_insert_line($fixlinenr, $fixedline); + $fixedline = $line; + $fixedline =~ s/^(.\s*)\{\s*/$1/; + fix_insert_line($fixlinenr, $fixedline); + } + } + +# +# Checks which are anchored on the added line. +# + +# check for malformed paths in #include statements (uses RAW line) + if ($rawline =~ m{^.\s*\#\s*include\s+[<"](.*)[">]}) { + my $path = $1; + if ($path =~ m{//}) { + ERROR("MALFORMED_INCLUDE", + "malformed #include filename\n" . $herecurr); + } + if ($path =~ "^uapi/" && $realfile =~ m@\binclude/uapi/@) { + ERROR("UAPI_INCLUDE", + "No #include in ...include/uapi/... should use a uapi/ path prefix\n" . $herecurr); + } + } + +# no C99 // comments + if ($line =~ m{//}) { + if (ERROR("C99_COMMENTS", + "do not use C99 // comments\n" . $herecurr) && + $fix) { + my $line = $fixed[$fixlinenr]; + if ($line =~ /\/\/(.*)$/) { + my $comment = trim($1); + $fixed[$fixlinenr] =~ s@\/\/(.*)$@/\* $comment \*/@; + } + } + } + # Remove C99 comments. + $line =~ s@//.*@@; + $opline =~ s@//.*@@; + +# EXPORT_SYMBOL should immediately follow the thing it is exporting, consider +# the whole statement. +#print "APW <$lines[$realline_next - 1]>\n"; + if (defined $realline_next && + exists $lines[$realline_next - 1] && + !defined $suppress_export{$realline_next} && + ($lines[$realline_next - 1] =~ /EXPORT_SYMBOL.*\((.*)\)/ || + $lines[$realline_next - 1] =~ /EXPORT_UNUSED_SYMBOL.*\((.*)\)/)) { + # Handle definitions which produce identifiers with + # a prefix: + # XXX(foo); + # EXPORT_SYMBOL(something_foo); + my $name = $1; + if ($stat =~ /^(?:.\s*}\s*\n)?.([A-Z_]+)\s*\(\s*($Ident)/ && + $name =~ /^${Ident}_$2/) { +#print "FOO C name<$name>\n"; + $suppress_export{$realline_next} = 1; + + } elsif ($stat !~ /(?: + \n.}\s*$| + ^.DEFINE_$Ident\(\Q$name\E\)| + ^.DECLARE_$Ident\(\Q$name\E\)| + ^.LIST_HEAD\(\Q$name\E\)| + ^.(?:$Storage\s+)?$Type\s*\(\s*\*\s*\Q$name\E\s*\)\s*\(| + \b\Q$name\E(?:\s+$Attribute)*\s*(?:;|=|\[|\() + )/x) { +#print "FOO A<$lines[$realline_next - 1]> stat<$stat> name<$name>\n"; + $suppress_export{$realline_next} = 2; + } else { + $suppress_export{$realline_next} = 1; + } + } + if (!defined $suppress_export{$linenr} && + $prevline =~ /^.\s*$/ && + ($line =~ /EXPORT_SYMBOL.*\((.*)\)/ || + $line =~ /EXPORT_UNUSED_SYMBOL.*\((.*)\)/)) { +#print "FOO B <$lines[$linenr - 1]>\n"; + $suppress_export{$linenr} = 2; + } + if (defined $suppress_export{$linenr} && + $suppress_export{$linenr} == 2) { + WARN("EXPORT_SYMBOL", + "EXPORT_SYMBOL(foo); should immediately follow its function/variable\n" . $herecurr); + } + +# check for global initialisers. + if ($line =~ /^\+$Type\s*$Ident(?:\s+$Modifier)*\s*=\s*($zero_initializer)\s*;/) { + if (ERROR("GLOBAL_INITIALISERS", + "do not initialise globals to $1\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/(^.$Type\s*$Ident(?:\s+$Modifier)*)\s*=\s*$zero_initializer\s*;/$1;/; + } + } +# check for static initialisers. + if ($line =~ /^\+.*\bstatic\s.*=\s*($zero_initializer)\s*;/) { + if (ERROR("INITIALISED_STATIC", + "do not initialise statics to $1\n" . + $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/(\bstatic\s.*?)\s*=\s*$zero_initializer\s*;/$1;/; + } + } + +# check for misordered declarations of char/short/int/long with signed/unsigned + while ($sline =~ m{(\b$TypeMisordered\b)}g) { + my $tmp = trim($1); + WARN("MISORDERED_TYPE", + "type '$tmp' should be specified in [[un]signed] [short|int|long|long long] order\n" . $herecurr); + } + +# check for static const char * arrays. + if ($line =~ /\bstatic\s+const\s+char\s*\*\s*(\w+)\s*\[\s*\]\s*=\s*/) { + WARN("STATIC_CONST_CHAR_ARRAY", + "static const char * array should probably be static const char * const\n" . + $herecurr); + } + +# check for static char foo[] = "bar" declarations. + if ($line =~ /\bstatic\s+char\s+(\w+)\s*\[\s*\]\s*=\s*"/) { + WARN("STATIC_CONST_CHAR_ARRAY", + "static char array declaration should probably be static const char\n" . + $herecurr); + } + +# check for const <foo> const where <foo> is not a pointer or array type + if ($sline =~ /\bconst\s+($BasicType)\s+const\b/) { + my $found = $1; + if ($sline =~ /\bconst\s+\Q$found\E\s+const\b\s*\*/) { + WARN("CONST_CONST", + "'const $found const *' should probably be 'const $found * const'\n" . $herecurr); + } elsif ($sline !~ /\bconst\s+\Q$found\E\s+const\s+\w+\s*\[/) { + WARN("CONST_CONST", + "'const $found const' should probably be 'const $found'\n" . $herecurr); + } + } + +# check for non-global char *foo[] = {"bar", ...} declarations. + if ($line =~ /^.\s+(?:static\s+|const\s+)?char\s+\*\s*\w+\s*\[\s*\]\s*=\s*\{/) { + WARN("STATIC_CONST_CHAR_ARRAY", + "char * array declaration might be better as static const\n" . + $herecurr); + } + +# check for sizeof(foo)/sizeof(foo[0]) that could be ARRAY_SIZE(foo) + if ($line =~ m@\bsizeof\s*\(\s*($Lval)\s*\)@) { + my $array = $1; + if ($line =~ m@\b(sizeof\s*\(\s*\Q$array\E\s*\)\s*/\s*sizeof\s*\(\s*\Q$array\E\s*\[\s*0\s*\]\s*\))@) { + my $array_div = $1; + if (WARN("ARRAY_SIZE", + "Prefer ARRAY_SIZE($array)\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\Q$array_div\E/ARRAY_SIZE($array)/; + } + } + } + +# check for function declarations without arguments like "int foo()" + if ($line =~ /(\b$Type\s+$Ident)\s*\(\s*\)/) { + if (ERROR("FUNCTION_WITHOUT_ARGS", + "Bad function definition - $1() should probably be $1(void)\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/(\b($Type)\s+($Ident))\s*\(\s*\)/$2 $3(void)/; + } + } + +# check for new typedefs, only function parameters and sparse annotations +# make sense. + if ($line =~ /\btypedef\s/ && + $line !~ /\btypedef\s+$Type\s*\(\s*\*?$Ident\s*\)\s*\(/ && + $line !~ /\btypedef\s+$Type\s+$Ident\s*\(/ && + $line !~ /\b$typeTypedefs\b/ && + $line !~ /\b__bitwise\b/) { + WARN("NEW_TYPEDEFS", + "do not add new typedefs\n" . $herecurr); + } + +# * goes on variable not on type + # (char*[ const]) + while ($line =~ m{(\($NonptrType(\s*(?:$Modifier\b\s*|\*\s*)+)\))}g) { + #print "AA<$1>\n"; + my ($ident, $from, $to) = ($1, $2, $2); + + # Should start with a space. + $to =~ s/^(\S)/ $1/; + # Should not end with a space. + $to =~ s/\s+$//; + # '*'s should not have spaces between. + while ($to =~ s/\*\s+\*/\*\*/) { + } + +## print "1: from<$from> to<$to> ident<$ident>\n"; + if ($from ne $to) { + if (ERROR("POINTER_LOCATION", + "\"(foo$from)\" should be \"(foo$to)\"\n" . $herecurr) && + $fix) { + my $sub_from = $ident; + my $sub_to = $ident; + $sub_to =~ s/\Q$from\E/$to/; + $fixed[$fixlinenr] =~ + s@\Q$sub_from\E@$sub_to@; + } + } + } + while ($line =~ m{(\b$NonptrType(\s*(?:$Modifier\b\s*|\*\s*)+)($Ident))}g) { + #print "BB<$1>\n"; + my ($match, $from, $to, $ident) = ($1, $2, $2, $3); + + # Should start with a space. + $to =~ s/^(\S)/ $1/; + # Should not end with a space. + $to =~ s/\s+$//; + # '*'s should not have spaces between. + while ($to =~ s/\*\s+\*/\*\*/) { + } + # Modifiers should have spaces. + $to =~ s/(\b$Modifier$)/$1 /; + +## print "2: from<$from> to<$to> ident<$ident>\n"; + if ($from ne $to && $ident !~ /^$Modifier$/) { + if (ERROR("POINTER_LOCATION", + "\"foo${from}bar\" should be \"foo${to}bar\"\n" . $herecurr) && + $fix) { + + my $sub_from = $match; + my $sub_to = $match; + $sub_to =~ s/\Q$from\E/$to/; + $fixed[$fixlinenr] =~ + s@\Q$sub_from\E@$sub_to@; + } + } + } + +# avoid BUG() or BUG_ON() + if ($line =~ /\b(?:BUG|BUG_ON)\b/) { + my $msg_level = \&WARN; + $msg_level = \&CHK if ($file); + &{$msg_level}("AVOID_BUG", + "Avoid crashing the kernel - try using WARN_ON & recovery code rather than BUG() or BUG_ON()\n" . $herecurr); + } + +# avoid LINUX_VERSION_CODE + if ($line =~ /\bLINUX_VERSION_CODE\b/) { + WARN("LINUX_VERSION_CODE", + "LINUX_VERSION_CODE should be avoided, code should be for the version to which it is merged\n" . $herecurr); + } + +# check for uses of printk_ratelimit + if ($line =~ /\bprintk_ratelimit\s*\(/) { + WARN("PRINTK_RATELIMITED", + "Prefer printk_ratelimited or pr_<level>_ratelimited to printk_ratelimit\n" . $herecurr); + } + +# printk should use KERN_* levels. Note that follow on printk's on the +# same line do not need a level, so we use the current block context +# to try and find and validate the current printk. In summary the current +# printk includes all preceding printk's which have no newline on the end. +# we assume the first bad printk is the one to report. + if ($line =~ /\bprintk\((?!KERN_)\s*"/) { + my $ok = 0; + for (my $ln = $linenr - 1; $ln >= $first_line; $ln--) { + #print "CHECK<$lines[$ln - 1]\n"; + # we have a preceding printk if it ends + # with "\n" ignore it, else it is to blame + if ($lines[$ln - 1] =~ m{\bprintk\(}) { + if ($rawlines[$ln - 1] !~ m{\\n"}) { + $ok = 1; + } + last; + } + } + if ($ok == 0) { + WARN("PRINTK_WITHOUT_KERN_LEVEL", + "printk() should include KERN_ facility level\n" . $herecurr); + } + } + + if ($line =~ /\bprintk\s*\(\s*KERN_([A-Z]+)/) { + my $orig = $1; + my $level = lc($orig); + $level = "warn" if ($level eq "warning"); + my $level2 = $level; + $level2 = "dbg" if ($level eq "debug"); + WARN("PREFER_PR_LEVEL", + "Prefer [subsystem eg: netdev]_$level2([subsystem]dev, ... then dev_$level2(dev, ... then pr_$level(... to printk(KERN_$orig ...\n" . $herecurr); + } + + if ($line =~ /\bpr_warning\s*\(/) { + if (WARN("PREFER_PR_LEVEL", + "Prefer pr_warn(... to pr_warning(...\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/\bpr_warning\b/pr_warn/; + } + } + + if ($line =~ /\bdev_printk\s*\(\s*KERN_([A-Z]+)/) { + my $orig = $1; + my $level = lc($orig); + $level = "warn" if ($level eq "warning"); + $level = "dbg" if ($level eq "debug"); + WARN("PREFER_DEV_LEVEL", + "Prefer dev_$level(... to dev_printk(KERN_$orig, ...\n" . $herecurr); + } + +# ENOSYS means "bad syscall nr" and nothing else. This will have a small +# number of false positives, but assembly files are not checked, so at +# least the arch entry code will not trigger this warning. + if ($line =~ /\bENOSYS\b/) { + WARN("ENOSYS", + "ENOSYS means 'invalid syscall nr' and nothing else\n" . $herecurr); + } + +# function brace can't be on same line, except for #defines of do while, +# or if closed on same line + if (($line=~/$Type\s*$Ident\(.*\).*\s*{/) and + !($line=~/\#\s*define.*do\s\{/) and !($line=~/}/)) { + if (ERROR("OPEN_BRACE", + "open brace '{' following function declarations go on the next line\n" . $herecurr) && + $fix) { + fix_delete_line($fixlinenr, $rawline); + my $fixed_line = $rawline; + $fixed_line =~ /(^..*$Type\s*$Ident\(.*\)\s*){(.*)$/; + my $line1 = $1; + my $line2 = $2; + fix_insert_line($fixlinenr, ltrim($line1)); + fix_insert_line($fixlinenr, "\+{"); + if ($line2 !~ /^\s*$/) { + fix_insert_line($fixlinenr, "\+\t" . trim($line2)); + } + } + } + +# open braces for enum, union and struct go on the same line. + if ($line =~ /^.\s*{/ && + $prevline =~ /^.\s*(?:typedef\s+)?(enum|union|struct)(?:\s+$Ident)?\s*$/) { + if (ERROR("OPEN_BRACE", + "open brace '{' following $1 go on the same line\n" . $hereprev) && + $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { + fix_delete_line($fixlinenr - 1, $prevrawline); + fix_delete_line($fixlinenr, $rawline); + my $fixedline = rtrim($prevrawline) . " {"; + fix_insert_line($fixlinenr, $fixedline); + $fixedline = $rawline; + $fixedline =~ s/^(.\s*)\{\s*/$1\t/; + if ($fixedline !~ /^\+\s*$/) { + fix_insert_line($fixlinenr, $fixedline); + } + } + } + +# missing space after union, struct or enum definition + if ($line =~ /^.\s*(?:typedef\s+)?(enum|union|struct)(?:\s+$Ident){1,2}[=\{]/) { + if (WARN("SPACING", + "missing space after $1 definition\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/^(.\s*(?:typedef\s+)?(?:enum|union|struct)(?:\s+$Ident){1,2})([=\{])/$1 $2/; + } + } + +# Function pointer declarations +# check spacing between type, funcptr, and args +# canonical declaration is "type (*funcptr)(args...)" + if ($line =~ /^.\s*($Declare)\((\s*)\*(\s*)($Ident)(\s*)\)(\s*)\(/) { + my $declare = $1; + my $pre_pointer_space = $2; + my $post_pointer_space = $3; + my $funcname = $4; + my $post_funcname_space = $5; + my $pre_args_space = $6; + +# the $Declare variable will capture all spaces after the type +# so check it for a missing trailing missing space but pointer return types +# don't need a space so don't warn for those. + my $post_declare_space = ""; + if ($declare =~ /(\s+)$/) { + $post_declare_space = $1; + $declare = rtrim($declare); + } + if ($declare !~ /\*$/ && $post_declare_space =~ /^$/) { + WARN("SPACING", + "missing space after return type\n" . $herecurr); + $post_declare_space = " "; + } + +# unnecessary space "type (*funcptr)(args...)" +# This test is not currently implemented because these declarations are +# equivalent to +# int foo(int bar, ...) +# and this is form shouldn't/doesn't generate a checkpatch warning. +# +# elsif ($declare =~ /\s{2,}$/) { +# WARN("SPACING", +# "Multiple spaces after return type\n" . $herecurr); +# } + +# unnecessary space "type ( *funcptr)(args...)" + if (defined $pre_pointer_space && + $pre_pointer_space =~ /^\s/) { + WARN("SPACING", + "Unnecessary space after function pointer open parenthesis\n" . $herecurr); + } + +# unnecessary space "type (* funcptr)(args...)" + if (defined $post_pointer_space && + $post_pointer_space =~ /^\s/) { + WARN("SPACING", + "Unnecessary space before function pointer name\n" . $herecurr); + } + +# unnecessary space "type (*funcptr )(args...)" + if (defined $post_funcname_space && + $post_funcname_space =~ /^\s/) { + WARN("SPACING", + "Unnecessary space after function pointer name\n" . $herecurr); + } + +# unnecessary space "type (*funcptr) (args...)" + if (defined $pre_args_space && + $pre_args_space =~ /^\s/) { + WARN("SPACING", + "Unnecessary space before function pointer arguments\n" . $herecurr); + } + + if (show_type("SPACING") && $fix) { + $fixed[$fixlinenr] =~ + s/^(.\s*)$Declare\s*\(\s*\*\s*$Ident\s*\)\s*\(/$1 . $declare . $post_declare_space . '(*' . $funcname . ')('/ex; + } + } + +# check for spacing round square brackets; allowed: +# 1. with a type on the left -- int [] a; +# 2. at the beginning of a line for slice initialisers -- [0...10] = 5, +# 3. inside a curly brace -- = { [0...10] = 5 } + while ($line =~ /(.*?\s)\[/g) { + my ($where, $prefix) = ($-[1], $1); + if ($prefix !~ /$Type\s+$/ && + ($where != 0 || $prefix !~ /^.\s+$/) && + $prefix !~ /[{,]\s+$/ && + $prefix !~ /:\s+$/) { + if (ERROR("BRACKET_SPACE", + "space prohibited before open square bracket '['\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/^(\+.*?)\s+\[/$1\[/; + } + } + } + +# check for spaces between functions and their parentheses. + while ($line =~ /($Ident)\s+\(/g) { + my $name = $1; + my $ctx_before = substr($line, 0, $-[1]); + my $ctx = "$ctx_before$name"; + + # Ignore those directives where spaces _are_ permitted. + if ($name =~ /^(?: + if|for|while|switch|return|case| + volatile|__volatile__| + __attribute__|format|__extension__| + asm|__asm__)$/x) + { + # cpp #define statements have non-optional spaces, ie + # if there is a space between the name and the open + # parenthesis it is simply not a parameter group. + } elsif ($ctx_before =~ /^.\s*\#\s*define\s*$/) { + + # cpp #elif statement condition may start with a ( + } elsif ($ctx =~ /^.\s*\#\s*elif\s*$/) { + + # If this whole things ends with a type its most + # likely a typedef for a function. + } elsif ($ctx =~ /$Type$/) { + + } else { + if (WARN("SPACING", + "space prohibited between function name and open parenthesis '('\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/\b$name\s+\(/$name\(/; + } + } + } + +# Check operator spacing. + if (!($line=~/\#\s*include/)) { + my $fixed_line = ""; + my $line_fixed = 0; + + my $ops = qr{ + <<=|>>=|<=|>=|==|!=| + \+=|-=|\*=|\/=|%=|\^=|\|=|&=| + =>|->|<<|>>|<|>|=|!|~| + &&|\|\||,|\^|\+\+|--|&|\||\+|-|\*|\/|%| + \?:|\?|: + }x; + my @elements = split(/($ops|;)/, $opline); + +## print("element count: <" . $#elements . ">\n"); +## foreach my $el (@elements) { +## print("el: <$el>\n"); +## } + + my @fix_elements = (); + my $off = 0; + + foreach my $el (@elements) { + push(@fix_elements, substr($rawline, $off, length($el))); + $off += length($el); + } + + $off = 0; + + my $blank = copy_spacing($opline); + my $last_after = -1; + + for (my $n = 0; $n < $#elements; $n += 2) { + + my $good = $fix_elements[$n] . $fix_elements[$n + 1]; + +## print("n: <$n> good: <$good>\n"); + + $off += length($elements[$n]); + + # Pick up the preceding and succeeding characters. + my $ca = substr($opline, 0, $off); + my $cc = ''; + if (length($opline) >= ($off + length($elements[$n + 1]))) { + $cc = substr($opline, $off + length($elements[$n + 1])); + } + my $cb = "$ca$;$cc"; + + my $a = ''; + $a = 'V' if ($elements[$n] ne ''); + $a = 'W' if ($elements[$n] =~ /\s$/); + $a = 'C' if ($elements[$n] =~ /$;$/); + $a = 'B' if ($elements[$n] =~ /(\[|\()$/); + $a = 'O' if ($elements[$n] eq ''); + $a = 'E' if ($ca =~ /^\s*$/); + + my $op = $elements[$n + 1]; + + my $c = ''; + if (defined $elements[$n + 2]) { + $c = 'V' if ($elements[$n + 2] ne ''); + $c = 'W' if ($elements[$n + 2] =~ /^\s/); + $c = 'C' if ($elements[$n + 2] =~ /^$;/); + $c = 'B' if ($elements[$n + 2] =~ /^(\)|\]|;)/); + $c = 'O' if ($elements[$n + 2] eq ''); + $c = 'E' if ($elements[$n + 2] =~ /^\s*\\$/); + } else { + $c = 'E'; + } + + my $ctx = "${a}x${c}"; + + my $at = "(ctx:$ctx)"; + + my $ptr = substr($blank, 0, $off) . "^"; + my $hereptr = "$hereline$ptr\n"; + + # Pull out the value of this operator. + my $op_type = substr($curr_values, $off + 1, 1); + + # Get the full operator variant. + my $opv = $op . substr($curr_vars, $off, 1); + + # Ignore operators passed as parameters. + if ($op_type ne 'V' && + $ca =~ /\s$/ && $cc =~ /^\s*[,\)]/) { + +# # Ignore comments +# } elsif ($op =~ /^$;+$/) { + + # ; should have either the end of line or a space or \ after it + } elsif ($op eq ';') { + if ($ctx !~ /.x[WEBC]/ && + $cc !~ /^\\/ && $cc !~ /^;/) { + if (ERROR("SPACING", + "space required after that '$op' $at\n" . $hereptr)) { + $good = $fix_elements[$n] . trim($fix_elements[$n + 1]) . " "; + $line_fixed = 1; + } + } + + # // is a comment + } elsif ($op eq '//') { + + # : when part of a bitfield + } elsif ($opv eq ':B') { + # skip the bitfield test for now + + # No spaces for: + # -> + } elsif ($op eq '->') { + if ($ctx =~ /Wx.|.xW/) { + if (ERROR("SPACING", + "spaces prohibited around that '$op' $at\n" . $hereptr)) { + $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); + if (defined $fix_elements[$n + 2]) { + $fix_elements[$n + 2] =~ s/^\s+//; + } + $line_fixed = 1; + } + } + + # , must not have a space before and must have a space on the right. + } elsif ($op eq ',') { + my $rtrim_before = 0; + my $space_after = 0; + if ($ctx =~ /Wx./) { + if (ERROR("SPACING", + "space prohibited before that '$op' $at\n" . $hereptr)) { + $line_fixed = 1; + $rtrim_before = 1; + } + } + if ($ctx !~ /.x[WEC]/ && $cc !~ /^}/ && $cc !~ /^\)/) { + if (ERROR("SPACING", + "space required after that '$op' $at\n" . $hereptr)) { + $line_fixed = 1; + $last_after = $n; + $space_after = 1; + } + } + if ($rtrim_before || $space_after) { + if ($rtrim_before) { + $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); + } else { + $good = $fix_elements[$n] . trim($fix_elements[$n + 1]); + } + if ($space_after) { + $good .= " "; + } + } + + # '*' as part of a type definition -- reported already. + } elsif ($opv eq '*_') { + #warn "'*' is part of type\n"; + + # unary operators should have a space before and + # none after. May be left adjacent to another + # unary operator, or a cast + } elsif ($op eq '!' || $op eq '~' || + $opv eq '*U' || $opv eq '-U' || + $opv eq '&U' || $opv eq '&&U') { + if ($ctx !~ /[WEBC]x./ && $ca !~ /(?:\)|!|~|\*|-|\&|\||\+\+|\-\-|\{)$/) { + if (ERROR("SPACING", + "space required before that '$op' $at\n" . $hereptr)) { + if ($n != $last_after + 2) { + $good = $fix_elements[$n] . " " . ltrim($fix_elements[$n + 1]); + $line_fixed = 1; + } + } + } + if ($op eq '*' && $cc =~/\s*$Modifier\b/) { + # A unary '*' may be const + + } elsif ($ctx =~ /.xW/) { + if (ERROR("SPACING", + "space prohibited after that '$op' $at\n" . $hereptr)) { + $good = $fix_elements[$n] . rtrim($fix_elements[$n + 1]); + if (defined $fix_elements[$n + 2]) { + $fix_elements[$n + 2] =~ s/^\s+//; + } + $line_fixed = 1; + } + } + + # unary ++ and unary -- are allowed no space on one side. + } elsif ($op eq '++' or $op eq '--') { + if ($ctx !~ /[WEOBC]x[^W]/ && $ctx !~ /[^W]x[WOBEC]/) { + if (ERROR("SPACING", + "space required one side of that '$op' $at\n" . $hereptr)) { + $good = $fix_elements[$n] . trim($fix_elements[$n + 1]) . " "; + $line_fixed = 1; + } + } + if ($ctx =~ /Wx[BE]/ || + ($ctx =~ /Wx./ && $cc =~ /^;/)) { + if (ERROR("SPACING", + "space prohibited before that '$op' $at\n" . $hereptr)) { + $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); + $line_fixed = 1; + } + } + if ($ctx =~ /ExW/) { + if (ERROR("SPACING", + "space prohibited after that '$op' $at\n" . $hereptr)) { + $good = $fix_elements[$n] . trim($fix_elements[$n + 1]); + if (defined $fix_elements[$n + 2]) { + $fix_elements[$n + 2] =~ s/^\s+//; + } + $line_fixed = 1; + } + } + + # << and >> may either have or not have spaces both sides + } elsif ($op eq '<<' or $op eq '>>' or + $op eq '&' or $op eq '^' or $op eq '|' or + $op eq '+' or $op eq '-' or + $op eq '*' or $op eq '/' or + $op eq '%') + { + if ($check) { + if (defined $fix_elements[$n + 2] && $ctx !~ /[EW]x[EW]/) { + if (CHK("SPACING", + "spaces preferred around that '$op' $at\n" . $hereptr)) { + $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " "; + $fix_elements[$n + 2] =~ s/^\s+//; + $line_fixed = 1; + } + } elsif (!defined $fix_elements[$n + 2] && $ctx !~ /Wx[OE]/) { + if (CHK("SPACING", + "space preferred before that '$op' $at\n" . $hereptr)) { + $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]); + $line_fixed = 1; + } + } + } elsif ($ctx =~ /Wx[^WCE]|[^WCE]xW/) { + if (ERROR("SPACING", + "need consistent spacing around '$op' $at\n" . $hereptr)) { + $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " "; + if (defined $fix_elements[$n + 2]) { + $fix_elements[$n + 2] =~ s/^\s+//; + } + $line_fixed = 1; + } + } + + # A colon needs no spaces before when it is + # terminating a case value or a label. + } elsif ($opv eq ':C' || $opv eq ':L') { + if ($ctx =~ /Wx./) { + if (ERROR("SPACING", + "space prohibited before that '$op' $at\n" . $hereptr)) { + $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); + $line_fixed = 1; + } + } + + # All the others need spaces both sides. + } elsif ($ctx !~ /[EWC]x[CWE]/) { + my $ok = 0; + + # Ignore email addresses <foo@bar> + if (($op eq '<' && + $cc =~ /^\S+\@\S+>/) || + ($op eq '>' && + $ca =~ /<\S+\@\S+$/)) + { + $ok = 1; + } + + # for asm volatile statements + # ignore a colon with another + # colon immediately before or after + if (($op eq ':') && + ($ca =~ /:$/ || $cc =~ /^:/)) { + $ok = 1; + } + + # messages are ERROR, but ?: are CHK + if ($ok == 0) { + my $msg_level = \&ERROR; + $msg_level = \&CHK if (($op eq '?:' || $op eq '?' || $op eq ':') && $ctx =~ /VxV/); + + if (&{$msg_level}("SPACING", + "spaces required around that '$op' $at\n" . $hereptr)) { + $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " "; + if (defined $fix_elements[$n + 2]) { + $fix_elements[$n + 2] =~ s/^\s+//; + } + $line_fixed = 1; + } + } + } + $off += length($elements[$n + 1]); + +## print("n: <$n> GOOD: <$good>\n"); + + $fixed_line = $fixed_line . $good; + } + + if (($#elements % 2) == 0) { + $fixed_line = $fixed_line . $fix_elements[$#elements]; + } + + if ($fix && $line_fixed && $fixed_line ne $fixed[$fixlinenr]) { + $fixed[$fixlinenr] = $fixed_line; + } + + + } + +# check for whitespace before a non-naked semicolon + if ($line =~ /^\+.*\S\s+;\s*$/) { + if (WARN("SPACING", + "space prohibited before semicolon\n" . $herecurr) && + $fix) { + 1 while $fixed[$fixlinenr] =~ + s/^(\+.*\S)\s+;/$1;/; + } + } + +# check for multiple assignments + if ($line =~ /^.\s*$Lval\s*=\s*$Lval\s*=(?!=)/) { + CHK("MULTIPLE_ASSIGNMENTS", + "multiple assignments should be avoided\n" . $herecurr); + } + +## # check for multiple declarations, allowing for a function declaration +## # continuation. +## if ($line =~ /^.\s*$Type\s+$Ident(?:\s*=[^,{]*)?\s*,\s*$Ident.*/ && +## $line !~ /^.\s*$Type\s+$Ident(?:\s*=[^,{]*)?\s*,\s*$Type\s*$Ident.*/) { +## +## # Remove any bracketed sections to ensure we do not +## # falsly report the parameters of functions. +## my $ln = $line; +## while ($ln =~ s/\([^\(\)]*\)//g) { +## } +## if ($ln =~ /,/) { +## WARN("MULTIPLE_DECLARATION", +## "declaring multiple variables together should be avoided\n" . $herecurr); +## } +## } + +#need space before brace following if, while, etc + if (($line =~ /\(.*\)\{/ && $line !~ /\($Type\)\{/) || + $line =~ /do\{/) { + if (ERROR("SPACING", + "space required before the open brace '{'\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/^(\+.*(?:do|\)))\{/$1 {/; + } + } + +## # check for blank lines before declarations +## if ($line =~ /^.\t+$Type\s+$Ident(?:\s*=.*)?;/ && +## $prevrawline =~ /^.\s*$/) { +## WARN("SPACING", +## "No blank lines before declarations\n" . $hereprev); +## } +## + +# closing brace should have a space following it when it has anything +# on the line + if ($line =~ /}(?!(?:,|;|\)))\S/) { + if (ERROR("SPACING", + "space required after that close brace '}'\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/}((?!(?:,|;|\)))\S)/} $1/; + } + } + +# check spacing on square brackets + if ($line =~ /\[\s/ && $line !~ /\[\s*$/) { + if (ERROR("SPACING", + "space prohibited after that open square bracket '['\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/\[\s+/\[/; + } + } + if ($line =~ /\s\]/) { + if (ERROR("SPACING", + "space prohibited before that close square bracket ']'\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/\s+\]/\]/; + } + } + +# check spacing on parentheses + if ($line =~ /\(\s/ && $line !~ /\(\s*(?:\\)?$/ && + $line !~ /for\s*\(\s+;/) { + if (ERROR("SPACING", + "space prohibited after that open parenthesis '('\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/\(\s+/\(/; + } + } + if ($line =~ /(\s+)\)/ && $line !~ /^.\s*\)/ && + $line !~ /for\s*\(.*;\s+\)/ && + $line !~ /:\s+\)/) { + if (ERROR("SPACING", + "space prohibited before that close parenthesis ')'\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/\s+\)/\)/; + } + } + +# check unnecessary parentheses around addressof/dereference single $Lvals +# ie: &(foo->bar) should be &foo->bar and *(foo->bar) should be *foo->bar + + while ($line =~ /(?:[^&]&\s*|\*)\(\s*($Ident\s*(?:$Member\s*)+)\s*\)/g) { + my $var = $1; + if (CHK("UNNECESSARY_PARENTHESES", + "Unnecessary parentheses around $var\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\(\s*\Q$var\E\s*\)/$var/; + } + } + +# check for unnecessary parentheses around function pointer uses +# ie: (foo->bar)(); should be foo->bar(); +# but not "if (foo->bar) (" to avoid some false positives + if ($line =~ /(\bif\s*|)(\(\s*$Ident\s*(?:$Member\s*)+\))[ \t]*\(/ && $1 !~ /^if/) { + my $var = $2; + if (CHK("UNNECESSARY_PARENTHESES", + "Unnecessary parentheses around function pointer $var\n" . $herecurr) && + $fix) { + my $var2 = deparenthesize($var); + $var2 =~ s/\s//g; + $fixed[$fixlinenr] =~ s/\Q$var\E/$var2/; + } + } + +# check for unnecessary parentheses around comparisons in if uses + if ($^V && $^V ge 5.10.0 && defined($stat) && + $stat =~ /(^.\s*if\s*($balanced_parens))/) { + my $if_stat = $1; + my $test = substr($2, 1, -1); + my $herectx; + while ($test =~ /(?:^|[^\w\&\!\~])+\s*\(\s*([\&\!\~]?\s*$Lval\s*(?:$Compare\s*$FuncArg)?)\s*\)/g) { + my $match = $1; + # avoid parentheses around potential macro args + next if ($match =~ /^\s*\w+\s*$/); + if (!defined($herectx)) { + $herectx = $here . "\n"; + my $cnt = statement_rawlines($if_stat); + for (my $n = 0; $n < $cnt; $n++) { + my $rl = raw_line($linenr, $n); + $herectx .= $rl . "\n"; + last if $rl =~ /^[ \+].*\{/; + } + } + CHK("UNNECESSARY_PARENTHESES", + "Unnecessary parentheses around '$match'\n" . $herectx); + } + } + +#goto labels aren't indented, allow a single space however + if ($line=~/^.\s+[A-Za-z\d_]+:(?![0-9]+)/ and + !($line=~/^. [A-Za-z\d_]+:/) and !($line=~/^.\s+default:/)) { + if (WARN("INDENTED_LABEL", + "labels should not be indented\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/^(.)\s+/$1/; + } + } + +# return is not a function + if (defined($stat) && $stat =~ /^.\s*return(\s*)\(/s) { + my $spacing = $1; + if ($^V && $^V ge 5.10.0 && + $stat =~ /^.\s*return\s*($balanced_parens)\s*;\s*$/) { + my $value = $1; + $value = deparenthesize($value); + if ($value =~ m/^\s*$FuncArg\s*(?:\?|$)/) { + ERROR("RETURN_PARENTHESES", + "return is not a function, parentheses are not required\n" . $herecurr); + } + } elsif ($spacing !~ /\s+/) { + ERROR("SPACING", + "space required before the open parenthesis '('\n" . $herecurr); + } + } + +# unnecessary return in a void function +# at end-of-function, with the previous line a single leading tab, then return; +# and the line before that not a goto label target like "out:" + if ($sline =~ /^[ \+]}\s*$/ && + $prevline =~ /^\+\treturn\s*;\s*$/ && + $linenr >= 3 && + $lines[$linenr - 3] =~ /^[ +]/ && + $lines[$linenr - 3] !~ /^[ +]\s*$Ident\s*:/) { + WARN("RETURN_VOID", + "void function return statements are not generally useful\n" . $hereprev); + } + +# if statements using unnecessary parentheses - ie: if ((foo == bar)) + if ($^V && $^V ge 5.10.0 && + $line =~ /\bif\s*((?:\(\s*){2,})/) { + my $openparens = $1; + my $count = $openparens =~ tr@\(@\(@; + my $msg = ""; + if ($line =~ /\bif\s*(?:\(\s*){$count,$count}$LvalOrFunc\s*($Compare)\s*$LvalOrFunc(?:\s*\)){$count,$count}/) { + my $comp = $4; #Not $1 because of $LvalOrFunc + $msg = " - maybe == should be = ?" if ($comp eq "=="); + WARN("UNNECESSARY_PARENTHESES", + "Unnecessary parentheses$msg\n" . $herecurr); + } + } + +# comparisons with a constant or upper case identifier on the left +# avoid cases like "foo + BAR < baz" +# only fix matches surrounded by parentheses to avoid incorrect +# conversions like "FOO < baz() + 5" being "misfixed" to "baz() > FOO + 5" + if ($^V && $^V ge 5.10.0 && + $line =~ /^\+(.*)\b($Constant|[A-Z_][A-Z0-9_]*)\s*($Compare)\s*($LvalOrFunc)/) { + my $lead = $1; + my $const = $2; + my $comp = $3; + my $to = $4; + my $newcomp = $comp; + if ($lead !~ /(?:$Operators|\.)\s*$/ && + $to !~ /^(?:Constant|[A-Z_][A-Z0-9_]*)$/ && + WARN("CONSTANT_COMPARISON", + "Comparisons should place the constant on the right side of the test\n" . $herecurr) && + $fix) { + if ($comp eq "<") { + $newcomp = ">"; + } elsif ($comp eq "<=") { + $newcomp = ">="; + } elsif ($comp eq ">") { + $newcomp = "<"; + } elsif ($comp eq ">=") { + $newcomp = "<="; + } + $fixed[$fixlinenr] =~ s/\(\s*\Q$const\E\s*$Compare\s*\Q$to\E\s*\)/($to $newcomp $const)/; + } + } + +# Return of what appears to be an errno should normally be negative + if ($sline =~ /\breturn(?:\s*\(+\s*|\s+)(E[A-Z]+)(?:\s*\)+\s*|\s*)[;:,]/) { + my $name = $1; + if ($name ne 'EOF' && $name ne 'ERROR') { + WARN("USE_NEGATIVE_ERRNO", + "return of an errno should typically be negative (ie: return -$1)\n" . $herecurr); + } + } + +# Need a space before open parenthesis after if, while etc + if ($line =~ /\b(if|while|for|switch)\(/) { + if (ERROR("SPACING", + "space required before the open parenthesis '('\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/\b(if|while|for|switch)\(/$1 \(/; + } + } + +# Check for illegal assignment in if conditional -- and check for trailing +# statements after the conditional. + if ($line =~ /do\s*(?!{)/) { + ($stat, $cond, $line_nr_next, $remain_next, $off_next) = + ctx_statement_block($linenr, $realcnt, 0) + if (!defined $stat); + my ($stat_next) = ctx_statement_block($line_nr_next, + $remain_next, $off_next); + $stat_next =~ s/\n./\n /g; + ##print "stat<$stat> stat_next<$stat_next>\n"; + + if ($stat_next =~ /^\s*while\b/) { + # If the statement carries leading newlines, + # then count those as offsets. + my ($whitespace) = + ($stat_next =~ /^((?:\s*\n[+-])*\s*)/s); + my $offset = + statement_rawlines($whitespace) - 1; + + $suppress_whiletrailers{$line_nr_next + + $offset} = 1; + } + } + if (!defined $suppress_whiletrailers{$linenr} && + defined($stat) && defined($cond) && + $line =~ /\b(?:if|while|for)\s*\(/ && $line !~ /^.\s*#/) { + my ($s, $c) = ($stat, $cond); + + if ($c =~ /\bif\s*\(.*[^<>!=]=[^=].*/s) { + ERROR("ASSIGN_IN_IF", + "do not use assignment in if condition\n" . $herecurr); + } + + # Find out what is on the end of the line after the + # conditional. + substr($s, 0, length($c), ''); + $s =~ s/\n.*//g; + $s =~ s/$;//g; # Remove any comments + if (length($c) && $s !~ /^\s*{?\s*\\*\s*$/ && + $c !~ /}\s*while\s*/) + { + # Find out how long the conditional actually is. + my @newlines = ($c =~ /\n/gs); + my $cond_lines = 1 + $#newlines; + my $stat_real = ''; + + $stat_real = raw_line($linenr, $cond_lines) + . "\n" if ($cond_lines); + if (defined($stat_real) && $cond_lines > 1) { + $stat_real = "[...]\n$stat_real"; + } + + ERROR("TRAILING_STATEMENTS", + "trailing statements should be on next line\n" . $herecurr . $stat_real); + } + } + +# Check for bitwise tests written as boolean + if ($line =~ / + (?: + (?:\[|\(|\&\&|\|\|) + \s*0[xX][0-9]+\s* + (?:\&\&|\|\|) + | + (?:\&\&|\|\|) + \s*0[xX][0-9]+\s* + (?:\&\&|\|\||\)|\]) + )/x) + { + WARN("HEXADECIMAL_BOOLEAN_TEST", + "boolean test with hexadecimal, perhaps just 1 \& or \|?\n" . $herecurr); + } + +# if and else should not have general statements after it + if ($line =~ /^.\s*(?:}\s*)?else\b(.*)/) { + my $s = $1; + $s =~ s/$;//g; # Remove any comments + if ($s !~ /^\s*(?:\sif|(?:{|)\s*\\?\s*$)/) { + ERROR("TRAILING_STATEMENTS", + "trailing statements should be on next line\n" . $herecurr); + } + } +# if should not continue a brace + if ($line =~ /}\s*if\b/) { + ERROR("TRAILING_STATEMENTS", + "trailing statements should be on next line (or did you mean 'else if'?)\n" . + $herecurr); + } +# case and default should not have general statements after them + if ($line =~ /^.\s*(?:case\s*.*|default\s*):/g && + $line !~ /\G(?: + (?:\s*$;*)(?:\s*{)?(?:\s*$;*)(?:\s*\\)?\s*$| + \s*return\s+ + )/xg) + { + ERROR("TRAILING_STATEMENTS", + "trailing statements should be on next line\n" . $herecurr); + } + + # Check for }<nl>else {, these must be at the same + # indent level to be relevant to each other. + if ($prevline=~/}\s*$/ and $line=~/^.\s*else\s*/ && + $previndent == $indent) { + if (ERROR("ELSE_AFTER_BRACE", + "else should follow close brace '}'\n" . $hereprev) && + $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { + fix_delete_line($fixlinenr - 1, $prevrawline); + fix_delete_line($fixlinenr, $rawline); + my $fixedline = $prevrawline; + $fixedline =~ s/}\s*$//; + if ($fixedline !~ /^\+\s*$/) { + fix_insert_line($fixlinenr, $fixedline); + } + $fixedline = $rawline; + $fixedline =~ s/^(.\s*)else/$1} else/; + fix_insert_line($fixlinenr, $fixedline); + } + } + + if ($prevline=~/}\s*$/ and $line=~/^.\s*while\s*/ && + $previndent == $indent) { + my ($s, $c) = ctx_statement_block($linenr, $realcnt, 0); + + # Find out what is on the end of the line after the + # conditional. + substr($s, 0, length($c), ''); + $s =~ s/\n.*//g; + + if ($s =~ /^\s*;/) { + if (ERROR("WHILE_AFTER_BRACE", + "while should follow close brace '}'\n" . $hereprev) && + $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { + fix_delete_line($fixlinenr - 1, $prevrawline); + fix_delete_line($fixlinenr, $rawline); + my $fixedline = $prevrawline; + my $trailing = $rawline; + $trailing =~ s/^\+//; + $trailing = trim($trailing); + $fixedline =~ s/}\s*$/} $trailing/; + fix_insert_line($fixlinenr, $fixedline); + } + } + } + +#Specific variable tests + while ($line =~ m{($Constant|$Lval)}g) { + my $var = $1; + +#gcc binary extension + if ($var =~ /^$Binary$/) { + if (WARN("GCC_BINARY_CONSTANT", + "Avoid gcc v4.3+ binary constant extension: <$var>\n" . $herecurr) && + $fix) { + my $hexval = sprintf("0x%x", oct($var)); + $fixed[$fixlinenr] =~ + s/\b$var\b/$hexval/; + } + } + +#CamelCase + if ($var !~ /^$Constant$/ && + $var =~ /[A-Z][a-z]|[a-z][A-Z]/ && +#Ignore Page<foo> variants + $var !~ /^(?:Clear|Set|TestClear|TestSet|)Page[A-Z]/ && +#Ignore SI style variants like nS, mV and dB (ie: max_uV, regulator_min_uA_show) + $var !~ /^(?:[a-z_]*?)_?[a-z][A-Z](?:_[a-z_]+)?$/ && +#Ignore some three character SI units explicitly, like MiB and KHz + $var !~ /^(?:[a-z_]*?)_?(?:[KMGT]iB|[KMGT]?Hz)(?:_[a-z_]+)?$/) { + while ($var =~ m{($Ident)}g) { + my $word = $1; + next if ($word !~ /[A-Z][a-z]|[a-z][A-Z]/); + if ($check) { + seed_camelcase_includes(); + if (!$file && !$camelcase_file_seeded) { + seed_camelcase_file($realfile); + $camelcase_file_seeded = 1; + } + } + if (!defined $camelcase{$word}) { + $camelcase{$word} = 1; + CHK("CAMELCASE", + "Avoid CamelCase: <$word>\n" . $herecurr); + } + } + } + } + +#no spaces allowed after \ in define + if ($line =~ /\#\s*define.*\\\s+$/) { + if (WARN("WHITESPACE_AFTER_LINE_CONTINUATION", + "Whitespace after \\ makes next lines useless\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\s+$//; + } + } + +# warn if <asm/foo.h> is #included and <linux/foo.h> is available and includes +# itself <asm/foo.h> (uses RAW line) + if ($tree && $rawline =~ m{^.\s*\#\s*include\s*\<asm\/(.*)\.h\>}) { + my $file = "$1.h"; + my $checkfile = "include/linux/$file"; + if (-f "$root/$checkfile" && + $realfile ne $checkfile && + $1 !~ /$allowed_asm_includes/) + { + my $asminclude = `grep -Ec "#include\\s+<asm/$file>" $root/$checkfile`; + if ($asminclude > 0) { + if ($realfile =~ m{^arch/}) { + CHK("ARCH_INCLUDE_LINUX", + "Consider using #include <linux/$file> instead of <asm/$file>\n" . $herecurr); + } else { + WARN("INCLUDE_LINUX", + "Use #include <linux/$file> instead of <asm/$file>\n" . $herecurr); + } + } + } + } + +# multi-statement macros should be enclosed in a do while loop, grab the +# first statement and ensure its the whole macro if its not enclosed +# in a known good container + if ($realfile !~ m@/vmlinux.lds.h$@ && + $line =~ /^.\s*\#\s*define\s*$Ident(\()?/) { + my $ln = $linenr; + my $cnt = $realcnt; + my ($off, $dstat, $dcond, $rest); + my $ctx = ''; + my $has_flow_statement = 0; + my $has_arg_concat = 0; + ($dstat, $dcond, $ln, $cnt, $off) = + ctx_statement_block($linenr, $realcnt, 0); + $ctx = $dstat; + #print "dstat<$dstat> dcond<$dcond> cnt<$cnt> off<$off>\n"; + #print "LINE<$lines[$ln-1]> len<" . length($lines[$ln-1]) . "\n"; + + $has_flow_statement = 1 if ($ctx =~ /\b(goto|return)\b/); + $has_arg_concat = 1 if ($ctx =~ /\#\#/ && $ctx !~ /\#\#\s*(?:__VA_ARGS__|args)\b/); + + $dstat =~ s/^.\s*\#\s*define\s+$Ident(\([^\)]*\))?\s*//; + my $define_args = $1; + my $define_stmt = $dstat; + my @def_args = (); + + if (defined $define_args && $define_args ne "") { + $define_args = substr($define_args, 1, length($define_args) - 2); + $define_args =~ s/\s*//g; + @def_args = split(",", $define_args); + } + + $dstat =~ s/$;//g; + $dstat =~ s/\\\n.//g; + $dstat =~ s/^\s*//s; + $dstat =~ s/\s*$//s; + + # Flatten any parentheses and braces + while ($dstat =~ s/\([^\(\)]*\)/1/ || + $dstat =~ s/\{[^\{\}]*\}/1/ || + $dstat =~ s/.\[[^\[\]]*\]/1/) + { + } + + # Flatten any obvious string concatentation. + while ($dstat =~ s/($String)\s*$Ident/$1/ || + $dstat =~ s/$Ident\s*($String)/$1/) + { + } + + # Make asm volatile uses seem like a generic function + $dstat =~ s/\b_*asm_*\s+_*volatile_*\b/asm_volatile/g; + + my $exceptions = qr{ + $Declare| + module_param_named| + MODULE_PARM_DESC| + DECLARE_PER_CPU| + DEFINE_PER_CPU| + __typeof__\(| + union| + struct| + \.$Ident\s*=\s*| + ^\"|\"$| + ^\[ + }x; + #print "REST<$rest> dstat<$dstat> ctx<$ctx>\n"; + + $ctx =~ s/\n*$//; + my $herectx = $here . "\n"; + my $stmt_cnt = statement_rawlines($ctx); + + for (my $n = 0; $n < $stmt_cnt; $n++) { + $herectx .= raw_line($linenr, $n) . "\n"; + } + + if ($dstat ne '' && + $dstat !~ /^(?:$Ident|-?$Constant),$/ && # 10, // foo(), + $dstat !~ /^(?:$Ident|-?$Constant);$/ && # foo(); + $dstat !~ /^[!~-]?(?:$Lval|$Constant)$/ && # 10 // foo() // !foo // ~foo // -foo // foo->bar // foo.bar->baz + $dstat !~ /^'X'$/ && $dstat !~ /^'XX'$/ && # character constants + $dstat !~ /$exceptions/ && + $dstat !~ /^\.$Ident\s*=/ && # .foo = + $dstat !~ /^(?:\#\s*$Ident|\#\s*$Constant)\s*$/ && # stringification #foo + $dstat !~ /^do\s*$Constant\s*while\s*$Constant;?$/ && # do {...} while (...); // do {...} while (...) + $dstat !~ /^for\s*$Constant$/ && # for (...) + $dstat !~ /^for\s*$Constant\s+(?:$Ident|-?$Constant)$/ && # for (...) bar() + $dstat !~ /^do\s*{/ && # do {... + $dstat !~ /^\(\{/ && # ({... + $ctx !~ /^.\s*#\s*define\s+TRACE_(?:SYSTEM|INCLUDE_FILE|INCLUDE_PATH)\b/) + { + if ($dstat =~ /^\s*if\b/) { + ERROR("MULTISTATEMENT_MACRO_USE_DO_WHILE", + "Macros starting with if should be enclosed by a do - while loop to avoid possible if/else logic defects\n" . "$herectx"); + } elsif ($dstat =~ /;/) { + ERROR("MULTISTATEMENT_MACRO_USE_DO_WHILE", + "Macros with multiple statements should be enclosed in a do - while loop\n" . "$herectx"); + } else { + WARN("COMPLEX_MACRO", + "Macros with complex values should be enclosed in parentheses\n" . "$herectx"); + } + + } + + # Make $define_stmt single line, comment-free, etc + my @stmt_array = split('\n', $define_stmt); + my $first = 1; + $define_stmt = ""; + foreach my $l (@stmt_array) { + $l =~ s/\\$//; + if ($first) { + $define_stmt = $l; + $first = 0; + } elsif ($l =~ /^[\+ ]/) { + $define_stmt .= substr($l, 1); + } + } + $define_stmt =~ s/$;//g; + $define_stmt =~ s/\s+/ /g; + $define_stmt = trim($define_stmt); + +# check if any macro arguments are reused (ignore '...' and 'type') + foreach my $arg (@def_args) { + next if ($arg =~ /\.\.\./); + next if ($arg =~ /^type$/i); + my $tmp_stmt = $define_stmt; + $tmp_stmt =~ s/\b(typeof|__typeof__|__builtin\w+|typecheck\s*\(\s*$Type\s*,|\#+)\s*\(*\s*$arg\s*\)*\b//g; + $tmp_stmt =~ s/\#+\s*$arg\b//g; + $tmp_stmt =~ s/\b$arg\s*\#\#//g; + my $use_cnt = $tmp_stmt =~ s/\b$arg\b//g; + if ($use_cnt > 1) { + CHK("MACRO_ARG_REUSE", + "Macro argument reuse '$arg' - possible side-effects?\n" . "$herectx"); + } +# check if any macro arguments may have other precedence issues + if ($tmp_stmt =~ m/($Operators)?\s*\b$arg\b\s*($Operators)?/m && + ((defined($1) && $1 ne ',') || + (defined($2) && $2 ne ','))) { + CHK("MACRO_ARG_PRECEDENCE", + "Macro argument '$arg' may be better as '($arg)' to avoid precedence issues\n" . "$herectx"); + } + } + +# check for macros with flow control, but without ## concatenation +# ## concatenation is commonly a macro that defines a function so ignore those + if ($has_flow_statement && !$has_arg_concat) { + my $herectx = $here . "\n"; + my $cnt = statement_rawlines($ctx); + + for (my $n = 0; $n < $cnt; $n++) { + $herectx .= raw_line($linenr, $n) . "\n"; + } + WARN("MACRO_WITH_FLOW_CONTROL", + "Macros with flow control statements should be avoided\n" . "$herectx"); + } + +# check for line continuations outside of #defines, preprocessor #, and asm + + } else { + if ($prevline !~ /^..*\\$/ && + $line !~ /^\+\s*\#.*\\$/ && # preprocessor + $line !~ /^\+.*\b(__asm__|asm)\b.*\\$/ && # asm + $line =~ /^\+.*\\$/) { + WARN("LINE_CONTINUATIONS", + "Avoid unnecessary line continuations\n" . $herecurr); + } + } + +# do {} while (0) macro tests: +# single-statement macros do not need to be enclosed in do while (0) loop, +# macro should not end with a semicolon + if ($^V && $^V ge 5.10.0 && + $realfile !~ m@/vmlinux.lds.h$@ && + $line =~ /^.\s*\#\s*define\s+$Ident(\()?/) { + my $ln = $linenr; + my $cnt = $realcnt; + my ($off, $dstat, $dcond, $rest); + my $ctx = ''; + ($dstat, $dcond, $ln, $cnt, $off) = + ctx_statement_block($linenr, $realcnt, 0); + $ctx = $dstat; + + $dstat =~ s/\\\n.//g; + $dstat =~ s/$;/ /g; + + if ($dstat =~ /^\+\s*#\s*define\s+$Ident\s*${balanced_parens}\s*do\s*{(.*)\s*}\s*while\s*\(\s*0\s*\)\s*([;\s]*)\s*$/) { + my $stmts = $2; + my $semis = $3; + + $ctx =~ s/\n*$//; + my $cnt = statement_rawlines($ctx); + my $herectx = $here . "\n"; + + for (my $n = 0; $n < $cnt; $n++) { + $herectx .= raw_line($linenr, $n) . "\n"; + } + + if (($stmts =~ tr/;/;/) == 1 && + $stmts !~ /^\s*(if|while|for|switch)\b/) { + WARN("SINGLE_STATEMENT_DO_WHILE_MACRO", + "Single statement macros should not use a do {} while (0) loop\n" . "$herectx"); + } + if (defined $semis && $semis ne "") { + WARN("DO_WHILE_MACRO_WITH_TRAILING_SEMICOLON", + "do {} while (0) macros should not be semicolon terminated\n" . "$herectx"); + } + } elsif ($dstat =~ /^\+\s*#\s*define\s+$Ident.*;\s*$/) { + $ctx =~ s/\n*$//; + my $cnt = statement_rawlines($ctx); + my $herectx = $here . "\n"; + + for (my $n = 0; $n < $cnt; $n++) { + $herectx .= raw_line($linenr, $n) . "\n"; + } + + WARN("TRAILING_SEMICOLON", + "macros should not use a trailing semicolon\n" . "$herectx"); + } + } + +# make sure symbols are always wrapped with VMLINUX_SYMBOL() ... +# all assignments may have only one of the following with an assignment: +# . +# ALIGN(...) +# VMLINUX_SYMBOL(...) + if ($realfile eq 'vmlinux.lds.h' && $line =~ /(?:(?:^|\s)$Ident\s*=|=\s*$Ident(?:\s|$))/) { + WARN("MISSING_VMLINUX_SYMBOL", + "vmlinux.lds.h needs VMLINUX_SYMBOL() around C-visible symbols\n" . $herecurr); + } + +# check for redundant bracing round if etc + if ($line =~ /(^.*)\bif\b/ && $1 !~ /else\s*$/) { + my ($level, $endln, @chunks) = + ctx_statement_full($linenr, $realcnt, 1); + #print "chunks<$#chunks> linenr<$linenr> endln<$endln> level<$level>\n"; + #print "APW: <<$chunks[1][0]>><<$chunks[1][1]>>\n"; + if ($#chunks > 0 && $level == 0) { + my @allowed = (); + my $allow = 0; + my $seen = 0; + my $herectx = $here . "\n"; + my $ln = $linenr - 1; + for my $chunk (@chunks) { + my ($cond, $block) = @{$chunk}; + + # If the condition carries leading newlines, then count those as offsets. + my ($whitespace) = ($cond =~ /^((?:\s*\n[+-])*\s*)/s); + my $offset = statement_rawlines($whitespace) - 1; + + $allowed[$allow] = 0; + #print "COND<$cond> whitespace<$whitespace> offset<$offset>\n"; + + # We have looked at and allowed this specific line. + $suppress_ifbraces{$ln + $offset} = 1; + + $herectx .= "$rawlines[$ln + $offset]\n[...]\n"; + $ln += statement_rawlines($block) - 1; + + substr($block, 0, length($cond), ''); + + $seen++ if ($block =~ /^\s*{/); + + #print "cond<$cond> block<$block> allowed<$allowed[$allow]>\n"; + if (statement_lines($cond) > 1) { + #print "APW: ALLOWED: cond<$cond>\n"; + $allowed[$allow] = 1; + } + if ($block =~/\b(?:if|for|while)\b/) { + #print "APW: ALLOWED: block<$block>\n"; + $allowed[$allow] = 1; + } + if (statement_block_size($block) > 1) { + #print "APW: ALLOWED: lines block<$block>\n"; + $allowed[$allow] = 1; + } + $allow++; + } + if ($seen) { + my $sum_allowed = 0; + foreach (@allowed) { + $sum_allowed += $_; + } + if ($sum_allowed == 0) { + WARN("BRACES", + "braces {} are not necessary for any arm of this statement\n" . $herectx); + } elsif ($sum_allowed != $allow && + $seen != $allow) { + CHK("BRACES", + "braces {} should be used on all arms of this statement\n" . $herectx); + } + } + } + } + if (!defined $suppress_ifbraces{$linenr - 1} && + $line =~ /\b(if|while|for|else)\b/) { + my $allowed = 0; + + # Check the pre-context. + if (substr($line, 0, $-[0]) =~ /(\}\s*)$/) { + #print "APW: ALLOWED: pre<$1>\n"; + $allowed = 1; + } + + my ($level, $endln, @chunks) = + ctx_statement_full($linenr, $realcnt, $-[0]); + + # Check the condition. + my ($cond, $block) = @{$chunks[0]}; + #print "CHECKING<$linenr> cond<$cond> block<$block>\n"; + if (defined $cond) { + substr($block, 0, length($cond), ''); + } + if (statement_lines($cond) > 1) { + #print "APW: ALLOWED: cond<$cond>\n"; + $allowed = 1; + } + if ($block =~/\b(?:if|for|while)\b/) { + #print "APW: ALLOWED: block<$block>\n"; + $allowed = 1; + } + if (statement_block_size($block) > 1) { + #print "APW: ALLOWED: lines block<$block>\n"; + $allowed = 1; + } + # Check the post-context. + if (defined $chunks[1]) { + my ($cond, $block) = @{$chunks[1]}; + if (defined $cond) { + substr($block, 0, length($cond), ''); + } + if ($block =~ /^\s*\{/) { + #print "APW: ALLOWED: chunk-1 block<$block>\n"; + $allowed = 1; + } + } + if ($level == 0 && $block =~ /^\s*\{/ && !$allowed) { + my $herectx = $here . "\n"; + my $cnt = statement_rawlines($block); + + for (my $n = 0; $n < $cnt; $n++) { + $herectx .= raw_line($linenr, $n) . "\n"; + } + + WARN("BRACES", + "braces {} are not necessary for single statement blocks\n" . $herectx); + } + } + +# check for single line unbalanced braces + if ($sline =~ /^.\s*\}\s*else\s*$/ || + $sline =~ /^.\s*else\s*\{\s*$/) { + CHK("BRACES", "Unbalanced braces around else statement\n" . $herecurr); + } + +# check for unnecessary blank lines around braces + if (($line =~ /^.\s*}\s*$/ && $prevrawline =~ /^.\s*$/)) { + if (CHK("BRACES", + "Blank lines aren't necessary before a close brace '}'\n" . $hereprev) && + $fix && $prevrawline =~ /^\+/) { + fix_delete_line($fixlinenr - 1, $prevrawline); + } + } + if (($rawline =~ /^.\s*$/ && $prevline =~ /^..*{\s*$/)) { + if (CHK("BRACES", + "Blank lines aren't necessary after an open brace '{'\n" . $hereprev) && + $fix) { + fix_delete_line($fixlinenr, $rawline); + } + } + +# no volatiles please + my $asm_volatile = qr{\b(__asm__|asm)\s+(__volatile__|volatile)\b}; + if ($line =~ /\bvolatile\b/ && $line !~ /$asm_volatile/) { + WARN("VOLATILE", + "Use of volatile is usually wrong: see Documentation/process/volatile-considered-harmful.rst\n" . $herecurr); + } + +# Check for user-visible strings broken across lines, which breaks the ability +# to grep for the string. Make exceptions when the previous string ends in a +# newline (multiple lines in one string constant) or '\t', '\r', ';', or '{' +# (common in inline assembly) or is a octal \123 or hexadecimal \xaf value + if ($line =~ /^\+\s*$String/ && + $prevline =~ /"\s*$/ && + $prevrawline !~ /(?:\\(?:[ntr]|[0-7]{1,3}|x[0-9a-fA-F]{1,2})|;\s*|\{\s*)"\s*$/) { + if (WARN("SPLIT_STRING", + "quoted string split across lines\n" . $hereprev) && + $fix && + $prevrawline =~ /^\+.*"\s*$/ && + $last_coalesced_string_linenr != $linenr - 1) { + my $extracted_string = get_quoted_string($line, $rawline); + my $comma_close = ""; + if ($rawline =~ /\Q$extracted_string\E(\s*\)\s*;\s*$|\s*,\s*)/) { + $comma_close = $1; + } + + fix_delete_line($fixlinenr - 1, $prevrawline); + fix_delete_line($fixlinenr, $rawline); + my $fixedline = $prevrawline; + $fixedline =~ s/"\s*$//; + $fixedline .= substr($extracted_string, 1) . trim($comma_close); + fix_insert_line($fixlinenr - 1, $fixedline); + $fixedline = $rawline; + $fixedline =~ s/\Q$extracted_string\E\Q$comma_close\E//; + if ($fixedline !~ /\+\s*$/) { + fix_insert_line($fixlinenr, $fixedline); + } + $last_coalesced_string_linenr = $linenr; + } + } + +# check for missing a space in a string concatenation + if ($prevrawline =~ /[^\\]\w"$/ && $rawline =~ /^\+[\t ]+"\w/) { + WARN('MISSING_SPACE', + "break quoted strings at a space character\n" . $hereprev); + } + +# check for an embedded function name in a string when the function is known +# This does not work very well for -f --file checking as it depends on patch +# context providing the function name or a single line form for in-file +# function declarations + if ($line =~ /^\+.*$String/ && + defined($context_function) && + get_quoted_string($line, $rawline) =~ /\b$context_function\b/ && + length(get_quoted_string($line, $rawline)) != (length($context_function) + 2)) { + WARN("EMBEDDED_FUNCTION_NAME", + "Prefer using '\"%s...\", __func__' to using '$context_function', this function's name, in a string\n" . $herecurr); + } + +# check for spaces before a quoted newline + if ($rawline =~ /^.*\".*\s\\n/) { + if (WARN("QUOTED_WHITESPACE_BEFORE_NEWLINE", + "unnecessary whitespace before a quoted newline\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/^(\+.*\".*)\s+\\n/$1\\n/; + } + + } + +# concatenated string without spaces between elements + if ($line =~ /$String[A-Z_]/ || $line =~ /[A-Za-z0-9_]$String/) { + CHK("CONCATENATED_STRING", + "Concatenated strings should use spaces between elements\n" . $herecurr); + } + +# uncoalesced string fragments + if ($line =~ /$String\s*"/) { + WARN("STRING_FRAGMENTS", + "Consecutive strings are generally better as a single string\n" . $herecurr); + } + +# check for non-standard and hex prefixed decimal printf formats + my $show_L = 1; #don't show the same defect twice + my $show_Z = 1; + while ($line =~ /(?:^|")([X\t]*)(?:"|$)/g) { + my $string = substr($rawline, $-[1], $+[1] - $-[1]); + $string =~ s/%%/__/g; + # check for %L + if ($show_L && $string =~ /%[\*\d\.\$]*L([diouxX])/) { + WARN("PRINTF_L", + "\%L$1 is non-standard C, use %ll$1\n" . $herecurr); + $show_L = 0; + } + # check for %Z + if ($show_Z && $string =~ /%[\*\d\.\$]*Z([diouxX])/) { + WARN("PRINTF_Z", + "%Z$1 is non-standard C, use %z$1\n" . $herecurr); + $show_Z = 0; + } + # check for 0x<decimal> + if ($string =~ /0x%[\*\d\.\$\Llzth]*[diou]/) { + ERROR("PRINTF_0XDECIMAL", + "Prefixing 0x with decimal output is defective\n" . $herecurr); + } + } + +# check for line continuations in quoted strings with odd counts of " + if ($rawline =~ /\\$/ && $rawline =~ tr/"/"/ % 2) { + WARN("LINE_CONTINUATIONS", + "Avoid line continuations in quoted strings\n" . $herecurr); + } + +# warn about #if 0 + if ($line =~ /^.\s*\#\s*if\s+0\b/) { + CHK("REDUNDANT_CODE", + "if this code is redundant consider removing it\n" . + $herecurr); + } + +# check for needless "if (<foo>) fn(<foo>)" uses + if ($prevline =~ /\bif\s*\(\s*($Lval)\s*\)/) { + my $tested = quotemeta($1); + my $expr = '\s*\(\s*' . $tested . '\s*\)\s*;'; + if ($line =~ /\b(kfree|usb_free_urb|debugfs_remove(?:_recursive)?|(?:kmem_cache|mempool|dma_pool)_destroy)$expr/) { + my $func = $1; + if (WARN('NEEDLESS_IF', + "$func(NULL) is safe and this check is probably not required\n" . $hereprev) && + $fix) { + my $do_fix = 1; + my $leading_tabs = ""; + my $new_leading_tabs = ""; + if ($lines[$linenr - 2] =~ /^\+(\t*)if\s*\(\s*$tested\s*\)\s*$/) { + $leading_tabs = $1; + } else { + $do_fix = 0; + } + if ($lines[$linenr - 1] =~ /^\+(\t+)$func\s*\(\s*$tested\s*\)\s*;\s*$/) { + $new_leading_tabs = $1; + if (length($leading_tabs) + 1 ne length($new_leading_tabs)) { + $do_fix = 0; + } + } else { + $do_fix = 0; + } + if ($do_fix) { + fix_delete_line($fixlinenr - 1, $prevrawline); + $fixed[$fixlinenr] =~ s/^\+$new_leading_tabs/\+$leading_tabs/; + } + } + } + } + +# check for unnecessary "Out of Memory" messages + if ($line =~ /^\+.*\b$logFunctions\s*\(/ && + $prevline =~ /^[ \+]\s*if\s*\(\s*(\!\s*|NULL\s*==\s*)?($Lval)(\s*==\s*NULL\s*)?\s*\)/ && + (defined $1 || defined $3) && + $linenr > 3) { + my $testval = $2; + my $testline = $lines[$linenr - 3]; + + my ($s, $c) = ctx_statement_block($linenr - 3, $realcnt, 0); +# print("line: <$line>\nprevline: <$prevline>\ns: <$s>\nc: <$c>\n\n\n"); + + if ($s =~ /(?:^|\n)[ \+]\s*(?:$Type\s*)?\Q$testval\E\s*=\s*(?:\([^\)]*\)\s*)?\s*(?:devm_)?(?:[kv][czm]alloc(?:_node|_array)?\b|kstrdup|kmemdup|(?:dev_)?alloc_skb)/) { + WARN("OOM_MESSAGE", + "Possible unnecessary 'out of memory' message\n" . $hereprev); + } + } + +# check for logging functions with KERN_<LEVEL> + if ($line !~ /printk(?:_ratelimited|_once)?\s*\(/ && + $line =~ /\b$logFunctions\s*\(.*\b(KERN_[A-Z]+)\b/) { + my $level = $1; + if (WARN("UNNECESSARY_KERN_LEVEL", + "Possible unnecessary $level\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\s*$level\s*//; + } + } + +# check for logging continuations + if ($line =~ /\bprintk\s*\(\s*KERN_CONT\b|\bpr_cont\s*\(/) { + WARN("LOGGING_CONTINUATION", + "Avoid logging continuation uses where feasible\n" . $herecurr); + } + +# check for mask then right shift without a parentheses + if ($^V && $^V ge 5.10.0 && + $line =~ /$LvalOrFunc\s*\&\s*($LvalOrFunc)\s*>>/ && + $4 !~ /^\&/) { # $LvalOrFunc may be &foo, ignore if so + WARN("MASK_THEN_SHIFT", + "Possible precedence defect with mask then right shift - may need parentheses\n" . $herecurr); + } + +# check for pointer comparisons to NULL + if ($^V && $^V ge 5.10.0) { + while ($line =~ /\b$LvalOrFunc\s*(==|\!=)\s*NULL\b/g) { + my $val = $1; + my $equal = "!"; + $equal = "" if ($4 eq "!="); + if (CHK("COMPARISON_TO_NULL", + "Comparison to NULL could be written \"${equal}${val}\"\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\b\Q$val\E\s*(?:==|\!=)\s*NULL\b/$equal$val/; + } + } + } + +# check for bad placement of section $InitAttribute (e.g.: __initdata) + if ($line =~ /(\b$InitAttribute\b)/) { + my $attr = $1; + if ($line =~ /^\+\s*static\s+(?:const\s+)?(?:$attr\s+)?($NonptrTypeWithAttr)\s+(?:$attr\s+)?($Ident(?:\[[^]]*\])?)\s*[=;]/) { + my $ptr = $1; + my $var = $2; + if ((($ptr =~ /\b(union|struct)\s+$attr\b/ && + ERROR("MISPLACED_INIT", + "$attr should be placed after $var\n" . $herecurr)) || + ($ptr !~ /\b(union|struct)\s+$attr\b/ && + WARN("MISPLACED_INIT", + "$attr should be placed after $var\n" . $herecurr))) && + $fix) { + $fixed[$fixlinenr] =~ s/(\bstatic\s+(?:const\s+)?)(?:$attr\s+)?($NonptrTypeWithAttr)\s+(?:$attr\s+)?($Ident(?:\[[^]]*\])?)\s*([=;])\s*/"$1" . trim(string_find_replace($2, "\\s*$attr\\s*", " ")) . " " . trim(string_find_replace($3, "\\s*$attr\\s*", "")) . " $attr" . ("$4" eq ";" ? ";" : " = ")/e; + } + } + } + +# check for $InitAttributeData (ie: __initdata) with const + if ($line =~ /\bconst\b/ && $line =~ /($InitAttributeData)/) { + my $attr = $1; + $attr =~ /($InitAttributePrefix)(.*)/; + my $attr_prefix = $1; + my $attr_type = $2; + if (ERROR("INIT_ATTRIBUTE", + "Use of const init definition must use ${attr_prefix}initconst\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/$InitAttributeData/${attr_prefix}initconst/; + } + } + +# check for $InitAttributeConst (ie: __initconst) without const + if ($line !~ /\bconst\b/ && $line =~ /($InitAttributeConst)/) { + my $attr = $1; + if (ERROR("INIT_ATTRIBUTE", + "Use of $attr requires a separate use of const\n" . $herecurr) && + $fix) { + my $lead = $fixed[$fixlinenr] =~ + /(^\+\s*(?:static\s+))/; + $lead = rtrim($1); + $lead = "$lead " if ($lead !~ /^\+$/); + $lead = "${lead}const "; + $fixed[$fixlinenr] =~ s/(^\+\s*(?:static\s+))/$lead/; + } + } + +# check for __read_mostly with const non-pointer (should just be const) + if ($line =~ /\b__read_mostly\b/ && + $line =~ /($Type)\s*$Ident/ && $1 !~ /\*\s*$/ && $1 =~ /\bconst\b/) { + if (ERROR("CONST_READ_MOSTLY", + "Invalid use of __read_mostly with const type\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\s+__read_mostly\b//; + } + } + +# don't use __constant_<foo> functions outside of include/uapi/ + if ($realfile !~ m@^include/uapi/@ && + $line =~ /(__constant_(?:htons|ntohs|[bl]e(?:16|32|64)_to_cpu|cpu_to_[bl]e(?:16|32|64)))\s*\(/) { + my $constant_func = $1; + my $func = $constant_func; + $func =~ s/^__constant_//; + if (WARN("CONSTANT_CONVERSION", + "$constant_func should be $func\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\b$constant_func\b/$func/g; + } + } + +# prefer usleep_range over udelay + if ($line =~ /\budelay\s*\(\s*(\d+)\s*\)/) { + my $delay = $1; + # ignore udelay's < 10, however + if (! ($delay < 10) ) { + CHK("USLEEP_RANGE", + "usleep_range is preferred over udelay; see Documentation/timers/timers-howto.txt\n" . $herecurr); + } + if ($delay > 2000) { + WARN("LONG_UDELAY", + "long udelay - prefer mdelay; see arch/arm/include/asm/delay.h\n" . $herecurr); + } + } + +# warn about unexpectedly long msleep's + if ($line =~ /\bmsleep\s*\((\d+)\);/) { + if ($1 < 20) { + WARN("MSLEEP", + "msleep < 20ms can sleep for up to 20ms; see Documentation/timers/timers-howto.txt\n" . $herecurr); + } + } + +# check for comparisons of jiffies + if ($line =~ /\bjiffies\s*$Compare|$Compare\s*jiffies\b/) { + WARN("JIFFIES_COMPARISON", + "Comparing jiffies is almost always wrong; prefer time_after, time_before and friends\n" . $herecurr); + } + +# check for comparisons of get_jiffies_64() + if ($line =~ /\bget_jiffies_64\s*\(\s*\)\s*$Compare|$Compare\s*get_jiffies_64\s*\(\s*\)/) { + WARN("JIFFIES_COMPARISON", + "Comparing get_jiffies_64() is almost always wrong; prefer time_after64, time_before64 and friends\n" . $herecurr); + } + +# warn about #ifdefs in C files +# if ($line =~ /^.\s*\#\s*if(|n)def/ && ($realfile =~ /\.c$/)) { +# print "#ifdef in C files should be avoided\n"; +# print "$herecurr"; +# $clean = 0; +# } + +# warn about spacing in #ifdefs + if ($line =~ /^.\s*\#\s*(ifdef|ifndef|elif)\s\s+/) { + if (ERROR("SPACING", + "exactly one space required after that #$1\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ + s/^(.\s*\#\s*(ifdef|ifndef|elif))\s{2,}/$1 /; + } + + } + +# check for spinlock_t definitions without a comment. + if ($line =~ /^.\s*(struct\s+mutex|spinlock_t)\s+\S+;/ || + $line =~ /^.\s*(DEFINE_MUTEX)\s*\(/) { + my $which = $1; + if (!ctx_has_comment($first_line, $linenr)) { + CHK("UNCOMMENTED_DEFINITION", + "$1 definition without comment\n" . $herecurr); + } + } +# check for memory barriers without a comment. + + my $barriers = qr{ + mb| + rmb| + wmb| + read_barrier_depends + }x; + my $barrier_stems = qr{ + mb__before_atomic| + mb__after_atomic| + store_release| + load_acquire| + store_mb| + (?:$barriers) + }x; + my $all_barriers = qr{ + (?:$barriers)| + smp_(?:$barrier_stems)| + virt_(?:$barrier_stems) + }x; + + if ($line =~ /\b(?:$all_barriers)\s*\(/) { + if (!ctx_has_comment($first_line, $linenr)) { + WARN("MEMORY_BARRIER", + "memory barrier without comment\n" . $herecurr); + } + } + + my $underscore_smp_barriers = qr{__smp_(?:$barrier_stems)}x; + + if ($realfile !~ m@^include/asm-generic/@ && + $realfile !~ m@/barrier\.h$@ && + $line =~ m/\b(?:$underscore_smp_barriers)\s*\(/ && + $line !~ m/^.\s*\#\s*define\s+(?:$underscore_smp_barriers)\s*\(/) { + WARN("MEMORY_BARRIER", + "__smp memory barriers shouldn't be used outside barrier.h and asm-generic\n" . $herecurr); + } + +# check for waitqueue_active without a comment. + if ($line =~ /\bwaitqueue_active\s*\(/) { + if (!ctx_has_comment($first_line, $linenr)) { + WARN("WAITQUEUE_ACTIVE", + "waitqueue_active without comment\n" . $herecurr); + } + } + +# check of hardware specific defines + if ($line =~ m@^.\s*\#\s*if.*\b(__i386__|__powerpc64__|__sun__|__s390x__)\b@ && $realfile !~ m@include/asm-@) { + CHK("ARCH_DEFINES", + "architecture specific defines should be avoided\n" . $herecurr); + } + +# check that the storage class is not after a type + if ($line =~ /\b($Type)\s+($Storage)\b/) { + WARN("STORAGE_CLASS", + "storage class '$2' should be located before type '$1'\n" . $herecurr); + } +# Check that the storage class is at the beginning of a declaration + if ($line =~ /\b$Storage\b/ && + $line !~ /^.\s*$Storage/ && + $line =~ /^.\s*(.+?)\$Storage\s/ && + $1 !~ /[\,\)]\s*$/) { + WARN("STORAGE_CLASS", + "storage class should be at the beginning of the declaration\n" . $herecurr); + } + +# check the location of the inline attribute, that it is between +# storage class and type. + if ($line =~ /\b$Type\s+$Inline\b/ || + $line =~ /\b$Inline\s+$Storage\b/) { + ERROR("INLINE_LOCATION", + "inline keyword should sit between storage class and type\n" . $herecurr); + } + +# Check for __inline__ and __inline, prefer inline + if ($realfile !~ m@\binclude/uapi/@ && + $line =~ /\b(__inline__|__inline)\b/) { + if (WARN("INLINE", + "plain inline is preferred over $1\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\b(__inline__|__inline)\b/inline/; + + } + } + +# Check for __attribute__ packed, prefer __packed + if ($realfile !~ m@\binclude/uapi/@ && + $line =~ /\b__attribute__\s*\(\s*\(.*\bpacked\b/) { + WARN("PREFER_PACKED", + "__packed is preferred over __attribute__((packed))\n" . $herecurr); + } + +# Check for __attribute__ aligned, prefer __aligned + if ($realfile !~ m@\binclude/uapi/@ && + $line =~ /\b__attribute__\s*\(\s*\(.*aligned/) { + WARN("PREFER_ALIGNED", + "__aligned(size) is preferred over __attribute__((aligned(size)))\n" . $herecurr); + } + +# Check for __attribute__ format(printf, prefer __printf + if ($realfile !~ m@\binclude/uapi/@ && + $line =~ /\b__attribute__\s*\(\s*\(\s*format\s*\(\s*printf/) { + if (WARN("PREFER_PRINTF", + "__printf(string-index, first-to-check) is preferred over __attribute__((format(printf, string-index, first-to-check)))\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\b__attribute__\s*\(\s*\(\s*format\s*\(\s*printf\s*,\s*(.*)\)\s*\)\s*\)/"__printf(" . trim($1) . ")"/ex; + + } + } + +# Check for __attribute__ format(scanf, prefer __scanf + if ($realfile !~ m@\binclude/uapi/@ && + $line =~ /\b__attribute__\s*\(\s*\(\s*format\s*\(\s*scanf\b/) { + if (WARN("PREFER_SCANF", + "__scanf(string-index, first-to-check) is preferred over __attribute__((format(scanf, string-index, first-to-check)))\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\b__attribute__\s*\(\s*\(\s*format\s*\(\s*scanf\s*,\s*(.*)\)\s*\)\s*\)/"__scanf(" . trim($1) . ")"/ex; + } + } + +# Check for __attribute__ weak, or __weak declarations (may have link issues) + if ($^V && $^V ge 5.10.0 && + $line =~ /(?:$Declare|$DeclareMisordered)\s*$Ident\s*$balanced_parens\s*(?:$Attribute)?\s*;/ && + ($line =~ /\b__attribute__\s*\(\s*\(.*\bweak\b/ || + $line =~ /\b__weak\b/)) { + ERROR("WEAK_DECLARATION", + "Using weak declarations can have unintended link defects\n" . $herecurr); + } + +# check for c99 types like uint8_t + if ($line =~ /\b($Declare)\s*$Ident\s*[=;,\[]/) { + my $type = $1; + if ($type =~ /\b($typeC99Typedefs)\b/) { + $type = $1; + my $kernel_type = 'u'; + $kernel_type = 's' if ($type =~ /^_*[si]/); + $type =~ /(\d+)/; + $kernel_type .= $1.'_t'; + WARN("PREFER_KERNEL_TYPES", + "Prefer kernel type '$kernel_type' over '$type'\n" . $herecurr) + } + } + +# check for cast of C90 native int or longer types constants + if ($line =~ /(\(\s*$C90_int_types\s*\)\s*)($Constant)\b/) { + my $cast = $1; + my $const = $2; + if (WARN("TYPECAST_INT_CONSTANT", + "Unnecessary typecast of c90 int constant\n" . $herecurr) && + $fix) { + my $suffix = ""; + my $newconst = $const; + $newconst =~ s/${Int_type}$//; + $suffix .= 'U' if ($cast =~ /\bunsigned\b/); + if ($cast =~ /\blong\s+long\b/) { + $suffix .= 'LL'; + } elsif ($cast =~ /\blong\b/) { + $suffix .= 'L'; + } + $fixed[$fixlinenr] =~ s/\Q$cast\E$const\b/$newconst$suffix/; + } + } + +# check for sizeof(&) + if ($line =~ /\bsizeof\s*\(\s*\&/) { + WARN("SIZEOF_ADDRESS", + "sizeof(& should be avoided\n" . $herecurr); + } + +# check for sizeof without parenthesis + if ($line =~ /\bsizeof\s+((?:\*\s*|)$Lval|$Type(?:\s+$Lval|))/) { + if (WARN("SIZEOF_PARENTHESIS", + "sizeof $1 should be sizeof($1)\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\bsizeof\s+((?:\*\s*|)$Lval|$Type(?:\s+$Lval|))/"sizeof(" . trim($1) . ")"/ex; + } + } + +# check for struct spinlock declarations + if ($line =~ /^.\s*\bstruct\s+spinlock\s+\w+\s*;/) { + WARN("USE_SPINLOCK_T", + "struct spinlock should be spinlock_t\n" . $herecurr); + } + +# check for seq_printf uses that could be seq_puts + if ($sline =~ /\bseq_printf\s*\(.*"\s*\)\s*;\s*$/) { + my $fmt = get_quoted_string($line, $rawline); + $fmt =~ s/%%//g; + if ($fmt !~ /%/) { + if (WARN("PREFER_SEQ_PUTS", + "Prefer seq_puts to seq_printf\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\bseq_printf\b/seq_puts/; + } + } + } + + # check for vsprintf extension %p<foo> misuses + if ($^V && $^V ge 5.10.0 && + defined $stat && + $stat =~ /^\+(?![^\{]*\{\s*).*\b(\w+)\s*\(.*$String\s*,/s && + $1 !~ /^_*volatile_*$/) { + my $bad_extension = ""; + my $lc = $stat =~ tr@\n@@; + $lc = $lc + $linenr; + for (my $count = $linenr; $count <= $lc; $count++) { + my $fmt = get_quoted_string($lines[$count - 1], raw_line($count, 0)); + $fmt =~ s/%%//g; + if ($fmt =~ /(\%[\*\d\.]*p(?![\WFfSsBKRraEhMmIiUDdgVCbGNO]).)/) { + $bad_extension = $1; + last; + } + } + if ($bad_extension ne "") { + my $stat_real = raw_line($linenr, 0); + for (my $count = $linenr + 1; $count <= $lc; $count++) { + $stat_real = $stat_real . "\n" . raw_line($count, 0); + } + WARN("VSPRINTF_POINTER_EXTENSION", + "Invalid vsprintf pointer extension '$bad_extension'\n" . "$here\n$stat_real\n"); + } + } + +# Check for misused memsets + if ($^V && $^V ge 5.10.0 && + defined $stat && + $stat =~ /^\+(?:.*?)\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*$FuncArg\s*\)/) { + + my $ms_addr = $2; + my $ms_val = $7; + my $ms_size = $12; + + if ($ms_size =~ /^(0x|)0$/i) { + ERROR("MEMSET", + "memset to 0's uses 0 as the 2nd argument, not the 3rd\n" . "$here\n$stat\n"); + } elsif ($ms_size =~ /^(0x|)1$/i) { + WARN("MEMSET", + "single byte memset is suspicious. Swapped 2nd/3rd argument?\n" . "$here\n$stat\n"); + } + } + +# Check for memcpy(foo, bar, ETH_ALEN) that could be ether_addr_copy(foo, bar) +# if ($^V && $^V ge 5.10.0 && +# defined $stat && +# $stat =~ /^\+(?:.*?)\bmemcpy\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) { +# if (WARN("PREFER_ETHER_ADDR_COPY", +# "Prefer ether_addr_copy() over memcpy() if the Ethernet addresses are __aligned(2)\n" . "$here\n$stat\n") && +# $fix) { +# $fixed[$fixlinenr] =~ s/\bmemcpy\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/ether_addr_copy($2, $7)/; +# } +# } + +# Check for memcmp(foo, bar, ETH_ALEN) that could be ether_addr_equal*(foo, bar) +# if ($^V && $^V ge 5.10.0 && +# defined $stat && +# $stat =~ /^\+(?:.*?)\bmemcmp\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) { +# WARN("PREFER_ETHER_ADDR_EQUAL", +# "Prefer ether_addr_equal() or ether_addr_equal_unaligned() over memcmp()\n" . "$here\n$stat\n") +# } + +# check for memset(foo, 0x0, ETH_ALEN) that could be eth_zero_addr +# check for memset(foo, 0xFF, ETH_ALEN) that could be eth_broadcast_addr +# if ($^V && $^V ge 5.10.0 && +# defined $stat && +# $stat =~ /^\+(?:.*?)\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) { +# +# my $ms_val = $7; +# +# if ($ms_val =~ /^(?:0x|)0+$/i) { +# if (WARN("PREFER_ETH_ZERO_ADDR", +# "Prefer eth_zero_addr over memset()\n" . "$here\n$stat\n") && +# $fix) { +# $fixed[$fixlinenr] =~ s/\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*,\s*ETH_ALEN\s*\)/eth_zero_addr($2)/; +# } +# } elsif ($ms_val =~ /^(?:0xff|255)$/i) { +# if (WARN("PREFER_ETH_BROADCAST_ADDR", +# "Prefer eth_broadcast_addr() over memset()\n" . "$here\n$stat\n") && +# $fix) { +# $fixed[$fixlinenr] =~ s/\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*,\s*ETH_ALEN\s*\)/eth_broadcast_addr($2)/; +# } +# } +# } + +# typecasts on min/max could be min_t/max_t + if ($^V && $^V ge 5.10.0 && + defined $stat && + $stat =~ /^\+(?:.*?)\b(min|max)\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\)/) { + if (defined $2 || defined $7) { + my $call = $1; + my $cast1 = deparenthesize($2); + my $arg1 = $3; + my $cast2 = deparenthesize($7); + my $arg2 = $8; + my $cast; + + if ($cast1 ne "" && $cast2 ne "" && $cast1 ne $cast2) { + $cast = "$cast1 or $cast2"; + } elsif ($cast1 ne "") { + $cast = $cast1; + } else { + $cast = $cast2; + } + WARN("MINMAX", + "$call() should probably be ${call}_t($cast, $arg1, $arg2)\n" . "$here\n$stat\n"); + } + } + +# check usleep_range arguments + if ($^V && $^V ge 5.10.0 && + defined $stat && + $stat =~ /^\+(?:.*?)\busleep_range\s*\(\s*($FuncArg)\s*,\s*($FuncArg)\s*\)/) { + my $min = $1; + my $max = $7; + if ($min eq $max) { + WARN("USLEEP_RANGE", + "usleep_range should not use min == max args; see Documentation/timers/timers-howto.txt\n" . "$here\n$stat\n"); + } elsif ($min =~ /^\d+$/ && $max =~ /^\d+$/ && + $min > $max) { + WARN("USLEEP_RANGE", + "usleep_range args reversed, use min then max; see Documentation/timers/timers-howto.txt\n" . "$here\n$stat\n"); + } + } + +# check for naked sscanf + if ($^V && $^V ge 5.10.0 && + defined $stat && + $line =~ /\bsscanf\b/ && + ($stat !~ /$Ident\s*=\s*sscanf\s*$balanced_parens/ && + $stat !~ /\bsscanf\s*$balanced_parens\s*(?:$Compare)/ && + $stat !~ /(?:$Compare)\s*\bsscanf\s*$balanced_parens/)) { + my $lc = $stat =~ tr@\n@@; + $lc = $lc + $linenr; + my $stat_real = raw_line($linenr, 0); + for (my $count = $linenr + 1; $count <= $lc; $count++) { + $stat_real = $stat_real . "\n" . raw_line($count, 0); + } + WARN("NAKED_SSCANF", + "unchecked sscanf return value\n" . "$here\n$stat_real\n"); + } + +# check for simple sscanf that should be kstrto<foo> + if ($^V && $^V ge 5.10.0 && + defined $stat && + $line =~ /\bsscanf\b/) { + my $lc = $stat =~ tr@\n@@; + $lc = $lc + $linenr; + my $stat_real = raw_line($linenr, 0); + for (my $count = $linenr + 1; $count <= $lc; $count++) { + $stat_real = $stat_real . "\n" . raw_line($count, 0); + } + if ($stat_real =~ /\bsscanf\b\s*\(\s*$FuncArg\s*,\s*("[^"]+")/) { + my $format = $6; + my $count = $format =~ tr@%@%@; + if ($count == 1 && + $format =~ /^"\%(?i:ll[udxi]|[udxi]ll|ll|[hl]h?[udxi]|[udxi][hl]h?|[hl]h?|[udxi])"$/) { + WARN("SSCANF_TO_KSTRTO", + "Prefer kstrto<type> to single variable sscanf\n" . "$here\n$stat_real\n"); + } + } + } + +# check for new externs in .h files. + if ($realfile =~ /\.h$/ && + $line =~ /^\+\s*(extern\s+)$Type\s*$Ident\s*\(/s) { + if (CHK("AVOID_EXTERNS", + "extern prototypes should be avoided in .h files\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/(.*)\bextern\b\s*(.*)/$1$2/; + } + } + +# check for new externs in .c files. + if ($realfile =~ /\.c$/ && defined $stat && + $stat =~ /^.\s*(?:extern\s+)?$Type\s+($Ident)(\s*)\(/s) + { + my $function_name = $1; + my $paren_space = $2; + + my $s = $stat; + if (defined $cond) { + substr($s, 0, length($cond), ''); + } + if ($s =~ /^\s*;/ && + $function_name ne 'uninitialized_var') + { + WARN("AVOID_EXTERNS", + "externs should be avoided in .c files\n" . $herecurr); + } + + if ($paren_space =~ /\n/) { + WARN("FUNCTION_ARGUMENTS", + "arguments for function declarations should follow identifier\n" . $herecurr); + } + + } elsif ($realfile =~ /\.c$/ && defined $stat && + $stat =~ /^.\s*extern\s+/) + { + WARN("AVOID_EXTERNS", + "externs should be avoided in .c files\n" . $herecurr); + } + +# check for function declarations that have arguments without identifier names + if (defined $stat && + $stat =~ /^.\s*(?:extern\s+)?$Type\s*$Ident\s*\(\s*([^{]+)\s*\)\s*;/s && + $1 ne "void") { + my $args = trim($1); + while ($args =~ m/\s*($Type\s*(?:$Ident|\(\s*\*\s*$Ident?\s*\)\s*$balanced_parens)?)/g) { + my $arg = trim($1); + if ($arg =~ /^$Type$/ && $arg !~ /enum\s+$Ident$/) { + WARN("FUNCTION_ARGUMENTS", + "function definition argument '$arg' should also have an identifier name\n" . $herecurr); + } + } + } + +# check for function definitions + if ($^V && $^V ge 5.10.0 && + defined $stat && + $stat =~ /^.\s*(?:$Storage\s+)?$Type\s*($Ident)\s*$balanced_parens\s*{/s) { + $context_function = $1; + +# check for multiline function definition with misplaced open brace + my $ok = 0; + my $cnt = statement_rawlines($stat); + my $herectx = $here . "\n"; + for (my $n = 0; $n < $cnt; $n++) { + my $rl = raw_line($linenr, $n); + $herectx .= $rl . "\n"; + $ok = 1 if ($rl =~ /^[ \+]\{/); + $ok = 1 if ($rl =~ /\{/ && $n == 0); + last if $rl =~ /^[ \+].*\{/; + } + if (!$ok) { + ERROR("OPEN_BRACE", + "open brace '{' following function definitions go on the next line\n" . $herectx); + } + } + +# checks for new __setup's + if ($rawline =~ /\b__setup\("([^"]*)"/) { + my $name = $1; + + if (!grep(/$name/, @setup_docs)) { + CHK("UNDOCUMENTED_SETUP", + "__setup appears un-documented -- check Documentation/admin-guide/kernel-parameters.rst\n" . $herecurr); + } + } + +# check for pointless casting of kmalloc return + if ($line =~ /\*\s*\)\s*[kv][czm]alloc(_node){0,1}\b/) { + WARN("UNNECESSARY_CASTS", + "unnecessary cast may hide bugs, see http://c-faq.com/malloc/mallocnocast.html\n" . $herecurr); + } + +# alloc style +# p = alloc(sizeof(struct foo), ...) should be p = alloc(sizeof(*p), ...) + if ($^V && $^V ge 5.10.0 && + $line =~ /\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*([kv][mz]alloc(?:_node)?)\s*\(\s*(sizeof\s*\(\s*struct\s+$Lval\s*\))/) { + CHK("ALLOC_SIZEOF_STRUCT", + "Prefer $3(sizeof(*$1)...) over $3($4...)\n" . $herecurr); + } + +# check for k[mz]alloc with multiplies that could be kmalloc_array/kcalloc + if ($^V && $^V ge 5.10.0 && + defined $stat && + $stat =~ /^\+\s*($Lval)\s*\=\s*(?:$balanced_parens)?\s*(k[mz]alloc)\s*\(\s*($FuncArg)\s*\*\s*($FuncArg)\s*,/) { + my $oldfunc = $3; + my $a1 = $4; + my $a2 = $10; + my $newfunc = "kmalloc_array"; + $newfunc = "kcalloc" if ($oldfunc eq "kzalloc"); + my $r1 = $a1; + my $r2 = $a2; + if ($a1 =~ /^sizeof\s*\S/) { + $r1 = $a2; + $r2 = $a1; + } + if ($r1 !~ /^sizeof\b/ && $r2 =~ /^sizeof\s*\S/ && + !($r1 =~ /^$Constant$/ || $r1 =~ /^[A-Z_][A-Z0-9_]*$/)) { + my $ctx = ''; + my $herectx = $here . "\n"; + my $cnt = statement_rawlines($stat); + for (my $n = 0; $n < $cnt; $n++) { + $herectx .= raw_line($linenr, $n) . "\n"; + } + if (WARN("ALLOC_WITH_MULTIPLY", + "Prefer $newfunc over $oldfunc with multiply\n" . $herectx) && + $cnt == 1 && + $fix) { + $fixed[$fixlinenr] =~ s/\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*(k[mz]alloc)\s*\(\s*($FuncArg)\s*\*\s*($FuncArg)/$1 . ' = ' . "$newfunc(" . trim($r1) . ', ' . trim($r2)/e; + } + } + } + +# check for krealloc arg reuse + if ($^V && $^V ge 5.10.0 && + $line =~ /\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*krealloc\s*\(\s*\1\s*,/) { + WARN("KREALLOC_ARG_REUSE", + "Reusing the krealloc arg is almost always a bug\n" . $herecurr); + } + +# check for alloc argument mismatch + if ($line =~ /\b(kcalloc|kmalloc_array)\s*\(\s*sizeof\b/) { + WARN("ALLOC_ARRAY_ARGS", + "$1 uses number as first arg, sizeof is generally wrong\n" . $herecurr); + } + +# check for multiple semicolons + if ($line =~ /;\s*;\s*$/) { + if (WARN("ONE_SEMICOLON", + "Statements terminations use 1 semicolon\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/(\s*;\s*){2,}$/;/g; + } + } + +# check for #defines like: 1 << <digit> that could be BIT(digit), it is not exported to uapi + if ($realfile !~ m@^include/uapi/@ && + $line =~ /#\s*define\s+\w+\s+\(?\s*1\s*([ulUL]*)\s*\<\<\s*(?:\d+|$Ident)\s*\)?/) { + my $ull = ""; + $ull = "_ULL" if (defined($1) && $1 =~ /ll/i); + if (CHK("BIT_MACRO", + "Prefer using the BIT$ull macro\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\(?\s*1\s*[ulUL]*\s*<<\s*(\d+|$Ident)\s*\)?/BIT${ull}($1)/; + } + } + +# check for #if defined CONFIG_<FOO> || defined CONFIG_<FOO>_MODULE + if ($line =~ /^\+\s*#\s*if\s+defined(?:\s*\(?\s*|\s+)(CONFIG_[A-Z_]+)\s*\)?\s*\|\|\s*defined(?:\s*\(?\s*|\s+)\1_MODULE\s*\)?\s*$/) { + my $config = $1; + if (WARN("PREFER_IS_ENABLED", + "Prefer IS_ENABLED(<FOO>) to CONFIG_<FOO> || CONFIG_<FOO>_MODULE\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] = "\+#if IS_ENABLED($config)"; + } + } + +# check for case / default statements not preceded by break/fallthrough/switch + if ($line =~ /^.\s*(?:case\s+(?:$Ident|$Constant)\s*|default):/) { + my $has_break = 0; + my $has_statement = 0; + my $count = 0; + my $prevline = $linenr; + while ($prevline > 1 && ($file || $count < 3) && !$has_break) { + $prevline--; + my $rline = $rawlines[$prevline - 1]; + my $fline = $lines[$prevline - 1]; + last if ($fline =~ /^\@\@/); + next if ($fline =~ /^\-/); + next if ($fline =~ /^.(?:\s*(?:case\s+(?:$Ident|$Constant)[\s$;]*|default):[\s$;]*)*$/); + $has_break = 1 if ($rline =~ /fall[\s_-]*(through|thru)/i); + next if ($fline =~ /^.[\s$;]*$/); + $has_statement = 1; + $count++; + $has_break = 1 if ($fline =~ /\bswitch\b|\b(?:break\s*;[\s$;]*$|return\b|goto\b|continue\b)/); + } + if (!$has_break && $has_statement) { + WARN("MISSING_BREAK", + "Possible switch case/default not preceded by break or fallthrough comment\n" . $herecurr); + } + } + +# check for switch/default statements without a break; + if ($^V && $^V ge 5.10.0 && + defined $stat && + $stat =~ /^\+[$;\s]*(?:case[$;\s]+\w+[$;\s]*:[$;\s]*|)*[$;\s]*\bdefault[$;\s]*:[$;\s]*;/g) { + my $ctx = ''; + my $herectx = $here . "\n"; + my $cnt = statement_rawlines($stat); + for (my $n = 0; $n < $cnt; $n++) { + $herectx .= raw_line($linenr, $n) . "\n"; + } + WARN("DEFAULT_NO_BREAK", + "switch default: should use break\n" . $herectx); + } + +# check for gcc specific __FUNCTION__ + if ($line =~ /\b__FUNCTION__\b/) { + if (WARN("USE_FUNC", + "__func__ should be used instead of gcc specific __FUNCTION__\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\b__FUNCTION__\b/__func__/g; + } + } + +# check for uses of __DATE__, __TIME__, __TIMESTAMP__ + while ($line =~ /\b(__(?:DATE|TIME|TIMESTAMP)__)\b/g) { + ERROR("DATE_TIME", + "Use of the '$1' macro makes the build non-deterministic\n" . $herecurr); + } + +# check for use of yield() + if ($line =~ /\byield\s*\(\s*\)/) { + WARN("YIELD", + "Using yield() is generally wrong. See yield() kernel-doc (sched/core.c)\n" . $herecurr); + } + +# check for comparisons against true and false + if ($line =~ /\+\s*(.*?)\b(true|false|$Lval)\s*(==|\!=)\s*(true|false|$Lval)\b(.*)$/i) { + my $lead = $1; + my $arg = $2; + my $test = $3; + my $otype = $4; + my $trail = $5; + my $op = "!"; + + ($arg, $otype) = ($otype, $arg) if ($arg =~ /^(?:true|false)$/i); + + my $type = lc($otype); + if ($type =~ /^(?:true|false)$/) { + if (("$test" eq "==" && "$type" eq "true") || + ("$test" eq "!=" && "$type" eq "false")) { + $op = ""; + } + + CHK("BOOL_COMPARISON", + "Using comparison to $otype is error prone\n" . $herecurr); + +## maybe suggesting a correct construct would better +## "Using comparison to $otype is error prone. Perhaps use '${lead}${op}${arg}${trail}'\n" . $herecurr); + + } + } + +# check for semaphores initialized locked + if ($line =~ /^.\s*sema_init.+,\W?0\W?\)/) { + WARN("CONSIDER_COMPLETION", + "consider using a completion\n" . $herecurr); + } + +# recommend kstrto* over simple_strto* and strict_strto* + if ($line =~ /\b((simple|strict)_(strto(l|ll|ul|ull)))\s*\(/) { + WARN("CONSIDER_KSTRTO", + "$1 is obsolete, use k$3 instead\n" . $herecurr); + } + +# check for __initcall(), use device_initcall() explicitly or more appropriate function please + if ($line =~ /^.\s*__initcall\s*\(/) { + WARN("USE_DEVICE_INITCALL", + "please use device_initcall() or more appropriate function instead of __initcall() (see include/linux/init.h)\n" . $herecurr); + } + +# check for various structs that are normally const (ops, kgdb, device_tree) +# and avoid what seem like struct definitions 'struct foo {' + if ($line !~ /\bconst\b/ && + $line =~ /\bstruct\s+($const_structs)\b(?!\s*\{)/) { + WARN("CONST_STRUCT", + "struct $1 should normally be const\n" . $herecurr); + } + +# use of NR_CPUS is usually wrong +# ignore definitions of NR_CPUS and usage to define arrays as likely right + if ($line =~ /\bNR_CPUS\b/ && + $line !~ /^.\s*\s*#\s*if\b.*\bNR_CPUS\b/ && + $line !~ /^.\s*\s*#\s*define\b.*\bNR_CPUS\b/ && + $line !~ /^.\s*$Declare\s.*\[[^\]]*NR_CPUS[^\]]*\]/ && + $line !~ /\[[^\]]*\.\.\.[^\]]*NR_CPUS[^\]]*\]/ && + $line !~ /\[[^\]]*NR_CPUS[^\]]*\.\.\.[^\]]*\]/) + { + WARN("NR_CPUS", + "usage of NR_CPUS is often wrong - consider using cpu_possible(), num_possible_cpus(), for_each_possible_cpu(), etc\n" . $herecurr); + } + +# Use of __ARCH_HAS_<FOO> or ARCH_HAVE_<BAR> is wrong. + if ($line =~ /\+\s*#\s*define\s+((?:__)?ARCH_(?:HAS|HAVE)\w*)\b/) { + ERROR("DEFINE_ARCH_HAS", + "#define of '$1' is wrong - use Kconfig variables or standard guards instead\n" . $herecurr); + } + +# likely/unlikely comparisons similar to "(likely(foo) > 0)" + if ($^V && $^V ge 5.10.0 && + $line =~ /\b((?:un)?likely)\s*\(\s*$FuncArg\s*\)\s*$Compare/) { + WARN("LIKELY_MISUSE", + "Using $1 should generally have parentheses around the comparison\n" . $herecurr); + } + +# whine mightly about in_atomic + if ($line =~ /\bin_atomic\s*\(/) { + if ($realfile =~ m@^drivers/@) { + ERROR("IN_ATOMIC", + "do not use in_atomic in drivers\n" . $herecurr); + } elsif ($realfile !~ m@^kernel/@) { + WARN("IN_ATOMIC", + "use of in_atomic() is incorrect outside core kernel code\n" . $herecurr); + } + } + +# whine about ACCESS_ONCE + if ($^V && $^V ge 5.10.0 && + $line =~ /\bACCESS_ONCE\s*$balanced_parens\s*(=(?!=))?\s*($FuncArg)?/) { + my $par = $1; + my $eq = $2; + my $fun = $3; + $par =~ s/^\(\s*(.*)\s*\)$/$1/; + if (defined($eq)) { + if (WARN("PREFER_WRITE_ONCE", + "Prefer WRITE_ONCE(<FOO>, <BAR>) over ACCESS_ONCE(<FOO>) = <BAR>\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\bACCESS_ONCE\s*\(\s*\Q$par\E\s*\)\s*$eq\s*\Q$fun\E/WRITE_ONCE($par, $fun)/; + } + } else { + if (WARN("PREFER_READ_ONCE", + "Prefer READ_ONCE(<FOO>) over ACCESS_ONCE(<FOO>)\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/\bACCESS_ONCE\s*\(\s*\Q$par\E\s*\)/READ_ONCE($par)/; + } + } + } + +# check for mutex_trylock_recursive usage + if ($line =~ /mutex_trylock_recursive/) { + ERROR("LOCKING", + "recursive locking is bad, do not use this ever.\n" . $herecurr); + } + +# check for lockdep_set_novalidate_class + if ($line =~ /^.\s*lockdep_set_novalidate_class\s*\(/ || + $line =~ /__lockdep_no_validate__\s*\)/ ) { + if ($realfile !~ m@^kernel/lockdep@ && + $realfile !~ m@^include/linux/lockdep@ && + $realfile !~ m@^drivers/base/core@) { + ERROR("LOCKDEP", + "lockdep_no_validate class is reserved for device->mutex.\n" . $herecurr); + } + } + + if ($line =~ /debugfs_create_\w+.*\b$mode_perms_world_writable\b/ || + $line =~ /DEVICE_ATTR.*\b$mode_perms_world_writable\b/) { + WARN("EXPORTED_WORLD_WRITABLE", + "Exporting world writable files is usually an error. Consider more restrictive permissions.\n" . $herecurr); + } + +# Mode permission misuses where it seems decimal should be octal +# This uses a shortcut match to avoid unnecessary uses of a slow foreach loop + if ($^V && $^V ge 5.10.0 && + defined $stat && + $line =~ /$mode_perms_search/) { + foreach my $entry (@mode_permission_funcs) { + my $func = $entry->[0]; + my $arg_pos = $entry->[1]; + + my $lc = $stat =~ tr@\n@@; + $lc = $lc + $linenr; + my $stat_real = raw_line($linenr, 0); + for (my $count = $linenr + 1; $count <= $lc; $count++) { + $stat_real = $stat_real . "\n" . raw_line($count, 0); + } + + my $skip_args = ""; + if ($arg_pos > 1) { + $arg_pos--; + $skip_args = "(?:\\s*$FuncArg\\s*,\\s*){$arg_pos,$arg_pos}"; + } + my $test = "\\b$func\\s*\\(${skip_args}($FuncArg(?:\\|\\s*$FuncArg)*)\\s*[,\\)]"; + if ($stat =~ /$test/) { + my $val = $1; + $val = $6 if ($skip_args ne ""); + if (($val =~ /^$Int$/ && $val !~ /^$Octal$/) || + ($val =~ /^$Octal$/ && length($val) ne 4)) { + ERROR("NON_OCTAL_PERMISSIONS", + "Use 4 digit octal (0777) not decimal permissions\n" . "$here\n" . $stat_real); + } + if ($val =~ /^$Octal$/ && (oct($val) & 02)) { + ERROR("EXPORTED_WORLD_WRITABLE", + "Exporting writable files is usually an error. Consider more restrictive permissions.\n" . "$here\n" . $stat_real); + } + } + } + } + +# check for uses of S_<PERMS> that could be octal for readability + if ($line =~ /\b$mode_perms_string_search\b/) { + my $val = ""; + my $oval = ""; + my $to = 0; + my $curpos = 0; + my $lastpos = 0; + while ($line =~ /\b(($mode_perms_string_search)\b(?:\s*\|\s*)?\s*)/g) { + $curpos = pos($line); + my $match = $2; + my $omatch = $1; + last if ($lastpos > 0 && ($curpos - length($omatch) != $lastpos)); + $lastpos = $curpos; + $to |= $mode_permission_string_types{$match}; + $val .= '\s*\|\s*' if ($val ne ""); + $val .= $match; + $oval .= $omatch; + } + $oval =~ s/^\s*\|\s*//; + $oval =~ s/\s*\|\s*$//; + my $octal = sprintf("%04o", $to); + if (WARN("SYMBOLIC_PERMS", + "Symbolic permissions '$oval' are not preferred. Consider using octal permissions '$octal'.\n" . $herecurr) && + $fix) { + $fixed[$fixlinenr] =~ s/$val/$octal/; + } + } + +# validate content of MODULE_LICENSE against list from include/linux/module.h + if ($line =~ /\bMODULE_LICENSE\s*\(\s*($String)\s*\)/) { + my $extracted_string = get_quoted_string($line, $rawline); + my $valid_licenses = qr{ + GPL| + GPL\ v2| + GPL\ and\ additional\ rights| + Dual\ BSD/GPL| + Dual\ MIT/GPL| + Dual\ MPL/GPL| + Proprietary + }x; + if ($extracted_string !~ /^"(?:$valid_licenses)"$/x) { + WARN("MODULE_LICENSE", + "unknown module license " . $extracted_string . "\n" . $herecurr); + } + } + } + + # If we have no input at all, then there is nothing to report on + # so just keep quiet. + if ($#rawlines == -1) { + exit(0); + } + + # In mailback mode only produce a report in the negative, for + # things that appear to be patches. + if ($mailback && ($clean == 1 || !$is_patch)) { + exit(0); + } + + # This is not a patch, and we are are in 'no-patch' mode so + # just keep quiet. + if (!$chk_patch && !$is_patch) { + exit(0); + } + + if (!$is_patch && $file !~ /cover-letter\.patch$/) { + ERROR("NOT_UNIFIED_DIFF", + "Does not appear to be a unified-diff format patch\n"); + } + if ($is_patch && $has_commit_log && $chk_signoff && $signoff == 0) { + ERROR("MISSING_SIGN_OFF", + "Missing Signed-off-by: line(s)\n"); + } + + print report_dump(); + if ($summary && !($clean == 1 && $quiet == 1)) { + print "$filename " if ($summary_file); + print "total: $cnt_error errors, $cnt_warn warnings, " . + (($check)? "$cnt_chk checks, " : "") . + "$cnt_lines lines checked\n"; + } + + if ($quiet == 0) { + # If there were any defects found and not already fixing them + if (!$clean and !$fix) { + print << "EOM" + +NOTE: For some of the reported defects, checkpatch may be able to + mechanically convert to the typical style using --fix or --fix-inplace. +EOM + } + # If there were whitespace errors which cleanpatch can fix + # then suggest that. + if ($rpt_cleaners) { + $rpt_cleaners = 0; + print << "EOM" + +NOTE: Whitespace errors detected. + You may wish to use scripts/cleanpatch or scripts/cleanfile +EOM + } + } + + if ($clean == 0 && $fix && + ("@rawlines" ne "@fixed" || + $#fixed_inserted >= 0 || $#fixed_deleted >= 0)) { + my $newfile = $filename; + $newfile .= ".EXPERIMENTAL-checkpatch-fixes" if (!$fix_inplace); + my $linecount = 0; + my $f; + + @fixed = fix_inserted_deleted_lines(\@fixed, \@fixed_inserted, \@fixed_deleted); + + open($f, '>', $newfile) + or die "$P: Can't open $newfile for write\n"; + foreach my $fixed_line (@fixed) { + $linecount++; + if ($file) { + if ($linecount > 3) { + $fixed_line =~ s/^\+//; + print $f $fixed_line . "\n"; + } + } else { + print $f $fixed_line . "\n"; + } + } + close($f); + + if (!$quiet) { + print << "EOM"; + +Wrote EXPERIMENTAL --fix correction(s) to '$newfile' + +Do _NOT_ trust the results written to this file. +Do _NOT_ submit these changes without inspecting them for correctness. + +This EXPERIMENTAL file is simply a convenience to help rewrite patches. +No warranties, expressed or implied... +EOM + } + } + + if ($quiet == 0) { + print "\n"; + if ($clean == 1) { + print "$vname has no obvious style problems and is ready for submission.\n"; + } else { + print "$vname has style problems, please review.\n"; + } + } + return $clean; +} diff --git a/stm32/system/Middlewares/OpenAMP/open-amp/scripts/ci/check_compliance.py b/stm32/system/Middlewares/OpenAMP/open-amp/scripts/ci/check_compliance.py old mode 100755 new mode 100644 diff --git a/stm32/system/Middlewares/OpenAMP/open-amp/scripts/do_checkpatch.sh b/stm32/system/Middlewares/OpenAMP/open-amp/scripts/do_checkpatch.sh old mode 100755 new mode 100644 diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Inc/usbd_audio.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Inc/usbd_audio.h index dbb71cd7..a082d190 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Inc/usbd_audio.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Inc/usbd_audio.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -58,7 +57,10 @@ extern "C" { #define AUDIO_FS_BINTERVAL 0x01U #endif /* AUDIO_FS_BINTERVAL */ +#ifndef AUDIO_OUT_EP #define AUDIO_OUT_EP 0x01U +#endif /* AUDIO_OUT_EP */ + #define USB_AUDIO_CONFIG_DESC_SIZ 0x6DU #define AUDIO_INTERFACE_DESC_SIZE 0x09U #define USB_AUDIO_DESC_SIZ 0x09U @@ -167,6 +169,115 @@ typedef struct int8_t (*PeriodicTC)(uint8_t *pbuf, uint32_t size, uint8_t cmd); int8_t (*GetState)(void); } USBD_AUDIO_ItfTypeDef; + +/* + * Audio Class specification release 1.0 + */ + +/* Table 4-2: Class-Specific AC Interface Header Descriptor */ +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubtype; + uint16_t bcdADC; + uint16_t wTotalLength; + uint8_t bInCollection; + uint8_t baInterfaceNr; +} __PACKED USBD_SpeakerIfDescTypeDef; + +/* Table 4-3: Input Terminal Descriptor */ +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubtype; + uint8_t bTerminalID; + uint16_t wTerminalType; + uint8_t bAssocTerminal; + uint8_t bNrChannels; + uint16_t wChannelConfig; + uint8_t iChannelNames; + uint8_t iTerminal; +} __PACKED USBD_SpeakerInDescTypeDef; + +/* USB Speaker Audio Feature Unit Descriptor */ +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubtype; + uint8_t bUnitID; + uint8_t bSourceID; + uint8_t bControlSize; + uint16_t bmaControls; + uint8_t iTerminal; +} __PACKED USBD_SpeakerFeatureDescTypeDef; + +/* Table 4-4: Output Terminal Descriptor */ +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubtype; + uint8_t bTerminalID; + uint16_t wTerminalType; + uint8_t bAssocTerminal; + uint8_t bSourceID; + uint8_t iTerminal; +} __PACKED USBD_SpeakerOutDescTypeDef; + +/* Table 4-19: Class-Specific AS Interface Descriptor */ +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubtype; + uint8_t bTerminalLink; + uint8_t bDelay; + uint16_t wFormatTag; +} __PACKED USBD_SpeakerStreamIfDescTypeDef; + +/* USB Speaker Audio Type III Format Interface Descriptor */ +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubtype; + uint8_t bFormatType; + uint8_t bNrChannels; + uint8_t bSubFrameSize; + uint8_t bBitResolution; + uint8_t bSamFreqType; + uint8_t tSamFreq2; + uint8_t tSamFreq1; + uint8_t tSamFreq0; +} USBD_SpeakerIIIFormatIfDescTypeDef; + +/* Table 4-17: Standard AC Interrupt Endpoint Descriptor */ +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bEndpointAddress; + uint8_t bmAttributes; + uint16_t wMaxPacketSize; + uint8_t bInterval; + uint8_t bRefresh; + uint8_t bSynchAddress; +} __PACKED USBD_SpeakerEndDescTypeDef; + +/* Table 4-21: Class-Specific AS Isochronous Audio Data Endpoint Descriptor */ +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptor; + uint8_t bmAttributes; + uint8_t bLockDelayUnits; + uint16_t wLockDelay; +} __PACKED USBD_SpeakerEndStDescTypeDef; + /** * @} */ @@ -198,6 +309,11 @@ uint8_t USBD_AUDIO_RegisterInterface(USBD_HandleTypeDef *pdev, USBD_AUDIO_ItfTypeDef *fops); void USBD_AUDIO_Sync(USBD_HandleTypeDef *pdev, AUDIO_OffsetTypeDef offset); + +#ifdef USE_USBD_COMPOSITE +uint32_t USBD_AUDIO_GetEpPcktSze(USBD_HandleTypeDef *pdev, uint8_t If, uint8_t Ep); +#endif /* USE_USBD_COMPOSITE */ + /** * @} */ @@ -214,5 +330,3 @@ void USBD_AUDIO_Sync(USBD_HandleTypeDef *pdev, AUDIO_OffsetTypeDef offset); /** * @} */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Inc/usbd_audio_if_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Inc/usbd_audio_if_template.h index 21cee0d7..02aa1b64 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Inc/usbd_audio_if_template.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Inc/usbd_audio_if_template.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -41,5 +40,3 @@ extern USBD_AUDIO_ItfTypeDef USBD_AUDIO_Template_fops; #endif #endif /* __USBD_AUDIO_IF_TEMPLATE_H */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Src/usbd_audio.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Src/usbd_audio.c index acaaf053..a0a09416 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Src/usbd_audio.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Src/usbd_audio.c @@ -4,6 +4,18 @@ * @author MCD Application Team * @brief This file provides the Audio core functions. * + * + ****************************************************************************** + * @attention + * + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** * @verbatim * * =================================================================== @@ -38,18 +50,6 @@ * * * @endverbatim - * - ****************************************************************************** - * @attention - * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> - * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * ****************************************************************************** */ @@ -93,11 +93,15 @@ EndBSPDependencies */ /** @defgroup USBD_AUDIO_Private_Macros * @{ */ -#define AUDIO_SAMPLE_FREQ(frq) (uint8_t)(frq), (uint8_t)((frq >> 8)), (uint8_t)((frq >> 16)) +#define AUDIO_SAMPLE_FREQ(frq) \ + (uint8_t)(frq), (uint8_t)((frq >> 8)), (uint8_t)((frq >> 16)) -#define AUDIO_PACKET_SZE(frq) (uint8_t)(((frq * 2U * 2U)/1000U) & 0xFFU), \ - (uint8_t)((((frq * 2U * 2U)/1000U) >> 8) & 0xFFU) +#define AUDIO_PACKET_SZE(frq) \ + (uint8_t)(((frq * 2U * 2U) / 1000U) & 0xFFU), (uint8_t)((((frq * 2U * 2U) / 1000U) >> 8) & 0xFFU) +#ifdef USE_USBD_COMPOSITE +#define AUDIO_PACKET_SZE_WORD(frq) (uint32_t)((((frq) * 2U * 2U)/1000U)) +#endif /* USE_USBD_COMPOSITE */ /** * @} */ @@ -111,9 +115,10 @@ static uint8_t USBD_AUDIO_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx); static uint8_t USBD_AUDIO_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); - +#ifndef USE_USBD_COMPOSITE static uint8_t *USBD_AUDIO_GetCfgDesc(uint16_t *length); static uint8_t *USBD_AUDIO_GetDeviceQualifierDesc(uint16_t *length); +#endif /* USE_USBD_COMPOSITE */ static uint8_t USBD_AUDIO_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum); static uint8_t USBD_AUDIO_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum); static uint8_t USBD_AUDIO_EP0_RxReady(USBD_HandleTypeDef *pdev); @@ -124,6 +129,7 @@ static uint8_t USBD_AUDIO_IsoINIncomplete(USBD_HandleTypeDef *pdev, uint8_t epnu static uint8_t USBD_AUDIO_IsoOutIncomplete(USBD_HandleTypeDef *pdev, uint8_t epnum); static void AUDIO_REQ_GetCurrent(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); static void AUDIO_REQ_SetCurrent(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); +static void *USBD_AUDIO_GetAudioHeaderDesc(uint8_t *pConfDesc); /** * @} @@ -145,12 +151,20 @@ USBD_ClassTypeDef USBD_AUDIO = USBD_AUDIO_SOF, USBD_AUDIO_IsoINIncomplete, USBD_AUDIO_IsoOutIncomplete, +#ifdef USE_USBD_COMPOSITE + NULL, + NULL, + NULL, + NULL, +#else USBD_AUDIO_GetCfgDesc, USBD_AUDIO_GetCfgDesc, USBD_AUDIO_GetCfgDesc, USBD_AUDIO_GetDeviceQualifierDesc, +#endif /* USE_USBD_COMPOSITE */ }; +#ifndef USE_USBD_COMPOSITE /* USB AUDIO device Configuration Descriptor */ __ALIGN_BEGIN static uint8_t USBD_AUDIO_CfgDesc[USB_AUDIO_CONFIG_DESC_SIZ] __ALIGN_END = { @@ -166,7 +180,7 @@ __ALIGN_BEGIN static uint8_t USBD_AUDIO_CfgDesc[USB_AUDIO_CONFIG_DESC_SIZ] __ALI 0xC0, /* bmAttributes: Bus Powered according to user configuration */ #else 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif +#endif /* USBD_SELF_POWERED */ USBD_MAX_POWER, /* MaxPower (mA) */ /* 09 byte*/ @@ -317,7 +331,9 @@ __ALIGN_BEGIN static uint8_t USBD_AUDIO_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIE 0x01, 0x00, }; +#endif /* USE_USBD_COMPOSITE */ +static uint8_t AUDIOOutEpAdd = AUDIO_OUT_EP; /** * @} */ @@ -343,24 +359,30 @@ static uint8_t USBD_AUDIO_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) if (haudio == NULL) { - pdev->pClassData = NULL; + pdev->pClassDataCmsit[pdev->classId] = NULL; return (uint8_t)USBD_EMEM; } - pdev->pClassData = (void *)haudio; + pdev->pClassDataCmsit[pdev->classId] = (void *)haudio; + pdev->pClassData = pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + AUDIOOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_ISOC, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (pdev->dev_speed == USBD_SPEED_HIGH) { - pdev->ep_out[AUDIO_OUT_EP & 0xFU].bInterval = AUDIO_HS_BINTERVAL; + pdev->ep_out[AUDIOOutEpAdd & 0xFU].bInterval = AUDIO_HS_BINTERVAL; } else /* LOW and FULL-speed endpoints */ { - pdev->ep_out[AUDIO_OUT_EP & 0xFU].bInterval = AUDIO_FS_BINTERVAL; + pdev->ep_out[AUDIOOutEpAdd & 0xFU].bInterval = AUDIO_FS_BINTERVAL; } /* Open EP OUT */ - (void)USBD_LL_OpenEP(pdev, AUDIO_OUT_EP, USBD_EP_TYPE_ISOC, AUDIO_OUT_PACKET); - pdev->ep_out[AUDIO_OUT_EP & 0xFU].is_used = 1U; + (void)USBD_LL_OpenEP(pdev, AUDIOOutEpAdd, USBD_EP_TYPE_ISOC, AUDIO_OUT_PACKET); + pdev->ep_out[AUDIOOutEpAdd & 0xFU].is_used = 1U; haudio->alt_setting = 0U; haudio->offset = AUDIO_OFFSET_UNKNOWN; @@ -369,15 +391,15 @@ static uint8_t USBD_AUDIO_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) haudio->rd_enable = 0U; /* Initialize the Audio output Hardware layer */ - if (((USBD_AUDIO_ItfTypeDef *)pdev->pUserData)->Init(USBD_AUDIO_FREQ, - AUDIO_DEFAULT_VOLUME, - 0U) != 0U) + if (((USBD_AUDIO_ItfTypeDef *)pdev->pUserData[pdev->classId])->Init(USBD_AUDIO_FREQ, + AUDIO_DEFAULT_VOLUME, + 0U) != 0U) { return (uint8_t)USBD_FAIL; } /* Prepare Out endpoint to receive 1st packet */ - (void)USBD_LL_PrepareReceive(pdev, AUDIO_OUT_EP, haudio->buffer, + (void)USBD_LL_PrepareReceive(pdev, AUDIOOutEpAdd, haudio->buffer, AUDIO_OUT_PACKET); return (uint8_t)USBD_OK; @@ -394,16 +416,22 @@ static uint8_t USBD_AUDIO_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) { UNUSED(cfgidx); +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + AUDIOOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_ISOC, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + /* Open EP OUT */ - (void)USBD_LL_CloseEP(pdev, AUDIO_OUT_EP); - pdev->ep_out[AUDIO_OUT_EP & 0xFU].is_used = 0U; - pdev->ep_out[AUDIO_OUT_EP & 0xFU].bInterval = 0U; + (void)USBD_LL_CloseEP(pdev, AUDIOOutEpAdd); + pdev->ep_out[AUDIOOutEpAdd & 0xFU].is_used = 0U; + pdev->ep_out[AUDIOOutEpAdd & 0xFU].bInterval = 0U; /* DeInit physical Interface components */ - if (pdev->pClassData != NULL) + if (pdev->pClassDataCmsit[pdev->classId] != NULL) { - ((USBD_AUDIO_ItfTypeDef *)pdev->pUserData)->DeInit(0U); - (void)USBD_free(pdev->pClassData); + ((USBD_AUDIO_ItfTypeDef *)pdev->pUserData[pdev->classId])->DeInit(0U); + (void)USBD_free(pdev->pClassDataCmsit[pdev->classId]); + pdev->pClassDataCmsit[pdev->classId] = NULL; pdev->pClassData = NULL; } @@ -426,7 +454,7 @@ static uint8_t USBD_AUDIO_Setup(USBD_HandleTypeDef *pdev, uint16_t status_info = 0U; USBD_StatusTypeDef ret = USBD_OK; - haudio = (USBD_AUDIO_HandleTypeDef *)pdev->pClassData; + haudio = (USBD_AUDIO_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (haudio == NULL) { @@ -471,10 +499,17 @@ static uint8_t USBD_AUDIO_Setup(USBD_HandleTypeDef *pdev, case USB_REQ_GET_DESCRIPTOR: if ((req->wValue >> 8) == AUDIO_DESCRIPTOR_TYPE) { - pbuf = USBD_AUDIO_CfgDesc + 18; - len = MIN(USB_AUDIO_DESC_SIZ, req->wLength); - - (void)USBD_CtlSendData(pdev, pbuf, len); + pbuf = (uint8_t *)USBD_AUDIO_GetAudioHeaderDesc(pdev->pConfDesc); + if (pbuf != NULL) + { + len = MIN(USB_AUDIO_DESC_SIZ, req->wLength); + (void)USBD_CtlSendData(pdev, pbuf, len); + } + else + { + USBD_CtlError(pdev, req); + ret = USBD_FAIL; + } } break; @@ -529,11 +564,10 @@ static uint8_t USBD_AUDIO_Setup(USBD_HandleTypeDef *pdev, return (uint8_t)ret; } - +#ifndef USE_USBD_COMPOSITE /** * @brief USBD_AUDIO_GetCfgDesc * return configuration descriptor - * @param speed : current device speed * @param length : pointer data length * @retval pointer to descriptor buffer */ @@ -543,7 +577,7 @@ static uint8_t *USBD_AUDIO_GetCfgDesc(uint16_t *length) return USBD_AUDIO_CfgDesc; } - +#endif /* USE_USBD_COMPOSITE */ /** * @brief USBD_AUDIO_DataIn * handle data IN Stage @@ -569,7 +603,7 @@ static uint8_t USBD_AUDIO_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) static uint8_t USBD_AUDIO_EP0_RxReady(USBD_HandleTypeDef *pdev) { USBD_AUDIO_HandleTypeDef *haudio; - haudio = (USBD_AUDIO_HandleTypeDef *)pdev->pClassData; + haudio = (USBD_AUDIO_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (haudio == NULL) { @@ -582,7 +616,7 @@ static uint8_t USBD_AUDIO_EP0_RxReady(USBD_HandleTypeDef *pdev) if (haudio->control.unit == AUDIO_OUT_STREAMING_CTRL) { - ((USBD_AUDIO_ItfTypeDef *)pdev->pUserData)->MuteCtl(haudio->control.data[0]); + ((USBD_AUDIO_ItfTypeDef *)pdev->pUserData[pdev->classId])->MuteCtl(haudio->control.data[0]); haudio->control.cmd = 0U; haudio->control.len = 0U; } @@ -620,6 +654,7 @@ static uint8_t USBD_AUDIO_SOF(USBD_HandleTypeDef *pdev) * @brief USBD_AUDIO_SOF * handle SOF event * @param pdev: device instance + * @param offset: audio offset * @retval status */ void USBD_AUDIO_Sync(USBD_HandleTypeDef *pdev, AUDIO_OffsetTypeDef offset) @@ -627,12 +662,12 @@ void USBD_AUDIO_Sync(USBD_HandleTypeDef *pdev, AUDIO_OffsetTypeDef offset) USBD_AUDIO_HandleTypeDef *haudio; uint32_t BufferSize = AUDIO_TOTAL_BUF_SIZE / 2U; - if (pdev->pClassData == NULL) + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return; } - haudio = (USBD_AUDIO_HandleTypeDef *)pdev->pClassData; + haudio = (USBD_AUDIO_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; haudio->offset = offset; @@ -678,8 +713,8 @@ void USBD_AUDIO_Sync(USBD_HandleTypeDef *pdev, AUDIO_OffsetTypeDef offset) if (haudio->offset == AUDIO_OFFSET_FULL) { - ((USBD_AUDIO_ItfTypeDef *)pdev->pUserData)->AudioCmd(&haudio->buffer[0], - BufferSize, AUDIO_CMD_PLAY); + ((USBD_AUDIO_ItfTypeDef *)pdev->pUserData[pdev->classId])->AudioCmd(&haudio->buffer[0], + BufferSize, AUDIO_CMD_PLAY); haudio->offset = AUDIO_OFFSET_NONE; } } @@ -724,35 +759,40 @@ static uint8_t USBD_AUDIO_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) uint16_t PacketSize; USBD_AUDIO_HandleTypeDef *haudio; - haudio = (USBD_AUDIO_HandleTypeDef *)pdev->pClassData; +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + AUDIOOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_ISOC, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + haudio = (USBD_AUDIO_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (haudio == NULL) { return (uint8_t)USBD_FAIL; } - if (epnum == AUDIO_OUT_EP) + if (epnum == AUDIOOutEpAdd) { /* Get received data packet length */ PacketSize = (uint16_t)USBD_LL_GetRxDataSize(pdev, epnum); /* Packet received Callback */ - ((USBD_AUDIO_ItfTypeDef *)pdev->pUserData)->PeriodicTC(&haudio->buffer[haudio->wr_ptr], - PacketSize, AUDIO_OUT_TC); + ((USBD_AUDIO_ItfTypeDef *)pdev->pUserData[pdev->classId])->PeriodicTC(&haudio->buffer[haudio->wr_ptr], + PacketSize, AUDIO_OUT_TC); /* Increment the Buffer pointer or roll it back when all buffers are full */ haudio->wr_ptr += PacketSize; - if (haudio->wr_ptr == AUDIO_TOTAL_BUF_SIZE) + if (haudio->wr_ptr >= AUDIO_TOTAL_BUF_SIZE) { /* All buffers are full: roll back */ haudio->wr_ptr = 0U; if (haudio->offset == AUDIO_OFFSET_UNKNOWN) { - ((USBD_AUDIO_ItfTypeDef *)pdev->pUserData)->AudioCmd(&haudio->buffer[0], - AUDIO_TOTAL_BUF_SIZE / 2U, - AUDIO_CMD_START); + ((USBD_AUDIO_ItfTypeDef *)pdev->pUserData[pdev->classId])->AudioCmd(&haudio->buffer[0], + AUDIO_TOTAL_BUF_SIZE / 2U, + AUDIO_CMD_START); haudio->offset = AUDIO_OFFSET_NONE; } } @@ -766,7 +806,7 @@ static uint8_t USBD_AUDIO_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) } /* Prepare Out endpoint to receive next audio packet */ - (void)USBD_LL_PrepareReceive(pdev, AUDIO_OUT_EP, + (void)USBD_LL_PrepareReceive(pdev, AUDIOOutEpAdd, &haudio->buffer[haudio->wr_ptr], AUDIO_OUT_PACKET); } @@ -777,14 +817,14 @@ static uint8_t USBD_AUDIO_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) /** * @brief AUDIO_Req_GetCurrent * Handles the GET_CUR Audio control request. - * @param pdev: instance + * @param pdev: device instance * @param req: setup class request * @retval status */ static void AUDIO_REQ_GetCurrent(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { USBD_AUDIO_HandleTypeDef *haudio; - haudio = (USBD_AUDIO_HandleTypeDef *)pdev->pClassData; + haudio = (USBD_AUDIO_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (haudio == NULL) { @@ -801,14 +841,14 @@ static void AUDIO_REQ_GetCurrent(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef /** * @brief AUDIO_Req_SetCurrent * Handles the SET_CUR Audio control request. - * @param pdev: instance + * @param pdev: device instance * @param req: setup class request * @retval status */ static void AUDIO_REQ_SetCurrent(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { USBD_AUDIO_HandleTypeDef *haudio; - haudio = (USBD_AUDIO_HandleTypeDef *)pdev->pClassData; + haudio = (USBD_AUDIO_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (haudio == NULL) { @@ -826,7 +866,7 @@ static void AUDIO_REQ_SetCurrent(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef } } - +#ifndef USE_USBD_COMPOSITE /** * @brief DeviceQualifierDescriptor * return Device Qualifier descriptor @@ -839,9 +879,10 @@ static uint8_t *USBD_AUDIO_GetDeviceQualifierDesc(uint16_t *length) return USBD_AUDIO_DeviceQualifierDesc; } - +#endif /* USE_USBD_COMPOSITE */ /** * @brief USBD_AUDIO_RegisterInterface + * @param pdev: device instance * @param fops: Audio interface callback * @retval status */ @@ -853,14 +894,65 @@ uint8_t USBD_AUDIO_RegisterInterface(USBD_HandleTypeDef *pdev, return (uint8_t)USBD_FAIL; } - pdev->pUserData = fops; + pdev->pUserData[pdev->classId] = fops; return (uint8_t)USBD_OK; } + +#ifdef USE_USBD_COMPOSITE /** - * @} + * @brief USBD_AUDIO_GetEpPcktSze + * @param pdev: device instance (reserved for future use) + * @param If: Interface number (reserved for future use) + * @param Ep: Endpoint number (reserved for future use) + * @retval status */ +uint32_t USBD_AUDIO_GetEpPcktSze(USBD_HandleTypeDef *pdev, uint8_t If, uint8_t Ep) +{ + uint32_t mps; + + UNUSED(pdev); + UNUSED(If); + UNUSED(Ep); + + mps = AUDIO_PACKET_SZE_WORD(USBD_AUDIO_FREQ); + /* Return the wMaxPacketSize value in Bytes (Freq(Samples)*2(Stereo)*2(HalfWord)) */ + return mps; +} +#endif /* USE_USBD_COMPOSITE */ + +/** + * @brief USBD_AUDIO_GetAudioHeaderDesc + * This function return the Audio descriptor + * @param pdev: device instance + * @param pConfDesc: pointer to Bos descriptor + * @retval pointer to the Audio AC Header descriptor + */ +static void *USBD_AUDIO_GetAudioHeaderDesc(uint8_t *pConfDesc) +{ + USBD_ConfigDescTypeDef *desc = (USBD_ConfigDescTypeDef *)(void *)pConfDesc; + USBD_DescHeaderTypeDef *pdesc = (USBD_DescHeaderTypeDef *)(void *)pConfDesc; + uint8_t *pAudioDesc = NULL; + uint16_t ptr; + + if (desc->wTotalLength > desc->bLength) + { + ptr = desc->bLength; + + while (ptr < desc->wTotalLength) + { + pdesc = USBD_GetNextDesc((uint8_t *)pdesc, &ptr); + if ((pdesc->bDescriptorType == AUDIO_INTERFACE_DESCRIPTOR_TYPE) && + (pdesc->bDescriptorSubType == AUDIO_CONTROL_HEADER)) + { + pAudioDesc = (uint8_t *)pdesc; + break; + } + } + } + return pAudioDesc; +} /** * @} @@ -871,4 +963,7 @@ uint8_t USBD_AUDIO_RegisterInterface(USBD_HandleTypeDef *pdev, * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ + +/** + * @} + */ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Src/usbd_audio_if_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Src/usbd_audio_if_template.c index 7c97c751..c9db989f 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Src/usbd_audio_if_template.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/AUDIO/Src/usbd_audio_if_template.c @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -197,5 +196,3 @@ static int8_t TEMPLATE_GetState(void) * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ - diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/BillBoard/Inc/usbd_billboard.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/BillBoard/Inc/usbd_billboard.h index 8bff7b35..80451404 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/BillBoard/Inc/usbd_billboard.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/BillBoard/Inc/usbd_billboard.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2019 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2019 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -141,7 +140,7 @@ extern USBD_ClassTypeDef USBD_BB; #if (USBD_CLASS_BOS_ENABLED == 1) void *USBD_BB_GetCapDesc(USBD_HandleTypeDef *pdev, uint8_t *buf); void *USBD_BB_GetAltModeDesc(USBD_HandleTypeDef *pdev, uint8_t *buf, uint8_t idx); -#endif +#endif /* (USBD_CLASS_BOS_ENABLED == 1) */ /** * @} @@ -159,5 +158,3 @@ void *USBD_BB_GetAltModeDesc(USBD_HandleTypeDef *pdev, uint8_t *buf, uint8_t idx /** * @} */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/BillBoard/Src/usbd_billboard.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/BillBoard/Src/usbd_billboard.c index e0bacaa0..f86c003b 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/BillBoard/Src/usbd_billboard.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/BillBoard/Src/usbd_billboard.c @@ -7,6 +7,17 @@ * - Initialization and Configuration of high and low layer * - Enumeration as BillBoard Device * - Error management + ****************************************************************************** + * @attention + * + * Copyright (c) 2019 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** * @verbatim * * =================================================================== @@ -23,17 +34,6 @@ * @endverbatim * ****************************************************************************** - * @attention - * - * <h2><center>© Copyright (c) 2019 STMicroelectronics. - * All rights reserved.</center></h2> - * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** */ /* Includes ------------------------------------------------------------------*/ @@ -93,7 +93,7 @@ static uint8_t *USBD_BB_GetOtherSpeedCfgDesc(uint16_t *length); #if (USBD_CLASS_BOS_ENABLED == 1) USBD_BB_DescHeader_t *USBD_BB_GetNextDesc(uint8_t *pbuf, uint16_t *ptr); -#endif +#endif /* USBD_CLASS_BOS_ENABLED */ @@ -122,7 +122,7 @@ USBD_ClassTypeDef USBD_BB = USBD_BB_GetDeviceQualifierDesc, #if (USBD_SUPPORT_USER_STRING_DESC == 1U) NULL, -#endif +#endif /* USBD_SUPPORT_USER_STRING_DESC */ }; /* USB Standard Device Qualifier Descriptor */ @@ -154,7 +154,7 @@ __ALIGN_BEGIN static uint8_t USBD_BB_CfgDesc[USB_BB_CONFIG_DESC_SIZ] __ALIGN_EN 0xC0, /* bmAttributes: Bus Powered according to user configuration */ #else 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif +#endif /* USBD_SELF_POWERED */ USBD_MAX_POWER, /* MaxPower (mA) */ /* 09 */ @@ -185,7 +185,7 @@ __ALIGN_BEGIN static uint8_t USBD_BB_OtherSpeedCfgDesc[USB_BB_CONFIG_DESC_SIZ] 0xC0, /* bmAttributes: Bus Powered according to user configuration */ #else 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif +#endif /* USBD_SELF_POWERED */ USBD_MAX_POWER, /* MaxPower (mA) */ /************** Descriptor of BillBoard interface ****************/ @@ -420,7 +420,7 @@ void *USBD_BB_GetCapDesc(USBD_HandleTypeDef *pdev, uint8_t *pBosDesc) UNUSED(pdev); USBD_BB_DescHeader_t *pdesc = (USBD_BB_DescHeader_t *)(void *)pBosDesc; - USBD_BosDescTypedef *desc = (USBD_BosDescTypedef *)(void *)pBosDesc; + USBD_BosDescTypeDef *desc = (USBD_BosDescTypeDef *)(void *)pBosDesc; USBD_BosBBCapDescTypedef *pCapDesc = NULL; uint16_t ptr; @@ -456,7 +456,7 @@ void *USBD_BB_GetAltModeDesc(USBD_HandleTypeDef *pdev, uint8_t *pBosDesc, uint8_ UNUSED(pdev); USBD_BB_DescHeader_t *pdesc = (USBD_BB_DescHeader_t *)(void *)pBosDesc; - USBD_BosDescTypedef *desc = (USBD_BosDescTypedef *)(void *)pBosDesc; + USBD_BosDescTypeDef *desc = (USBD_BosDescTypeDef *)(void *)pBosDesc; USBD_BB_AltModeCapDescTypeDef *pAltModDesc = NULL; uint8_t cnt = 0U; uint16_t ptr; @@ -485,7 +485,7 @@ void *USBD_BB_GetAltModeDesc(USBD_HandleTypeDef *pdev, uint8_t *pBosDesc, uint8_ } return (void *)pAltModDesc; } -#endif +#endif /* USBD_CLASS_BOS_ENABLED */ /** * @} @@ -500,5 +500,3 @@ void *USBD_BB_GetAltModeDesc(USBD_HandleTypeDef *pdev, uint8_t *pBosDesc, uint8_ /** * @} */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid.h new file mode 100644 index 00000000..d3b96a53 --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid.h @@ -0,0 +1,373 @@ +/** + ****************************************************************************** + * @file usbd_ccid.h + * @author MCD Application Team + * @brief header file for the usbd_ccid.c file. + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Define to prevent recursive inclusion -------------------------------------*/ +#ifndef __USBD_CCID_H +#define __USBD_CCID_H + +#ifdef __cplusplus +extern "C" { +#endif + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_ioreq.h" + +/** @addtogroup STM32_USB_DEVICE_LIBRARY + * @{ + */ + +/** @defgroup usbd_cdc + * @brief This file is the Header file for usbd_ccid.c + * @{ + */ + + +/** @defgroup usbd_cdc_Exported_Defines + * @{ + */ +#ifndef CCID_IN_EP +#define CCID_IN_EP 0x81U /* EP1 for data IN */ +#endif /* CCID_IN_EP */ + +#ifndef CCID_OUT_EP +#define CCID_OUT_EP 0x01U /* EP1 for data OUT */ +#endif /* CCID_OUT_EP */ + +#ifndef CCID_CMD_EP +#define CCID_CMD_EP 0x82U /* EP2 for CCID commands */ +#endif /* CCID_CMD_EP */ + +#ifndef CCID_CMD_HS_BINTERVAL +#define CCID_CMD_HS_BINTERVAL 0x10U +#endif /* CCID_CMD_HS_BINTERVAL */ + +#ifndef CCID_CMD_FS_BINTERVAL +#define CCID_CMD_FS_BINTERVAL 0x10U +#endif /* CCID_CMD_FS_BINTERVAL */ + + +#define CCID_DATA_HS_MAX_PACKET_SIZE 512U /* Endpoint IN & OUT Packet size */ +#define CCID_DATA_FS_MAX_PACKET_SIZE 64U /* Endpoint IN & OUT Packet size */ +#define CCID_CMD_PACKET_SIZE 8U /* Control Endpoint Packet size */ + +#define USB_CCID_CONFIG_DESC_SIZ 93U +#define CCID_DATA_HS_IN_PACKET_SIZE CCID_DATA_HS_MAX_PACKET_SIZE +#define CCID_DATA_HS_OUT_PACKET_SIZE CCID_DATA_HS_MAX_PACKET_SIZE + +#define CCID_DATA_FS_IN_PACKET_SIZE CCID_DATA_FS_MAX_PACKET_SIZE +#define CCID_DATA_FS_OUT_PACKET_SIZE CCID_DATA_FS_MAX_PACKET_SIZE + +/*---------------------------------------------------------------------*/ +/* CCID definitions */ +/*---------------------------------------------------------------------*/ +#define CCID_SEND_ENCAPSULATED_COMMAND 0x00U +#define CCID_GET_ENCAPSULATED_RESPONSE 0x01U +#define CCID_SET_COMM_FEATURE 0x02U +#define CCID_GET_COMM_FEATURE 0x03U +#define CCID_CLEAR_COMM_FEATURE 0x04U +#define CCID_SET_LINE_CODING 0x20U +#define CCID_GET_LINE_CODING 0x21U +#define CCID_SET_CONTROL_LINE_STATE 0x22U +#define CCID_SEND_BREAK 0x23U + +/*---------------------------------------------------------------------*/ +#define REQUEST_ABORT 0x01U +#define REQUEST_GET_CLOCK_FREQUENCIES 0x02U +#define REQUEST_GET_DATA_RATES 0x03U + +/*---------------------------------------------------------------------*/ +/* The Smart Card Device Class Descriptor definitions */ +/*---------------------------------------------------------------------*/ + +#define CCID_INTERFACE_DESC_SIZE 0x09U +#define USB_DEVICE_CLASS_CCID 0x0BU +#define CCID_CLASS_DESC_SIZE 0x36U +#define CCID_DESC_TYPE 0x21U + +#ifndef CCID_VOLTAGE_SUPP +#define CCID_VOLTAGE_SUPP 0x07U +#endif /* CCID_VOLTAGE_SUPP */ +#ifndef USBD_CCID_PROTOCOL +#define USBD_CCID_PROTOCOL 0x03U +#endif /* USBD_CCID_PROTOCOL */ +#ifndef USBD_CCID_DEFAULT_CLOCK_FREQ +#define USBD_CCID_DEFAULT_CLOCK_FREQ 3600U +#endif /* USBD_CCID_DEFAULT_CLOCK_FREQ */ +#ifndef USBD_CCID_MAX_CLOCK_FREQ +#define USBD_CCID_MAX_CLOCK_FREQ USBD_CCID_DEFAULT_CLOCK_FREQ +#endif /* USBD_CCID_MAX_CLOCK_FREQ */ +#ifndef USBD_CCID_DEFAULT_DATA_RATE +#define USBD_CCID_DEFAULT_DATA_RATE 9677U +#endif /* USBD_CCID_DEFAULT_DATA_RATE */ +#ifndef USBD_CCID_MAX_DATA_RATE +#define USBD_CCID_MAX_DATA_RATE USBD_CCID_DEFAULT_DATA_RATE +#endif /* USBD_CCID_MAX_DATA_RATE */ +#ifndef USBD_CCID_MAX_INF_FIELD_SIZE +#define USBD_CCID_MAX_INF_FIELD_SIZE 254U +#endif /* USBD_CCID_MAX_INF_FIELD_SIZE */ +#ifndef CCID_MAX_BLOCK_SIZE_HEADER +#define CCID_MAX_BLOCK_SIZE_HEADER 271U +#endif /* CCID_MAX_BLOCK_SIZE_HEADER */ + +#define TPDU_EXCHANGE 0x01U +#define SHORT_APDU_EXCHANGE 0x02U +#define EXTENDED_APDU_EXCHANGE 0x04U +#define CHARACTER_EXCHANGE 0x00U + +#ifndef EXCHANGE_LEVEL_FEATURE +#define EXCHANGE_LEVEL_FEATURE TPDU_EXCHANGE +#endif /* EXCHANGE_LEVEL_FEATURE */ + +#define CCID_ENDPOINT_DESC_SIZE 0x07U + +#ifndef CCID_EP0_BUFF_SIZ +#define CCID_EP0_BUFF_SIZ 64U +#endif /* CCID_EP0_BUFF_SIZ */ + +#ifndef CCID_BULK_EPIN_SIZE +#define CCID_BULK_EPIN_SIZE 64U +#endif /* CCID_BULK_EPIN_SIZE */ + +#define CCID_INT_BUFF_SIZ 2U +/*---------------------------------------------------------------------*/ +/* + * CCID Class specification revision 1.1 + * Command Pipe. Bulk Messages + */ + +/* CCID Bulk Out Command definitions */ +#define PC_TO_RDR_ICCPOWERON 0x62U +#define PC_TO_RDR_ICCPOWEROFF 0x63U +#define PC_TO_RDR_GETSLOTSTATUS 0x65U +#define PC_TO_RDR_XFRBLOCK 0x6FU +#define PC_TO_RDR_GETPARAMETERS 0x6CU +#define PC_TO_RDR_RESETPARAMETERS 0x6DU +#define PC_TO_RDR_SETPARAMETERS 0x61U +#define PC_TO_RDR_ESCAPE 0x6BU +#define PC_TO_RDR_ICCCLOCK 0x6EU +#define PC_TO_RDR_T0APDU 0x6AU +#define PC_TO_RDR_SECURE 0x69U +#define PC_TO_RDR_MECHANICAL 0x71U +#define PC_TO_RDR_ABORT 0x72U +#define PC_TO_RDR_SETDATARATEANDCLOCKFREQUENCY 0x73U + +/* CCID Bulk In Command definitions */ +#define RDR_TO_PC_DATABLOCK 0x80U +#define RDR_TO_PC_SLOTSTATUS 0x81U +#define RDR_TO_PC_PARAMETERS 0x82U +#define RDR_TO_PC_ESCAPE 0x83U +#define RDR_TO_PC_DATARATEANDCLOCKFREQUENCY 0x84U + +/* CCID Interrupt In Command definitions */ +#define RDR_TO_PC_NOTIFYSLOTCHANGE 0x50U +#define RDR_TO_PC_HARDWAREERROR 0x51U + +/* Bulk-only Command Block Wrapper */ +#define ABDATA_SIZE 261U +#define CCID_CMD_HEADER_SIZE 10U +#define CCID_RESPONSE_HEADER_SIZE 10U + +/* Number of SLOTS. For single card, this value is 1 */ +#define CCID_NUMBER_OF_SLOTS 1U + +#define CARD_SLOT_FITTED 1U +#define CARD_SLOT_REMOVED 0U + +#define OFFSET_INT_BMESSAGETYPE 0x00U +#define OFFSET_INT_BMSLOTICCSTATE 0x01U +#define SLOT_ICC_PRESENT 0x01U +/* LSb : (0b = no ICC present, 1b = ICC present) */ +#define SLOT_ICC_CHANGE 0x02U +/* MSb : (0b = no change, 1b = change) */ + + +/** + * @} + */ + + +/** @defgroup USBD_CORE_Exported_TypesDefinitions + * @{ + */ + +/** + * @} + */ + +typedef struct +{ + uint32_t bitrate; + uint8_t format; + uint8_t paritytype; + uint8_t datatype; +} USBD_CCID_LineCodingTypeDef; + +typedef struct +{ + uint8_t bMessageType; /* Offset = 0*/ + uint32_t dwLength; /* Offset = 1, The length field (dwLength) is the length + of the message not including the 10-byte header.*/ + uint8_t bSlot; /* Offset = 5*/ + uint8_t bSeq; /* Offset = 6*/ + uint8_t bSpecific_0; /* Offset = 7*/ + uint8_t bSpecific_1; /* Offset = 8*/ + uint8_t bSpecific_2; /* Offset = 9*/ + uint8_t abData [ABDATA_SIZE]; /* Offset = 10, For reference, the absolute + maximum block size for a TPDU T=0 block is 260 bytes + (5 bytes command; 255 bytes data), + or for a TPDU T=1 block is 259 bytes, + or for a short APDU T=1 block is 261 bytes, + or for an extended APDU T=1 block is 65544 bytes.*/ +} __PACKED USBD_CCID_BulkOut_DataTypeDef; + +typedef struct +{ + uint8_t bMessageType; /* Offset = 0 */ + uint32_t dwLength; /* Offset = 1 */ + uint8_t bSlot; /* Offset = 5, Same as Bulk-OUT message */ + uint8_t bSeq; /* Offset = 6, Same as Bulk-OUT message */ + uint8_t bStatus; /* Offset = 7, Slot status as defined in section 6.2.6 */ + uint8_t bError; /* Offset = 8, Slot error as defined in section 6.2.6 */ + uint8_t bSpecific; /* Offset = 9 */ + uint8_t abData[ABDATA_SIZE]; /* Offset = 10 */ + uint16_t u16SizeToSend; +} __PACKED USBD_CCID_BulkIn_DataTypeDef; + +typedef struct +{ + __IO uint8_t SlotStatus; + __IO uint8_t SlotStatusChange; +} USBD_CCID_SlotStatusTypeDef; + + +typedef struct +{ + __IO uint8_t bAbortRequestFlag; + __IO uint8_t bSeq; + __IO uint8_t bSlot; +} USBD_CCID_ParamTypeDef; + +/* + * CCID Class specification revision 1.1 + * Smart Card Device Class Descriptor Table + */ + +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint16_t bcdCCID; + uint8_t bMaxSlotIndex; + uint8_t bVoltageSupport; + uint32_t dwProtocols; + uint32_t dwDefaultClock; + uint32_t dwMaximumClock; + uint8_t bNumClockSupported; + uint32_t dwDataRate; + uint32_t dwMaxDataRate; + uint8_t bNumDataRatesSupported; + uint32_t dwMaxIFSD; + uint32_t dwSynchProtocols; + uint32_t dwMechanical; + uint32_t dwFeatures; + uint32_t dwMaxCCIDMessageLength; + uint8_t bClassGetResponse; + uint8_t bClassEnvelope; + uint16_t wLcdLayout; + uint8_t bPINSupport; + uint8_t bMaxCCIDBusySlots; +} __PACKED USBD_CCID_DescTypeDef; + +typedef struct +{ + uint8_t data[CCID_DATA_HS_MAX_PACKET_SIZE / 4U]; /* Force 32-bit alignment */ + uint32_t UsbMessageLength; + uint8_t UsbIntData[CCID_CMD_PACKET_SIZE]; /* Buffer for the Interrupt In Data */ + uint32_t alt_setting; + + USBD_CCID_BulkIn_DataTypeDef UsbBlkInData; /* Buffer for the Out Data */ + USBD_CCID_BulkOut_DataTypeDef UsbBlkOutData; /* Buffer for the In Data */ + USBD_CCID_SlotStatusTypeDef SlotStatus; + USBD_CCID_ParamTypeDef USBD_CCID_Param; + + __IO uint32_t MaxPcktLen; + __IO uint8_t blkt_state; /* Bulk transfer state */ + + uint16_t slot_nb; + uint16_t seq_nb; +} USBD_CCID_HandleTypeDef; + +/** @defgroup USBD_CORE_Exported_Macros + * @{ + */ + +/** + * @} + */ + +/** @defgroup USBD_CORE_Exported_Variables + * @{ + */ + +extern USBD_ClassTypeDef USBD_CCID; +#define USBD_CCID_CLASS &USBD_CCID +/** + * @} + */ + +/** @defgroup USB_CORE_Exported_Functions + * @{ + */ +typedef struct _USBD_CCID_Itf +{ + uint8_t (* Init)(USBD_HandleTypeDef *pdev); + uint8_t (* DeInit)(USBD_HandleTypeDef *pdev); + uint8_t (* Control)(uint8_t req, uint8_t *pbuf, uint16_t *length); + uint8_t (* Response_SendData)(USBD_HandleTypeDef *pdev, uint8_t *buf, uint16_t len); + uint8_t (* Send_Process)(uint8_t *Command, uint8_t *Data); + uint8_t (* SetSlotStatus)(USBD_HandleTypeDef *pdev); +} USBD_CCID_ItfTypeDef; + +/** + * @} + */ +/** @defgroup USB_CORE_Exported_Functions + * @{ + */ +uint8_t USBD_CCID_RegisterInterface(USBD_HandleTypeDef *pdev, + USBD_CCID_ItfTypeDef *fops); + +uint8_t USBD_CCID_IntMessage(USBD_HandleTypeDef *pdev); + + +/** + * @} + */ + +#ifdef __cplusplus +} +#endif + +#endif /* __USBD_CCID_H */ +/** + * @} + */ + +/** + * @} + */ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_cmd.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_cmd.h new file mode 100644 index 00000000..91e27207 --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_cmd.h @@ -0,0 +1,222 @@ +/** + ****************************************************************************** + * @file usbd_ccid_cmd.h + * @author MCD Application Team + * @brief header file for the usbd_ccid_cmd.c file. + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Define to prevent recursive inclusion -------------------------------------*/ +#ifndef __USBD_CCID_CMD_H +#define __USBD_CCID_CMD_H + +#ifdef __cplusplus +extern "C" { +#endif + +/* Includes ------------------------------------------------------------------*/ +#ifndef __USBD_CCID_IF_H +#include "usbd_ccid_if_template.h" +#endif /* __USBD_CCID_IF_H */ + +#ifndef __USBD_CCID_SC_IF_H +#include "usbd_ccid_sc_if_template.h" +#endif /* __USBD_CCID_SC_IF_H */ + + +/* Exported types ------------------------------------------------------------*/ +/* Exported macro ------------------------------------------------------------*/ + +/******************************************************************************/ +/* ERROR CODES for USB Bulk In Messages : bError */ +/******************************************************************************/ + +#define SLOT_NO_ERROR 0x81U +#define SLOTERROR_UNKNOWN 0x82U +/*----------------------------------------------------------------------------*/ + +/* Index of not supported / incorrect message parameter : 7Fh to 01h */ +/* These Values are used for Return Types between Firmware Layers */ +/* + Failure of a command + The CCID cannot parse one parameter or the ICC is not supporting one parameter. + Then the Slot Error register contains the index of the first bad parameter as a + positive number (1-127). For instance, if the CCID receives an ICC command to + an unimplemented slot, then the Slot Error register shall be set to 5 (index of bSlot field) */ + +/* + * CCID Class specification revision 1.1 + */ + +/* Following Parameters used in PC_to_RDR_XfrBlock */ +#define SLOTERROR_BAD_LENTGH 0x01U +#define SLOTERROR_BAD_SLOT 0x05U +#define SLOTERROR_BAD_POWERSELECT 0x07U +#define SLOTERROR_BAD_PROTOCOLNUM 0x07U +#define SLOTERROR_BAD_CLOCKCOMMAND 0x07U +#define SLOTERROR_BAD_ABRFU_3B 0x07U +#define SLOTERROR_BAD_BMCHANGES 0x07U +#define SLOTERROR_BAD_BFUNCTION_MECHANICAL 0x07U +#define SLOTERROR_BAD_ABRFU_2B 0x08U +#define SLOTERROR_BAD_LEVELPARAMETER 0x08U +#define SLOTERROR_BAD_FIDI 0x0AU +#define SLOTERROR_BAD_T01CONVCHECKSUM 0x0BU +#define SLOTERROR_BAD_GUARDTIME 0x0CU +#define SLOTERROR_BAD_WAITINGINTEGER 0x0DU +#define SLOTERROR_BAD_CLOCKSTOP 0x0EU +#define SLOTERROR_BAD_IFSC 0x0FU +#define SLOTERROR_BAD_NAD 0x10U +#define SLOTERROR_BAD_DWLENGTH 0x08U + + +/*---------- Table 6.2-2 Slot error register when bmCommandStatus = 1 */ +#define SLOTERROR_CMD_ABORTED 0xFFU +#define SLOTERROR_ICC_MUTE 0xFEU +#define SLOTERROR_XFR_PARITY_ERROR 0xFDU +#define SLOTERROR_XFR_OVERRUN 0xFCU +#define SLOTERROR_HW_ERROR 0xFBU +#define SLOTERROR_BAD_ATR_TS 0xF8U +#define SLOTERROR_BAD_ATR_TCK 0xF7U +#define SLOTERROR_ICC_PROTOCOL_NOT_SUPPORTED 0xF6U +#define SLOTERROR_ICC_CLASS_NOT_SUPPORTED 0xF5U +#define SLOTERROR_PROCEDURE_BYTE_CONFLICT 0xF4U +#define SLOTERROR_DEACTIVATED_PROTOCOL 0xF3U +#define SLOTERROR_BUSY_WITH_AUTO_SEQUENCE 0xF2U +#define SLOTERROR_PIN_TIMEOUT 0xF0U +#define SLOTERROR_PIN_CANCELLED 0xEFU +#define SLOTERROR_CMD_SLOT_BUSY 0xE0U +#define SLOTERROR_CMD_NOT_SUPPORTED 0x00U + +/* Following Parameters used in PC_to_RDR_ResetParameters */ +/* DEFAULT_FIDI_VALUE */ +#ifndef DEFAULT_FIDI +#define DEFAULT_FIDI 0x11U +#endif /* DEFAULT_FIDI */ +#ifndef DEFAULT_T01CONVCHECKSUM +#define DEFAULT_T01CONVCHECKSUM 0x00U +#endif /* DEFAULT_T01CONVCHECKSUM */ +#ifndef DEFAULT_EXTRA_GUARDTIME +#define DEFAULT_EXTRA_GUARDTIME 0x00U +#endif /* DEFAULT_EXTRA_GUARDTIME */ +#ifndef DEFAULT_WAITINGINTEGER +#define DEFAULT_WAITINGINTEGER 0x0AU +#endif /* DEFAULT_WAITINGINTEGER */ +#ifndef DEFAULT_CLOCKSTOP +#define DEFAULT_CLOCKSTOP 0x00U +#endif /* DEFAULT_CLOCKSTOP */ +#ifndef DEFAULT_IFSC +#define DEFAULT_IFSC 0x20U +#endif /* DEFAULT_IFSC */ +#ifndef DEFAULT_NAD +#define DEFAULT_NAD 0x00U +#endif /* DEFAULT_NAD */ + +/* Following Parameters used in PC_to_RDR_IccPowerOn */ +#define VOLTAGE_SELECTION_AUTOMATIC 0xFFU +#define VOLTAGE_SELECTION_3V 0x02U +#define VOLTAGE_SELECTION_5V 0x01U +#define VOLTAGE_SELECTION_1V8 0x03U + +/* +Offset=0 bmICCStatus 2 bit 0, 1, 2 + 0 - An ICC is present and active (power is on and stable, RST is inactive) + 1 - An ICC is present and inactive (not activated or shut down by hardware error) + 2 - No ICC is present + 3 - RFU +Offset=0 bmRFU 4 bits 0 RFU +Offset=6 bmCommandStatus 2 bits 0, 1, 2 + 0 - Processed without error + 1 - Failed (error code provided by the error register) + 2 - Time Extension is requested + 3 - RFU + */ + +#define BM_ICC_PRESENT_ACTIVE 0x00U +#define BM_ICC_PRESENT_INACTIVE 0x01U +#define BM_ICC_NO_ICC_PRESENT 0x02U + +#define BM_COMMAND_STATUS_OFFSET 0x06U +#define BM_COMMAND_STATUS_NO_ERROR 0x00U +#define BM_COMMAND_STATUS_FAILED (0x01U << BM_COMMAND_STATUS_OFFSET) +#define BM_COMMAND_STATUS_TIME_EXTN (0x02 << BM_COMMAND_STATUS_OFFSET) + + +#if (ATR_T01 == 0) +#define SIZE_OF_ATR 19U +#else +#define SIZE_OF_ATR 15U +#endif /* (ATR_T01 == 0) */ + +/* defines for the CCID_CMD Layers */ +#define LEN_PROTOCOL_STRUCT_T0 5U +#define LEN_PROTOCOL_STRUCT_T1 7U + +#define BPROTOCOL_NUM_T0 0U +#define BPROTOCOL_NUM_T1 1U + +/************************************************************************************/ +/* ERROR CODES for RDR_TO_PC_HARDWAREERROR Message : bHardwareErrorCode */ +/************************************************************************************/ + +#define HARDWAREERRORCODE_OVERCURRENT 0x01U +#define HARDWAREERRORCODE_VOLTAGEERROR 0x02U +#define HARDWAREERRORCODE_OVERCURRENT_IT 0x04U +#define HARDWAREERRORCODE_VOLTAGEERROR_IT 0x08U + + + +#define CHK_PARAM_SLOT 0x01U +#define CHK_PARAM_DWLENGTH 0x02U +#define CHK_PARAM_ABRFU2 0x04U +#define CHK_PARAM_ABRFU3 0x08U +#define CHK_PARAM_CARD_PRESENT 0x10U +#define CHK_PARAM_ABORT 0x20U +#define CHK_ACTIVE_STATE 0x40U + + +/* Exported functions ------------------------------------------------------- */ +uint8_t PC_to_RDR_IccPowerOn(USBD_HandleTypeDef *pdev); +uint8_t PC_to_RDR_IccPowerOff(USBD_HandleTypeDef *pdev); +uint8_t PC_to_RDR_GetSlotStatus(USBD_HandleTypeDef *pdev); +uint8_t PC_to_RDR_XfrBlock(USBD_HandleTypeDef *pdev); +uint8_t PC_to_RDR_GetParameters(USBD_HandleTypeDef *pdev); +uint8_t PC_to_RDR_ResetParameters(USBD_HandleTypeDef *pdev); +uint8_t PC_to_RDR_SetParameters(USBD_HandleTypeDef *pdev); +uint8_t PC_to_RDR_Escape(USBD_HandleTypeDef *pdev); +uint8_t PC_to_RDR_IccClock(USBD_HandleTypeDef *pdev); +uint8_t PC_to_RDR_Abort(USBD_HandleTypeDef *pdev); +uint8_t PC_TO_RDR_T0Apdu(USBD_HandleTypeDef *pdev); +uint8_t PC_TO_RDR_Mechanical(USBD_HandleTypeDef *pdev); +uint8_t PC_TO_RDR_SetDataRateAndClockFrequency(USBD_HandleTypeDef *pdev); +uint8_t PC_TO_RDR_Secure(USBD_HandleTypeDef *pdev); + +void RDR_to_PC_DataBlock(uint8_t errorCode, USBD_HandleTypeDef *pdev); +void RDR_to_PC_NotifySlotChange(USBD_HandleTypeDef *pdev); +void RDR_to_PC_SlotStatus(uint8_t errorCode, USBD_HandleTypeDef *pdev); +void RDR_to_PC_Parameters(uint8_t errorCode, USBD_HandleTypeDef *pdev); +void RDR_to_PC_Escape(uint8_t errorCode, USBD_HandleTypeDef *pdev); +void RDR_to_PC_DataRateAndClockFrequency(uint8_t errorCode, USBD_HandleTypeDef *pdev); + +void CCID_UpdSlotStatus(USBD_HandleTypeDef *pdev, uint8_t slotStatus); +void CCID_UpdSlotChange(USBD_HandleTypeDef *pdev, uint8_t changeStatus); +uint8_t CCID_IsSlotStatusChange(USBD_HandleTypeDef *pdev); +uint8_t CCID_CmdAbort(USBD_HandleTypeDef *pdev, uint8_t slot, uint8_t seq); +uint8_t USBD_CCID_Transfer_Data_Request(USBD_HandleTypeDef *pdev, + uint8_t *dataPointer, uint16_t dataLen); + + +#ifdef __cplusplus +} +#endif + +#endif /* __USBD_CCID_CMD_H */ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_if_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_if_template.h new file mode 100644 index 00000000..af756896 --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_if_template.h @@ -0,0 +1,76 @@ +/** + ****************************************************************************** + * @file usbd_ccid_if_template.h + * @author MCD Application Team + * @brief header file for the usbd_ccid_if_template.c file. + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Define to prevent recursive inclusion -------------------------------------*/ +#ifndef __USBD_CCID_IF_TEMPLATE_H +#define __USBD_CCID_IF_TEMPLATE_H + +#ifdef __cplusplus +extern "C" { +#endif + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_ccid.h" +#include "usbd_ccid_cmd.h" + +#ifndef __USBD_CCID_SMARTCARD_H +#include "usbd_ccid_smartcard_template.h" +#endif /* __USBD_CCID_SMARTCARD_H */ + +/* Exported defines ----------------------------------------------------------*/ + +/*****************************************************************************/ +/*********************** CCID Bulk Transfer State machine ********************/ +/*****************************************************************************/ +#define CCID_STATE_IDLE 0U +#define CCID_STATE_DATA_OUT 1U +#define CCID_STATE_RECEIVE_DATA 2U +#define CCID_STATE_SEND_RESP 3U +#define CCID_STATE_DATAIN 4U +#define CCID_STATE_UNCORRECT_LENGTH 5U + +#define DIR_IN 0U +#define DIR_OUT 1U +#define BOTH_DIR 2U + +/************ Value of the Interrupt transfer status to set ******************/ +#define INTRSTATUS_COMPLETE 1U +#define INTRSTATUS_RESET 0U +/************** slot change status *******************************************/ +#define SLOTSTATUS_CHANGED 1U +#define SLOTSTATUS_RESET 0U + +/* Exported types ------------------------------------------------------------*/ +extern USBD_HandleTypeDef USBD_Device; + +/* CCID Interface callback */ +extern USBD_CCID_ItfTypeDef USBD_CCID_If_fops; + +/* Exported macros -----------------------------------------------------------*/ +/* Exported variables --------------------------------------------------------*/ +/* Exported functions ------------------------------------------------------- */ + +/** + * @} + */ + +#ifdef __cplusplus +} +#endif + +#endif /* __USBD_CCID_IF_TEMPLATE_H */ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_sc_if_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_sc_if_template.h new file mode 100644 index 00000000..0072f8be --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_sc_if_template.h @@ -0,0 +1,100 @@ +/** + ****************************************************************************** + * @file usbd_ccid_sc_if_template.h + * @author MCD Application Team + * @brief header file for the usbd_ccid_sc_if_template.c file. + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Define to prevent recursive inclusion -------------------------------------*/ +#ifndef __USBD_CCID_SC_IF_TEMPLATE_H +#define __USBD_CCID_SC_IF_TEMPLATE_H + +#ifdef __cplusplus +extern "C" { +#endif + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_ccid.h" +#include "usbd_ccid_cmd.h" + +#ifndef __USBD_CCID_SMARTCARD_H +#include "usbd_ccid_smartcard_template.h" +#endif /* __USBD_CCID_SMARTCARD_H */ + +/* Exported constants --------------------------------------------------------*/ +/* Exported types ------------------------------------------------------------*/ +typedef struct +{ + uint8_t voltage; /* Voltage for the Card Already Selected */ + uint8_t USART_GuardTime; + uint8_t SC_A2R_FiDi; + uint8_t SC_hostFiDi; + uint8_t USART_DefaultGuardTime; + uint32_t USART_BaudRate; +} SC_Param_t; + +#pragma pack(1) +typedef struct +{ + uint8_t bmFindexDindex; + uint8_t bmTCCKST0; + uint8_t bGuardTimeT0; + uint8_t bWaitingIntegerT0; + uint8_t bClockStop; + uint8_t bIfsc; + uint8_t bNad; +} Protocol_01_DataTypeDef; +#pragma pack() + +extern Protocol_01_DataTypeDef ProtocolData; +extern SC_Param_t SC_Param; + +/* Exported macro ------------------------------------------------------------*/ +#define MAX_EXTRA_GUARD_TIME (0xFF - DEFAULT_EXTRA_GUARDTIME) + +/* Following macros are used for SC_XferBlock command */ +#define XFER_BLK_SEND_DATA 1U /* Command is for issuing the data */ +#define XFER_BLK_RECEIVE_DATA 2U /* Command is for receiving the data */ +#define XFER_BLK_NO_DATA 3U /* Command type is No data exchange */ + +/* Exported functions ------------------------------------------------------- */ +/* APPLICATION LAYER ---------------------------------------------------------*/ +void SC_Itf_InitParams(void); +void SC_Itf_IccPowerOn(uint8_t voltage); +void SC_Itf_IccPowerOff(void); +uint8_t SC_GetState(void); +uint8_t SC_Itf_XferBlock(uint8_t *ptrBlock, uint32_t blockLen, + uint16_t expectedLen, + USBD_CCID_BulkIn_DataTypeDef *CCID_BulkIn_Data); + +uint8_t SC_Itf_SetParams(Protocol_01_DataTypeDef *pPtr, uint8_t T_01); +uint8_t SC_Itf_Escape(uint8_t *escapePtr, uint32_t escapeLen, + uint8_t *responseBuff, uint32_t *responseLen); + +uint8_t SC_Itf_SetClock(uint8_t bClockCommand); +uint8_t SC_Itf_T0Apdu(uint8_t bmChanges, uint8_t bClassGetResponse, + uint8_t bClassEnvelope); + +uint8_t SC_Itf_Mechanical(uint8_t bFunction); +uint8_t SC_Itf_SetDataRateAndClockFrequency(uint32_t dwClockFrequency, + uint32_t dwDataRate); + +uint8_t SC_Itf_Secure(uint32_t dwLength, uint8_t bBWI, uint16_t wLevelParameter, + uint8_t *pbuf, uint32_t *returnLen); + +#ifdef __cplusplus +} +#endif + +#endif /* __USBD_CCID_SC_IF_TEMPLATE_H */ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_smartcard_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_smartcard_template.h new file mode 100644 index 00000000..7730180a --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Inc/usbd_ccid_smartcard_template.h @@ -0,0 +1,279 @@ +/** + ****************************************************************************** + * @file usbd_ccid_smartcard_template.h + * @author MCD Application Team + * @brief header file for the usbd_ccid_smartcard_template.c file. + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Define to prevent recursive inclusion -------------------------------------*/ +#ifndef __USBD_CCID_SMARTCARD_TEMPLATE_H +#define __USBD_CCID_SMARTCARD_TEMPLATE_H + +#ifdef __cplusplus +extern "C" { +#endif + + +/* Includes ------------------------------------------------------------------*/ +#ifndef __USBD_CCID_IF_H +#include "usbd_ccid_if_template.h" +#endif /* __USBD_CCID_IF_H */ + +/* Exported constants --------------------------------------------------------*/ +#define T0_PROTOCOL 0x00U /* T0 protocol */ +#define T1_PROTOCOL 0x01U /* T1 protocol */ +#define DIRECT 0x3BU /* Direct bit convention */ +#define INDIRECT 0x3FU /* Indirect bit convention */ +#define SETUP_LENGTH 20U +#define HIST_LENGTH 20U + +#define SC_TRANSMIT_TIMEOUT 200U /* Direction to transmit */ +#define MAX_PROTOCOLLEVEL 7U /* Maximum levels of protocol */ +#define MAX_INTERFACEBYTE 4U /* Maximum number of interface bytes per protocol */ +#define LC_MAX 24U +#define SC_RECEIVE_TIMEOUT 0x8000U /* Direction to reader */ + +/* T=1 protocol constants */ +#define T1_I_BLOCK 0x00U /* PCB (I-block: b8 = 0) */ +#define T1_R_BLOCK 0x80U /* PCB (R-block: b8 b7 = 10) */ +#define T1_S_BLOCK 0xC0U /* PCB (S-block: b8 b7 = 11) */ + +/* I block */ +#define T1_I_SEQ_SHIFT 6U /* N(S) position (bit 7) */ + +/* R block */ +#define T1_IS_ERROR(pcb) ((pcb) & 0x0FU) +#define T1_EDC_ERROR 0x01U /* [b6..b1] = 0-N(R)-0001 */ +#define T1_OTHER_ERROR 0x02U /* [b6..b1] = 0-N(R)-0010 */ +#define T1_R_SEQ_SHIFT 4U /* N(R) position (b5) */ + +/* S block */ +#define T1_S_RESPONSE 0x20U /* If response: set bit b6, if request reset b6 in PCB S-Block */ +#define T1_S_RESYNC 0x00U /* RESYNCH: b6->b1: 000000 of PCB S-Block */ +#define T1_S_IFS 0x01U /* IFS: b6->b1: 000001 of PCB S-Block */ +#define T1_S_ABORT 0x02U /* ABORT: b6->b1: 000010 of PCB S-Block */ +#define T1_S_WTX 0x03U /* WTX: b6->b1: 000011 of PCB S-Block */ + +#define NAD 0U /* NAD byte position in the block */ +#define PCB 1U /* PCB byte position in the block */ +#define LEN 2U /* LEN byte position in the block */ +#define DATA 3U /* The position of the first byte of INF field in the block */ + +/* Modifiable parameters */ +#define SAD 0x0U /* Source address: reader (allowed values 0 -> 7) */ +#define DAD 0x0U /* Destination address: card (allowed values 0 -> 7) */ +#define IFSD_VALUE 254U /* Max length of INF field Supported by the reader */ +#define SC_FILE_SIZE 0x100U /* File size */ +#define SC_FILE_ID 0x0001U /* File identifier */ +#define SC_CLASS 0x00U + +/* Constant parameters */ +#define INS_SELECT_FILE 0xA4U /* Select file instruction */ +#define INS_READ_FILE 0xB0U /* Read file instruction */ +#define INS_WRITE_FILE 0xD6U /* Write file instruction */ +#define TRAILER_LENGTH 2U /* Trailer length (SW1 and SW2: 2 bytes) */ + +#define SC_T1_RECEIVE_SUCCESS 0U +#define SC_T1_BWT_TIMEOUT 1U +#define SC_T1_CWT_TIMEOUT 2U + +#define DEFAULT_FIDI_VALUE 0x11U +#define PPS_REQUEST 0xFFU + +/* SC Tree Structure ----------------------------------------------------------- + MasterFile + ________|___________ + | | | + System UserData Note +------------------------------------------------------------------------------*/ + +/* SC ADPU Command: Operation Code -------------------------------------------*/ +#define SC_CLA_NAME 0x00U + +/*------------------------ Data Area Management Commands ---------------------*/ +#define SC_SELECT_FILE 0xA4U +#define SC_GET_RESPONCE 0xC0U +#define SC_STATUS 0xF2U +#define SC_UPDATE_BINARY 0xD6U +#define SC_READ_BINARY 0xB0U +#define SC_WRITE_BINARY 0xD0U +#define SC_UPDATE_RECORD 0xDCU +#define SC_READ_RECORD 0xB2U + +/*-------------------------- Administrative Commands -------------------------*/ +#define SC_CREATE_FILE 0xE0U + +/*-------------------------- Safety Management Commands ----------------------*/ +#define SC_VERIFY 0x20U +#define SC_CHANGE 0x24U +#define SC_DISABLE 0x26U +#define SC_ENABLE 0x28U +#define SC_UNBLOCK 0x2CU +#define SC_EXTERNAL_AUTH 0x82U +#define SC_GET_CHALLENGE 0x84U + +/*-------------------------- Smartcard Interface Byte-------------------------*/ +#define SC_INTERFACEBYTE_TA 0U /* Interface byte TA(i) */ +#define SC_INTERFACEBYTE_TB 1U /* Interface byte TB(i) */ +#define SC_INTERFACEBYTE_TC 2U /* Interface byte TC(i) */ +#define SC_INTERFACEBYTE_TD 3U /* Interface byte TD(i) */ + +/*-------------------------- Answer to reset Commands ------------------------*/ +#define SC_GET_A2R 0x00U + +/* SC STATUS: Status Code ----------------------------------------------------*/ +#define SC_EF_SELECTED 0x9FU +#define SC_DF_SELECTED 0x9FU +#define SC_OP_TERMINATED 0x9000U + +/* Smartcard Voltage */ +#define SC_VOLTAGE_5V 0x00U +#define SC_VOLTAGE_3V 0x01U +#define SC_VOLTAGE_NOINIT 0xFFU +/*----------------- ATR Protocole supported ----------------------------------*/ +#define ATR_T01 0x00U + + +/* Exported types ------------------------------------------------------------*/ +typedef enum +{ + SC_POWER_ON = 0x00, + SC_RESET_LOW = 0x01, + SC_RESET_HIGH = 0x02, + SC_ACTIVE = 0x03, + SC_ACTIVE_ON_T0 = 0x04, + SC_ACTIVE_ON_T1 = 0x05, + SC_POWER_OFF = 0x06, + SC_NO_INIT = 0x07 + +} SC_State; + +/* Interface Byte structure - TA(i), TB(i), TC(i) and TD(i) ------------------*/ +typedef struct +{ + uint8_t Status; /* The Presence of the Interface byte */ + uint8_t Value; /* The Value of the Interface byte */ +} SC_InterfaceByteTypeDef; + +/* Protocol Level structure - ------------------------------------------------*/ +typedef struct +{ + SC_InterfaceByteTypeDef InterfaceByte[MAX_INTERFACEBYTE]; /* The Values of the Interface byte + TA(i), TB(i), TC(i)and TD(i) */ +} SC_ProtocolLevelTypeDef; + +/* ATR structure - Answer To Reset -------------------------------------------*/ +typedef struct +{ + uint8_t TS; /* Bit Convention Direct/Indirect */ + uint8_t T0; /* Each bit in the high nibble = Presence of the further interface byte; + Low nibble = Number of historical byte */ + SC_ProtocolLevelTypeDef T[MAX_PROTOCOLLEVEL]; /* Setup array */ + uint8_t Historical[HIST_LENGTH]; /* Historical array */ + uint8_t Tlength; /* Setup array dimension */ + uint8_t Hlength; /* Historical array dimension */ + uint8_t TCK; +} SC_ATRTypeDef; + +/* ADPU-Header command structure ---------------------------------------------*/ +typedef struct +{ + uint8_t CLA; /* Command class */ + uint8_t INS; /* Operation code */ + uint8_t P1; /* Selection Mode */ + uint8_t P2; /* Selection Option */ +} SC_HeaderTypeDef; + +/* ADPU-Body command structure -----------------------------------------------*/ +typedef struct +{ + uint8_t LC; /* Data field length */ + uint8_t Data[LC_MAX]; /* Command parameters */ + uint8_t LE; /* Expected length of data to be returned */ +} SC_BodyTypeDef; + +/* ADPU Command structure ----------------------------------------------------*/ +typedef struct +{ + SC_HeaderTypeDef Header; + SC_BodyTypeDef Body; +} SC_ADPU_CommandsTypeDef; + +/* SC response structure -----------------------------------------------------*/ +typedef struct +{ + uint8_t Data[LC_MAX]; /* Data returned from the card */ + uint8_t SW1; /* Command Processing status */ + uint8_t SW2; /* Command Processing qualification */ +} SC_ADPU_ResponseTypeDef; + +/* SC Command Status -----------------------------------------------------*/ +typedef enum +{ + SC_CS_FAILED = 0x00, + SC_CS_PIN_ENABLED = 0x01, + SC_CS_PIN_VERIFIED = 0x02, + SC_CS_READ = 0x03, + SC_CS_PIN_CHANGED = 0x04 + +} SC_Command_State; +/* SC Response Status -----------------------------------------------------*/ +typedef enum +{ + REP_OK = 0x00, + REP_NOT_OK = 0x01, + REP_NOT_SUPP = 0x02, + REP_ENABLED = 0x03, + REP_CHANGE = 0x04 + +} REP_Command_t; +/* Conforming of Command with ICC APP -----------------------------------------------------*/ +typedef enum +{ + Command_OK = 0x00, + Command_NOT_OK = 0x01, + +} Command_State_t; + +typedef enum +{ + SC_DISABLED = 0U, + SC_ENABLED = !SC_DISABLED +} SCPowerState; + +/* Exported macro ------------------------------------------------------------*/ +/* Exported functions ------------------------------------------------------- */ +/* APPLICATION LAYER ---------------------------------------------------------*/ +void SC_Handler(SC_State *SCState, SC_ADPU_CommandsTypeDef *SC_ADPU, SC_ADPU_ResponseTypeDef *SC_Response); +void SC_PowerCmd(SCPowerState NewState); +void SC_ParityErrorHandler(void); +void SC_PTSConfig(void); +uint8_t SC_Detect(void); +uint32_t SC_GetDTableValue(uint32_t idx); +void SC_VoltageConfig(uint32_t SC_Voltage); +void SC_SetState(SC_State scState); +void SC_IOConfig(void); + +extern uint8_t SC_ATR_Table[40]; +extern SC_ATRTypeDef SC_A2R; +extern SC_ADPU_ResponseTypeDef SC_Response; + +extern uint8_t ProtocolNUM_OUT; +extern SC_ADPU_CommandsTypeDef SC_ADPU; + +#ifdef __cplusplus +} +#endif + +#endif /* __USBD_CCID_SMARTCARD_TEMPLATE_H */ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid.c new file mode 100644 index 00000000..4c77b135 --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid.c @@ -0,0 +1,969 @@ +/** + ****************************************************************************** + * @file usbd_ccid.c + * @author MCD Application Team + * @brief This file provides the high layer firmware functions to manage + * all the functionalities of the USB CCID Class: + * + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + * @verbatim + * + * =================================================================== + * CCID Class Driver Description + * =================================================================== + * This module manages the Specification for Integrated Circuit(s) + * Cards Interface Revision 1.1 + * This driver implements the following aspects of the specification: + * - Device descriptor management + * - Configuration descriptor management + * - Enumeration as CCID device with 2 data endpoints (IN and OUT) and 1 command endpoint (IN) + * and enumeration for each implemented memory interface + * - Bulk OUT/IN data Transfers + * - Requests management + * + * @endverbatim + * + ****************************************************************************** + */ + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_ccid.h" +#include "usbd_ccid_cmd.h" +#include "usbd_ctlreq.h" +/** @addtogroup STM32_USB_DEVICE_LIBRARY + * @{ + */ + +/** @defgroup USBD_CCID + * @brief usbd core module + * @{ + */ + +/** @defgroup USBD_CCID_Private_TypesDefinitions + * @{ + */ +/** + * @} + */ + + +/** @defgroup USBD_CCID_Private_Defines + * @{ + */ +/** + * @} + */ + + +/** @defgroup USBD_CCID_Private_Macros + * @{ + */ + +/** + * @} + */ + + +/** @defgroup USBD_CCID_Private_FunctionPrototypes + * @{ + */ +static uint8_t USBD_CCID_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx); +static uint8_t USBD_CCID_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx); +static uint8_t USBD_CCID_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); +static uint8_t USBD_CCID_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum); +static uint8_t USBD_CCID_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum); +static uint8_t USBD_CCID_DispatchCommand(USBD_HandleTypeDef *pdev); +static uint8_t USBD_CCID_ReceiveCmdHeader(USBD_HandleTypeDef *pdev, + uint8_t *pDst, uint16_t u8length); +#ifndef USE_USBD_COMPOSITE +static uint8_t *USBD_CCID_GetHSCfgDesc(uint16_t *length); +static uint8_t *USBD_CCID_GetFSCfgDesc(uint16_t *length); +static uint8_t *USBD_CCID_GetOtherSpeedCfgDesc(uint16_t *length); +static uint8_t *USBD_CCID_GetDeviceQualifierDescriptor(uint16_t *length); +#endif /* USE_USBD_COMPOSITE */ + +/** + * @} + */ + +/** @defgroup USBD_CCID_Private_Variables + * @{ + */ + +static uint8_t CCIDInEpAdd = CCID_IN_EP; +static uint8_t CCIDOutEpAdd = CCID_OUT_EP; +static uint8_t CCIDCmdEpAdd = CCID_CMD_EP; + + +/* CCID interface class callbacks structure */ +USBD_ClassTypeDef USBD_CCID = +{ + USBD_CCID_Init, + USBD_CCID_DeInit, + USBD_CCID_Setup, + NULL, /*EP0_TxSent*/ + NULL, /*EP0_RxReady*/ + USBD_CCID_DataIn, + USBD_CCID_DataOut, + NULL, /*SOF */ + NULL, /*ISOIn*/ + NULL, /*ISOOut*/ +#ifdef USE_USBD_COMPOSITE + NULL, + NULL, + NULL, + NULL, +#else + USBD_CCID_GetHSCfgDesc, + USBD_CCID_GetFSCfgDesc, + USBD_CCID_GetOtherSpeedCfgDesc, + USBD_CCID_GetDeviceQualifierDescriptor, +#endif /* USE_USBD_COMPOSITE */ +}; + +#ifndef USE_USBD_COMPOSITE + +/* USB CCID device Configuration Descriptor */ +__ALIGN_BEGIN static uint8_t USBD_CCID_CfgDesc[USB_CCID_CONFIG_DESC_SIZ] __ALIGN_END = +{ + /* Configuration Descriptor */ + 0x09, /* bLength: Configuration Descriptor size */ + USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ + USB_CCID_CONFIG_DESC_SIZ, /* wTotalLength:no of returned bytes */ + 0x00, + 0x01, /* bNumInterfaces: 1 interface */ + 0x01, /* bConfigurationValue: */ + 0x00, /* iConfiguration: */ +#if (USBD_SELF_POWERED == 1U) + 0xC0, /* bmAttributes: Bus Powered according to user configuration */ +#else + 0x80, /* bmAttributes: Bus Powered according to user configuration */ +#endif /* USBD_SELF_POWERED */ + USBD_MAX_POWER, /* MaxPower (mA) */ + + /******************** CCID **** interface ********************/ + CCID_INTERFACE_DESC_SIZE, /* bLength: Interface Descriptor size */ + USB_DESC_TYPE_INTERFACE, /* bDescriptorType: */ + 0x00, /* bInterfaceNumber: Number of Interface */ + 0x00, /* bAlternateSetting: Alternate setting */ + 0x03, /* bNumEndpoints: 3 endpoints used */ + USB_DEVICE_CLASS_CCID, /* bInterfaceClass: user's interface for CCID */ + 0x00, /* bInterfaceSubClass : No subclass, + can be changed but no description in USB 2.0 Spec */ + 0x00, /* nInterfaceProtocol : None */ + 0x00, /* iInterface */ + + /******************* CCID class descriptor ********************/ + CCID_CLASS_DESC_SIZE, /* bLength: CCID Descriptor size */ + CCID_DESC_TYPE, /* bDescriptorType: Functional Descriptor type. */ + 0x10, /* bcdCCID(LSB): CCID Class Spec release number (1.1) */ + 0x01, /* bcdCCID(MSB) */ + + 0x00, /* bMaxSlotIndex :highest available slot on this device */ + CCID_VOLTAGE_SUPP, /* bVoltageSupport: bVoltageSupport: 5v, 3v and 1.8v */ + LOBYTE(USBD_CCID_PROTOCOL), /* dwProtocols: supports T=0 and T=1 */ + HIBYTE(USBD_CCID_PROTOCOL), + 0x00, + 0x00, + LOBYTE(USBD_CCID_DEFAULT_CLOCK_FREQ), /* dwDefaultClock: 3.6Mhz */ + HIBYTE(USBD_CCID_DEFAULT_CLOCK_FREQ), + 0x00, + 0x00, + LOBYTE(USBD_CCID_MAX_CLOCK_FREQ), /* dwMaximumClock */ + HIBYTE(USBD_CCID_MAX_CLOCK_FREQ), + 0x00, + 0x00, + 0x00, /* bNumClockSupported */ + LOBYTE(USBD_CCID_DEFAULT_DATA_RATE), /* dwDataRate: 9677 bps */ + HIBYTE(USBD_CCID_DEFAULT_DATA_RATE), + 0x00, + 0x00, + + LOBYTE(USBD_CCID_MAX_DATA_RATE), /* dwMaxDataRate */ + HIBYTE(USBD_CCID_MAX_DATA_RATE), + 0x00, + 0x00, + 0x35, /* bNumDataRatesSupported */ + + LOBYTE(USBD_CCID_MAX_INF_FIELD_SIZE), /* dwMaxIFSD: maximum IFSD supported for T=1 */ + HIBYTE(USBD_CCID_MAX_INF_FIELD_SIZE), + 0x00, + 0x00, + 0x00, 0x00, 0x00, 0x00, /* dwSynchProtocols */ + 0x00, 0x00, 0x00, 0x00, /* dwMechanical: no special characteristics */ + + 0xBA, 0x04, EXCHANGE_LEVEL_FEATURE, 0x00, /* dwFeatures */ + LOBYTE(CCID_MAX_BLOCK_SIZE_HEADER), /* dwMaxCCIDMessageLength: Maximum block size + header*/ + HIBYTE(CCID_MAX_BLOCK_SIZE_HEADER), + 0x00, + 0x00, + 0x00, /* bClassGetResponse*/ + 0x00, /* bClassEnvelope */ + 0x00, 0x00, /* wLcdLayout : 0000h no LCD. */ + 0x03, /* bPINSupport : PIN verification and PIN modification */ + 0x01, /* bMaxCCIDBusySlots */ + + /******************** CCID Endpoints ********************/ + CCID_ENDPOINT_DESC_SIZE, /* Endpoint descriptor length = 7 */ + USB_DESC_TYPE_ENDPOINT, /* Endpoint descriptor type */ + CCID_IN_EP, /* Endpoint address (IN, address 1) */ + USBD_EP_TYPE_BULK, /* Bulk endpoint type */ + + LOBYTE(CCID_DATA_FS_MAX_PACKET_SIZE), + HIBYTE(CCID_DATA_FS_MAX_PACKET_SIZE), + 0x00, /* Polling interval in milliseconds */ + CCID_ENDPOINT_DESC_SIZE, /* Endpoint descriptor length = 7 */ + USB_DESC_TYPE_ENDPOINT, /* Endpoint descriptor type */ + CCID_OUT_EP, /* Endpoint address (OUT, address 1) */ + USBD_EP_TYPE_BULK, /* Bulk endpoint type */ + + LOBYTE(CCID_DATA_FS_MAX_PACKET_SIZE), + HIBYTE(CCID_DATA_FS_MAX_PACKET_SIZE), + 0x00, /* Polling interval in milliseconds */ + CCID_ENDPOINT_DESC_SIZE, /* bLength: Endpoint Descriptor size */ + USB_DESC_TYPE_ENDPOINT, /* bDescriptorType:*/ + CCID_CMD_EP, /* bEndpointAddress: Endpoint Address (IN) */ + USBD_EP_TYPE_INTR, /* bmAttributes: Interrupt endpoint */ + LOBYTE(CCID_CMD_PACKET_SIZE), + HIBYTE(CCID_CMD_PACKET_SIZE), + CCID_CMD_FS_BINTERVAL /* Polling interval in milliseconds */ +}; + +/* USB Standard Device Descriptor */ +__ALIGN_BEGIN static uint8_t USBD_CCID_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_DESC] __ALIGN_END = +{ + USB_LEN_DEV_QUALIFIER_DESC, + USB_DESC_TYPE_DEVICE_QUALIFIER, + 0x00, + 0x02, + 0x00, + 0x00, + 0x00, + 0x40, + 0x01, + 0x00, +}; +#endif /* USE_USBD_COMPOSITE */ + +/** + * @} + */ + +/** @defgroup USBD_CCID_Private_Functions + * @{ + */ + +/** + * @brief USBD_CCID_Init + * Initialize the CCID interface + * @param pdev: device instance + * @param cfgidx: Configuration index + * @retval status + */ +static uint8_t USBD_CCID_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) +{ + USBD_CCID_HandleTypeDef *hccid; + UNUSED(cfgidx); + + /* Allocate CCID structure */ + hccid = (USBD_CCID_HandleTypeDef *)USBD_malloc(sizeof(USBD_CCID_HandleTypeDef)); + + if (hccid == NULL) + { + pdev->pClassDataCmsit[pdev->classId] = NULL; + return (uint8_t)USBD_EMEM; + } + + pdev->pClassDataCmsit[pdev->classId] = (void *)hccid; + pdev->pClassData = pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + CCIDInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + CCIDOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + CCIDCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + /* Init the CCID parameters into a state where it can receive a new command message */ + hccid->USBD_CCID_Param.bAbortRequestFlag = 0U; + hccid->USBD_CCID_Param.bSeq = 0U; + hccid->USBD_CCID_Param.bSlot = 0U; + hccid->MaxPcktLen = (pdev->dev_speed == USBD_SPEED_HIGH) ? \ + CCID_DATA_HS_MAX_PACKET_SIZE : CCID_DATA_FS_MAX_PACKET_SIZE; + + /* Open EP IN */ + (void)USBD_LL_OpenEP(pdev, CCIDInEpAdd, USBD_EP_TYPE_BULK, (uint16_t)hccid->MaxPcktLen); + pdev->ep_in[CCIDInEpAdd & 0xFU].is_used = 1U; + + /* Open EP OUT */ + (void)USBD_LL_OpenEP(pdev, CCIDOutEpAdd, USBD_EP_TYPE_BULK, (uint16_t)hccid->MaxPcktLen); + pdev->ep_out[CCIDOutEpAdd & 0xFU].is_used = 1U; + + /* Open INTR EP IN */ + (void)USBD_LL_OpenEP(pdev, CCIDCmdEpAdd, + USBD_EP_TYPE_INTR, CCID_CMD_PACKET_SIZE); + pdev->ep_in[CCIDCmdEpAdd & 0xFU].is_used = 1U; + + /* Init physical Interface components */ + ((USBD_CCID_ItfTypeDef *)pdev->pUserData[pdev->classId])->Init(pdev); + + /* Prepare Out endpoint to receive next packet */ + (void)USBD_LL_PrepareReceive(pdev, CCIDOutEpAdd, + hccid->data, hccid->MaxPcktLen); + + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_CCID_DeInit + * DeInitialize the CCID layer + * @param pdev: device instance + * @param cfgidx: Configuration index + * @retval status + */ +static uint8_t USBD_CCID_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) +{ + UNUSED(cfgidx); + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + CCIDInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + CCIDOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + CCIDCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + /* Close EP IN */ + (void)USBD_LL_CloseEP(pdev, CCIDInEpAdd); + pdev->ep_in[CCIDInEpAdd & 0xFU].is_used = 0U; + + /* Close EP OUT */ + (void)USBD_LL_CloseEP(pdev, CCIDOutEpAdd); + pdev->ep_out[CCIDOutEpAdd & 0xFU].is_used = 0U; + + /* Close EP Command */ + (void)USBD_LL_CloseEP(pdev, CCIDCmdEpAdd); + pdev->ep_in[CCIDCmdEpAdd & 0xFU].is_used = 0U; + + /* DeInit physical Interface components */ + if (pdev->pClassDataCmsit[pdev->classId] != NULL) + { + ((USBD_CCID_ItfTypeDef *)pdev->pUserData[pdev->classId])->DeInit(pdev); + (void)USBD_free(pdev->pClassDataCmsit[pdev->classId]); + pdev->pClassDataCmsit[pdev->classId] = NULL; + pdev->pClassData = NULL; + } + + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_CCID_Setup + * Handle the CCID specific requests + * @param pdev: instance + * @param req: usb requests + * @retval status + */ +static uint8_t USBD_CCID_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_CCID_ItfTypeDef *hCCIDitf = (USBD_CCID_ItfTypeDef *)pdev->pUserData[pdev->classId]; + USBD_StatusTypeDef ret = USBD_OK; + uint8_t ifalt = 0U; + uint16_t status_info = 0U; + uint16_t len; + + switch (req->bmRequest & USB_REQ_TYPE_MASK) + { + /* Class request */ + case USB_REQ_TYPE_CLASS : + if (req->wLength != 0U) + { + len = MIN(CCID_EP0_BUFF_SIZ, req->wLength); + if ((req->bmRequest & 0x80U) != 0U) + { + hCCIDitf->Control(req->bRequest, hccid->data, &len); + (void)USBD_CtlSendData(pdev, hccid->data, len); + } + else + { + (void)USBD_CtlPrepareRx(pdev, hccid->data, len); + } + } + else + { + len = 0U; + hCCIDitf->Control(req->bRequest, (uint8_t *)&req->wValue, &len); + } + break; + + /* Interface & Endpoint request */ + case USB_REQ_TYPE_STANDARD: + switch (req->bRequest) + { + case USB_REQ_GET_STATUS: + if (pdev->dev_state == USBD_STATE_CONFIGURED) + { + (void)USBD_CtlSendData(pdev, (uint8_t *)&status_info, 2U); + } + else + { + USBD_CtlError(pdev, req); + ret = USBD_FAIL; + } + break; + + case USB_REQ_GET_INTERFACE: + if (pdev->dev_state == USBD_STATE_CONFIGURED) + { + (void)USBD_CtlSendData(pdev, &ifalt, 1U); + } + else + { + USBD_CtlError(pdev, req); + ret = USBD_FAIL; + } + break; + + case USB_REQ_SET_INTERFACE: + if (pdev->dev_state != USBD_STATE_CONFIGURED) + { + USBD_CtlError(pdev, req); + ret = USBD_FAIL; + } + break; + + case USB_REQ_CLEAR_FEATURE: + break; + + default: + USBD_CtlError(pdev, req); + ret = USBD_FAIL; + break; + } + break; + + default: + USBD_CtlError(pdev, req); + ret = USBD_FAIL; + break; + } + + return (uint8_t)ret; +} + +/** + * @brief USBD_CCID_DataIn + * Data sent on non-control IN endpoint + * @param pdev: device instance + * @param epnum: endpoint number + * @retval status + */ +static uint8_t USBD_CCID_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + CCIDInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + CCIDCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + if (epnum == (CCIDInEpAdd & 0x7FU)) + { + /* Filter the epnum by masking with 0x7f (mask of IN Direction) */ + + /*************** Handle Bulk Transfer IN data completion *****************/ + + switch (hccid->blkt_state) + { + case CCID_STATE_SEND_RESP: + + /* won't wait ack to avoid missing a command */ + hccid->blkt_state = CCID_STATE_IDLE; + + /* Prepare EP to Receive Cmd */ + (void)USBD_LL_PrepareReceive(pdev, CCID_OUT_EP, + hccid->data, hccid->MaxPcktLen); + break; + + default: + break; + } + } + else if (epnum == (CCIDCmdEpAdd & 0x7FU)) + { + /* Filter the epnum by masking with 0x7f (mask of IN Direction) */ + + /*************** Handle Interrupt Transfer IN data completion *****************/ + + (void)USBD_CCID_IntMessage(pdev); + } + else + { + return (uint8_t)USBD_FAIL; + } + + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_CCID_DataOut + * Data received on non-control Out endpoint + * @param pdev: device instance + * @param epnum: endpoint number + * @retval status + */ +static uint8_t USBD_CCID_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint16_t CurrPcktLen; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + CCIDOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + if (hccid == NULL) + { + return (uint8_t)USBD_EMEM; + } + + if (epnum == CCIDOutEpAdd) + { + CurrPcktLen = (uint16_t)USBD_GetRxCount(pdev, epnum); + + switch (hccid->blkt_state) + { + case CCID_STATE_IDLE: + + if (CurrPcktLen >= (uint16_t)CCID_CMD_HEADER_SIZE) + { + hccid->UsbMessageLength = CurrPcktLen; /* Store for future use */ + + /* Fill CCID_BulkOut Data Buffer from USB Buffer */ + (void)USBD_CCID_ReceiveCmdHeader(pdev, (uint8_t *)&hccid->UsbBlkOutData.bMessageType, + (uint16_t)CurrPcktLen); + + /* + Refer : 6 CCID Messages + The response messages always contain the exact same slot number, + and sequence number fields from the header that was contained in + the Bulk-OUT command message. + */ + hccid->UsbBlkInData.bSlot = hccid->UsbBlkOutData.bSlot; + hccid->UsbBlkInData.bSeq = hccid->UsbBlkOutData.bSeq; + + if (CurrPcktLen < hccid->MaxPcktLen) + { + /* Short message, less than the EP Out Size, execute the command, + if parameter like dwLength is too big, the appropriate command will + give an error */ + (void)USBD_CCID_DispatchCommand(pdev); + } + else + { + /* Check if length of data to be sent by host is > buffer size */ + if (hccid->UsbBlkOutData.dwLength > (uint32_t)ABDATA_SIZE) + { + /* Too long data received.... Error ! */ + hccid->blkt_state = CCID_STATE_UNCORRECT_LENGTH; + } + + else + { + /* Expect more data on OUT EP */ + hccid->blkt_state = CCID_STATE_RECEIVE_DATA; + + /* Prepare EP to Receive next Cmd */ + (void)USBD_LL_PrepareReceive(pdev, CCID_OUT_EP, + hccid->data, hccid->MaxPcktLen); + + } /* if (CurrPcktLen == CCID_DATA_MAX_PACKET_SIZE) ends */ + } /* if (CurrPcktLen >= CCID_DATA_MAX_PACKET_SIZE) ends */ + } /* if (CurrPcktLen >= CCID_CMD_HEADER_SIZE) ends */ + else + { + if (CurrPcktLen == 0x00U) /* Zero Length Packet Received */ + { + hccid->blkt_state = CCID_STATE_IDLE; + } + } + + break; + + case CCID_STATE_RECEIVE_DATA: + hccid->UsbMessageLength += CurrPcktLen; + + if (CurrPcktLen < hccid->MaxPcktLen) + { + /* Short message, less than the EP Out Size, execute the command, + if parameter like dwLength is too big, the appropriate command will + give an error */ + + /* Full command is received, process the Command */ + (void)USBD_CCID_ReceiveCmdHeader(pdev, (uint8_t *)&hccid->UsbBlkOutData.bMessageType, + (uint16_t)CurrPcktLen); + + (void)USBD_CCID_DispatchCommand(pdev); + } + else if (CurrPcktLen == hccid->MaxPcktLen) + { + if (hccid->UsbMessageLength < (hccid->UsbBlkOutData.dwLength + (uint32_t)CCID_CMD_HEADER_SIZE)) + { + (void)USBD_CCID_ReceiveCmdHeader(pdev, (uint8_t *)&hccid->UsbBlkOutData.bMessageType, + (uint16_t)CurrPcktLen); /* Copy data */ + + /* Prepare EP to Receive next Cmd */ + (void)USBD_LL_PrepareReceive(pdev, CCID_OUT_EP, + hccid->data, hccid->MaxPcktLen); + } + else if (hccid->UsbMessageLength == (hccid->UsbBlkOutData.dwLength + (uint32_t)CCID_CMD_HEADER_SIZE)) + { + /* Full command is received, process the Command */ + (void)USBD_CCID_ReceiveCmdHeader(pdev, (uint8_t *)&hccid->UsbBlkOutData.bMessageType, + (uint16_t)CurrPcktLen); + + (void)USBD_CCID_DispatchCommand(pdev); + } + else + { + /* Too long data received.... Error ! */ + hccid->blkt_state = CCID_STATE_UNCORRECT_LENGTH; + } + } + else + { + /* Too long data received.... Error ! */ + hccid->blkt_state = CCID_STATE_UNCORRECT_LENGTH; + } + break; + + case CCID_STATE_UNCORRECT_LENGTH: + hccid->blkt_state = CCID_STATE_IDLE; + break; + + default: + break; + } + } + else + { + return (uint8_t)USBD_FAIL; + } + + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_CCID_DispatchCommand + * Parse the commands and Process command + * @param pdev: device instance + * @retval status value + */ +static uint8_t USBD_CCID_DispatchCommand(USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint8_t errorCode; + + switch (hccid->UsbBlkOutData.bMessageType) + { + case PC_TO_RDR_ICCPOWERON: + errorCode = PC_to_RDR_IccPowerOn(pdev); + RDR_to_PC_DataBlock(errorCode, pdev); + break; + + case PC_TO_RDR_ICCPOWEROFF: + errorCode = PC_to_RDR_IccPowerOff(pdev); + RDR_to_PC_SlotStatus(errorCode, pdev); + break; + + case PC_TO_RDR_GETSLOTSTATUS: + errorCode = PC_to_RDR_GetSlotStatus(pdev); + RDR_to_PC_SlotStatus(errorCode, pdev); + break; + + case PC_TO_RDR_XFRBLOCK: + errorCode = PC_to_RDR_XfrBlock(pdev); + RDR_to_PC_DataBlock(errorCode, pdev); + break; + + case PC_TO_RDR_GETPARAMETERS: + errorCode = PC_to_RDR_GetParameters(pdev); + RDR_to_PC_Parameters(errorCode, pdev); + break; + + case PC_TO_RDR_RESETPARAMETERS: + errorCode = PC_to_RDR_ResetParameters(pdev); + RDR_to_PC_Parameters(errorCode, pdev); + break; + + case PC_TO_RDR_SETPARAMETERS: + errorCode = PC_to_RDR_SetParameters(pdev); + RDR_to_PC_Parameters(errorCode, pdev); + break; + + case PC_TO_RDR_ESCAPE: + errorCode = PC_to_RDR_Escape(pdev); + RDR_to_PC_Escape(errorCode, pdev); + break; + + case PC_TO_RDR_ICCCLOCK: + errorCode = PC_to_RDR_IccClock(pdev); + RDR_to_PC_SlotStatus(errorCode, pdev); + break; + + case PC_TO_RDR_ABORT: + errorCode = PC_to_RDR_Abort(pdev); + RDR_to_PC_SlotStatus(errorCode, pdev); + break; + + case PC_TO_RDR_T0APDU: + errorCode = PC_TO_RDR_T0Apdu(pdev); + RDR_to_PC_SlotStatus(errorCode, pdev); + break; + + case PC_TO_RDR_MECHANICAL: + errorCode = PC_TO_RDR_Mechanical(pdev); + RDR_to_PC_SlotStatus(errorCode, pdev); + break; + + case PC_TO_RDR_SETDATARATEANDCLOCKFREQUENCY: + errorCode = PC_TO_RDR_SetDataRateAndClockFrequency(pdev); + RDR_to_PC_DataRateAndClockFrequency(errorCode, pdev); + break; + + case PC_TO_RDR_SECURE: + errorCode = PC_TO_RDR_Secure(pdev); + RDR_to_PC_DataBlock(errorCode, pdev); + break; + + default: + RDR_to_PC_SlotStatus(SLOTERROR_CMD_NOT_SUPPORTED, pdev); + break; + } + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_CCID_Transfer_Data_Request + * Prepare the request response to be sent to the host + * @param pdev: device instance + * @param dataPointer: Pointer to the data buffer to send + * @param dataLen : number of bytes to send + * @retval status value + */ +uint8_t USBD_CCID_Transfer_Data_Request(USBD_HandleTypeDef *pdev, + uint8_t *dataPointer, uint16_t dataLen) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_CCID_ItfTypeDef *hCCIDitf = (USBD_CCID_ItfTypeDef *)pdev->pUserData[pdev->classId]; + + UNUSED(dataPointer); + + hccid->blkt_state = CCID_STATE_SEND_RESP; + hccid->UsbMessageLength = (uint32_t)dataLen; /* Store for future use */ + + /* use the header declared size packet must be well formed */ + hCCIDitf->Response_SendData(pdev, (uint8_t *)&hccid->UsbBlkInData, + (uint16_t)MIN(CCID_DATA_FS_MAX_PACKET_SIZE, hccid->UsbMessageLength)); + + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_CCID_ReceiveCmdHeader + * Receive the Data from USB BulkOut Buffer to Pointer + * @param pdev: device instance + * @param pDst: destination address to copy the buffer + * @param u8length: length of data to copy + * @retval status + */ +static uint8_t USBD_CCID_ReceiveCmdHeader(USBD_HandleTypeDef *pdev, + uint8_t *pDst, uint16_t u8length) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint8_t *pdst = pDst; + uint32_t Counter; + + for (Counter = 0U; Counter < u8length; Counter++) + { + *pdst = hccid->data[Counter]; + pdst++; + } + + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_CCID_IntMessage + * Send the Interrupt-IN data to the host + * @param pdev: device instance + * @retval None + */ +uint8_t USBD_CCID_IntMessage(USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + CCIDCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + /* Check if there is change in Smartcard Slot status */ + if (CCID_IsSlotStatusChange(pdev) != 0U) + { + /* Check Slot Status is changed. Card is Removed/Fitted */ + RDR_to_PC_NotifySlotChange(pdev); + + /* Set the Slot status */ + ((USBD_CCID_ItfTypeDef *)pdev->pUserData[pdev->classId])->SetSlotStatus(pdev); + + (void)USBD_LL_Transmit(pdev, CCIDCmdEpAdd, hccid->UsbIntData, 2U); + } + else + { + /* Set the Slot status */ + ((USBD_CCID_ItfTypeDef *)pdev->pUserData[pdev->classId])->SetSlotStatus(pdev); + } + + return (uint8_t)USBD_OK; +} + +#ifndef USE_USBD_COMPOSITE +/** + * @brief USBD_CCID_GetHSCfgDesc + * Return configuration descriptor + * @param length pointer data length + * @retval pointer to descriptor buffer + */ +static uint8_t *USBD_CCID_GetHSCfgDesc(uint16_t *length) +{ + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CCID_CfgDesc, CCID_IN_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CCID_CfgDesc, CCID_OUT_EP); + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_CCID_CfgDesc, CCID_CMD_EP); + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = CCID_DATA_HS_MAX_PACKET_SIZE; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = CCID_DATA_HS_MAX_PACKET_SIZE; + } + + if (pEpCmdDesc != NULL) + { + pEpCmdDesc->bInterval = CCID_CMD_HS_BINTERVAL; + } + + *length = (uint16_t)sizeof(USBD_CCID_CfgDesc); + return USBD_CCID_CfgDesc; +} +/** + * @brief USBD_CCID_GetFSCfgDesc + * Return configuration descriptor + * @param length : pointer data length + * @retval pointer to descriptor buffer + */ +static uint8_t *USBD_CCID_GetFSCfgDesc(uint16_t *length) +{ + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CCID_CfgDesc, CCID_IN_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CCID_CfgDesc, CCID_OUT_EP); + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_CCID_CfgDesc, CCID_CMD_EP); + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = CCID_DATA_FS_MAX_PACKET_SIZE; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = CCID_DATA_FS_MAX_PACKET_SIZE; + } + + if (pEpCmdDesc != NULL) + { + pEpCmdDesc->bInterval = CCID_CMD_FS_BINTERVAL; + } + + *length = (uint16_t)sizeof(USBD_CCID_CfgDesc); + return USBD_CCID_CfgDesc; +} + +/** + * @brief USBD_CCID_GetOtherSpeedCfgDesc + * Return configuration descriptor + * @param length : pointer data length + * @retval pointer to descriptor buffer + */ +static uint8_t *USBD_CCID_GetOtherSpeedCfgDesc(uint16_t *length) +{ + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CCID_CfgDesc, CCID_IN_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CCID_CfgDesc, CCID_OUT_EP); + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_CCID_CfgDesc, CCID_CMD_EP); + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = CCID_DATA_FS_MAX_PACKET_SIZE; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = CCID_DATA_FS_MAX_PACKET_SIZE; + } + + if (pEpCmdDesc != NULL) + { + pEpCmdDesc->bInterval = CCID_CMD_FS_BINTERVAL; + } + + *length = (uint16_t)sizeof(USBD_CCID_CfgDesc); + return USBD_CCID_CfgDesc; +} + +/** + * @brief USBD_CCID_GetDeviceQualifierDescriptor + * return Device Qualifier descriptor + * @param length : pointer data length + * @retval pointer to descriptor buffer + */ +static uint8_t *USBD_CCID_GetDeviceQualifierDescriptor(uint16_t *length) +{ + *length = (uint16_t)(sizeof(USBD_CCID_DeviceQualifierDesc)); + return USBD_CCID_DeviceQualifierDesc; +} +#endif /* USE_USBD_COMPOSITE */ + +/** + * @brief USBD_CCID_RegisterInterface + * @param pdev: device instance + * @param fops: CD Interface callback + * @retval status + */ +uint8_t USBD_CCID_RegisterInterface(USBD_HandleTypeDef *pdev, + USBD_CCID_ItfTypeDef *fops) +{ + if (fops == NULL) + { + return (uint8_t)USBD_FAIL; + } + + pdev->pUserData[pdev->classId] = fops; + + return (uint8_t)USBD_OK; +} + +/** + * @} + */ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_cmd.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_cmd.c new file mode 100644 index 00000000..b91cb1f1 --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_cmd.c @@ -0,0 +1,1099 @@ +/** + ****************************************************************************** + * @file usbd_ccid_cmd.c + * @author MCD Application Team + * @brief CCID command (Bulk-OUT Messages / Bulk-IN Messages) handling + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_ccid.h" +#include "usbd_ccid_cmd.h" + +/* Private typedef -----------------------------------------------------------*/ +/* Private define ------------------------------------------------------------*/ +/* Private macro -------------------------------------------------------------*/ +/* Private function prototypes -----------------------------------------------*/ +static uint8_t CCID_CheckCommandParams(USBD_HandleTypeDef *pdev, uint32_t param_type); +static void CCID_UpdateCommandStatus(USBD_HandleTypeDef *pdev, uint8_t cmd_status, uint8_t icc_status); + +/* Private functions ---------------------------------------------------------*/ + +/******************************************************************************/ +/* BULK OUT ROUTINES */ +/******************************************************************************/ + +/** + * @brief PC_to_RDR_IccPowerOn + * PC_TO_RDR_ICCPOWERON message execution, apply voltage and get ATR + * @param pdev: device instance + * @retval status of the command execution + */ +uint8_t PC_to_RDR_IccPowerOn(USBD_HandleTypeDef *pdev) +{ + /* Apply the ICC VCC + Fills the Response buffer with ICC ATR + This Command is returned with RDR_to_PC_DataBlock(); + */ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint8_t voltage; + uint8_t sc_voltage = 0U; + uint8_t index; + uint8_t error; + + hccid->UsbBlkInData.dwLength = 0U; /* Reset Number of Bytes in abData */ + + error = CCID_CheckCommandParams(pdev, CHK_PARAM_SLOT | CHK_PARAM_DWLENGTH | + CHK_PARAM_ABRFU2 | CHK_PARAM_CARD_PRESENT | + CHK_PARAM_ABORT); + if (error != 0U) + { + return error; + } + + /* Voltage that is applied to the ICC + 00h Automatic Voltage Selection + 01h 5.0 volts + 02h 3.0 volts + 03h 1.8 volts + */ + /* UsbBlkOutData.bSpecific_0 Contains bPowerSelect */ + voltage = hccid->UsbBlkOutData.bSpecific_0; + if (voltage >= VOLTAGE_SELECTION_1V8) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + return SLOTERROR_BAD_POWERSELECT; /* The Voltage specified is out of Spec */ + } + + /* Correct Voltage Requested by the Host */ + if ((voltage == VOLTAGE_SELECTION_AUTOMATIC) || (voltage == VOLTAGE_SELECTION_3V)) + { + sc_voltage = SC_VOLTAGE_3V; + } + else + { + if (voltage == VOLTAGE_SELECTION_5V) + { + sc_voltage = SC_VOLTAGE_5V; + } + } + SC_Itf_IccPowerOn(sc_voltage); + + /* Check if the Card has come to Active State*/ + error = CCID_CheckCommandParams(pdev, (uint32_t)CHK_ACTIVE_STATE); + + if (error != 0U) + { + /* Check if Voltage is not Automatic */ + if (voltage != 0U) + { + /* If Specific Voltage requested by Host i.e 3V or 5V*/ + return error; + } + else + { + /* Automatic Voltage selection requested by Host */ + + if (sc_voltage != SC_VOLTAGE_5V) + { + /* If voltage selected was Automatic and 5V is not yet tried */ + sc_voltage = SC_VOLTAGE_5V; + SC_Itf_IccPowerOn(sc_voltage); + + /* Check again the State */ + error = CCID_CheckCommandParams(pdev, (uint32_t)CHK_ACTIVE_STATE); + if (error != 0U) + { + return error; + } + } + else + { + /* Voltage requested from Host was 5V already*/ + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_INACTIVE)); + return error; + } + } /* Voltage Selection was automatic */ + } /* If Active State */ + + /* ATR is received, No Error Condition Found */ + hccid->UsbBlkInData.dwLength = SIZE_OF_ATR; + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_NO_ERROR), (BM_ICC_PRESENT_ACTIVE)); + + for (index = 0U; index < SIZE_OF_ATR; index++) + { + /* Copy the ATR to the Response Buffer */ + hccid->UsbBlkInData.abData[index] = SC_ATR_Table[index]; + } + + return SLOT_NO_ERROR; +} + +/** + * @brief PC_to_RDR_IccPowerOff + * Icc VCC is switched Off + * @param pdev: device instance + * @retval error: status of the command execution + */ +uint8_t PC_to_RDR_IccPowerOff(USBD_HandleTypeDef *pdev) +{ + /* The response to this command is the RDR_to_PC_SlotStatus*/ + uint8_t error; + + error = CCID_CheckCommandParams(pdev, CHK_PARAM_SLOT | CHK_PARAM_ABRFU3 | + CHK_PARAM_DWLENGTH); + if (error != 0U) + { + return error; + } + /* Command is ok, Check for Card Presence */ + if (SC_Detect() != 0U) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_NO_ERROR), (BM_ICC_PRESENT_INACTIVE)); + } + else + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_NO_ERROR), (BM_ICC_NO_ICC_PRESENT)); + } + + /* Power OFF the card */ + SC_Itf_IccPowerOff(); + + return SLOT_NO_ERROR; +} + +/** + * @brief PC_to_RDR_GetSlotStatus + * Provides the Slot status to the host + * @param pdev: device instance + * @retval status of the command execution + */ +uint8_t PC_to_RDR_GetSlotStatus(USBD_HandleTypeDef *pdev) +{ + uint8_t error; + + error = CCID_CheckCommandParams(pdev, CHK_PARAM_SLOT | CHK_PARAM_DWLENGTH | + CHK_PARAM_CARD_PRESENT | CHK_PARAM_ABRFU3); + if (error != 0U) + { + return error; + } + + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_NO_ERROR), (BM_ICC_PRESENT_ACTIVE)); + return SLOT_NO_ERROR; +} + + +/** + * @brief PC_to_RDR_XfrBlock + * Handles the Block transfer from Host. + * Response to this command message is the RDR_to_PC_DataBlock + * @param pdev: device instance + * @retval status of the command execution + */ +uint8_t PC_to_RDR_XfrBlock(USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint16_t expectedLength; + + uint8_t error; + + error = CCID_CheckCommandParams(pdev, CHK_PARAM_SLOT | CHK_PARAM_CARD_PRESENT | + CHK_PARAM_ABRFU3 | CHK_PARAM_ABORT | CHK_ACTIVE_STATE); + if (error != 0U) + { + return error; + } + if (hccid->UsbBlkOutData.dwLength > ABDATA_SIZE) + { + /* Check amount of Data Sent by Host is > than memory allocated ? */ + + return SLOTERROR_BAD_DWLENGTH; + } + + /* wLevelParameter = Size of expected data to be returned by the + bulk-IN endpoint */ + expectedLength = (uint8_t)((hccid->UsbBlkOutData.bSpecific_2 << 8) | + hccid->UsbBlkOutData.bSpecific_1); + + hccid->UsbBlkInData.dwLength = (uint16_t)expectedLength; + + error = SC_Itf_XferBlock(&(hccid->UsbBlkOutData.abData[0]), + hccid->UsbBlkOutData.dwLength, + expectedLength, &hccid->UsbBlkInData); + + if (error != SLOT_NO_ERROR) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + } + else + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_NO_ERROR), (BM_ICC_PRESENT_ACTIVE)); + error = SLOT_NO_ERROR; + } + + return error; +} + +/** + * @brief PC_to_RDR_GetParameters + * Provides the ICC parameters to the host + * Response to this command message is the RDR_to_PC_Parameters + * @param pdev: device instance + * @retval status of the command execution + */ +uint8_t PC_to_RDR_GetParameters(USBD_HandleTypeDef *pdev) +{ + uint8_t error; + + error = CCID_CheckCommandParams(pdev, CHK_PARAM_SLOT | CHK_PARAM_DWLENGTH | + CHK_PARAM_CARD_PRESENT | CHK_PARAM_ABRFU3); + if (error != 0U) + { + return error; + } + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_NO_ERROR), (BM_ICC_PRESENT_ACTIVE)); + + return SLOT_NO_ERROR; +} + +/** + * @brief PC_to_RDR_ResetParameters + * Set the ICC parameters to the default + * Response to this command message is the RDR_to_PC_Parameters + * @param pdev: device instance + * @retval status of the command execution + */ +uint8_t PC_to_RDR_ResetParameters(USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint8_t error; + + error = CCID_CheckCommandParams(pdev, CHK_PARAM_SLOT | CHK_PARAM_DWLENGTH | + CHK_PARAM_CARD_PRESENT | CHK_PARAM_ABRFU3 | + CHK_ACTIVE_STATE); + if (error != 0U) + { + return error; + } + /* This command resets the slot parameters to their default values */ + hccid->UsbBlkOutData.abData[0] = DEFAULT_FIDI; + hccid->UsbBlkOutData.abData[1] = DEFAULT_T01CONVCHECKSUM; + hccid->UsbBlkOutData.abData[2] = DEFAULT_EXTRA_GUARDTIME; + hccid->UsbBlkOutData.abData[3] = DEFAULT_WAITINGINTEGER; + hccid->UsbBlkOutData.abData[4] = DEFAULT_CLOCKSTOP; + hccid->UsbBlkOutData.abData[5] = 0x00U; + hccid->UsbBlkOutData.abData[6] = 0x00U; + + (void)USBD_memcpy(&ProtocolData, (void const *)(&hccid->UsbBlkOutData.abData[0]), + sizeof(ProtocolData)); + + error = SC_Itf_SetParams(&ProtocolData, ProtocolNUM_OUT); + + if (error != SLOT_NO_ERROR) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + } + else + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_NO_ERROR), (BM_ICC_PRESENT_ACTIVE)); + error = SLOT_NO_ERROR; + } + + return error; +} +/** + * @brief PC_to_RDR_SetParameters + * Set the ICC parameters to the host defined parameters + * Response to this command message is the RDR_to_PC_Parameters + * @param pdev: device instance + * @retval status of the command execution + */ +uint8_t PC_to_RDR_SetParameters(USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint8_t error; + + error = CCID_CheckCommandParams(pdev, CHK_PARAM_SLOT | CHK_PARAM_CARD_PRESENT | + CHK_PARAM_ABRFU2 | CHK_ACTIVE_STATE); + if (error != 0U) + { + return error; + } + error = SLOT_NO_ERROR; + + /* for Protocol T=0 (dwLength=00000005h) */ + /* for Protocol T=1 (dwLength=00000007h) */ + if (((hccid->UsbBlkOutData.dwLength == 5U) && (hccid->UsbBlkOutData.bSpecific_0 != 0U)) + || ((hccid->UsbBlkOutData.dwLength == 7U) && (hccid->UsbBlkOutData.bSpecific_0 != 1U))) + { + error = SLOTERROR_BAD_PROTOCOLNUM; + } + if (hccid->UsbBlkOutData.abData[4] != DEFAULT_CLOCKSTOP) + { + error = SLOTERROR_BAD_CLOCKSTOP; + } + if (error != SLOT_NO_ERROR) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + } + + (void)USBD_memcpy(&ProtocolData, (void const *)(&hccid->UsbBlkOutData.abData[0]), + sizeof(ProtocolData)); + + error = SC_Itf_SetParams(&ProtocolData, ProtocolNUM_OUT); + + if (error != SLOT_NO_ERROR) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + } + else + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_NO_ERROR), (BM_ICC_PRESENT_ACTIVE)); + error = SLOT_NO_ERROR; + } + + return error; +} +/** + * @brief PC_to_RDR_Escape + * Execute the Escape command. This is user specific Implementation + * Response to this command message is the RDR_to_PC_Escape + * @param pdev: device instance + * @retval status of the command execution + */ +uint8_t PC_to_RDR_Escape(USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint8_t error; + uint32_t size; + + error = CCID_CheckCommandParams(pdev, CHK_PARAM_SLOT | CHK_PARAM_CARD_PRESENT | + CHK_PARAM_ABRFU3 | CHK_PARAM_ABORT | CHK_ACTIVE_STATE); + + if (error != 0U) + { + return error; + } + error = SC_Itf_Escape(&hccid->UsbBlkOutData.abData[0], hccid->UsbBlkOutData.dwLength, + &hccid->UsbBlkInData.abData[0], &size); + + hccid->UsbBlkInData.dwLength = size; + + if (error != SLOT_NO_ERROR) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + } + else + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_NO_ERROR), (BM_ICC_PRESENT_ACTIVE)); + } + + return error; +} +/** + * @brief PC_to_RDR_IccClock + * Execute the Clock specific command from host + * Response to this command message is the RDR_to_PC_SlotStatus + * @param pdev: device instance + * @retval status of the command execution + */ +uint8_t PC_to_RDR_IccClock(USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint8_t error; + + error = CCID_CheckCommandParams(pdev, CHK_PARAM_SLOT | CHK_PARAM_CARD_PRESENT | + CHK_PARAM_ABRFU2 | CHK_PARAM_DWLENGTH | CHK_ACTIVE_STATE); + if (error != 0U) + { + return error; + } + /* bClockCommand : + 00h restarts Clock + 01h Stops Clock in the state shown in the bClockStop field */ + if (hccid->UsbBlkOutData.bSpecific_0 > 1U) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + return SLOTERROR_BAD_CLOCKCOMMAND; + } + + error = SC_Itf_SetClock(hccid->UsbBlkOutData.bSpecific_0); + + if (error != SLOT_NO_ERROR) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + } + else + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_NO_ERROR), (BM_ICC_PRESENT_ACTIVE)); + } + + return error; +} +/** + * @brief PC_to_RDR_Abort + * Execute the Abort command from host, This stops all Bulk transfers + * from host and ICC + * Response to this command message is the RDR_to_PC_SlotStatus + * @param pdev: device instance + * @retval status of the command execution + */ +uint8_t PC_to_RDR_Abort(USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint8_t error; + + error = CCID_CheckCommandParams(pdev, CHK_PARAM_SLOT | CHK_PARAM_ABRFU3 | + CHK_PARAM_DWLENGTH); + if (error != 0U) + { + return error; + } + (void)CCID_CmdAbort(pdev, hccid->UsbBlkOutData.bSlot, hccid->UsbBlkOutData.bSeq); + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_NO_ERROR), (BM_ICC_PRESENT_ACTIVE)); + return SLOT_NO_ERROR; +} + +/** + * @brief CCID_CmdAbort + * Execute the Abort command from Bulk EP or from Control EP, + * This stops all Bulk transfers from host and ICC + * @param pdev: device instance + * @param slot: slot number that host wants to abort + * @param seq : Seq number for PC_to_RDR_Abort + * @retval status of the command execution + */ +uint8_t CCID_CmdAbort(USBD_HandleTypeDef *pdev, uint8_t slot, uint8_t seq) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + + uint8_t BSlot = hccid->USBD_CCID_Param.bSlot; + /* This function is called for REQUEST_ABORT & PC_to_RDR_Abort */ + + if (slot >= CCID_NUMBER_OF_SLOTS) + { + /* error from CLASS_REQUEST*/ + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_NO_ICC_PRESENT)); + return SLOTERROR_BAD_SLOT; + } + + if (hccid->USBD_CCID_Param.bAbortRequestFlag == 1U) + { + /* Abort Command was already received from ClassReq or PC_to_RDR */ + if ((hccid->USBD_CCID_Param.bSeq == seq) && (BSlot == slot)) + { + /* CLASS Specific request is already Received, Reset the abort flag */ + hccid->USBD_CCID_Param.bAbortRequestFlag = 0; + } + } + else + { + /* Abort Command was NOT received from ClassReq or PC_to_RDR, + so save them for next ABORT command to verify */ + hccid->USBD_CCID_Param.bAbortRequestFlag = 1U; + hccid->USBD_CCID_Param.bSeq = seq; + hccid->USBD_CCID_Param.bSlot = slot; + } + + return 0; +} + +/** + * @brief PC_TO_RDR_T0Apdu + * Execute the PC_TO_RDR_T0APDU command from host + * Response to this command message is the RDR_to_PC_SlotStatus + * @param pdev: device instance + * @retval status of the command execution + */ +uint8_t PC_TO_RDR_T0Apdu(USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint8_t error; + + error = CCID_CheckCommandParams(pdev, CHK_PARAM_SLOT | CHK_PARAM_CARD_PRESENT | + CHK_PARAM_DWLENGTH | CHK_PARAM_ABORT); + if (error != 0U) + { + return error; + } + if (hccid->UsbBlkOutData.bSpecific_0 > 0x03U) + { + /* Bit 0 is associated with bClassGetResponse + Bit 1 is associated with bClassEnvelope */ + + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + return SLOTERROR_BAD_BMCHANGES; + } + + error = SC_Itf_T0Apdu(hccid->UsbBlkOutData.bSpecific_0, + hccid->UsbBlkOutData.bSpecific_1, + hccid->UsbBlkOutData.bSpecific_2); + + if (error != SLOT_NO_ERROR) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + } + else + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_NO_ERROR), (BM_ICC_PRESENT_ACTIVE)); + } + + return error; +} + +/** + * @brief PC_TO_RDR_Mechanical + * Execute the PC_TO_RDR_MECHANICAL command from host + * Response to this command message is the RDR_to_PC_SlotStatus + * @param pdev: device instance + * @retval status of the command execution + */ +uint8_t PC_TO_RDR_Mechanical(USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint8_t error; + + error = CCID_CheckCommandParams(pdev, CHK_PARAM_SLOT | CHK_PARAM_CARD_PRESENT | + CHK_PARAM_ABRFU2 | CHK_PARAM_DWLENGTH); + if (error != 0U) + { + return error; + } + if (hccid->UsbBlkOutData.bSpecific_0 > 0x05U) + { + /* + 01h Accept Card + 02h Eject Card + 03h Capture Card + 04h Lock Card + 05h Unlock Card + */ + + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + return SLOTERROR_BAD_BFUNCTION_MECHANICAL; + } + + error = SC_Itf_Mechanical(hccid->UsbBlkOutData.bSpecific_0); + + if (error != SLOT_NO_ERROR) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + } + else + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_NO_ERROR), (BM_ICC_PRESENT_ACTIVE)); + } + + return error; +} + +/** + * @brief PC_TO_RDR_SetDataRateAndClockFrequency + * Set the required Card Frequency and Data rate from the host. + * Response to this command message is the + * RDR_to_PC_DataRateAndClockFrequency + * @param pdev: device instance + * @retval status of the command execution + */ +uint8_t PC_TO_RDR_SetDataRateAndClockFrequency(USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint8_t error; + uint32_t clockFrequency; + uint32_t dataRate; + uint32_t temp; + + error = CCID_CheckCommandParams(pdev, CHK_PARAM_SLOT | CHK_PARAM_CARD_PRESENT | + CHK_PARAM_ABRFU3); + if (error != 0U) + { + return error; + } + if (hccid->UsbBlkOutData.dwLength != 0x08U) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + return SLOTERROR_BAD_LENTGH; + } + + /* HERE we avoiding to an unaligned memory access*/ + temp = (uint32_t)(hccid->UsbBlkOutData.abData[0]) & 0x000000FFU; + clockFrequency = temp; + + temp = (uint32_t)(hccid->UsbBlkOutData.abData[1]) & 0x000000FFU; + clockFrequency |= temp << 8; + + temp = (uint32_t)(hccid->UsbBlkOutData.abData[2]) & 0x000000FFU; + clockFrequency |= temp << 16; + + temp = (uint32_t)(hccid->UsbBlkOutData.abData[3]) & 0x000000FFU; + clockFrequency |= temp << 24; + + temp = (uint32_t)(hccid->UsbBlkOutData.abData[4]) & 0x000000FFU; + dataRate = temp; + + temp = (uint32_t)(hccid->UsbBlkOutData.abData[5]) & 0x000000FFU; + dataRate |= temp << 8; + + temp = (uint32_t)(hccid->UsbBlkOutData.abData[6]) & 0x000000FFU; + dataRate |= temp << 16; + + temp = (uint32_t)(hccid->UsbBlkOutData.abData[7]) & 0x000000FFU; + dataRate |= temp << 24; + + error = SC_Itf_SetDataRateAndClockFrequency(clockFrequency, dataRate); + hccid->UsbBlkInData.bError = error; + + if (error != SLOT_NO_ERROR) + { + hccid->UsbBlkInData.dwLength = 0; + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + } + else + { + hccid->UsbBlkInData.dwLength = 8; + + (hccid->UsbBlkInData.abData[0]) = (uint8_t)(clockFrequency & 0x000000FFU) ; + + (hccid->UsbBlkInData.abData[1]) = (uint8_t)((clockFrequency & 0x0000FF00U) >> 8); + + (hccid->UsbBlkInData.abData[2]) = (uint8_t)((clockFrequency & 0x00FF0000U) >> 16); + + (hccid->UsbBlkInData.abData[3]) = (uint8_t)((clockFrequency & 0xFF000000U) >> 24); + + (hccid->UsbBlkInData.abData[4]) = (uint8_t)(dataRate & 0x000000FFU) ; + + (hccid->UsbBlkInData.abData[5]) = (uint8_t)((dataRate & 0x0000FF00U) >> 8); + + (hccid->UsbBlkInData.abData[6]) = (uint8_t)((dataRate & 0x00FF0000U) >> 16); + + (hccid->UsbBlkInData.abData[7]) = (uint8_t)((dataRate & 0xFF000000U) >> 24); + + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_NO_ERROR), (BM_ICC_PRESENT_ACTIVE)); + } + + return error; +} + +/** + * @brief PC_TO_RDR_Secure + * Execute the Secure Command from the host. + * Response to this command message is the RDR_to_PC_DataBlock + * @param pdev: device instance + * @retval status of the command execution + */ +uint8_t PC_TO_RDR_Secure(USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint8_t error; + uint8_t bBWI; + uint16_t wLevelParameter; + uint32_t responseLen; + + hccid->UsbBlkInData.dwLength = 0; + + error = CCID_CheckCommandParams(pdev, CHK_PARAM_SLOT | CHK_PARAM_CARD_PRESENT | + CHK_PARAM_ABORT); + + if (error != 0U) + { + return error; + } + bBWI = hccid->UsbBlkOutData.bSpecific_0; + wLevelParameter = (hccid->UsbBlkOutData.bSpecific_1 + ((uint16_t)hccid->UsbBlkOutData.bSpecific_2 << 8)); + + if ((EXCHANGE_LEVEL_FEATURE == TPDU_EXCHANGE) || + (EXCHANGE_LEVEL_FEATURE == SHORT_APDU_EXCHANGE)) + { + /* TPDU level & short APDU level, wLevelParameter is RFU, = 0000h */ + if (wLevelParameter != 0U) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + error = SLOTERROR_BAD_LEVELPARAMETER; + return error; + } + } + + error = SC_Itf_Secure(hccid->UsbBlkOutData.dwLength, bBWI, wLevelParameter, + &(hccid->UsbBlkOutData.abData[0]), &responseLen); + + hccid->UsbBlkInData.dwLength = responseLen; + + if (error != SLOT_NO_ERROR) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + } + else + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_NO_ERROR), (BM_ICC_PRESENT_ACTIVE)); + } + + return error; +} + +/******************************************************************************/ +/* BULK IN ROUTINES */ +/******************************************************************************/ + +/** + * @brief RDR_to_PC_DataBlock + * Provide the data block response to the host + * Response for PC_to_RDR_IccPowerOn, PC_to_RDR_XfrBlock + * @param errorCode: code to be returned to the host + * @param pdev: device instance + * @retval None + */ +void RDR_to_PC_DataBlock(uint8_t errorCode, USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint32_t length = CCID_RESPONSE_HEADER_SIZE; + + hccid->UsbBlkInData.bMessageType = RDR_TO_PC_DATABLOCK; + hccid->UsbBlkInData.bError = 0U; + hccid->UsbBlkInData.bSpecific = 0U; /* bChainParameter */ + + if (errorCode == SLOT_NO_ERROR) + { + length += hccid->UsbBlkInData.dwLength; /* Length Specified in Command */ + } + + (void)USBD_CCID_Transfer_Data_Request(pdev, (uint8_t *)&hccid->UsbBlkInData, (uint16_t)length); +} + +/** + * @brief RDR_to_PC_SlotStatus + * Provide the Slot status response to the host + * Response for PC_to_RDR_IccPowerOff + * PC_to_RDR_GetSlotStatus + * PC_to_RDR_IccClock + * PC_to_RDR_T0APDU + * PC_to_RDR_Mechanical + * Also the device sends this response message when it has completed + * aborting a slot after receiving both the Class Specific ABORT request + * and PC_to_RDR_Abort command message. + * @param errorCode: code to be returned to the host + * @param pdev: device instance + * @retval None + */ +void RDR_to_PC_SlotStatus(uint8_t errorCode, USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint16_t length = CCID_RESPONSE_HEADER_SIZE; + + hccid->UsbBlkInData.bMessageType = RDR_TO_PC_SLOTSTATUS; + hccid->UsbBlkInData.dwLength = 0U; + hccid->UsbBlkInData.bError = 0U; + hccid->UsbBlkInData.bSpecific = 0U; /* bClockStatus = 00h Clock running + 01h Clock stopped in state L + 02h Clock stopped in state H + 03h Clock stopped in an unknown state */ + + if (errorCode == SLOT_NO_ERROR) + { + length += (uint16_t)hccid->UsbBlkInData.dwLength; + } + + (void)USBD_CCID_Transfer_Data_Request(pdev, (uint8_t *)(&hccid->UsbBlkInData), length); +} + +/** + * @brief RDR_to_PC_Parameters + * Provide the data block response to the host + * Response for PC_to_RDR_GetParameters, PC_to_RDR_ResetParameters + * PC_to_RDR_SetParameters + * @param errorCode: code to be returned to the host + * @param pdev: device instance + * @retval None + */ +void RDR_to_PC_Parameters(uint8_t errorCode, USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint16_t length = CCID_RESPONSE_HEADER_SIZE; + + hccid->UsbBlkInData.bMessageType = RDR_TO_PC_PARAMETERS; + hccid->UsbBlkInData.bError = 0U; + + if (errorCode == SLOT_NO_ERROR) + { + if (ProtocolNUM_OUT == 0x00U) + { + hccid->UsbBlkInData.dwLength = LEN_PROTOCOL_STRUCT_T0; + length += (uint16_t)hccid->UsbBlkInData.dwLength; + } + else + { + hccid->UsbBlkInData.dwLength = LEN_PROTOCOL_STRUCT_T1; + length += (uint16_t)hccid->UsbBlkInData.dwLength; + } + } + else + { + hccid->UsbBlkInData.dwLength = 0; + } + + hccid->UsbBlkInData.abData[0] = ProtocolData.bmFindexDindex; + hccid->UsbBlkInData.abData[1] = ProtocolData.bmTCCKST0; + hccid->UsbBlkInData.abData[2] = ProtocolData.bGuardTimeT0; + hccid->UsbBlkInData.abData[3] = ProtocolData.bWaitingIntegerT0; + hccid->UsbBlkInData.abData[4] = ProtocolData.bClockStop; + hccid->UsbBlkInData.abData[5] = ProtocolData.bIfsc; + hccid->UsbBlkInData.abData[6] = ProtocolData.bNad; + + /* bProtocolNum */ + if (ProtocolNUM_OUT == 0x00U) + { + hccid->UsbBlkInData.bSpecific = BPROTOCOL_NUM_T0; + } + else + { + hccid->UsbBlkInData.bSpecific = BPROTOCOL_NUM_T1; + } + (void)USBD_CCID_Transfer_Data_Request(pdev, (uint8_t *)(&hccid->UsbBlkInData), length); +} + +/** + * @brief RDR_to_PC_Escape + * Provide the Escaped data block response to the host + * Response for PC_to_RDR_Escape + * @param errorCode: code to be returned to the host + * @param pdev: device instance + * @retval None + */ +void RDR_to_PC_Escape(uint8_t errorCode, USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint32_t length = CCID_RESPONSE_HEADER_SIZE; + + hccid->UsbBlkInData.bMessageType = RDR_TO_PC_ESCAPE; + + hccid->UsbBlkInData.bSpecific = 0U; /* Reserved for Future Use */ + hccid->UsbBlkInData.bError = errorCode; + + if (errorCode == SLOT_NO_ERROR) + { + length += hccid->UsbBlkInData.dwLength; /* Length Specified in Command */ + } + + (void)USBD_CCID_Transfer_Data_Request(pdev, (uint8_t *)(&hccid->UsbBlkInData), (uint16_t)length); +} + +/** + * @brief RDR_to_PC_DataRateAndClockFrequency + * Provide the Clock and Data Rate information to host + * Response for PC_TO_RDR_SetDataRateAndClockFrequency + * @param errorCode: code to be returned to the host + * @param pdev: device instance + * @retval None + */ +void RDR_to_PC_DataRateAndClockFrequency(uint8_t errorCode, USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint32_t length = CCID_RESPONSE_HEADER_SIZE; + + hccid->UsbBlkInData.bMessageType = RDR_TO_PC_DATARATEANDCLOCKFREQUENCY; + hccid->UsbBlkInData.bError = errorCode; + hccid->UsbBlkInData.bSpecific = 0U; /* Reserved for Future Use */ + + if (errorCode == SLOT_NO_ERROR) + { + length += hccid->UsbBlkInData.dwLength; /* Length Specified in Command */ + } + + (void)USBD_CCID_Transfer_Data_Request(pdev, (uint8_t *)(&hccid->UsbBlkInData), (uint16_t)length); +} + +/** + * @brief RDR_to_PC_NotifySlotChange + * Interrupt message to be sent to the host, Checks the card presence + * status and update the buffer accordingly + * @param pdev: device instance + * @retval None + */ +void RDR_to_PC_NotifySlotChange(USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + + hccid->UsbIntData[OFFSET_INT_BMESSAGETYPE] = RDR_TO_PC_NOTIFYSLOTCHANGE; + + if (SC_Detect() != 0U) + { + /* + SLOT_ICC_PRESENT 0x01 : LSb : (0b = no ICC present, 1b = ICC present) + SLOT_ICC_CHANGE 0x02 : MSb : (0b = no change, 1b = change). + */ + hccid->UsbIntData[OFFSET_INT_BMSLOTICCSTATE] = SLOT_ICC_PRESENT | SLOT_ICC_CHANGE; + } + else + { + hccid->UsbIntData[OFFSET_INT_BMSLOTICCSTATE] = SLOT_ICC_CHANGE; + + /* Power OFF the card */ + SC_Itf_IccPowerOff(); + } +} + + +/** + * @brief CCID_UpdSlotStatus + * Updates the variable for the slot status + * @param pdev: device instance + * @param slotStatus : slot status from the calling function + * @retval None + */ +void CCID_UpdSlotStatus(USBD_HandleTypeDef *pdev, uint8_t slotStatus) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + hccid->SlotStatus.SlotStatus = slotStatus; +} + +/** + * @brief CCID_UpdSlotChange + * Updates the variable for the slot change status + * @param pdev: device instance + * @param changeStatus : slot change status from the calling function + * @retval None + */ +void CCID_UpdSlotChange(USBD_HandleTypeDef *pdev, uint8_t changeStatus) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + hccid->SlotStatus.SlotStatusChange = changeStatus; +} + +/** + * @brief CCID_IsSlotStatusChange + * Provides the value of the variable for the slot change status + * @param pdev: device instance + * @retval slot change status + */ +uint8_t CCID_IsSlotStatusChange(USBD_HandleTypeDef *pdev) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + return hccid->SlotStatus.SlotStatusChange; +} + +/** + * @brief CCID_UpdateCommandStatus + * Updates the variable for the BulkIn status + * @param pdev: device instance + * @param cmd_status : Command change status from the calling function + * @param icc_status : Slot change status from the calling function + * @retval None + */ +static void CCID_UpdateCommandStatus(USBD_HandleTypeDef *pdev, uint8_t cmd_status, uint8_t icc_status) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + hccid->UsbBlkInData.bStatus = (cmd_status | icc_status); +} +/** + * @brief CCID_CheckCommandParams + * Checks the specific parameters requested by the function and update + * status accordingly. This function is called from all + * PC_to_RDR functions + * @param pdev: device instance + * @param param_type : Parameter enum to be checked by calling function + * @retval status + */ +static uint8_t CCID_CheckCommandParams(USBD_HandleTypeDef *pdev, uint32_t param_type) +{ + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint32_t parameter; + uint8_t GetState = SC_GetState(); + + hccid->UsbBlkInData.bStatus = BM_ICC_PRESENT_ACTIVE | BM_COMMAND_STATUS_NO_ERROR; + + parameter = (uint32_t)param_type; + + if ((parameter & CHK_PARAM_SLOT) != 0U) + { + /* + The slot number (bSlot) identifies which ICC slot is being addressed + by the message*/ + + /* SLOT Number is 0 onwards, so always < CCID_NUMBER_OF_SLOTs */ + /* Error Condition !!! */ + if (hccid->UsbBlkOutData.bSlot >= CCID_NUMBER_OF_SLOTS) + { + /* Slot requested is more than supported by Firmware */ + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_NO_ICC_PRESENT)); + return SLOTERROR_BAD_SLOT; + } + } + + if ((parameter & CHK_PARAM_CARD_PRESENT) != 0U) + { + /* Commands Parameters ok, Check the Card Status */ + if (SC_Detect() == 0U) + { + /* Card is Not detected */ + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_NO_ICC_PRESENT)); + return SLOTERROR_ICC_MUTE; + } + } + + /* Check that DwLength is 0 */ + if ((parameter & CHK_PARAM_DWLENGTH) != 0U) + { + if (hccid->UsbBlkOutData.dwLength != 0U) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + return SLOTERROR_BAD_LENTGH; + } + } + + /* abRFU 2 : Reserved for Future Use*/ + if ((parameter & CHK_PARAM_ABRFU2) != 0U) + { + + if ((hccid->UsbBlkOutData.bSpecific_1 != 0U) || (hccid->UsbBlkOutData.bSpecific_2 != 0U)) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + return SLOTERROR_BAD_ABRFU_2B; /* bSpecific_1 */ + } + } + + if ((parameter & CHK_PARAM_ABRFU3) != 0U) + { + /* abRFU 3 : Reserved for Future Use*/ + if ((hccid->UsbBlkOutData.bSpecific_0 != 0U) || + (hccid->UsbBlkOutData.bSpecific_1 != 0U) || + (hccid->UsbBlkOutData.bSpecific_2 != 0U)) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_ACTIVE)); + return SLOTERROR_BAD_ABRFU_3B; + } + } + + if ((parameter & CHK_PARAM_ABORT) != 0U) + { + if (hccid->USBD_CCID_Param.bAbortRequestFlag != 0U) + { + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_INACTIVE)); + return SLOTERROR_CMD_ABORTED; + } + } + + if ((parameter & CHK_ACTIVE_STATE) != 0U) + { + /* Commands Parameters ok, Check the Card Status */ + /* Card is detected */ + + if ((GetState != (uint8_t)SC_ACTIVE_ON_T0) && (GetState != (uint8_t)SC_ACTIVE_ON_T1)) + { + /* Check that from Lower Layers, the SmartCard come to known state */ + CCID_UpdateCommandStatus(pdev, (BM_COMMAND_STATUS_FAILED), (BM_ICC_PRESENT_INACTIVE)); + return SLOTERROR_HW_ERROR; + } + } + + return 0U; +} diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_if_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_if_template.c new file mode 100644 index 00000000..ea3f1628 --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_if_template.c @@ -0,0 +1,270 @@ +/** + ****************************************************************************** + * @file usbd_ccid_if_template.c + * @author MCD Application Team + * @brief This file provides all the functions for USB Interface for CCID + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_ccid.h" +#include "usbd_ccid_if_template.h" + +/* Private typedef -----------------------------------------------------------*/ +/* Private define ------------------------------------------------------------*/ +/* Private macro -------------------------------------------------------------*/ +/* Private variables ---------------------------------------------------------*/ +static REP_Command_t REP_command; + +/* Private function prototypes -----------------------------------------------*/ +static uint8_t CCID_Init(USBD_HandleTypeDef *pdev); +static uint8_t CCID_DeInit(USBD_HandleTypeDef *pdev); +static uint8_t CCID_ControlReq(uint8_t req, uint8_t *pbuf, uint16_t *length); +static uint8_t CCID_Response_SendData(USBD_HandleTypeDef *pdev, uint8_t *buf, uint16_t len); +static uint8_t CCID_Send_Process(uint8_t *Command, uint8_t *Data); +static uint8_t CCID_Response_Process(void); +static uint8_t CCID_SetSlotStatus(USBD_HandleTypeDef *pdev); + +/* Private functions ---------------------------------------------------------*/ + +/** + * @} + */ + +USBD_CCID_ItfTypeDef USBD_CCID_If_fops = +{ + CCID_Init, + CCID_DeInit, + CCID_ControlReq, + CCID_Response_SendData, + CCID_Send_Process, + CCID_SetSlotStatus, +}; + +/** + * @brief CCID_Init + * Initialize the CCID USB Layer + * @param pdev: device instance + * @retval status value + */ +uint8_t CCID_Init(USBD_HandleTypeDef *pdev) +{ +#ifdef USE_USBD_COMPOSITE + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#else + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassData; +#endif /* USE_USBD_COMPOSITE */ + + /* CCID Related Initialization */ + + hccid->blkt_state = CCID_STATE_IDLE; + + return (uint8_t)USBD_OK; +} + +/** + * @brief CCID_DeInit + * Uninitialize the CCID Machine + * @param pdev: device instance + * @retval status value + */ +uint8_t CCID_DeInit(USBD_HandleTypeDef *pdev) +{ +#ifdef USE_USBD_COMPOSITE + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#else + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassData; +#endif /* USE_USBD_COMPOSITE */ + + hccid->blkt_state = CCID_STATE_IDLE; + + return (uint8_t)USBD_OK; +} + +/** + * @brief CCID_ControlReq + * Manage the CCID class requests + * @param Cmd: Command code + * @param Buf: Buffer containing command data (request parameters) + * @param Len: Number of data to be sent (in bytes) + * @retval Result of the operation: USBD_OK if all operations are OK else USBD_FAIL + */ +static uint8_t CCID_ControlReq(uint8_t req, uint8_t *pbuf, uint16_t *length) +{ +#ifdef USE_USBD_COMPOSITE + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)USBD_Device.pClassDataCmsit[USBD_Device.classId]; +#else + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)USBD_Device.pClassData; +#endif /* USE_USBD_COMPOSITE */ + + UNUSED(length); + + switch (req) + { + case REQUEST_ABORT: + /* The wValue field contains the slot number (bSlot) in the low byte + and the sequence number (bSeq) in the high byte.*/ + hccid->slot_nb = ((uint16_t) * pbuf & 0x0fU); + hccid->seq_nb = (((uint16_t) * pbuf & 0xf0U) >> 8); + + if (CCID_CmdAbort(&USBD_Device, (uint8_t)hccid->slot_nb, (uint8_t)hccid->seq_nb) != 0U) + { + /* If error is returned by lower layer : + Generally Slot# may not have matched */ + return (int8_t)USBD_FAIL; + } + break; + + case REQUEST_GET_CLOCK_FREQUENCIES: + + /* User have to fill the pbuf with the GetClockFrequency data buffer */ + + break; + + case REQUEST_GET_DATA_RATES: + + /* User have to fill the pbuf with the GetDataRates data buffer */ + + break; + + default: + break; + } + + UNUSED(pbuf); + + return ((int8_t)USBD_OK); +} + +/** + * @brief CCID_Response_SendData + * Send the data on bulk-in EP + * @param pdev: device instance + * @param buf: pointer to data buffer + * @param len: Data Length + * @retval status value + */ +uint8_t CCID_Response_SendData(USBD_HandleTypeDef *pdev, uint8_t *buf, uint16_t len) +{ + (void)USBD_LL_Transmit(pdev, CCID_IN_EP, buf, len); + return (uint8_t)USBD_OK; +} + +/** + * @brief CCID_SEND_Process + * @param Command: pointer to a buffer containing command header + * @param Data: pointer to a buffer containing data sent from Host + * @retval Result of the operation: USBD_OK if all operations are OK else USBD_FAIL + */ +static uint8_t CCID_Send_Process(uint8_t *Command, uint8_t *Data) +{ + Command_State_t Command_State = Command_NOT_OK; + + /* Initialize ICC APP header */ + uint8_t SC_Command[5] = {0}; + UNUSED(Data); + UNUSED(Command_State); + UNUSED(SC_Command); + + /* Start SC Demo ---------------------------------------------------------*/ + switch (Command[1]) /* type of instruction */ + { + case SC_ENABLE: + /* Add your code here */ + break; + + case SC_VERIFY: + /* Add your code here */ + break; + + case SC_READ_BINARY : + /* Add your code here */ + break; + + case SC_CHANGE : + /* Add your code here */ + break; + + default: + break; + } + + /* check if Command header is OK */ + (void)CCID_Response_Process(); /* Get ICC response */ + + return ((uint8_t)USBD_OK); +} + +/** + * @brief CCID_Response_Process + * @param None + * @retval Result of the operation: USBD_OK if all operations are OK else USBD_FAIL + */ +static uint8_t CCID_Response_Process(void) +{ + switch (REP_command) + { + case REP_OK: + /* Add your code here */ + break; + + case REP_NOT_OK : + /* Add your code here */ + break; + + case REP_NOT_SUPP : + /* Add your code here */ + break; + + case REP_ENABLED : + /* Add your code here */ + break; + + case REP_CHANGE : + /* Add your code here */ + break; + + default: + break; + } + + return ((uint8_t)USBD_OK); +} + +/** + * @brief CCID_SetSlotStatus + * Set Slot Status of the Interrupt Transfer + * @param pdev: device instance + * @retval status + */ +uint8_t CCID_SetSlotStatus(USBD_HandleTypeDef *pdev) +{ + /* Get the CCID handler pointer */ +#ifdef USE_USBD_COMPOSITE + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#else + USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassData; +#endif /* USE_USBD_COMPOSITE */ + + if ((hccid->SlotStatus.SlotStatus) == 1U) /* Transfer Complete Status + of previous Interrupt transfer */ + { + /* Add your code here */ + } + else + { + /* Add your code here */ + } + + return (uint8_t)USBD_OK; +} diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_sc_if_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_sc_if_template.c new file mode 100644 index 00000000..fddf2d04 --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_sc_if_template.c @@ -0,0 +1,473 @@ +/** + ****************************************************************************** + * @file usbd_ccid_sc_if_template.c + * @author MCD Application Team + * @brief SmartCard Interface file + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_ccid_sc_if_template.h" + +/* Private typedef -----------------------------------------------------------*/ +/* Private define ------------------------------------------------------------*/ +/* Private macro -------------------------------------------------------------*/ +/* Private variables ---------------------------------------------------------*/ +/* State Machine for the SmartCard Interface */ +static SC_State SCState = SC_POWER_OFF; + +/* APDU Transport Structures */ +SC_ADPU_CommandsTypeDef SC_ADPU; +SC_ADPU_ResponseTypeDef SC_Response; +SC_Param_t SC_Param; +Protocol_01_DataTypeDef ProtocolData; + +/* Extern variables ----------------------------------------------------------*/ +/* Private function prototypes -----------------------------------------------*/ +static void SC_SaveVoltage(uint8_t voltage); +static void SC_Itf_UpdateParams(void); + +/* Private functions ---------------------------------------------------------*/ +/** + * @brief SC_Itf_IccPowerOn Manages the Warm and Cold Reset + and get the Answer to Reset from ICC + * @param voltage: required by host + * @retval None + */ +void SC_Itf_IccPowerOn(uint8_t voltage) +{ + SCState = SC_POWER_ON; + SC_ADPU.Header.CLA = 0x00U; + SC_ADPU.Header.INS = SC_GET_A2R; + SC_ADPU.Header.P1 = 0x00U; + SC_ADPU.Header.P2 = 0x00U; + SC_ADPU.Body.LC = 0x00U; + + /* Power ON the card */ + SC_PowerCmd(SC_ENABLED); + + /* Configure the Voltage, Even if IO is still not configured */ + SC_VoltageConfig(voltage); + + while ((SCState != SC_ACTIVE_ON_T0) && (SCState != SC_ACTIVE_ON_T1) + && (SCState != SC_NO_INIT)) + { + /* If Either The Card has become Active or Become De-Active */ + SC_Handler(&SCState, &SC_ADPU, &SC_Response); + } + + if ((SCState == SC_ACTIVE_ON_T0) || (SCState == SC_ACTIVE_ON_T1)) + { + SC_Itf_UpdateParams(); + /* Apply the Procedure Type Selection (PTS) */ + SC_PTSConfig(); + + /* Save Voltage for Future use */ + SC_SaveVoltage(voltage); + } + + return; +} + +/** + * @brief SC_Itf_IccPowerOff Power OFF the card + * @param None + * @retval None + */ +void SC_Itf_IccPowerOff(void) +{ + SC_PowerCmd(SC_DISABLED); + SC_SetState(SC_POWER_OFF); + + return; +} + +/** + * @brief Initialize the parameters structures to the default value + * @param None + * @retval None + */ +void SC_Itf_InitParams(void) +{ + /* + FI, the reference to a clock rate conversion factor + over the bits b8 to b5 + - DI, the reference to a baud rate adjustment factor + over the bits b4 to bl + */ + SC_Param.SC_A2R_FiDi = DEFAULT_FIDI; + SC_Param.SC_hostFiDi = DEFAULT_FIDI; + + ProtocolData.bmFindexDindex = DEFAULT_FIDI; + + /* Placeholder, Ignored */ + /* 0 = Direct, first byte of the ICC ATR data. */ + ProtocolData.bmTCCKST0 = DEFAULT_T01CONVCHECKSUM; + + /* Extra GuardTime = 0 etu */ + ProtocolData.bGuardTimeT0 = DEFAULT_EXTRA_GUARDTIME; + ProtocolData.bWaitingIntegerT0 = DEFAULT_WAITINGINTEGER; + ProtocolData.bClockStop = 0U; /* Stopping the Clock is not allowed */ + + /*T=1 protocol */ + ProtocolData.bIfsc = DEFAULT_IFSC; + ProtocolData.bNad = DEFAULT_NAD; + + return; +} + +/** + * @brief Save the A2R Parameters for further usage + * @param None + * @retval None + */ +static void SC_Itf_UpdateParams(void) +{ + /* + FI, the reference to a clock rate conversion factor + over the bits b8 to b5 + DI, the reference to a baud rate adjustment factor + over the bits b4 to bl + */ + SC_Param.SC_A2R_FiDi = SC_A2R.T[0].InterfaceByte[0].Value; + SC_Param.SC_hostFiDi = SC_A2R.T[0].InterfaceByte[0].Value; + + ProtocolData.bmFindexDindex = SC_A2R.T[0].InterfaceByte[0].Value; + + return; +} + +/** + * @brief SC_Itf_SetParams + * Set the parameters for CCID/USART interface + * @param pPtr: pointer to buffer containing the + * parameters to be set in USART + * @param T_01: type of protocol, T=1 or T=0 + * @retval status value + */ +uint8_t SC_Itf_SetParams(Protocol_01_DataTypeDef *pPtr, uint8_t T_01) +{ + /* uint16_t guardTime; */ /* Keep it 16b for handling 8b additions */ + uint32_t fi_new; + uint32_t di_new; + Protocol_01_DataTypeDef New_DataStructure; + fi_new = pPtr->bmFindexDindex; + di_new = pPtr->bmFindexDindex; + + New_DataStructure.bmTCCKST0 = pPtr->bmTCCKST0; + + New_DataStructure.bGuardTimeT0 = pPtr->bGuardTimeT0; + New_DataStructure.bWaitingIntegerT0 = pPtr->bWaitingIntegerT0; + New_DataStructure.bClockStop = pPtr->bClockStop; + if (T_01 == 0x01U) + { + New_DataStructure.bIfsc = pPtr->bIfsc; + New_DataStructure.bNad = pPtr->bNad; + } + else + { + New_DataStructure.bIfsc = 0x00U; + New_DataStructure.bNad = 0x00U; + } + + /* Check for the FIDI Value set by Host */ + di_new &= (uint8_t)0x0F; + if (SC_GetDTableValue(di_new) == 0U) + { + return SLOTERROR_BAD_FIDI; + } + + fi_new >>= 4U; + fi_new &= 0x0FU; + + if (SC_GetDTableValue(fi_new) == 0U) + { + return SLOTERROR_BAD_FIDI; + } + + if ((T_01 == 0x00U) + && (New_DataStructure.bmTCCKST0 != 0x00U) + && (New_DataStructure.bmTCCKST0 != 0x02U)) + { + return SLOTERROR_BAD_T01CONVCHECKSUM; + } + + if ((T_01 == 0x01U) + && (New_DataStructure.bmTCCKST0 != 0x10U) + && (New_DataStructure.bmTCCKST0 != 0x11U) + && (New_DataStructure.bmTCCKST0 != 0x12U) + && (New_DataStructure.bmTCCKST0 != 0x13U)) + { + return SLOTERROR_BAD_T01CONVCHECKSUM; + } + + if ((New_DataStructure.bWaitingIntegerT0 >= 0xA0U) + && ((New_DataStructure.bmTCCKST0 & 0x10U) == 0x10U)) + { + return SLOTERROR_BAD_WAITINGINTEGER; + } + if ((New_DataStructure.bClockStop != 0x00U) + && (New_DataStructure.bClockStop != 0x03U)) + { + return SLOTERROR_BAD_CLOCKSTOP; + } + if (New_DataStructure.bNad != 0x00U) + { + return SLOTERROR_BAD_NAD; + } + /* Put Total GuardTime in USART Settings */ + /* USART_SetGuardTime(SC_USART, (uint8_t)(guardTime + DEFAULT_EXTRA_GUARDTIME)); */ + + /* Save Extra GuardTime Value */ + ProtocolData.bGuardTimeT0 = New_DataStructure.bGuardTimeT0; + ProtocolData.bmTCCKST0 = New_DataStructure.bmTCCKST0; + ProtocolData.bWaitingIntegerT0 = New_DataStructure.bWaitingIntegerT0; + ProtocolData.bClockStop = New_DataStructure.bClockStop; + ProtocolData.bIfsc = New_DataStructure.bIfsc; + ProtocolData.bNad = New_DataStructure.bNad; + + /* Save New bmFindexDindex */ + SC_Param.SC_hostFiDi = pPtr->bmFindexDindex; + SC_PTSConfig(); + + ProtocolData.bmFindexDindex = pPtr->bmFindexDindex; + + return SLOT_NO_ERROR; +} + +/** + * @brief SC_Itf_Escape function from the host + * This is user implementable + * @param ptrEscape: pointer to buffer containing the Escape data + * @param escapeLen: length of escaped data + * @param responseBuff: pointer containing escape buffer response + * @param responseLen: length of escape response buffer + * @retval status value + */ +uint8_t SC_Itf_Escape(uint8_t *ptrEscape, uint32_t escapeLen, + uint8_t *responseBuff, uint32_t *responseLen) +{ + UNUSED(ptrEscape); + UNUSED(escapeLen); + UNUSED(responseBuff); + UNUSED(responseLen); + + /* Manufacturer specific implementation ... */ + /* + uint32_t idx; + uint8_t *pResBuff = responseBuff; + uint8_t *pEscape = ptrEscape; + + for(idx = 0; idx < escapeLen; idx++) + { + *pResBuff = *pEscape; + pResBuff++; + pEscape++; + } + + *responseLen = escapeLen; + */ + return SLOT_NO_ERROR; +} + +/** + * @brief SC_Itf_SetClock function to define Clock Status request from the host. + * This is user implementable + * @param bClockCommand: Clock status from the host + * @retval status value + */ +uint8_t SC_Itf_SetClock(uint8_t bClockCommand) +{ + /* bClockCommand + 00h restarts Clock + 01h Stops Clock in the state shown in the bClockStop + field of the PC_to_RDR_SetParameters command + and RDR_to_PC_Parameters message.*/ + + if (bClockCommand == 0U) + { + /* 00h restarts Clock : Since Clock is always running, PASS this command */ + return SLOT_NO_ERROR; + } + else + { + if (bClockCommand == 1U) + { + return SLOTERROR_BAD_CLOCKCOMMAND; + } + } + + return SLOTERROR_CMD_NOT_SUPPORTED; +} + +/** + * @brief SC_Itf_XferBlock function from the host. + * This is user implementable + * @param ptrBlock : Pointer containing the data from host + * @param blockLen : length of block data for the data transfer + * @param expectedLen: expected length of data transfer + * @param CCID_BulkIn_Data: Pointer containing the CCID Bulk In Data Structure + * @retval status value + */ +uint8_t SC_Itf_XferBlock(uint8_t *ptrBlock, uint32_t blockLen, uint16_t expectedLen, + USBD_CCID_BulkIn_DataTypeDef *CCID_BulkIn_Data) +{ + uint8_t ErrorCode = SLOT_NO_ERROR; + UNUSED(CCID_BulkIn_Data); + UNUSED(expectedLen); + UNUSED(blockLen); + UNUSED(ptrBlock); + + if (ProtocolNUM_OUT == 0x00U) + { + /* Add your code here */ + } + + if (ProtocolNUM_OUT == 0x01U) + { + /* Add your code here */ + } + + if (ErrorCode != SLOT_NO_ERROR) + { + return ErrorCode; + } + + return ErrorCode; +} + + +/** + * @brief SC_Itf_T0Apdu + Class Specific Request from the host to provide supported data rates + * This is Optional function & user implementable + * @param bmChanges : value specifying which parameter is valid in + * command among next bClassGetResponse, bClassEnvelope + * @param bClassGetResponse : Value to force the class byte of the + * header in a Get Response command. + * @param bClassEnvelope : Value to force the class byte of the header + * in a Envelope command. + * @retval status value + */ +uint8_t SC_Itf_T0Apdu(uint8_t bmChanges, uint8_t bClassGetResponse, + uint8_t bClassEnvelope) +{ + UNUSED(bClassEnvelope); + UNUSED(bClassGetResponse); + + /* User have to fill the pbuf with the GetDataRates data buffer */ + + if (bmChanges == 0U) + { + /* Bit cleared indicates that the associated field is not significant and + that default behaviour defined in CCID class descriptor is selected */ + return SLOT_NO_ERROR; + } + + return SLOTERROR_CMD_NOT_SUPPORTED; +} + +/** + * @brief SC_Itf_Mechanical + Mechanical Function being requested by Host + * This is Optional function & user implementable + * @param bFunction : value corresponds to the mechanical function + * being requested by host + * @retval status value + */ +uint8_t SC_Itf_Mechanical(uint8_t bFunction) +{ + UNUSED(bFunction); + + return SLOTERROR_CMD_NOT_SUPPORTED; +} + +/** + * @brief SC_Itf_SetDataRateAndClockFrequency + * Set the Clock and data Rate of the Interface + * This is Optional function & user implementable + * @param dwClockFrequency : value of clock in kHz requested by host + * @param dwDataRate : value of data rate requested by host + * @retval status value + */ +uint8_t SC_Itf_SetDataRateAndClockFrequency(uint32_t dwClockFrequency, + uint32_t dwDataRate) +{ + /* User have to fill the pbuf with the GetDataRates data buffer */ + + if ((dwDataRate == USBD_CCID_DEFAULT_DATA_RATE) && + (dwClockFrequency == USBD_CCID_DEFAULT_CLOCK_FREQ)) + { + return SLOT_NO_ERROR; + } + + return SLOTERROR_CMD_NOT_SUPPORTED; +} + +/** + * @brief SC_Itf_Secure + * Process the Secure command + * This is Optional function & user implementable + * @param dwLength : length of data from the host + * @param bBWI : Block Waiting Timeout sent by host + * @param wLevelParameter : Parameters sent by host + * @param pbuf : buffer containing the data + * @param returnLen : Length of data expected to return + * @retval status value + */ +uint8_t SC_Itf_Secure(uint32_t dwLength, uint8_t bBWI, uint16_t wLevelParameter, + uint8_t *pbuf, uint32_t *returnLen) +{ + UNUSED(pbuf); + UNUSED(wLevelParameter); + UNUSED(bBWI); + UNUSED(dwLength); + *returnLen = 0U; + + return SLOTERROR_CMD_NOT_SUPPORTED; +} + +/** + * @brief SC_SaveVoltage + Saves the voltage value to be saved for further usage + * @param voltage: voltage value to be saved for further usage + * @retval None + */ +static void SC_SaveVoltage(uint8_t voltage) +{ + SC_Param.voltage = voltage; + + return; +} + +/** + * @brief Provides the value of SCState variable + * @param None + * @retval uint8_t SCState + */ +uint8_t SC_GetState(void) +{ + return (uint8_t)SCState; +} + +/** + * @brief Set the value of SCState variable to Off + * @param scState: value of SCState to be updated + * @retval None + */ +void SC_SetState(SC_State scState) +{ + SCState = scState; + + return; +} diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_smartcard_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_smartcard_template.c new file mode 100644 index 00000000..059aef4e --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CCID/Src/usbd_ccid_smartcard_template.c @@ -0,0 +1,486 @@ +/** + ****************************************************************************** + * @file usbd_ccid_smartcard_template.c + * @author MCD Application Team + * @brief This file provides all the Smartcard firmware functions. + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/** @addtogroup usbd_ccid_Smartcard + * @{ + */ + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_ccid_smartcard_template.h" + +/* Private typedef -----------------------------------------------------------*/ +/* Private define ------------------------------------------------------------*/ +/* Private macro -------------------------------------------------------------*/ +/* Private variables ---------------------------------------------------------*/ +/* Directories & Files ID */ +/*The following Directories & Files ID can take any of following Values and can + be used in the smartcard application */ +/* +const uint8_t MasterRoot[2] = {0x3F, 0x00}; +const uint8_t GSMDir[2] = {0x7F, 0x20}; +const uint8_t ICCID[2] = {0x2F, 0xE2}; +const uint8_t IMSI[2] = {0x6F, 0x07}; + +__IO uint8_t ICCID_Content[10] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; + +uint32_t CHV1Status = 0U; + +uint8_t CHV1[8] = {'0', '0', '0', '0', '0', '0', '0', '0'}; +__IO uint8_t IMSI_Content[9] = {0x01, 0x02, 0x04, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; +*/ + +/* F Table: Clock Rate Conversion Table from ISO/IEC 7816-3 */ +/* static uint32_t F_Table[16] = {372, 372, 558, 744, 1116, 1488, 1860, 0, 0, 512, 768, + 1024, 1536, 2048, 0, 0 + }; */ + + +/* D Table: Baud Rate Adjustment Factor Table from ISO/IEC 7816-3 */ +static uint32_t D_Table[16] = {0, 1, 2, 4, 8, 16, 32, 64, 12, 20, 0, 0, 0, 0, 0, 0}; + +/* Global variables definition and initialization ----------------------------*/ +SC_ATRTypeDef SC_A2R; +uint8_t SC_ATR_Table[40]; +uint8_t ProtocolNUM_OUT; + +/* Private function prototypes -----------------------------------------------*/ +static void SC_Init(void); +static void SC_DeInit(void); +static void SC_AnswerReq(SC_State *SC_state, uint8_t *card, uint8_t length); /* Ask ATR */ +static uint8_t SC_decode_Answer2reset(uint8_t *card); /* Decode ATR */ +static void SC_SendData(SC_ADPU_CommandsTypeDef *SCADPU, SC_ADPU_ResponseTypeDef *SC_ResponseStatus); +/* static void SC_Reset(GPIO_PinState ResetState); */ + +/* Private functions ---------------------------------------------------------*/ + +/** + * @brief Handles all Smartcard states and serves to send and receive all + * communication data between Smartcard and reader. + * @param SCState: pointer to an SC_State enumeration that will contain the + * Smartcard state. + * @param SC_ADPU: pointer to an SC_ADPU_Commands structure that will be initialized. + * @param SC_Response: pointer to a SC_ADPU_Response structure which will be initialized. + * @retval None + */ +void SC_Handler(SC_State *SCState, SC_ADPU_CommandsTypeDef *SC_ADPU, SC_ADPU_ResponseTypeDef *SC_Response) +{ + uint32_t i; + uint32_t j; + + switch (*SCState) + { + case SC_POWER_ON: + if (SC_ADPU->Header.INS == SC_GET_A2R) + { + /* Smartcard initialization */ + SC_Init(); + + /* Reset Data from SC buffer */ + for (i = 0U; i < 40U; i++) + { + SC_ATR_Table[i] = 0; + } + + /* Reset SC_A2R Structure */ + SC_A2R.TS = 0U; + SC_A2R.T0 = 0U; + + for (i = 0U; i < MAX_PROTOCOLLEVEL; i++) + { + for (j = 0U; j < MAX_INTERFACEBYTE; j++) + { + SC_A2R.T[i].InterfaceByte[j].Status = 0U; + SC_A2R.T[i].InterfaceByte[j].Value = 0U; + } + } + + for (i = 0U; i < HIST_LENGTH; i++) + { + SC_A2R.Historical[i] = 0U; + } + + SC_A2R.Tlength = 0U; + SC_A2R.Hlength = 0U; + + /* Next State */ + *SCState = SC_RESET_LOW; + } + break; + + case SC_RESET_LOW: + if (SC_ADPU->Header.INS == SC_GET_A2R) + { + /* If card is detected then Power ON, Card Reset and wait for an answer) */ + if (SC_Detect() != 0U) + { + while (((*SCState) != SC_POWER_OFF) && ((*SCState) != SC_ACTIVE)) + { + SC_AnswerReq(SCState, &SC_ATR_Table[0], 40U); /* Check for answer to reset */ + } + } + else + { + (*SCState) = SC_POWER_OFF; + } + } + break; + + case SC_ACTIVE: + if (SC_ADPU->Header.INS == SC_GET_A2R) + { + uint8_t protocol = SC_decode_Answer2reset(&SC_ATR_Table[0]); + if (protocol == T0_PROTOCOL) + { + (*SCState) = SC_ACTIVE_ON_T0; + ProtocolNUM_OUT = T0_PROTOCOL; + } + else if (protocol == T1_PROTOCOL) + { + (*SCState) = SC_ACTIVE_ON_T1; + ProtocolNUM_OUT = T1_PROTOCOL; + } + else + { + (*SCState) = SC_POWER_OFF; + } + } + break; + + case SC_ACTIVE_ON_T0: + /* process commands other than ATR */ + SC_SendData(SC_ADPU, SC_Response); + break; + + case SC_ACTIVE_ON_T1: + /* process commands other than ATR */ + SC_SendData(SC_ADPU, SC_Response); + break; + + case SC_POWER_OFF: + SC_DeInit(); /* Disable Smartcard interface */ + break; + + default: + (*SCState) = SC_POWER_OFF; + break; + } +} + +/** + * @brief Enables or disables the power to the Smartcard. + * @param NewState: new state of the Smartcard power supply. + * This parameter can be: SC_ENABLED or SC_DISABLED. + * @retval None + */ +void SC_PowerCmd(SCPowerState NewState) +{ + UNUSED(NewState); + /* enable or disable smartcard pin */ + + return; +} + +/** + * @brief Sets or clears the Smartcard reset pin. + * @param ResetState: this parameter specifies the state of the Smartcard + * reset pin. BitVal must be one of the BitAction enum values: + * @arg Bit_RESET: to clear the port pin. + * @arg Bit_SET: to set the port pin. + * @retval None + */ +/* static void SC_Reset(GPIO_PinState ResetState) +{ + UNUSED(ResetState); + + return; +} +*/ + + +/** + * @brief Resends the byte that failed to be received (by the Smartcard) correctly. + * @param None + * @retval None + */ + +void SC_ParityErrorHandler(void) +{ + /* Add your code here */ + + return; +} + +/** + * @brief Configures the IO speed (BaudRate) communication. + * @param None + * @retval None + */ + +void SC_PTSConfig(void) +{ + /* Add your code here */ + + return; +} + + +/** + * @brief Manages the Smartcard transport layer: send APDU commands and receives + * the APDU response. + * @param SC_ADPU: pointer to a SC_ADPU_Commands structure which will be initialized. + * @param SC_Response: pointer to a SC_ADPU_Response structure which will be initialized. + * @retval None + */ +static void SC_SendData(SC_ADPU_CommandsTypeDef *SCADPU, SC_ADPU_ResponseTypeDef *SC_ResponseStatus) +{ + uint8_t i; + uint8_t SC_Command[5]; + uint8_t SC_DATA[LC_MAX]; + + UNUSED(SCADPU); + + /* Reset response buffer */ + for (i = 0U; i < LC_MAX; i++) + { + SC_ResponseStatus->Data[i] = 0U; + SC_DATA[i] = 0U; + } + + /* User to add code here */ + + /* send command to ICC and get response status */ + USBD_CCID_If_fops.Send_Process((uint8_t *)&SC_Command, (uint8_t *)&SC_DATA); + +} + +/** + * @brief SC_AnswerReq + Requests the reset answer from card. + * @param SC_state: pointer to an SC_State enumeration that will contain the Smartcard state. + * @param atr_buffer: pointer to a buffer which will contain the card ATR. + * @param length: maximum ATR length + * @retval None + */ +static void SC_AnswerReq(SC_State *SC_state, uint8_t *atr_buffer, uint8_t length) +{ + UNUSED(length); + UNUSED(atr_buffer); + + /* to be implemented by USER */ + switch (*SC_state) + { + case SC_RESET_LOW: + /* Check response with reset low */ + (*SC_state) = SC_ACTIVE; + break; + + case SC_ACTIVE: + break; + case SC_RESET_HIGH: + /* Check response with reset high */ + + break; + + case SC_POWER_OFF: + /* Close Connection if no answer received */ + + break; + + default: + (*SC_state) = SC_RESET_LOW; + break; + } + + return; +} + +/** + * @brief SC_decode_Answer2reset + Decodes the Answer to reset received from card. + * @param card: pointer to the buffer containing the card ATR. + * @retval None + */ +static uint8_t SC_decode_Answer2reset(uint8_t *card) +{ + uint32_t i = 0U; + uint32_t flag = 0U; + uint32_t protocol; + uint8_t index = 0U; + uint8_t level = 0U; + + /******************************TS/T0 Decode************************************/ + index++; + SC_A2R.TS = card[index]; /* Initial character */ + + index++; + SC_A2R.T0 = card[index]; /* Format character */ + + /*************************Historical Table Length Decode***********************/ + SC_A2R.Hlength = SC_A2R.T0 & 0x0FU; + + /******************************Protocol Level(1) Decode************************/ + /* Check TD(1) if present */ + if ((SC_A2R.T0 & 0x80U) == 0x80U) + { + flag = 1U; + } + + /* Each bits in the T0 high nibble(b8 to b5) equal to 1 indicates the presence + of a further interface byte */ + for (i = 0U; i < 4U; i++) + { + if ((((SC_A2R.T0 & 0xF0U) >> (4U + i)) & 0x1U) != 0U) + { + SC_A2R.T[level].InterfaceByte[i].Status = 1U; + index++; + SC_A2R.T[level].InterfaceByte[i].Value = card[index]; + SC_A2R.Tlength++; + } + } + + /*****************************T Decode*****************************************/ + if (SC_A2R.T[level].InterfaceByte[3].Status == 1U) + { + /* Only the protocol(parameter T) present in TD(1) is detected + if two or more values of parameter T are present in TD(1), TD(2)..., so the + firmware should be updated to support them */ + protocol = (uint8_t)(SC_A2R.T[level].InterfaceByte[SC_INTERFACEBYTE_TD].Value & 0x0FU); + } + else + { + protocol = 0U; + } + + /* Protocol Level Increment */ + /******************************Protocol Level(n>1) Decode**********************/ + while (flag != 0U) + { + if ((SC_A2R.T[level].InterfaceByte[SC_INTERFACEBYTE_TD].Value & 0x80U) == 0x80U) + { + flag = 1U; + } + else + { + flag = 0U; + } + /* Each bits in the high nibble(b8 to b5) for the TD(i) equal to 1 indicates + the presence of a further interface byte */ + for (i = 0U; i < 4U; i++) + { + if ((((SC_A2R.T[level].InterfaceByte[SC_INTERFACEBYTE_TD].Value & 0xF0U) >> (4U + i)) & 0x1U) != 0U) + { + SC_A2R.T[level + 1U].InterfaceByte[i].Status = 1U; + index++; + SC_A2R.T[level + 1U].InterfaceByte[i].Value = card[index]; + SC_A2R.Tlength++; + } + } + level++; + } + + for (i = 0U; i < SC_A2R.Hlength; i++) + { + SC_A2R.Historical[i] = card[i + 2U + SC_A2R.Tlength]; + } + /*************************************TCK Decode*******************************/ + SC_A2R.TCK = card[SC_A2R.Hlength + 2U + SC_A2R.Tlength]; + + return (uint8_t)protocol; +} + +/** + * @brief Initializes all peripheral used for Smartcard interface. + * @param None + * @retval None + */ +static void SC_Init(void) +{ + /* + Add your initialization code here + */ + + return; +} + + +/** + * @brief Deinitializes all resources used by the Smartcard interface. + * @param None + * @retval None + */ +static void SC_DeInit(void) +{ + /* + Add your deinitialization code here + */ + + return; +} + +/** + * @brief Configures the card power voltage. + * @param SC_Voltage: specifies the card power voltage. + * This parameter can be one of the following values: + * @arg SC_VOLTAGE_5V: 5V cards. + * @arg SC_VOLTAGE_3V: 3V cards. + * @retval None + */ +void SC_VoltageConfig(uint32_t SC_Voltage) +{ + UNUSED(SC_Voltage); + /* Add your code here */ + + return; +} + +/** + * @brief Configures GPIO hardware resources used for Samrtcard. + * @param None + * @retval None + */ +void SC_IOConfig(void) +{ + /* Add your code here */ + + return; +} + +/** + * @brief Detects whether the Smartcard is present or not. + * @param None. + * @retval 1 - Smartcard inserted + * 0 - Smartcard not inserted + */ +uint8_t SC_Detect(void) +{ + uint8_t PIN_State = 0U; + + /* Add your code here */ + + return PIN_State; +} + +/** + * @brief Get the Right Value from the D_Table Index + * @param idx : Index to Read from the Table + * @retval Value read from the Table + */ +uint32_t SC_GetDTableValue(uint32_t idx) +{ + return D_Table[idx]; +} diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Inc/usbd_cdc.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Inc/usbd_cdc.h index cb5c6d8a..aeac6bf5 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Inc/usbd_cdc.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Inc/usbd_cdc.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -41,9 +40,15 @@ extern "C" { /** @defgroup usbd_cdc_Exported_Defines * @{ */ +#ifndef CDC_IN_EP #define CDC_IN_EP 0x81U /* EP1 for data IN */ +#endif /* CDC_IN_EP */ +#ifndef CDC_OUT_EP #define CDC_OUT_EP 0x01U /* EP1 for data OUT */ +#endif /* CDC_OUT_EP */ +#ifndef CDC_CMD_EP #define CDC_CMD_EP 0x82U /* EP2 for CDC commands */ +#endif /* CDC_CMD_EP */ #ifndef CDC_HS_BINTERVAL #define CDC_HS_BINTERVAL 0x10U @@ -149,12 +154,17 @@ extern USBD_ClassTypeDef USBD_CDC; uint8_t USBD_CDC_RegisterInterface(USBD_HandleTypeDef *pdev, USBD_CDC_ItfTypeDef *fops); +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CDC_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, + uint32_t length, uint8_t ClassId); +uint8_t USBD_CDC_TransmitPacket(USBD_HandleTypeDef *pdev, uint8_t ClassId); +#else uint8_t USBD_CDC_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, uint32_t length); - +uint8_t USBD_CDC_TransmitPacket(USBD_HandleTypeDef *pdev); +#endif /* USE_USBD_COMPOSITE */ uint8_t USBD_CDC_SetRxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff); uint8_t USBD_CDC_ReceivePacket(USBD_HandleTypeDef *pdev); -uint8_t USBD_CDC_TransmitPacket(USBD_HandleTypeDef *pdev); /** * @} */ @@ -172,4 +182,3 @@ uint8_t USBD_CDC_TransmitPacket(USBD_HandleTypeDef *pdev); * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Inc/usbd_cdc_if_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Inc/usbd_cdc_if_template.h index 158ad40c..6a1c0abf 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Inc/usbd_cdc_if_template.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Inc/usbd_cdc_if_template.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -42,4 +41,3 @@ extern USBD_CDC_ItfTypeDef USBD_CDC_Template_fops; #endif /* __USBD_CDC_IF_TEMPLATE_H */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Src/usbd_cdc.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Src/usbd_cdc.c index d6ed6ce3..ad8da254 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Src/usbd_cdc.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Src/usbd_cdc.c @@ -10,6 +10,17 @@ * - Command IN transfer (class requests management) * - Error management * + ****************************************************************************** + * @attention + * + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** * @verbatim * * =================================================================== @@ -37,17 +48,6 @@ * @endverbatim * ****************************************************************************** - * @attention - * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> - * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** */ /* BSPDependencies @@ -105,13 +105,14 @@ static uint8_t USBD_CDC_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *re static uint8_t USBD_CDC_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum); static uint8_t USBD_CDC_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum); static uint8_t USBD_CDC_EP0_RxReady(USBD_HandleTypeDef *pdev); - +#ifndef USE_USBD_COMPOSITE static uint8_t *USBD_CDC_GetFSCfgDesc(uint16_t *length); static uint8_t *USBD_CDC_GetHSCfgDesc(uint16_t *length); static uint8_t *USBD_CDC_GetOtherSpeedCfgDesc(uint16_t *length); -static uint8_t *USBD_CDC_GetOtherSpeedCfgDesc(uint16_t *length); uint8_t *USBD_CDC_GetDeviceQualifierDescriptor(uint16_t *length); +#endif /* USE_USBD_COMPOSITE */ +#ifndef USE_USBD_COMPOSITE /* USB Standard Device Descriptor */ __ALIGN_BEGIN static uint8_t USBD_CDC_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_DESC] __ALIGN_END = { @@ -126,7 +127,7 @@ __ALIGN_BEGIN static uint8_t USBD_CDC_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_ 0x01, 0x00, }; - +#endif /* USE_USBD_COMPOSITE */ /** * @} */ @@ -149,113 +150,22 @@ USBD_ClassTypeDef USBD_CDC = NULL, NULL, NULL, +#ifdef USE_USBD_COMPOSITE + NULL, + NULL, + NULL, + NULL, +#else USBD_CDC_GetHSCfgDesc, USBD_CDC_GetFSCfgDesc, USBD_CDC_GetOtherSpeedCfgDesc, USBD_CDC_GetDeviceQualifierDescriptor, +#endif /* USE_USBD_COMPOSITE */ }; +#ifndef USE_USBD_COMPOSITE /* USB CDC device Configuration Descriptor */ -__ALIGN_BEGIN static uint8_t USBD_CDC_CfgHSDesc[USB_CDC_CONFIG_DESC_SIZ] __ALIGN_END = -{ - /* Configuration Descriptor */ - 0x09, /* bLength: Configuration Descriptor size */ - USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ - USB_CDC_CONFIG_DESC_SIZ, /* wTotalLength:no of returned bytes */ - 0x00, - 0x02, /* bNumInterfaces: 2 interfaces */ - 0x01, /* bConfigurationValue: Configuration value */ - 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ -#if (USBD_SELF_POWERED == 1U) - 0xC0, /* bmAttributes: Bus Powered according to user configuration */ -#else - 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif - USBD_MAX_POWER, /* MaxPower (mA) */ - - /*---------------------------------------------------------------------------*/ - - /* Interface Descriptor */ - 0x09, /* bLength: Interface Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: Interface */ - 0x00, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x01, /* bNumEndpoints: One endpoints used */ - 0x02, /* bInterfaceClass: Communication Interface Class */ - 0x02, /* bInterfaceSubClass: Abstract Control Model */ - 0x01, /* bInterfaceProtocol: Common AT commands */ - 0x00, /* iInterface */ - - /* Header Functional Descriptor */ - 0x05, /* bLength: Endpoint Descriptor size */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x00, /* bDescriptorSubtype: Header Func Desc */ - 0x10, /* bcdCDC: spec release number */ - 0x01, - - /* Call Management Functional Descriptor */ - 0x05, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x01, /* bDescriptorSubtype: Call Management Func Desc */ - 0x00, /* bmCapabilities: D0+D1 */ - 0x01, /* bDataInterface: 1 */ - - /* ACM Functional Descriptor */ - 0x04, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x02, /* bDescriptorSubtype: Abstract Control Management desc */ - 0x02, /* bmCapabilities */ - - /* Union Functional Descriptor */ - 0x05, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x06, /* bDescriptorSubtype: Union func desc */ - 0x00, /* bMasterInterface: Communication class interface */ - 0x01, /* bSlaveInterface0: Data Class Interface */ - - /* Endpoint 2 Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_CMD_EP, /* bEndpointAddress */ - 0x03, /* bmAttributes: Interrupt */ - LOBYTE(CDC_CMD_PACKET_SIZE), /* wMaxPacketSize */ - HIBYTE(CDC_CMD_PACKET_SIZE), - CDC_HS_BINTERVAL, /* bInterval */ - /*---------------------------------------------------------------------------*/ - - /* Data class interface descriptor */ - 0x09, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: */ - 0x01, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x02, /* bNumEndpoints: Two endpoints used */ - 0x0A, /* bInterfaceClass: CDC */ - 0x00, /* bInterfaceSubClass */ - 0x00, /* bInterfaceProtocol */ - 0x00, /* iInterface */ - - /* Endpoint OUT Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_OUT_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - LOBYTE(CDC_DATA_HS_MAX_PACKET_SIZE), /* wMaxPacketSize */ - HIBYTE(CDC_DATA_HS_MAX_PACKET_SIZE), - 0x00, /* bInterval */ - - /* Endpoint IN Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_IN_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - LOBYTE(CDC_DATA_HS_MAX_PACKET_SIZE), /* wMaxPacketSize */ - HIBYTE(CDC_DATA_HS_MAX_PACKET_SIZE), - 0x00 /* bInterval */ -}; - - -/* USB CDC device Configuration Descriptor */ -__ALIGN_BEGIN static uint8_t USBD_CDC_CfgFSDesc[USB_CDC_CONFIG_DESC_SIZ] __ALIGN_END = +__ALIGN_BEGIN static uint8_t USBD_CDC_CfgDesc[USB_CDC_CONFIG_DESC_SIZ] __ALIGN_END = { /* Configuration Descriptor */ 0x09, /* bLength: Configuration Descriptor size */ @@ -264,12 +174,13 @@ __ALIGN_BEGIN static uint8_t USBD_CDC_CfgFSDesc[USB_CDC_CONFIG_DESC_SIZ] __ALIGN 0x00, 0x02, /* bNumInterfaces: 2 interfaces */ 0x01, /* bConfigurationValue: Configuration value */ - 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ + 0x00, /* iConfiguration: Index of string descriptor + describing the configuration */ #if (USBD_SELF_POWERED == 1U) 0xC0, /* bmAttributes: Bus Powered according to user configuration */ #else 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif +#endif /* USBD_SELF_POWERED */ USBD_MAX_POWER, /* MaxPower (mA) */ /*---------------------------------------------------------------------------*/ @@ -352,102 +263,11 @@ __ALIGN_BEGIN static uint8_t USBD_CDC_CfgFSDesc[USB_CDC_CONFIG_DESC_SIZ] __ALIGN HIBYTE(CDC_DATA_FS_MAX_PACKET_SIZE), 0x00 /* bInterval */ }; +#endif /* USE_USBD_COMPOSITE */ -__ALIGN_BEGIN static uint8_t USBD_CDC_OtherSpeedCfgDesc[USB_CDC_CONFIG_DESC_SIZ] __ALIGN_END = -{ - 0x09, /* bLength: Configuration Descriptor size */ - USB_DESC_TYPE_OTHER_SPEED_CONFIGURATION, - USB_CDC_CONFIG_DESC_SIZ, - 0x00, - 0x02, /* bNumInterfaces: 2 interfaces */ - 0x01, /* bConfigurationValue */ - 0x04, /* iConfiguration */ -#if (USBD_SELF_POWERED == 1U) - 0xC0, /* bmAttributes: Bus Powered according to user configuration */ -#else - 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif - USBD_MAX_POWER, /* MaxPower (mA) */ - - /* Interface Descriptor */ - 0x09, /* bLength: Interface Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: Interface */ - /* Interface descriptor type */ - 0x00, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x01, /* bNumEndpoints: One endpoints used */ - 0x02, /* bInterfaceClass: Communication Interface Class */ - 0x02, /* bInterfaceSubClass: Abstract Control Model */ - 0x01, /* bInterfaceProtocol: Common AT commands */ - 0x00, /* iInterface */ - - /* Header Functional Descriptor */ - 0x05, /* bLength: Endpoint Descriptor size */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x00, /* bDescriptorSubtype: Header Func Desc */ - 0x10, /* bcdCDC: spec release number */ - 0x01, - - /* Call Management Functional Descriptor */ - 0x05, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x01, /* bDescriptorSubtype: Call Management Func Desc */ - 0x00, /* bmCapabilities */ - 0x01, /* bDataInterface */ - - /* ACM Functional Descriptor */ - 0x04, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x02, /* bDescriptorSubtype: Abstract Control Management desc */ - 0x02, /* bmCapabilities */ - - /* Union Functional Descriptor */ - 0x05, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x06, /* bDescriptorSubtype: Union func desc */ - 0x00, /* bMasterInterface: Communication class interface */ - 0x01, /* bSlaveInterface0: Data Class Interface */ - - /* Endpoint 2 Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_CMD_EP, /* bEndpointAddress */ - 0x03, /* bmAttributes: Interrupt */ - LOBYTE(CDC_CMD_PACKET_SIZE), /* wMaxPacketSize */ - HIBYTE(CDC_CMD_PACKET_SIZE), - CDC_FS_BINTERVAL, /* bInterval */ - - /*---------------------------------------------------------------------------*/ - - /*Data class interface descriptor*/ - 0x09, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: */ - 0x01, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x02, /* bNumEndpoints: Two endpoints used */ - 0x0A, /* bInterfaceClass: CDC */ - 0x00, /* bInterfaceSubClass */ - 0x00, /* bInterfaceProtocol */ - 0x00, /* iInterface */ - - /*Endpoint OUT Descriptor*/ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_OUT_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - 0x40, /* wMaxPacketSize */ - 0x00, - 0x00, /* bInterval */ - - /*Endpoint IN Descriptor*/ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_IN_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - 0x40, /* wMaxPacketSize */ - 0x00, - 0x00 /* bInterval */ -}; +static uint8_t CDCInEpAdd = CDC_IN_EP; +static uint8_t CDCOutEpAdd = CDC_OUT_EP; +static uint8_t CDCCmdEpAdd = CDC_CMD_EP; /** * @} @@ -473,68 +293,85 @@ static uint8_t USBD_CDC_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) if (hcdc == NULL) { - pdev->pClassData = NULL; + pdev->pClassDataCmsit[pdev->classId] = NULL; return (uint8_t)USBD_EMEM; } - pdev->pClassData = (void *)hcdc; + (void)USBD_memset(hcdc, 0, sizeof(USBD_CDC_HandleTypeDef)); + + pdev->pClassDataCmsit[pdev->classId] = (void *)hcdc; + pdev->pClassData = pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + CDCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + CDCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + CDCCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (pdev->dev_speed == USBD_SPEED_HIGH) { /* Open EP IN */ - (void)USBD_LL_OpenEP(pdev, CDC_IN_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, CDCInEpAdd, USBD_EP_TYPE_BULK, CDC_DATA_HS_IN_PACKET_SIZE); - pdev->ep_in[CDC_IN_EP & 0xFU].is_used = 1U; + pdev->ep_in[CDCInEpAdd & 0xFU].is_used = 1U; /* Open EP OUT */ - (void)USBD_LL_OpenEP(pdev, CDC_OUT_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, CDCOutEpAdd, USBD_EP_TYPE_BULK, CDC_DATA_HS_OUT_PACKET_SIZE); - pdev->ep_out[CDC_OUT_EP & 0xFU].is_used = 1U; + pdev->ep_out[CDCOutEpAdd & 0xFU].is_used = 1U; /* Set bInterval for CDC CMD Endpoint */ - pdev->ep_in[CDC_CMD_EP & 0xFU].bInterval = CDC_HS_BINTERVAL; + pdev->ep_in[CDCCmdEpAdd & 0xFU].bInterval = CDC_HS_BINTERVAL; } else { /* Open EP IN */ - (void)USBD_LL_OpenEP(pdev, CDC_IN_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, CDCInEpAdd, USBD_EP_TYPE_BULK, CDC_DATA_FS_IN_PACKET_SIZE); - pdev->ep_in[CDC_IN_EP & 0xFU].is_used = 1U; + pdev->ep_in[CDCInEpAdd & 0xFU].is_used = 1U; /* Open EP OUT */ - (void)USBD_LL_OpenEP(pdev, CDC_OUT_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, CDCOutEpAdd, USBD_EP_TYPE_BULK, CDC_DATA_FS_OUT_PACKET_SIZE); - pdev->ep_out[CDC_OUT_EP & 0xFU].is_used = 1U; + pdev->ep_out[CDCOutEpAdd & 0xFU].is_used = 1U; /* Set bInterval for CMD Endpoint */ - pdev->ep_in[CDC_CMD_EP & 0xFU].bInterval = CDC_FS_BINTERVAL; + pdev->ep_in[CDCCmdEpAdd & 0xFU].bInterval = CDC_FS_BINTERVAL; } /* Open Command IN EP */ - (void)USBD_LL_OpenEP(pdev, CDC_CMD_EP, USBD_EP_TYPE_INTR, CDC_CMD_PACKET_SIZE); - pdev->ep_in[CDC_CMD_EP & 0xFU].is_used = 1U; + (void)USBD_LL_OpenEP(pdev, CDCCmdEpAdd, USBD_EP_TYPE_INTR, CDC_CMD_PACKET_SIZE); + pdev->ep_in[CDCCmdEpAdd & 0xFU].is_used = 1U; + + hcdc->RxBuffer = NULL; /* Init physical Interface components */ - ((USBD_CDC_ItfTypeDef *)pdev->pUserData)->Init(); + ((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->Init(); /* Init Xfer states */ hcdc->TxState = 0U; hcdc->RxState = 0U; + if (hcdc->RxBuffer == NULL) + { + return (uint8_t)USBD_EMEM; + } + if (pdev->dev_speed == USBD_SPEED_HIGH) { /* Prepare Out endpoint to receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, CDC_OUT_EP, hcdc->RxBuffer, + (void)USBD_LL_PrepareReceive(pdev, CDCOutEpAdd, hcdc->RxBuffer, CDC_DATA_HS_OUT_PACKET_SIZE); } else { /* Prepare Out endpoint to receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, CDC_OUT_EP, hcdc->RxBuffer, + (void)USBD_LL_PrepareReceive(pdev, CDCOutEpAdd, hcdc->RxBuffer, CDC_DATA_FS_OUT_PACKET_SIZE); } @@ -552,24 +389,33 @@ static uint8_t USBD_CDC_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) { UNUSED(cfgidx); + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this CDC class instance */ + CDCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + CDCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + CDCCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + /* Close EP IN */ - (void)USBD_LL_CloseEP(pdev, CDC_IN_EP); - pdev->ep_in[CDC_IN_EP & 0xFU].is_used = 0U; + (void)USBD_LL_CloseEP(pdev, CDCInEpAdd); + pdev->ep_in[CDCInEpAdd & 0xFU].is_used = 0U; /* Close EP OUT */ - (void)USBD_LL_CloseEP(pdev, CDC_OUT_EP); - pdev->ep_out[CDC_OUT_EP & 0xFU].is_used = 0U; + (void)USBD_LL_CloseEP(pdev, CDCOutEpAdd); + pdev->ep_out[CDCOutEpAdd & 0xFU].is_used = 0U; /* Close Command IN EP */ - (void)USBD_LL_CloseEP(pdev, CDC_CMD_EP); - pdev->ep_in[CDC_CMD_EP & 0xFU].is_used = 0U; - pdev->ep_in[CDC_CMD_EP & 0xFU].bInterval = 0U; + (void)USBD_LL_CloseEP(pdev, CDCCmdEpAdd); + pdev->ep_in[CDCCmdEpAdd & 0xFU].is_used = 0U; + pdev->ep_in[CDCCmdEpAdd & 0xFU].bInterval = 0U; /* DeInit physical Interface components */ - if (pdev->pClassData != NULL) + if (pdev->pClassDataCmsit[pdev->classId] != NULL) { - ((USBD_CDC_ItfTypeDef *)pdev->pUserData)->DeInit(); - (void)USBD_free(pdev->pClassData); + ((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->DeInit(); + (void)USBD_free(pdev->pClassDataCmsit[pdev->classId]); + pdev->pClassDataCmsit[pdev->classId] = NULL; pdev->pClassData = NULL; } @@ -586,7 +432,7 @@ static uint8_t USBD_CDC_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) static uint8_t USBD_CDC_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; uint16_t len; uint8_t ifalt = 0U; uint16_t status_info = 0U; @@ -604,9 +450,9 @@ static uint8_t USBD_CDC_Setup(USBD_HandleTypeDef *pdev, { if ((req->bmRequest & 0x80U) != 0U) { - ((USBD_CDC_ItfTypeDef *)pdev->pUserData)->Control(req->bRequest, - (uint8_t *)hcdc->data, - req->wLength); + ((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->Control(req->bRequest, + (uint8_t *)hcdc->data, + req->wLength); len = MIN(CDC_REQ_MAX_DATA_SIZE, req->wLength); (void)USBD_CtlSendData(pdev, (uint8_t *)hcdc->data, len); @@ -621,8 +467,8 @@ static uint8_t USBD_CDC_Setup(USBD_HandleTypeDef *pdev, } else { - ((USBD_CDC_ItfTypeDef *)pdev->pUserData)->Control(req->bRequest, - (uint8_t *)req, 0U); + ((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->Control(req->bRequest, + (uint8_t *)req, 0U); } break; @@ -690,20 +536,20 @@ static uint8_t USBD_CDC_Setup(USBD_HandleTypeDef *pdev, static uint8_t USBD_CDC_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) { USBD_CDC_HandleTypeDef *hcdc; - PCD_HandleTypeDef *hpcd = pdev->pData; + PCD_HandleTypeDef *hpcd = (PCD_HandleTypeDef *)pdev->pData; - if (pdev->pClassData == NULL) + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_FAIL; } - hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; + hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; - if ((pdev->ep_in[epnum].total_length > 0U) && - ((pdev->ep_in[epnum].total_length % hpcd->IN_ep[epnum].maxpacket) == 0U)) + if ((pdev->ep_in[epnum & 0xFU].total_length > 0U) && + ((pdev->ep_in[epnum & 0xFU].total_length % hpcd->IN_ep[epnum & 0xFU].maxpacket) == 0U)) { /* Update the packet total length */ - pdev->ep_in[epnum].total_length = 0U; + pdev->ep_in[epnum & 0xFU].total_length = 0U; /* Send ZLP */ (void)USBD_LL_Transmit(pdev, epnum, NULL, 0U); @@ -712,9 +558,9 @@ static uint8_t USBD_CDC_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) { hcdc->TxState = 0U; - if (((USBD_CDC_ItfTypeDef *)pdev->pUserData)->TransmitCplt != NULL) + if (((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->TransmitCplt != NULL) { - ((USBD_CDC_ItfTypeDef *)pdev->pUserData)->TransmitCplt(hcdc->TxBuffer, &hcdc->TxLength, epnum); + ((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->TransmitCplt(hcdc->TxBuffer, &hcdc->TxLength, epnum); } } @@ -730,9 +576,9 @@ static uint8_t USBD_CDC_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) */ static uint8_t USBD_CDC_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) { - USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; - if (pdev->pClassData == NULL) + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_FAIL; } @@ -743,7 +589,7 @@ static uint8_t USBD_CDC_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) /* USB data will be immediately processed, this allow next USB traffic being NAKed till the end of the application Xfer */ - ((USBD_CDC_ItfTypeDef *)pdev->pUserData)->Receive(hcdc->RxBuffer, &hcdc->RxLength); + ((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->Receive(hcdc->RxBuffer, &hcdc->RxLength); return (uint8_t)USBD_OK; } @@ -756,64 +602,115 @@ static uint8_t USBD_CDC_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) */ static uint8_t USBD_CDC_EP0_RxReady(USBD_HandleTypeDef *pdev) { - USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hcdc == NULL) { return (uint8_t)USBD_FAIL; } - if ((pdev->pUserData != NULL) && (hcdc->CmdOpCode != 0xFFU)) + if ((pdev->pUserData[pdev->classId] != NULL) && (hcdc->CmdOpCode != 0xFFU)) { - ((USBD_CDC_ItfTypeDef *)pdev->pUserData)->Control(hcdc->CmdOpCode, - (uint8_t *)hcdc->data, - (uint16_t)hcdc->CmdLength); + ((USBD_CDC_ItfTypeDef *)pdev->pUserData[pdev->classId])->Control(hcdc->CmdOpCode, + (uint8_t *)hcdc->data, + (uint16_t)hcdc->CmdLength); hcdc->CmdOpCode = 0xFFU; } return (uint8_t)USBD_OK; } - +#ifndef USE_USBD_COMPOSITE /** * @brief USBD_CDC_GetFSCfgDesc * Return configuration descriptor - * @param speed : current device speed * @param length : pointer data length * @retval pointer to descriptor buffer */ static uint8_t *USBD_CDC_GetFSCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_CDC_CfgFSDesc); + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_CDC_CfgDesc, CDC_CMD_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CDC_CfgDesc, CDC_OUT_EP); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CDC_CfgDesc, CDC_IN_EP); + + if (pEpCmdDesc != NULL) + { + pEpCmdDesc->bInterval = CDC_FS_BINTERVAL; + } - return USBD_CDC_CfgFSDesc; + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = CDC_DATA_FS_MAX_PACKET_SIZE; + } + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = CDC_DATA_FS_MAX_PACKET_SIZE; + } + + *length = (uint16_t)sizeof(USBD_CDC_CfgDesc); + return USBD_CDC_CfgDesc; } /** * @brief USBD_CDC_GetHSCfgDesc * Return configuration descriptor - * @param speed : current device speed * @param length : pointer data length * @retval pointer to descriptor buffer */ static uint8_t *USBD_CDC_GetHSCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_CDC_CfgHSDesc); + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_CDC_CfgDesc, CDC_CMD_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CDC_CfgDesc, CDC_OUT_EP); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CDC_CfgDesc, CDC_IN_EP); - return USBD_CDC_CfgHSDesc; + if (pEpCmdDesc != NULL) + { + pEpCmdDesc->bInterval = CDC_HS_BINTERVAL; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = CDC_DATA_HS_MAX_PACKET_SIZE; + } + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = CDC_DATA_HS_MAX_PACKET_SIZE; + } + + *length = (uint16_t)sizeof(USBD_CDC_CfgDesc); + return USBD_CDC_CfgDesc; } /** * @brief USBD_CDC_GetOtherSpeedCfgDesc * Return configuration descriptor - * @param speed : current device speed * @param length : pointer data length * @retval pointer to descriptor buffer */ static uint8_t *USBD_CDC_GetOtherSpeedCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_CDC_OtherSpeedCfgDesc); + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_CDC_CfgDesc, CDC_CMD_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CDC_CfgDesc, CDC_OUT_EP); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CDC_CfgDesc, CDC_IN_EP); - return USBD_CDC_OtherSpeedCfgDesc; + if (pEpCmdDesc != NULL) + { + pEpCmdDesc->bInterval = CDC_FS_BINTERVAL; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = CDC_DATA_FS_MAX_PACKET_SIZE; + } + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = CDC_DATA_FS_MAX_PACKET_SIZE; + } + + *length = (uint16_t)sizeof(USBD_CDC_CfgDesc); + return USBD_CDC_CfgDesc; } /** @@ -828,7 +725,7 @@ uint8_t *USBD_CDC_GetDeviceQualifierDescriptor(uint16_t *length) return USBD_CDC_DeviceQualifierDesc; } - +#endif /* USE_USBD_COMPOSITE */ /** * @brief USBD_CDC_RegisterInterface * @param pdev: device instance @@ -843,21 +740,31 @@ uint8_t USBD_CDC_RegisterInterface(USBD_HandleTypeDef *pdev, return (uint8_t)USBD_FAIL; } - pdev->pUserData = fops; + pdev->pUserData[pdev->classId] = fops; return (uint8_t)USBD_OK; } + /** * @brief USBD_CDC_SetTxBuffer * @param pdev: device instance * @param pbuff: Tx Buffer + * @param length: length of data to be sent + * @param ClassId: The Class ID * @retval status */ +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CDC_SetTxBuffer(USBD_HandleTypeDef *pdev, + uint8_t *pbuff, uint32_t length, uint8_t ClassId) +{ + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[ClassId]; +#else uint8_t USBD_CDC_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, uint32_t length) { - USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#endif /* USE_USBD_COMPOSITE */ if (hcdc == NULL) { @@ -878,7 +785,7 @@ uint8_t USBD_CDC_SetTxBuffer(USBD_HandleTypeDef *pdev, */ uint8_t USBD_CDC_SetRxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff) { - USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hcdc == NULL) { @@ -890,18 +797,32 @@ uint8_t USBD_CDC_SetRxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff) return (uint8_t)USBD_OK; } + /** * @brief USBD_CDC_TransmitPacket * Transmit packet on IN endpoint * @param pdev: device instance + * @param ClassId: The Class ID * @retval status */ +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CDC_TransmitPacket(USBD_HandleTypeDef *pdev, uint8_t ClassId) +{ + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[ClassId]; +#else uint8_t USBD_CDC_TransmitPacket(USBD_HandleTypeDef *pdev) { - USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#endif /* USE_USBD_COMPOSITE */ + USBD_StatusTypeDef ret = USBD_BUSY; - if (pdev->pClassData == NULL) +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + CDCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, ClassId); +#endif /* USE_USBD_COMPOSITE */ + + if (hcdc == NULL) { return (uint8_t)USBD_FAIL; } @@ -912,10 +833,10 @@ uint8_t USBD_CDC_TransmitPacket(USBD_HandleTypeDef *pdev) hcdc->TxState = 1U; /* Update the packet total length */ - pdev->ep_in[CDC_IN_EP & 0xFU].total_length = hcdc->TxLength; + pdev->ep_in[CDCInEpAdd & 0xFU].total_length = hcdc->TxLength; /* Transmit next packet */ - (void)USBD_LL_Transmit(pdev, CDC_IN_EP, hcdc->TxBuffer, hcdc->TxLength); + (void)USBD_LL_Transmit(pdev, CDCInEpAdd, hcdc->TxBuffer, hcdc->TxLength); ret = USBD_OK; } @@ -931,9 +852,14 @@ uint8_t USBD_CDC_TransmitPacket(USBD_HandleTypeDef *pdev) */ uint8_t USBD_CDC_ReceivePacket(USBD_HandleTypeDef *pdev) { - USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassData; + USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + CDCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ - if (pdev->pClassData == NULL) + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_FAIL; } @@ -941,13 +867,13 @@ uint8_t USBD_CDC_ReceivePacket(USBD_HandleTypeDef *pdev) if (pdev->dev_speed == USBD_SPEED_HIGH) { /* Prepare Out endpoint to receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, CDC_OUT_EP, hcdc->RxBuffer, + (void)USBD_LL_PrepareReceive(pdev, CDCOutEpAdd, hcdc->RxBuffer, CDC_DATA_HS_OUT_PACKET_SIZE); } else { /* Prepare Out endpoint to receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, CDC_OUT_EP, hcdc->RxBuffer, + (void)USBD_LL_PrepareReceive(pdev, CDCOutEpAdd, hcdc->RxBuffer, CDC_DATA_FS_OUT_PACKET_SIZE); } @@ -965,4 +891,3 @@ uint8_t USBD_CDC_ReceivePacket(USBD_HandleTypeDef *pdev) * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Src/usbd_cdc_if_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Src/usbd_cdc_if_template.c index f863ce1b..df16834b 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Src/usbd_cdc_if_template.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Src/usbd_cdc_if_template.c @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -128,6 +127,8 @@ static int8_t TEMPLATE_DeInit(void) */ static int8_t TEMPLATE_Control(uint8_t cmd, uint8_t *pbuf, uint16_t length) { + UNUSED(length); + switch (cmd) { case CDC_SEND_ENCAPSULATED_COMMAND: @@ -244,5 +245,3 @@ static int8_t TEMPLATE_TransmitCplt(uint8_t *Buf, uint32_t *Len, uint8_t epnum) * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ - diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Inc/usbd_cdc_ecm.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Inc/usbd_cdc_ecm.h index 1cd71fa1..a69c4fd2 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Inc/usbd_cdc_ecm.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Inc/usbd_cdc_ecm.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2019 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2019 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -43,10 +42,15 @@ extern "C" { */ /* Comment this define in order to disable the CDC ECM Notification pipe */ - +#ifndef CDC_ECM_IN_EP #define CDC_ECM_IN_EP 0x81U /* EP1 for data IN */ +#endif /* CDC_ECM_IN_EP */ +#ifndef CDC_ECM_OUT_EP #define CDC_ECM_OUT_EP 0x01U /* EP1 for data OUT */ +#endif /* CDC_ECM_OUT_EP */ +#ifndef CDC_ECM_CMD_EP #define CDC_ECM_CMD_EP 0x82U /* EP2 for CDC ECM commands */ +#endif /* CDC_ECM_CMD_EP */ #ifndef CDC_ECM_CMD_ITF_NBR #define CDC_ECM_CMD_ITF_NBR 0x00U /* Command Interface Number 0 */ @@ -173,6 +177,26 @@ typedef struct uint8_t data[8]; } USBD_CDC_ECM_NotifTypeDef; +/* + * ECM Class specification revision 1.2 + * Table 3: Ethernet Networking Functional Descriptor + */ + +typedef struct +{ + uint8_t bFunctionLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubType; + uint8_t iMacAddress; + uint8_t bEthernetStatistics3; + uint8_t bEthernetStatistics2; + uint8_t bEthernetStatistics1; + uint8_t bEthernetStatistics0; + uint16_t wMaxSegmentSize; + uint16_t bNumberMCFiltes; + uint8_t bNumberPowerFiltes; +} __PACKED USBD_ECMFuncDescTypeDef; + typedef struct { uint32_t data[CDC_ECM_DATA_BUFFER_SIZE / 4U]; /* Force 32-bit alignment */ @@ -194,12 +218,6 @@ typedef struct USBD_CDC_ECM_NotifTypeDef Req; } USBD_CDC_ECM_HandleTypeDef; -typedef enum -{ - NETWORK_CONNECTION = 0x00, - RESPONSE_AVAILABLE = 0x01, - CONNECTION_SPEED_CHANGE = 0x2A -} USBD_CDC_ECM_NotifCodeTypeDef; /** @defgroup USBD_CORE_Exported_Macros * @{ @@ -225,17 +243,21 @@ extern USBD_ClassTypeDef USBD_CDC_ECM; uint8_t USBD_CDC_ECM_RegisterInterface(USBD_HandleTypeDef *pdev, USBD_CDC_ECM_ItfTypeDef *fops); -uint8_t USBD_CDC_ECM_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, - uint32_t length); - uint8_t USBD_CDC_ECM_SetRxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff); uint8_t USBD_CDC_ECM_ReceivePacket(USBD_HandleTypeDef *pdev); +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CDC_ECM_TransmitPacket(USBD_HandleTypeDef *pdev, uint8_t ClassId); +uint8_t USBD_CDC_ECM_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, + uint32_t length, uint8_t ClassId); +#else uint8_t USBD_CDC_ECM_TransmitPacket(USBD_HandleTypeDef *pdev); - +uint8_t USBD_CDC_ECM_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, + uint32_t length); +#endif /* USE_USBD_COMPOSITE */ uint8_t USBD_CDC_ECM_SendNotification(USBD_HandleTypeDef *pdev, - USBD_CDC_ECM_NotifCodeTypeDef Notif, + USBD_CDC_NotifCodeTypeDef Notif, uint16_t bVal, uint8_t *pData); /** * @} @@ -254,4 +276,3 @@ uint8_t USBD_CDC_ECM_SendNotification(USBD_HandleTypeDef *pdev, * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Inc/usbd_cdc_ecm_if_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Inc/usbd_cdc_ecm_if_template.h index 7ce8714d..c08f0f3f 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Inc/usbd_cdc_ecm_if_template.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Inc/usbd_cdc_ecm_if_template.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2019 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2019 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -79,4 +78,3 @@ extern USBD_CDC_ECM_ItfTypeDef USBD_CDC_ECM_fops; /* Exported functions ------------------------------------------------------- */ #endif /* __USBD_CDC_ECM_IF_H */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Src/usbd_cdc_ecm.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Src/usbd_cdc_ecm.c index 4b34d7c9..ec632526 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Src/usbd_cdc_ecm.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Src/usbd_cdc_ecm.c @@ -13,13 +13,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2019 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2019 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -35,7 +34,7 @@ EndBSPDependencies */ #ifndef __USBD_CDC_ECM_IF_H #include "usbd_cdc_ecm_if_template.h" -#endif +#endif /* __USBD_CDC_ECM_IF_H */ /** @addtogroup STM32_USB_DEVICE_LIBRARY @@ -84,19 +83,20 @@ static uint8_t USBD_CDC_ECM_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum); static uint8_t USBD_CDC_ECM_EP0_RxReady(USBD_HandleTypeDef *pdev); static uint8_t USBD_CDC_ECM_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); - +#ifndef USE_USBD_COMPOSITE static uint8_t *USBD_CDC_ECM_GetFSCfgDesc(uint16_t *length); static uint8_t *USBD_CDC_ECM_GetHSCfgDesc(uint16_t *length); static uint8_t *USBD_CDC_ECM_GetOtherSpeedCfgDesc(uint16_t *length); static uint8_t *USBD_CDC_ECM_GetOtherSpeedCfgDesc(uint16_t *length); - +#endif /* USE_USBD_COMPOSITE */ #if (USBD_SUPPORT_USER_STRING_DESC == 1U) static uint8_t *USBD_CDC_ECM_USRStringDescriptor(USBD_HandleTypeDef *pdev, uint8_t index, uint16_t *length); -#endif - +#endif /* USBD_SUPPORT_USER_STRING_DESC */ +#ifndef USE_USBD_COMPOSITE uint8_t *USBD_CDC_ECM_GetDeviceQualifierDescriptor(uint16_t *length); - +#endif /* USE_USBD_COMPOSITE */ +#ifndef USE_USBD_COMPOSITE /* USB Standard Device Descriptor */ __ALIGN_BEGIN static uint8_t USBD_CDC_ECM_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_DESC] __ALIGN_END = { @@ -111,7 +111,7 @@ __ALIGN_BEGIN static uint8_t USBD_CDC_ECM_DeviceQualifierDesc[USB_LEN_DEV_QUALIF 0x01, 0x00, }; - +#endif /* USE_USBD_COMPOSITE */ static uint32_t ConnSpeedTab[2] = {CDC_ECM_CONNECT_SPEED_UPSTREAM, CDC_ECM_CONNECT_SPEED_DOWNSTREAM }; @@ -138,129 +138,25 @@ USBD_ClassTypeDef USBD_CDC_ECM = NULL, NULL, NULL, +#ifdef USE_USBD_COMPOSITE + NULL, + NULL, + NULL, + NULL, +#else USBD_CDC_ECM_GetHSCfgDesc, USBD_CDC_ECM_GetFSCfgDesc, USBD_CDC_ECM_GetOtherSpeedCfgDesc, USBD_CDC_ECM_GetDeviceQualifierDescriptor, +#endif /* USE_USBD_COMPOSITE */ #if (USBD_SUPPORT_USER_STRING_DESC == 1U) USBD_CDC_ECM_USRStringDescriptor, -#endif -}; - -/* USB CDC_ECM device Configuration Descriptor */ -__ALIGN_BEGIN static uint8_t USBD_CDC_ECM_CfgHSDesc[] __ALIGN_END = -{ - /* Configuration Descriptor */ - 0x09, /* bLength: Configuration Descriptor size */ - USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ - LOBYTE(CDC_ECM_CONFIG_DESC_SIZ), /* wTotalLength:no of returned bytes */ - HIBYTE(CDC_ECM_CONFIG_DESC_SIZ), - 0x02, /* bNumInterfaces: 2 interfaces */ - 0x01, /* bConfigurationValue: Configuration value */ - 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ -#if (USBD_SELF_POWERED == 1U) - 0xC0, /* bmAttributes: Bus Powered according to user configuration */ -#else - 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif - USBD_MAX_POWER, /* MaxPower (mA) */ - - /*---------------------------------------------------------------------------*/ - - /* IAD descriptor */ - 0x08, /* bLength */ - 0x0B, /* bDescriptorType */ - 0x00, /* bFirstInterface */ - 0x02, /* bInterfaceCount */ - 0x02, /* bFunctionClass (Wireless Controller) */ - 0x06, /* bFunctionSubClass */ - 0x00, /* bFunctionProtocol */ - 0x00, /* iFunction */ - - /* Interface Descriptor */ - 0x09, /* bLength: Interface Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: Interface descriptor type */ - CDC_ECM_CMD_ITF_NBR, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x01, /* bNumEndpoints: One endpoint used */ - 0x02, /* bInterfaceClass: Communication Interface Class */ - 0x06, /* bInterfaceSubClass: Ethernet Control Model */ - 0x00, /* bInterfaceProtocol: No specific protocol required */ - 0x00, /* iInterface */ - - /* Header Functional Descriptor */ - 0x05, /* bLength: Endpoint Descriptor size */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x00, /* bDescriptorSubtype: Header functional descriptor */ - 0x10, /* bcd CDC_ECM: spec release number: 1.10 */ - 0x01, - - /* CDC_ECM Functional Descriptor */ - 0x0D, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x0F, /* Ethernet Networking functional descriptor subtype */ - CDC_ECM_MAC_STRING_INDEX, /* Device's MAC string index */ - CDC_ECM_ETH_STATS_BYTE3, /* Ethernet statistics byte 3 (bitmap) */ - CDC_ECM_ETH_STATS_BYTE2, /* Ethernet statistics byte 2 (bitmap) */ - CDC_ECM_ETH_STATS_BYTE1, /* Ethernet statistics byte 1 (bitmap) */ - CDC_ECM_ETH_STATS_BYTE0, /* Ethernet statistics byte 0 (bitmap) */ - LOBYTE(CDC_ECM_ETH_MAX_SEGSZE), - HIBYTE(CDC_ECM_ETH_MAX_SEGSZE), /* wMaxSegmentSize: Ethernet Maximum Segment size, typically 1514 bytes */ - LOBYTE(CDC_ECM_ETH_NBR_MACFILTERS), - HIBYTE(CDC_ECM_ETH_NBR_MACFILTERS), /* wNumberMCFilters: the number of multicast filters */ - CDC_ECM_ETH_NBR_PWRFILTERS, /* bNumberPowerFilters: the number of wakeup power filters */ - - /* Union Functional Descriptor */ - 0x05, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x06, /* bDescriptorSubtype: Union functional descriptor */ - 0x00, /* bMasterInterface: Communication class interface */ - 0x01, /* bSlaveInterface0: Data Class Interface */ - - /* Communication Endpoint Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_ECM_CMD_EP, /* bEndpointAddress */ - 0x03, /* bmAttributes: Interrupt */ - LOBYTE(CDC_ECM_CMD_PACKET_SIZE), /* wMaxPacketSize */ - HIBYTE(CDC_ECM_CMD_PACKET_SIZE), - CDC_ECM_HS_BINTERVAL, /* bInterval */ - - /*----------------------*/ - - /* Data class interface descriptor */ - 0x09, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: */ - CDC_ECM_COM_ITF_NBR, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x02, /* bNumEndpoints: Two endpoints used */ - 0x0A, /* bInterfaceClass: CDC */ - 0x00, /* bInterfaceSubClass */ - 0x00, /* bInterfaceProtocol */ - 0x00, /* iInterface */ - - /* Endpoint OUT Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_ECM_OUT_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - LOBYTE(CDC_ECM_DATA_HS_MAX_PACKET_SIZE), /* wMaxPacketSize */ - HIBYTE(CDC_ECM_DATA_HS_MAX_PACKET_SIZE), - 0x00, /* bInterval */ - - /* Endpoint IN Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_ECM_IN_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - LOBYTE(CDC_ECM_DATA_HS_MAX_PACKET_SIZE), /* wMaxPacketSize */ - HIBYTE(CDC_ECM_DATA_HS_MAX_PACKET_SIZE), - 0x00 /* bInterval */ +#endif /* USBD_SUPPORT_USER_STRING_DESC */ }; - +#ifndef USE_USBD_COMPOSITE /* USB CDC_ECM device Configuration Descriptor */ -__ALIGN_BEGIN static uint8_t USBD_CDC_ECM_CfgFSDesc[] __ALIGN_END = +__ALIGN_BEGIN static uint8_t USBD_CDC_ECM_CfgDesc[] __ALIGN_END = { /* Configuration Descriptor */ 0x09, /* bLength: Configuration Descriptor size */ @@ -269,12 +165,13 @@ __ALIGN_BEGIN static uint8_t USBD_CDC_ECM_CfgFSDesc[] __ALIGN_END = HIBYTE(CDC_ECM_CONFIG_DESC_SIZ), 0x02, /* bNumInterfaces: 2 interfaces */ 0x01, /* bConfigurationValue: Configuration value */ - 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ + 0x00, /* iConfiguration: Index of string descriptor + describing the configuration */ #if (USBD_SELF_POWERED == 1U) 0xC0, /* bmAttributes: Bus Powered according to user configuration */ #else 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif +#endif /* USBD_SELF_POWERED */ USBD_MAX_POWER, /* MaxPower (mA) */ /*---------------------------------------------------------------------------*/ @@ -369,115 +266,11 @@ __ALIGN_BEGIN static uint8_t USBD_CDC_ECM_CfgFSDesc[] __ALIGN_END = HIBYTE(CDC_ECM_DATA_FS_MAX_PACKET_SIZE), 0x00 /* bInterval */ } ; +#endif /* USE_USBD_COMPOSITE */ -__ALIGN_BEGIN static uint8_t USBD_CDC_ECM_OtherSpeedCfgDesc[] __ALIGN_END = -{ - /* Configuration Descriptor */ - 0x09, /* bLength: Configuration Descriptor size */ - USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ - LOBYTE(CDC_ECM_CONFIG_DESC_SIZ), /* wTotalLength */ - HIBYTE(CDC_ECM_CONFIG_DESC_SIZ), - 0x02, /* bNumInterfaces: 2 interfaces */ - 0x01, /* bConfigurationValue: Configuration value */ - 0x04, /* iConfiguration: Index of string descriptor describing the configuration */ -#if (USBD_SELF_POWERED == 1U) - 0xC0, /* bmAttributes: Bus Powered according to user configuration */ -#else - 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif - USBD_MAX_POWER, /* MaxPower (mA) */ - - /*--------------------------------------- ------------------------------------*/ - /* IAD descriptor */ - 0x08, /* bLength */ - 0x0B, /* bDescriptorType */ - 0x00, /* bFirstInterface */ - 0x02, /* bInterfaceCount */ - 0x02, /* bFunctionClass (Wireless Controller) */ - 0x06, /* bFunctionSubClass */ - 0x00, /* bFunctionProtocol */ - 0x00, /* iFunction */ - - /* Interface Descriptor */ - 0x09, /* bLength: Interface Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: Interface descriptor type */ - 0x00, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x01, /* bNumEndpoints: One endpoint used */ - 0x02, /* bInterfaceClass: Communication Interface Class */ - 0x06, /* bInterfaceSubClass: Ethernet Control Model */ - 0x00, /* bInterfaceProtocol: No specific protocol required */ - 0x00, /* iInterface: */ - - /* Header Functional Descriptor */ - 0x05, /* bLength: Endpoint Descriptor size */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x00, /* bDescriptorSubtype: Header functional descriptor */ - 0x10, /* bcd CDC_ECM : spec release number: 1.20 */ - 0x01, - - /* CDC_ECM Functional Descriptor */ - 0x0D, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x0F, /* Ethernet Networking functional descriptor subtype */ - CDC_ECM_MAC_STRING_INDEX, /* Device's MAC string index */ - CDC_ECM_ETH_STATS_BYTE3, /* Ethernet statistics byte 3 (bitmap) */ - CDC_ECM_ETH_STATS_BYTE2, /* Ethernet statistics byte 2 (bitmap) */ - CDC_ECM_ETH_STATS_BYTE1, /* Ethernet statistics byte 1 (bitmap) */ - CDC_ECM_ETH_STATS_BYTE0, /* Ethernet statistics byte 0 (bitmap) */ - LOBYTE(CDC_ECM_ETH_MAX_SEGSZE), - HIBYTE(CDC_ECM_ETH_MAX_SEGSZE), /* wMaxSegmentSize: Ethernet Maximum Segment size, typically 1514 bytes */ - LOBYTE(CDC_ECM_ETH_NBR_MACFILTERS), - HIBYTE(CDC_ECM_ETH_NBR_MACFILTERS), /* wNumberMCFilters: the number of multicast filters */ - CDC_ECM_ETH_NBR_PWRFILTERS, /* bNumberPowerFilters: the number of wakeup power filters */ - - /* Union Functional Descriptor */ - 0x05, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x06, /* bDescriptorSubtype: Union functional descriptor */ - 0x00, /* bMasterInterface: Communication class interface */ - 0x01, /* bSlaveInterface0: Data Class Interface */ - - /* Communication Endpoint Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_ECM_CMD_EP, /* bEndpointAddress */ - 0x03, /* bmAttributes: Interrupt */ - LOBYTE(CDC_ECM_CMD_PACKET_SIZE), /* wMaxPacketSize */ - HIBYTE(CDC_ECM_CMD_PACKET_SIZE), - CDC_ECM_FS_BINTERVAL, /* bInterval */ - - /*----------------------*/ - - /* Data class interface descriptor */ - 0x09, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: interface */ - 0x01, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x02, /* bNumEndpoints: Two endpoints used */ - 0x0A, /* bInterfaceClass: CDC */ - 0x00, /* bInterfaceSubClass */ - 0x00, /* bInterfaceProtocol */ - 0x00, /* iInterface: */ - - /* Endpoint OUT Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_ECM_OUT_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - 0x40, /* wMaxPacketSize */ - 0x00, - 0x00, /* bInterval */ - - /* Endpoint IN Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_ECM_IN_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - 0x40, /* wMaxPacketSize */ - 0x00, - 0x00 /* bInterval */ -}; +static uint8_t ECMInEpAdd = CDC_ECM_IN_EP; +static uint8_t ECMOutEpAdd = CDC_ECM_OUT_EP; +static uint8_t ECMCmdEpAdd = CDC_ECM_CMD_EP; /** * @} @@ -500,57 +293,69 @@ static uint8_t USBD_CDC_ECM_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) USBD_CDC_ECM_HandleTypeDef *hcdc; +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + ECMInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + ECMOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + ECMCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + hcdc = (USBD_CDC_ECM_HandleTypeDef *)USBD_malloc(sizeof(USBD_CDC_ECM_HandleTypeDef)); if (hcdc == NULL) { - pdev->pClassData = NULL; + pdev->pClassDataCmsit[pdev->classId] = NULL; return (uint8_t)USBD_EMEM; } - pdev->pClassData = (void *)hcdc; + (void)USBD_memset(hcdc, 0, sizeof(USBD_CDC_ECM_HandleTypeDef)); + + pdev->pClassDataCmsit[pdev->classId] = (void *)hcdc; + pdev->pClassData = pdev->pClassDataCmsit[pdev->classId]; if (pdev->dev_speed == USBD_SPEED_HIGH) { /* Open EP IN */ - (void)USBD_LL_OpenEP(pdev, CDC_ECM_IN_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, ECMInEpAdd, USBD_EP_TYPE_BULK, CDC_ECM_DATA_HS_IN_PACKET_SIZE); - pdev->ep_in[CDC_ECM_IN_EP & 0xFU].is_used = 1U; + pdev->ep_in[ECMInEpAdd & 0xFU].is_used = 1U; /* Open EP OUT */ - (void)USBD_LL_OpenEP(pdev, CDC_ECM_OUT_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, ECMOutEpAdd, USBD_EP_TYPE_BULK, CDC_ECM_DATA_HS_OUT_PACKET_SIZE); - pdev->ep_out[CDC_ECM_OUT_EP & 0xFU].is_used = 1U; + pdev->ep_out[ECMOutEpAdd & 0xFU].is_used = 1U; /* Set bInterval for CDC ECM CMD Endpoint */ - pdev->ep_in[CDC_ECM_CMD_EP & 0xFU].bInterval = CDC_ECM_HS_BINTERVAL; + pdev->ep_in[ECMCmdEpAdd & 0xFU].bInterval = CDC_ECM_HS_BINTERVAL; } else { /* Open EP IN */ - (void)USBD_LL_OpenEP(pdev, CDC_ECM_IN_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, ECMInEpAdd, USBD_EP_TYPE_BULK, CDC_ECM_DATA_FS_IN_PACKET_SIZE); - pdev->ep_in[CDC_ECM_IN_EP & 0xFU].is_used = 1U; + pdev->ep_in[ECMInEpAdd & 0xFU].is_used = 1U; /* Open EP OUT */ - (void)USBD_LL_OpenEP(pdev, CDC_ECM_OUT_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, ECMOutEpAdd, USBD_EP_TYPE_BULK, CDC_ECM_DATA_FS_OUT_PACKET_SIZE); - pdev->ep_out[CDC_ECM_OUT_EP & 0xFU].is_used = 1U; + pdev->ep_out[ECMOutEpAdd & 0xFU].is_used = 1U; /* Set bInterval for CDC ECM CMD Endpoint */ - pdev->ep_in[CDC_ECM_CMD_EP & 0xFU].bInterval = CDC_ECM_FS_BINTERVAL; + pdev->ep_in[ECMCmdEpAdd & 0xFU].bInterval = CDC_ECM_FS_BINTERVAL; } /* Open Command IN EP */ - (void)USBD_LL_OpenEP(pdev, CDC_ECM_CMD_EP, USBD_EP_TYPE_INTR, CDC_ECM_CMD_PACKET_SIZE); - pdev->ep_in[CDC_ECM_CMD_EP & 0xFU].is_used = 1U; + (void)USBD_LL_OpenEP(pdev, ECMCmdEpAdd, USBD_EP_TYPE_INTR, CDC_ECM_CMD_PACKET_SIZE); + pdev->ep_in[ECMCmdEpAdd & 0xFU].is_used = 1U; + + hcdc->RxBuffer = NULL; /* Init physical Interface components */ - ((USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData)->Init(); + ((USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData[pdev->classId])->Init(); /* Init Xfer states */ hcdc->TxState = 0U; @@ -559,10 +364,16 @@ static uint8_t USBD_CDC_ECM_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) hcdc->TxLength = 0U; hcdc->LinkStatus = 0U; hcdc->NotificationStatus = 0U; - hcdc->MaxPcktLen = (pdev->dev_speed == USBD_SPEED_HIGH) ? CDC_ECM_DATA_HS_MAX_PACKET_SIZE : CDC_ECM_DATA_FS_MAX_PACKET_SIZE; + hcdc->MaxPcktLen = (pdev->dev_speed == USBD_SPEED_HIGH) ? CDC_ECM_DATA_HS_MAX_PACKET_SIZE : \ + CDC_ECM_DATA_FS_MAX_PACKET_SIZE; + + if (hcdc->RxBuffer == NULL) + { + return (uint8_t)USBD_EMEM; + } /* Prepare Out endpoint to receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, CDC_ECM_OUT_EP, hcdc->RxBuffer, hcdc->MaxPcktLen); + (void)USBD_LL_PrepareReceive(pdev, ECMOutEpAdd, hcdc->RxBuffer, hcdc->MaxPcktLen); return (uint8_t)USBD_OK; } @@ -578,24 +389,32 @@ static uint8_t USBD_CDC_ECM_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) { UNUSED(cfgidx); +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + ECMInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + ECMOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + ECMCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + /* Close EP IN */ - (void)USBD_LL_CloseEP(pdev, CDC_ECM_IN_EP); - pdev->ep_in[CDC_ECM_IN_EP & 0xFU].is_used = 0U; + (void)USBD_LL_CloseEP(pdev, ECMInEpAdd); + pdev->ep_in[ECMInEpAdd & 0xFU].is_used = 0U; /* Close EP OUT */ - (void)USBD_LL_CloseEP(pdev, CDC_ECM_OUT_EP); - pdev->ep_out[CDC_ECM_OUT_EP & 0xFU].is_used = 0U; + (void)USBD_LL_CloseEP(pdev, ECMOutEpAdd); + pdev->ep_out[ECMOutEpAdd & 0xFU].is_used = 0U; /* Close Command IN EP */ - (void)USBD_LL_CloseEP(pdev, CDC_ECM_CMD_EP); - pdev->ep_in[CDC_ECM_CMD_EP & 0xFU].is_used = 0U; - pdev->ep_in[CDC_ECM_CMD_EP & 0xFU].bInterval = 0U; + (void)USBD_LL_CloseEP(pdev, ECMCmdEpAdd); + pdev->ep_in[ECMCmdEpAdd & 0xFU].is_used = 0U; + pdev->ep_in[ECMCmdEpAdd & 0xFU].bInterval = 0U; /* DeInit physical Interface components */ - if (pdev->pClassData != NULL) + if (pdev->pClassDataCmsit[pdev->classId] != NULL) { - ((USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData)->DeInit(); - USBD_free(pdev->pClassData); + ((USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData[pdev->classId])->DeInit(); + USBD_free(pdev->pClassDataCmsit[pdev->classId]); + pdev->pClassDataCmsit[pdev->classId] = NULL; pdev->pClassData = NULL; } @@ -609,11 +428,10 @@ static uint8_t USBD_CDC_ECM_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) * @param req: usb requests * @retval status */ -static uint8_t USBD_CDC_ECM_Setup(USBD_HandleTypeDef *pdev, - USBD_SetupReqTypedef *req) +static uint8_t USBD_CDC_ECM_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *) pdev->pClassData; - USBD_CDC_ECM_ItfTypeDef *EcmInterface = (USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData; + USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *) pdev->pClassDataCmsit[pdev->classId]; + USBD_CDC_ECM_ItfTypeDef *EcmInterface = (USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData[pdev->classId]; USBD_StatusTypeDef ret = USBD_OK; uint16_t len; uint16_t status_info = 0U; @@ -631,8 +449,7 @@ static uint8_t USBD_CDC_ECM_Setup(USBD_HandleTypeDef *pdev, { if ((req->bmRequest & 0x80U) != 0U) { - EcmInterface->Control(req->bRequest, - (uint8_t *)hcdc->data, req->wLength); + EcmInterface->Control(req->bRequest, (uint8_t *)hcdc->data, req->wLength); len = MIN(CDC_ECM_DATA_BUFFER_SIZE, req->wLength); (void)USBD_CtlSendData(pdev, (uint8_t *)hcdc->data, len); @@ -714,21 +531,26 @@ static uint8_t USBD_CDC_ECM_Setup(USBD_HandleTypeDef *pdev, */ static uint8_t USBD_CDC_ECM_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) { - USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassData; - PCD_HandleTypeDef *hpcd = pdev->pData; + USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + PCD_HandleTypeDef *hpcd = (PCD_HandleTypeDef *)pdev->pData; - if (pdev->pClassData == NULL) +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + ECMInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_FAIL; } - if (epnum == (CDC_ECM_IN_EP & 0x7FU)) + if (epnum == (ECMInEpAdd & 0x7FU)) { - if ((pdev->ep_in[epnum].total_length > 0U) && - ((pdev->ep_in[epnum].total_length % hpcd->IN_ep[epnum].maxpacket) == 0U)) + if ((pdev->ep_in[epnum & 0xFU].total_length > 0U) && + ((pdev->ep_in[epnum & 0xFU].total_length % hpcd->IN_ep[epnum & 0xFU].maxpacket) == 0U)) { /* Update the packet total length */ - pdev->ep_in[epnum].total_length = 0U; + pdev->ep_in[epnum & 0xFU].total_length = 0U; /* Send ZLP */ (void)USBD_LL_Transmit(pdev, epnum, NULL, 0U); @@ -736,18 +558,18 @@ static uint8_t USBD_CDC_ECM_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) else { hcdc->TxState = 0U; - if (((USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData)->TransmitCplt != NULL) + if (((USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData[pdev->classId])->TransmitCplt != NULL) { - ((USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData)->TransmitCplt(hcdc->TxBuffer, &hcdc->TxLength, epnum); + ((USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData[pdev->classId])->TransmitCplt(hcdc->TxBuffer, + &hcdc->TxLength, epnum); } } } - else if (epnum == (CDC_ECM_CMD_EP & 0x7FU)) + else if (epnum == (ECMCmdEpAdd & 0x7FU)) { if (hcdc->NotificationStatus != 0U) { - (void)USBD_CDC_ECM_SendNotification(pdev, CONNECTION_SPEED_CHANGE, - 0U, (uint8_t *)ConnSpeedTab); + (void)USBD_CDC_ECM_SendNotification(pdev, CONNECTION_SPEED_CHANGE, 0U, (uint8_t *)ConnSpeedTab); hcdc->NotificationStatus = 0U; } @@ -769,15 +591,20 @@ static uint8_t USBD_CDC_ECM_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) */ static uint8_t USBD_CDC_ECM_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) { - USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassData; + USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; uint32_t CurrPcktLen; - if (pdev->pClassData == NULL) +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + ECMOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_FAIL; } - if (epnum == CDC_ECM_OUT_EP) + if (epnum == ECMOutEpAdd) { /* Get the received data length */ CurrPcktLen = USBD_LL_GetRxDataSize(pdev, epnum); @@ -793,12 +620,12 @@ static uint8_t USBD_CDC_ECM_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) /* Process data by application (ie. copy to app buffer or notify user) hcdc->RxLength must be reset to zero at the end of the call of this function */ - ((USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData)->Receive(hcdc->RxBuffer, &hcdc->RxLength); + ((USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData[pdev->classId])->Receive(hcdc->RxBuffer, &hcdc->RxLength); } else { /* Prepare Out endpoint to receive next packet in current/new frame */ - (void)USBD_LL_PrepareReceive(pdev, CDC_ECM_OUT_EP, + (void)USBD_LL_PrepareReceive(pdev, ECMOutEpAdd, (uint8_t *)(hcdc->RxBuffer + hcdc->RxLength), hcdc->MaxPcktLen); } @@ -819,64 +646,115 @@ static uint8_t USBD_CDC_ECM_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) */ static uint8_t USBD_CDC_ECM_EP0_RxReady(USBD_HandleTypeDef *pdev) { - USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassData; + USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hcdc == NULL) { return (uint8_t)USBD_FAIL; } - if ((pdev->pUserData != NULL) && (hcdc->CmdOpCode != 0xFFU)) + if ((pdev->pUserData[pdev->classId] != NULL) && (hcdc->CmdOpCode != 0xFFU)) { - ((USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData)->Control(hcdc->CmdOpCode, - (uint8_t *)hcdc->data, - (uint16_t)hcdc->CmdLength); + ((USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData[pdev->classId])->Control(hcdc->CmdOpCode, + (uint8_t *)hcdc->data, + (uint16_t)hcdc->CmdLength); hcdc->CmdOpCode = 0xFFU; } return (uint8_t)USBD_OK; } - +#ifndef USE_USBD_COMPOSITE /** * @brief USBD_CDC_ECM_GetFSCfgDesc * Return configuration descriptor - * @param speed : current device speed * @param length : pointer data length * @retval pointer to descriptor buffer */ static uint8_t *USBD_CDC_ECM_GetFSCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_CDC_ECM_CfgFSDesc); + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_CDC_ECM_CfgDesc, CDC_ECM_CMD_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CDC_ECM_CfgDesc, CDC_ECM_OUT_EP); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CDC_ECM_CfgDesc, CDC_ECM_IN_EP); + + if (pEpCmdDesc != NULL) + { + pEpCmdDesc->bInterval = CDC_ECM_FS_BINTERVAL; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = CDC_ECM_DATA_FS_MAX_PACKET_SIZE; + } + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = CDC_ECM_DATA_FS_MAX_PACKET_SIZE; + } - return USBD_CDC_ECM_CfgFSDesc; + *length = (uint16_t) sizeof(USBD_CDC_ECM_CfgDesc); + return USBD_CDC_ECM_CfgDesc; } /** * @brief USBD_CDC_ECM_GetHSCfgDesc * Return configuration descriptor - * @param speed : current device speed * @param length : pointer data length * @retval pointer to descriptor buffer */ static uint8_t *USBD_CDC_ECM_GetHSCfgDesc(uint16_t *length) { - *length = (uint16_t) sizeof(USBD_CDC_ECM_CfgHSDesc); + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_CDC_ECM_CfgDesc, CDC_ECM_CMD_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CDC_ECM_CfgDesc, CDC_ECM_OUT_EP); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CDC_ECM_CfgDesc, CDC_ECM_IN_EP); - return USBD_CDC_ECM_CfgHSDesc; + if (pEpCmdDesc != NULL) + { + pEpCmdDesc->bInterval = CDC_ECM_HS_BINTERVAL; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = CDC_ECM_DATA_HS_MAX_PACKET_SIZE; + } + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = CDC_ECM_DATA_HS_MAX_PACKET_SIZE; + } + + *length = (uint16_t) sizeof(USBD_CDC_ECM_CfgDesc); + return USBD_CDC_ECM_CfgDesc; } /** * @brief USBD_CDC_ECM_GetCfgDesc * Return configuration descriptor - * @param speed : current device speed * @param length : pointer data length * @retval pointer to descriptor buffer */ static uint8_t *USBD_CDC_ECM_GetOtherSpeedCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_CDC_ECM_OtherSpeedCfgDesc); + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_CDC_ECM_CfgDesc, CDC_ECM_CMD_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CDC_ECM_CfgDesc, CDC_ECM_OUT_EP); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CDC_ECM_CfgDesc, CDC_ECM_IN_EP); + + if (pEpCmdDesc != NULL) + { + pEpCmdDesc->bInterval = CDC_ECM_FS_BINTERVAL; + } - return USBD_CDC_ECM_OtherSpeedCfgDesc; + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = CDC_ECM_DATA_FS_MAX_PACKET_SIZE; + } + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = CDC_ECM_DATA_FS_MAX_PACKET_SIZE; + } + + *length = (uint16_t) sizeof(USBD_CDC_ECM_CfgDesc); + return USBD_CDC_ECM_CfgDesc; } /** @@ -891,7 +769,7 @@ uint8_t *USBD_CDC_ECM_GetDeviceQualifierDescriptor(uint16_t *length) return USBD_CDC_ECM_DeviceQualifierDesc; } - +#endif /* USE_USBD_COMPOSITE */ /** * @brief USBD_CDC_ECM_RegisterInterface * @param pdev: device instance @@ -906,7 +784,7 @@ uint8_t USBD_CDC_ECM_RegisterInterface(USBD_HandleTypeDef *pdev, return (uint8_t)USBD_FAIL; } - pdev->pUserData = fops; + pdev->pUserData[pdev->classId] = fops; return (uint8_t)USBD_OK; } @@ -915,7 +793,7 @@ uint8_t USBD_CDC_ECM_RegisterInterface(USBD_HandleTypeDef *pdev, /** * @brief USBD_CDC_ECM_USRStringDescriptor * Manages the transfer of user string descriptors. - * @param speed : current device speed + * @param pdev: device instance * @param index: descriptor index * @param length : pointer data length * @retval pointer to the descriptor table or NULL if the descriptor is not supported. @@ -928,7 +806,10 @@ static uint8_t *USBD_CDC_ECM_USRStringDescriptor(USBD_HandleTypeDef *pdev, uint8 /* Check if the requested string interface is supported */ if (index == CDC_ECM_MAC_STRING_INDEX) { - USBD_GetString((uint8_t *)((USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData)->pStrDesc, USBD_StrDesc, length); + USBD_GetString((uint8_t *)((USBD_CDC_ECM_ItfTypeDef *)pdev->pUserData[pdev->classId])->pStrDesc, + USBD_StrDesc, + length); + return USBD_StrDesc; } /* Not supported Interface Descriptor index */ @@ -937,17 +818,25 @@ static uint8_t *USBD_CDC_ECM_USRStringDescriptor(USBD_HandleTypeDef *pdev, uint8 return NULL; } } -#endif +#endif /* USBD_SUPPORT_USER_STRING_DESC */ /** * @brief USBD_CDC_ECM_SetTxBuffer * @param pdev: device instance * @param pbuff: Tx Buffer + * @param length: Tx Buffer length + * @param ClassId: The Class ID * @retval status */ +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CDC_ECM_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, uint32_t length, uint8_t ClassId) +{ + USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassDataCmsit[ClassId]; +#else uint8_t USBD_CDC_ECM_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, uint32_t length) { - USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassData; + USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#endif /* USE_USBD_COMPOSITE */ if (hcdc == NULL) { @@ -969,7 +858,7 @@ uint8_t USBD_CDC_ECM_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, uint3 */ uint8_t USBD_CDC_ECM_SetRxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff) { - USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassData; + USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hcdc == NULL) { @@ -981,18 +870,32 @@ uint8_t USBD_CDC_ECM_SetRxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff) return (uint8_t)USBD_OK; } + /** * @brief USBD_CDC_ECM_TransmitPacket * Transmit packet on IN endpoint * @param pdev: device instance + * @param ClassId: The Class ID * @retval status */ +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CDC_ECM_TransmitPacket(USBD_HandleTypeDef *pdev, uint8_t ClassId) +{ + USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassDataCmsit[ClassId]; +#else uint8_t USBD_CDC_ECM_TransmitPacket(USBD_HandleTypeDef *pdev) { - USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassData; + USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#endif /* USE_USBD_COMPOSITE */ + USBD_StatusTypeDef ret = USBD_BUSY; - if (pdev->pClassData == NULL) +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + ECMInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, ClassId); +#endif /* USE_USBD_COMPOSITE */ + + if (hcdc == NULL) { return (uint8_t)USBD_FAIL; } @@ -1003,10 +906,10 @@ uint8_t USBD_CDC_ECM_TransmitPacket(USBD_HandleTypeDef *pdev) hcdc->TxState = 1U; /* Update the packet total length */ - pdev->ep_in[CDC_ECM_IN_EP & 0xFU].total_length = hcdc->TxLength; + pdev->ep_in[ECMInEpAdd & 0xFU].total_length = hcdc->TxLength; /* Transmit next packet */ - (void)USBD_LL_Transmit(pdev, CDC_ECM_IN_EP, hcdc->TxBuffer, hcdc->TxLength); + (void)USBD_LL_Transmit(pdev, ECMInEpAdd, hcdc->TxBuffer, hcdc->TxLength); ret = USBD_OK; } @@ -1023,15 +926,20 @@ uint8_t USBD_CDC_ECM_TransmitPacket(USBD_HandleTypeDef *pdev) */ uint8_t USBD_CDC_ECM_ReceivePacket(USBD_HandleTypeDef *pdev) { - USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassData; + USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + ECMOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ - if (pdev->pClassData == NULL) + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_FAIL; } /* Prepare Out endpoint to receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, CDC_ECM_OUT_EP, hcdc->RxBuffer, hcdc->MaxPcktLen); + (void)USBD_LL_PrepareReceive(pdev, ECMOutEpAdd, hcdc->RxBuffer, hcdc->MaxPcktLen); return (uint8_t)USBD_OK; } @@ -1046,12 +954,12 @@ uint8_t USBD_CDC_ECM_ReceivePacket(USBD_HandleTypeDef *pdev) * @retval status */ uint8_t USBD_CDC_ECM_SendNotification(USBD_HandleTypeDef *pdev, - USBD_CDC_ECM_NotifCodeTypeDef Notif, + USBD_CDC_NotifCodeTypeDef Notif, uint16_t bVal, uint8_t *pData) { uint32_t Idx; uint32_t ReqSize = 0U; - USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassData; + USBD_CDC_ECM_HandleTypeDef *hcdc = (USBD_CDC_ECM_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; USBD_StatusTypeDef ret = USBD_OK; if (hcdc == NULL) @@ -1059,6 +967,11 @@ uint8_t USBD_CDC_ECM_SendNotification(USBD_HandleTypeDef *pdev, return (uint8_t)USBD_FAIL; } +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + ECMCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + /* Initialize the request fields */ (hcdc->Req).bmRequest = CDC_ECM_BMREQUEST_TYPE_ECM; (hcdc->Req).bRequest = (uint8_t)Notif; @@ -1112,7 +1025,7 @@ uint8_t USBD_CDC_ECM_SendNotification(USBD_HandleTypeDef *pdev, /* Transmit notification packet */ if (ReqSize != 0U) { - (void)USBD_LL_Transmit(pdev, CDC_ECM_CMD_EP, (uint8_t *) &(hcdc->Req), ReqSize); + (void)USBD_LL_Transmit(pdev, ECMCmdEpAdd, (uint8_t *)&hcdc->Req, ReqSize); } return (uint8_t)ret; @@ -1131,4 +1044,3 @@ uint8_t USBD_CDC_ECM_SendNotification(USBD_HandleTypeDef *pdev, * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Src/usbd_cdc_ecm_if_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Src/usbd_cdc_ecm_if_template.c index cab00565..ebe2d0b5 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Src/usbd_cdc_ecm_if_template.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_ECM/Src/usbd_cdc_ecm_if_template.c @@ -6,20 +6,19 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2019 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2019 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ /* Includes ------------------------------------------------------------------*/ -#include "main.h" +#include "usbd_cdc_ecm_if_template.h" /* Include here LwIP files if used @@ -30,16 +29,17 @@ /* Private define ------------------------------------------------------------*/ /* Private macro -------------------------------------------------------------*/ /* Private variables ---------------------------------------------------------*/ - +/* Received Data over USB are stored in this buffer */ #if defined ( __ICCARM__ ) /*!< IAR Compiler */ #pragma data_alignment=4 -#endif -__ALIGN_BEGIN static uint8_t UserRxBuffer[CDC_ECM_ETH_MAX_SEGSZE + 100]__ALIGN_END; /* Received Data over USB are stored in this buffer */ +#endif /* ( __ICCARM__ ) */ +__ALIGN_BEGIN static uint8_t UserRxBuffer[CDC_ECM_ETH_MAX_SEGSZE + 100]__ALIGN_END; +/* Transmitted Data over CDC_ECM (CDC_ECM interface) are stored in this buffer */ #if defined ( __ICCARM__ ) /*!< IAR Compiler */ #pragma data_alignment=4 -#endif -__ALIGN_BEGIN static uint8_t UserTxBuffer[CDC_ECM_ETH_MAX_SEGSZE + 100]__ALIGN_END; /* Received Data over CDC_ECM (CDC_ECM interface) are stored in this buffer */ +#endif /* ( __ICCARM__ ) */ +__ALIGN_BEGIN static uint8_t UserTxBuffer[CDC_ECM_ETH_MAX_SEGSZE + 100]__ALIGN_END; static uint8_t CDC_ECMInitialized = 0U; @@ -85,7 +85,11 @@ static int8_t CDC_ECM_Itf_Init(void) } /* Set Application Buffers */ +#ifdef USE_USBD_COMPOSITE + (void)USBD_CDC_ECM_SetTxBuffer(&USBD_Device, UserTxBuffer, 0U, 0U); +#else (void)USBD_CDC_ECM_SetTxBuffer(&USBD_Device, UserTxBuffer, 0U); +#endif /* USE_USBD_COMPOSITE */ (void)USBD_CDC_ECM_SetRxBuffer(&USBD_Device, UserRxBuffer); return (0); @@ -99,7 +103,12 @@ static int8_t CDC_ECM_Itf_Init(void) */ static int8_t CDC_ECM_Itf_DeInit(void) { +#ifdef USE_USBD_COMPOSITE + USBD_CDC_ECM_HandleTypeDef *hcdc_cdc_ecm = (USBD_CDC_ECM_HandleTypeDef *) \ + (USBD_Device.pClassDataCmsit[USBD_Device.classId]); +#else USBD_CDC_ECM_HandleTypeDef *hcdc_cdc_ecm = (USBD_CDC_ECM_HandleTypeDef *)(USBD_Device.pClassData); +#endif /* USE_USBD_COMPOSITE */ /* Notify application layer that link is down */ hcdc_cdc_ecm->LinkStatus = 0U; @@ -117,7 +126,12 @@ static int8_t CDC_ECM_Itf_DeInit(void) */ static int8_t CDC_ECM_Itf_Control(uint8_t cmd, uint8_t *pbuf, uint16_t length) { +#ifdef USE_USBD_COMPOSITE + USBD_CDC_ECM_HandleTypeDef *hcdc_cdc_ecm = (USBD_CDC_ECM_HandleTypeDef *) \ + (USBD_Device.pClassDataCmsit[USBD_Device.classId]); +#else USBD_CDC_ECM_HandleTypeDef *hcdc_cdc_ecm = (USBD_CDC_ECM_HandleTypeDef *)(USBD_Device.pClassData); +#endif /* USE_USBD_COMPOSITE */ switch (cmd) { @@ -188,7 +202,12 @@ static int8_t CDC_ECM_Itf_Control(uint8_t cmd, uint8_t *pbuf, uint16_t length) static int8_t CDC_ECM_Itf_Receive(uint8_t *Buf, uint32_t *Len) { /* Get the CDC_ECM handler pointer */ +#ifdef USE_USBD_COMPOSITE + USBD_CDC_ECM_HandleTypeDef *hcdc_cdc_ecm = (USBD_CDC_ECM_HandleTypeDef *) \ + (USBD_Device.pClassDataCmsit[USBD_Device.classId]); +#else USBD_CDC_ECM_HandleTypeDef *hcdc_cdc_ecm = (USBD_CDC_ECM_HandleTypeDef *)(USBD_Device.pClassData); +#endif /* USE_USBD_COMPOSITE */ /* Call Eth buffer processing */ hcdc_cdc_ecm->RxState = 1U; @@ -230,9 +249,18 @@ static int8_t CDC_ECM_Itf_TransmitCplt(uint8_t *Buf, uint32_t *Len, uint8_t epnu static int8_t CDC_ECM_Itf_Process(USBD_HandleTypeDef *pdev) { /* Get the CDC_ECM handler pointer */ +#ifdef USE_USBD_COMPOSITE + USBD_CDC_ECM_HandleTypeDef *hcdc_cdc_ecm = (USBD_CDC_ECM_HandleTypeDef *)(pdev->pClassDataCmsit[pdev->classId]); +#else USBD_CDC_ECM_HandleTypeDef *hcdc_cdc_ecm = (USBD_CDC_ECM_HandleTypeDef *)(pdev->pClassData); +#endif /* USE_USBD_COMPOSITE */ + + if (hcdc_cdc_ecm == NULL) + { + return (-1); + } - if ((hcdc_cdc_ecm != NULL) && (hcdc_cdc_ecm->LinkStatus != 0U)) + if (hcdc_cdc_ecm->LinkStatus != 0U) { /* Read a received packet from the Ethernet buffers and send it @@ -244,4 +272,3 @@ static int8_t CDC_ECM_Itf_Process(USBD_HandleTypeDef *pdev) return (0); } -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Inc/usbd_cdc_rndis.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Inc/usbd_cdc_rndis.h index a2fb4323..50d18c42 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Inc/usbd_cdc_rndis.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Inc/usbd_cdc_rndis.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2019 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -41,10 +40,15 @@ extern "C" { /** @defgroup usbd_cdc_rndis_Exported_Defines * @{ */ - +#ifndef CDC_RNDIS_IN_EP #define CDC_RNDIS_IN_EP 0x81U /* EP1 for data IN */ +#endif /* CDC_RNDIS_IN_EP */ +#ifndef CDC_RNDIS_OUT_EP #define CDC_RNDIS_OUT_EP 0x01U /* EP1 for data OUT */ +#endif /* CDC_RNDIS_OUT_EP */ +#ifndef CDC_RNDIS_CMD_EP #define CDC_RNDIS_CMD_EP 0x82U /* EP2 for CDC_RNDIS commands */ +#endif /* CDC_RNDIS_CMD_EP */ #ifndef CDC_RNDIS_CMD_ITF_NBR #define CDC_RNDIS_CMD_ITF_NBR 0x00U /* Command Interface Number 0 */ @@ -265,15 +269,6 @@ typedef struct __IO uint32_t PacketFilter; } USBD_CDC_RNDIS_HandleTypeDef; - -typedef enum -{ - NETWORK_CONNECTION = 0x00, - RESPONSE_AVAILABLE = 0x01, - CONNECTION_SPEED_CHANGE = 0x2A -} USBD_CDC_RNDIS_NotifCodeTypeDef; - - /* Messages Sent by the Host ---------------------*/ /* Type define for a CDC_RNDIS Initialize command message */ @@ -498,16 +493,20 @@ extern USBD_ClassTypeDef USBD_CDC_RNDIS; */ uint8_t USBD_CDC_RNDIS_SetRxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff); uint8_t USBD_CDC_RNDIS_ReceivePacket(USBD_HandleTypeDef *pdev); -uint8_t USBD_CDC_RNDIS_TransmitPacket(USBD_HandleTypeDef *pdev); uint8_t USBD_CDC_RNDIS_RegisterInterface(USBD_HandleTypeDef *pdev, USBD_CDC_RNDIS_ItfTypeDef *fops); - +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CDC_RNDIS_TransmitPacket(USBD_HandleTypeDef *pdev, uint8_t ClassId); +uint8_t USBD_CDC_RNDIS_SetTxBuffer(USBD_HandleTypeDef *pdev, + uint8_t *pbuff, uint32_t length, uint8_t ClassId); +#else +uint8_t USBD_CDC_RNDIS_TransmitPacket(USBD_HandleTypeDef *pdev); uint8_t USBD_CDC_RNDIS_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, uint32_t length); - +#endif /* USE_USBD_COMPOSITE */ uint8_t USBD_CDC_RNDIS_SendNotification(USBD_HandleTypeDef *pdev, - USBD_CDC_RNDIS_NotifCodeTypeDef Notif, + USBD_CDC_NotifCodeTypeDef Notif, uint16_t bVal, uint8_t *pData); /** * @} @@ -526,4 +525,3 @@ uint8_t USBD_CDC_RNDIS_SendNotification(USBD_HandleTypeDef *pdev, * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Inc/usbd_cdc_rndis_if_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Inc/usbd_cdc_rndis_if_template.h index 3cf02716..14474c62 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Inc/usbd_cdc_rndis_if_template.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Inc/usbd_cdc_rndis_if_template.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2019 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2019 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -57,4 +56,3 @@ extern USBD_CDC_RNDIS_ItfTypeDef USBD_CDC_RNDIS_fops; #endif /* __USBD_CDC_RNDIS_IF_H */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Src/usbd_cdc_rndis.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Src/usbd_cdc_rndis.c index a7c2cbc7..f04f2c22 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Src/usbd_cdc_rndis.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Src/usbd_cdc_rndis.c @@ -10,6 +10,17 @@ * - Command IN transfer (class requests management) * - Error management * + ****************************************************************************** + * @attention + * + * Copyright (c) 2019 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** * @verbatim * * =================================================================== @@ -37,17 +48,6 @@ * @endverbatim * ****************************************************************************** - * @attention - * - * <h2><center>© Copyright (c) 2019 STMicroelectronics. - * All rights reserved.</center></h2> - * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** */ /* Includes ------------------------------------------------------------------*/ @@ -56,7 +56,7 @@ #ifndef __USBD_CDC_RNDIS_IF_H #include "usbd_cdc_rndis_if_template.h" -#endif +#endif /* __USBD_CDC_RNDIS_IF_H */ /** @addtogroup STM32_USB_DEVICE_LIBRARY * @{ */ @@ -105,17 +105,18 @@ static uint8_t USBD_CDC_RNDIS_Setup(USBD_HandleTypeDef *pdev, static uint8_t USBD_CDC_RNDIS_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum); static uint8_t USBD_CDC_RNDIS_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum); static uint8_t USBD_CDC_RNDIS_EP0_RxReady(USBD_HandleTypeDef *pdev); +#ifndef USE_USBD_COMPOSITE static uint8_t *USBD_CDC_RNDIS_GetFSCfgDesc(uint16_t *length); static uint8_t *USBD_CDC_RNDIS_GetHSCfgDesc(uint16_t *length); static uint8_t *USBD_CDC_RNDIS_GetOtherSpeedCfgDesc(uint16_t *length); static uint8_t *USBD_CDC_RNDIS_GetOtherSpeedCfgDesc(uint16_t *length); - +#endif /* USE_USBD_COMPOSITE */ #if (USBD_SUPPORT_USER_STRING_DESC == 1U) static uint8_t *USBD_CDC_RNDIS_USRStringDescriptor(USBD_HandleTypeDef *pdev, uint8_t index, uint16_t *length); -#endif - +#endif /* USBD_SUPPORT_USER_STRING_DESC */ +#ifndef USE_USBD_COMPOSITE uint8_t *USBD_CDC_RNDIS_GetDeviceQualifierDescriptor(uint16_t *length); - +#endif /* USE_USBD_COMPOSITE */ /* CDC_RNDIS Internal messages parsing and construction functions */ static uint8_t USBD_CDC_RNDIS_MsgParsing(USBD_HandleTypeDef *pdev, uint8_t *RxBuff); @@ -129,6 +130,7 @@ static uint8_t USBD_CDC_RNDIS_ProcessPacketMsg(USBD_HandleTypeDef *pdev, USBD_CD static uint8_t USBD_CDC_RNDIS_ProcessUnsupportedMsg(USBD_HandleTypeDef *pdev, USBD_CDC_RNDIS_CtrlMsgTypeDef *Msg); /* USB Standard Device Descriptor */ +#ifndef USE_USBD_COMPOSITE __ALIGN_BEGIN static uint8_t USBD_CDC_RNDIS_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_DESC] __ALIGN_END = { USB_LEN_DEV_QUALIFIER_DESC, @@ -142,7 +144,7 @@ __ALIGN_BEGIN static uint8_t USBD_CDC_RNDIS_DeviceQualifierDesc[USB_LEN_DEV_QUAL 0x01, 0x00, }; - +#endif /* USE_USBD_COMPOSITE */ static uint8_t MAC_StrDesc[6] = {CDC_RNDIS_MAC_ADDR0, CDC_RNDIS_MAC_ADDR1, CDC_RNDIS_MAC_ADDR2, CDC_RNDIS_MAC_ADDR3, CDC_RNDIS_MAC_ADDR4, CDC_RNDIS_MAC_ADDR5 }; @@ -175,126 +177,25 @@ USBD_ClassTypeDef USBD_CDC_RNDIS = NULL, NULL, NULL, +#ifdef USE_USBD_COMPOSITE + NULL, + NULL, + NULL, + NULL, +#else USBD_CDC_RNDIS_GetHSCfgDesc, USBD_CDC_RNDIS_GetFSCfgDesc, USBD_CDC_RNDIS_GetOtherSpeedCfgDesc, USBD_CDC_RNDIS_GetDeviceQualifierDescriptor, +#endif /* USE_USBD_COMPOSITE */ #if (USBD_SUPPORT_USER_STRING_DESC == 1U) USBD_CDC_RNDIS_USRStringDescriptor, -#endif -}; - -/* USB CDC_RNDIS device Configuration Descriptor */ -__ALIGN_BEGIN static uint8_t USBD_CDC_RNDIS_CfgHSDesc[] __ALIGN_END = -{ - /* Configuration Descriptor */ - 0x09, /* bLength: Configuration Descriptor size */ - USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ - LOBYTE(CDC_RNDIS_CONFIG_DESC_SIZ), /* wTotalLength: Total size of the Config descriptor */ - HIBYTE(CDC_RNDIS_CONFIG_DESC_SIZ), - 0x02, /* bNumInterfaces: 2 interfaces */ - 0x01, /* bConfigurationValue: Configuration value */ - 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ -#if (USBD_SELF_POWERED == 1U) - 0xC0, /* bmAttributes: Bus Powered according to user configuration */ -#else - 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif - USBD_MAX_POWER, /* MaxPower (mA) */ - - /*---------------------------------------------------------------------------*/ - /* IAD descriptor */ - 0x08, /* bLength */ - 0x0B, /* bDescriptorType */ - 0x00, /* bFirstInterface */ - 0x02, /* bInterfaceCount */ - 0xE0, /* bFunctionClass (Wireless Controller) */ - 0x01, /* bFunctionSubClass */ - 0x03, /* bFunctionProtocol */ - 0x00, /* iFunction */ - - /*---------------------------------------------------------------------------*/ - /* Interface Descriptor */ - 0x09, /* bLength: Interface Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: Interface descriptor type */ - CDC_RNDIS_CMD_ITF_NBR, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x01, /* bNumEndpoints: One endpoint used */ - 0x02, /* bInterfaceClass: Communication Interface Class */ - 0x02, /* bInterfaceSubClass:Abstract Control Model */ - 0xFF, /* bInterfaceProtocol: Common AT commands */ - 0x00, /* iInterface */ - - /* Header Functional Descriptor */ - 0x05, /* bLength: Endpoint Descriptor size */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x00, /* bDescriptorSubtype: Header functional descriptor */ - 0x10, /* bcdCDC: spec release number: 1.20 */ - 0x01, - - /* Call Management Functional Descriptor */ - 0x05, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x01, /* bDescriptorSubtype: Call Management Func Desc */ - 0x00, /* bmCapabilities: D0+D1 */ - CDC_RNDIS_COM_ITF_NBR, /* bDataInterface: 1 */ - - /* ACM Functional Descriptor */ - 0x04, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x02, /* bDescriptorSubtype: Abstract Control Management desc */ - 0x00, /* bmCapabilities */ - - /* Union Functional Descriptor */ - 0x05, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x06, /* bDescriptorSubtype: Union functional descriptor */ - CDC_RNDIS_CMD_ITF_NBR, /* bMasterInterface: Communication class interface */ - CDC_RNDIS_COM_ITF_NBR, /* bSlaveInterface0: Data Class Interface */ - - /* Notification Endpoint Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_RNDIS_CMD_EP, /* bEndpointAddress */ - 0x03, /* bmAttributes: Interrupt */ - LOBYTE(CDC_RNDIS_CMD_PACKET_SIZE), /* wMaxPacketSize */ - HIBYTE(CDC_RNDIS_CMD_PACKET_SIZE), - CDC_RNDIS_HS_BINTERVAL, /* bInterval */ - - /*---------------------------------------------------------------------------*/ - /* Data class interface descriptor */ - 0x09, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: */ - CDC_RNDIS_COM_ITF_NBR, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x02, /* bNumEndpoints: Two endpoints used */ - 0x0A, /* bInterfaceClass: CDC */ - 0x00, /* bInterfaceSubClass */ - 0x00, /* bInterfaceProtocol */ - 0x00, /* iInterface */ - - /* Endpoint OUT Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_RNDIS_OUT_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - LOBYTE(CDC_RNDIS_DATA_HS_MAX_PACKET_SIZE), /* wMaxPacketSize */ - HIBYTE(CDC_RNDIS_DATA_HS_MAX_PACKET_SIZE), - 0x00, /* bInterval */ - - /* Endpoint IN Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_RNDIS_IN_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - LOBYTE(CDC_RNDIS_DATA_HS_MAX_PACKET_SIZE), /* wMaxPacketSize: */ - HIBYTE(CDC_RNDIS_DATA_HS_MAX_PACKET_SIZE), - 0x00 /* bInterval */ +#endif /* USBD_SUPPORT_USER_STRING_DESC */ }; - +#ifndef USE_USBD_COMPOSITE /* USB CDC device Configuration Descriptor */ -__ALIGN_BEGIN static uint8_t USBD_CDC_RNDIS_CfgFSDesc[] __ALIGN_END = +__ALIGN_BEGIN static uint8_t USBD_CDC_RNDIS_CfgDesc[] __ALIGN_END = { /* Configuration Descriptor */ 0x09, /* bLength: Configuration Descriptor size */ @@ -303,12 +204,13 @@ __ALIGN_BEGIN static uint8_t USBD_CDC_RNDIS_CfgFSDesc[] __ALIGN_END = HIBYTE(CDC_RNDIS_CONFIG_DESC_SIZ), 0x02, /* bNumInterfaces: 2 interfaces */ 0x01, /* bConfigurationValue: Configuration value */ - 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ + 0x00, /* iConfiguration: Index of string descriptor + describing the configuration */ #if (USBD_SELF_POWERED == 1U) 0xC0, /* bmAttributes: Bus Powered according to user configuration */ #else 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif +#endif /* USBD_SELF_POWERED */ USBD_MAX_POWER, /* MaxPower (mA) */ /*---------------------------------------------------------------------------*/ @@ -400,114 +302,11 @@ __ALIGN_BEGIN static uint8_t USBD_CDC_RNDIS_CfgFSDesc[] __ALIGN_END = HIBYTE(CDC_RNDIS_DATA_FS_MAX_PACKET_SIZE), 0x00 /* bInterval */ } ; +#endif /* USE_USBD_COMPOSITE */ -__ALIGN_BEGIN static uint8_t USBD_CDC_RNDIS_OtherSpeedCfgDesc[] __ALIGN_END = -{ - /* Configuration Descriptor */ - 0x09, /* bLength: Configuration Descriptor size */ - USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ - LOBYTE(CDC_RNDIS_CONFIG_DESC_SIZ), /* wTotalLength:no of returned bytes */ - HIBYTE(CDC_RNDIS_CONFIG_DESC_SIZ), - 0x02, /* bNumInterfaces: 2 interfaces */ - 0x01, /* bConfigurationValue: Configuration value */ - 0x04, /* iConfiguration: Index of string descriptor describing the configuration */ -#if (USBD_SELF_POWERED == 1U) - 0xC0, /* bmAttributes: Bus Powered according to user configuration */ -#else - 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif - USBD_MAX_POWER, /* MaxPower (mA) */ - - /*---------------------------------------------------------------------------*/ - /* IAD descriptor */ - 0x08, /* bLength */ - 0x0B, /* bDescriptorType */ - 0x00, /* bFirstInterface */ - 0x02, /* bInterfaceCount */ - 0xE0, /* bFunctionClass (Wireless Controller) */ - 0x01, /* bFunctionSubClass */ - 0x03, /* bFunctionProtocol */ - 0x00, /* iFunction */ - - /*---------------------------------------------------------------------------*/ - /* Interface Descriptor */ - 0x09, /* bLength: Interface Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: Interface descriptor type */ - CDC_RNDIS_CMD_ITF_NBR, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x01, /* bNumEndpoints: One endpoint used */ - 0x02, /* bInterfaceClass: Communication Interface Class */ - 0x02, /* bInterfaceSubClass:Abstract Control Model */ - 0xFF, /* bInterfaceProtocol: Common AT commands */ - 0x00, /* iInterface */ - - /* Header Functional Descriptor */ - 0x05, /* bLength: Endpoint Descriptor size */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x00, /* bDescriptorSubtype: Header functional descriptor */ - 0x10, /* bcdCDC: spec release number: 1.20 */ - 0x01, - - /* Call Management Functional Descriptor */ - 0x05, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x01, /* bDescriptorSubtype: Call Management Func Desc */ - 0x00, /* bmCapabilities: D0+D1 */ - CDC_RNDIS_COM_ITF_NBR, /* bDataInterface: 1 */ - - /* ACM Functional Descriptor */ - 0x04, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x02, /* bDescriptorSubtype: Abstract Control Management desc */ - 0x00, /* bmCapabilities */ - - /* Union Functional Descriptor */ - 0x05, /* bFunctionLength */ - 0x24, /* bDescriptorType: CS_INTERFACE */ - 0x06, /* bDescriptorSubtype: Union functional descriptor */ - CDC_RNDIS_CMD_ITF_NBR, /* bMasterInterface: Communication class interface */ - CDC_RNDIS_COM_ITF_NBR, /* bSlaveInterface0: Data Class Interface */ - - /* Communication Endpoint Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_RNDIS_CMD_EP, /* bEndpointAddress */ - 0x03, /* bmAttributes: Interrupt */ - LOBYTE(CDC_RNDIS_CMD_PACKET_SIZE), /* wMaxPacketSize */ - HIBYTE(CDC_RNDIS_CMD_PACKET_SIZE), - CDC_RNDIS_FS_BINTERVAL, /* bInterval */ - - /*---------------------------------------------------------------------------*/ - /* Data class interface descriptor */ - 0x09, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: */ - CDC_RNDIS_COM_ITF_NBR, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x02, /* bNumEndpoints: Two endpoints used */ - 0x0A, /* bInterfaceClass: CDC */ - 0x00, /* bInterfaceSubClass */ - 0x00, /* bInterfaceProtocol */ - 0x00, /* iInterface */ - - /* Endpoint OUT Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_RNDIS_OUT_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - 0x40, /* wMaxPacketSize */ - 0x00, - 0x00, /* bInterval */ - - /* Endpoint IN Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - CDC_RNDIS_IN_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - 0x40, /* wMaxPacketSize */ - 0x00, - 0x00 /* bInterval */ -}; - +static uint8_t RNDISInEpAdd = CDC_RNDIS_IN_EP; +static uint8_t RNDISOutEpAdd = CDC_RNDIS_OUT_EP; +static uint8_t RNDISCmdEpAdd = CDC_RNDIS_CMD_EP; static const uint32_t CDC_RNDIS_SupportedOIDs[] = { @@ -556,55 +355,67 @@ static uint8_t USBD_CDC_RNDIS_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)USBD_malloc(sizeof(USBD_CDC_RNDIS_HandleTypeDef)); +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + RNDISInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + RNDISOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + RNDISCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + if (hcdc == NULL) { - pdev->pClassData = NULL; + pdev->pClassDataCmsit[pdev->classId] = NULL; return (uint8_t)USBD_EMEM; } - pdev->pClassData = (void *)hcdc; + (void)USBD_memset(hcdc, 0, sizeof(USBD_CDC_RNDIS_HandleTypeDef)); + + pdev->pClassDataCmsit[pdev->classId] = (void *)hcdc; + pdev->pClassData = pdev->pClassDataCmsit[pdev->classId]; if (pdev->dev_speed == USBD_SPEED_HIGH) { /* Open EP IN */ - (void)USBD_LL_OpenEP(pdev, CDC_RNDIS_IN_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, RNDISInEpAdd, USBD_EP_TYPE_BULK, CDC_RNDIS_DATA_HS_IN_PACKET_SIZE); - pdev->ep_in[CDC_RNDIS_IN_EP & 0xFU].is_used = 1U; + pdev->ep_in[RNDISInEpAdd & 0xFU].is_used = 1U; /* Open EP OUT */ - (void)USBD_LL_OpenEP(pdev, CDC_RNDIS_OUT_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, RNDISOutEpAdd, USBD_EP_TYPE_BULK, CDC_RNDIS_DATA_HS_OUT_PACKET_SIZE); - pdev->ep_out[CDC_RNDIS_OUT_EP & 0xFU].is_used = 1U; + pdev->ep_out[RNDISOutEpAdd & 0xFU].is_used = 1U; /* Set bInterval for CDC RNDIS CMD Endpoint */ - pdev->ep_in[CDC_RNDIS_CMD_EP & 0xFU].bInterval = CDC_RNDIS_HS_BINTERVAL; + pdev->ep_in[RNDISCmdEpAdd & 0xFU].bInterval = CDC_RNDIS_HS_BINTERVAL; } else { /* Open EP IN */ - (void)USBD_LL_OpenEP(pdev, CDC_RNDIS_IN_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, RNDISInEpAdd, USBD_EP_TYPE_BULK, CDC_RNDIS_DATA_FS_IN_PACKET_SIZE); - pdev->ep_in[CDC_RNDIS_IN_EP & 0xFU].is_used = 1U; + pdev->ep_in[RNDISInEpAdd & 0xFU].is_used = 1U; /* Open EP OUT */ - (void)USBD_LL_OpenEP(pdev, CDC_RNDIS_OUT_EP, USBD_EP_TYPE_BULK, + (void)USBD_LL_OpenEP(pdev, RNDISOutEpAdd, USBD_EP_TYPE_BULK, CDC_RNDIS_DATA_FS_OUT_PACKET_SIZE); - pdev->ep_out[CDC_RNDIS_OUT_EP & 0xFU].is_used = 1U; + pdev->ep_out[RNDISOutEpAdd & 0xFU].is_used = 1U; /* Set bInterval for CDC RNDIS CMD Endpoint */ - pdev->ep_in[CDC_RNDIS_CMD_EP & 0xFU].bInterval = CDC_RNDIS_FS_BINTERVAL; + pdev->ep_in[RNDISCmdEpAdd & 0xFU].bInterval = CDC_RNDIS_FS_BINTERVAL; } /* Open Command IN EP */ - (void)USBD_LL_OpenEP(pdev, CDC_RNDIS_CMD_EP, USBD_EP_TYPE_INTR, CDC_RNDIS_CMD_PACKET_SIZE); - pdev->ep_in[CDC_RNDIS_CMD_EP & 0xFU].is_used = 1U; + (void)USBD_LL_OpenEP(pdev, RNDISCmdEpAdd, USBD_EP_TYPE_INTR, CDC_RNDIS_CMD_PACKET_SIZE); + pdev->ep_in[RNDISCmdEpAdd & 0xFU].is_used = 1U; + + hcdc->RxBuffer = NULL; /* Init physical Interface components */ - ((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData)->Init(); + ((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData[pdev->classId])->Init(); /* Init the CDC_RNDIS state */ hcdc->State = CDC_RNDIS_STATE_BUS_INITIALIZED; @@ -616,10 +427,16 @@ static uint8_t USBD_CDC_RNDIS_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) hcdc->TxLength = 0U; hcdc->LinkStatus = 0U; hcdc->NotificationStatus = 0U; - hcdc->MaxPcktLen = (pdev->dev_speed == USBD_SPEED_HIGH) ? CDC_RNDIS_DATA_HS_MAX_PACKET_SIZE : CDC_RNDIS_DATA_FS_MAX_PACKET_SIZE; + hcdc->MaxPcktLen = (pdev->dev_speed == USBD_SPEED_HIGH) ? CDC_RNDIS_DATA_HS_MAX_PACKET_SIZE : \ + CDC_RNDIS_DATA_FS_MAX_PACKET_SIZE; + + if (hcdc->RxBuffer == NULL) + { + return (uint8_t)USBD_EMEM; + } /* Prepare Out endpoint to receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, CDC_RNDIS_OUT_EP, + (void)USBD_LL_PrepareReceive(pdev, RNDISOutEpAdd, hcdc->RxBuffer, hcdc->MaxPcktLen); return (uint8_t)USBD_OK; @@ -636,24 +453,32 @@ static uint8_t USBD_CDC_RNDIS_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) { UNUSED(cfgidx); +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + RNDISInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + RNDISOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + RNDISCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + /* Close EP IN */ - (void)USBD_LL_CloseEP(pdev, CDC_RNDIS_IN_EP); - pdev->ep_in[CDC_RNDIS_IN_EP & 0xFU].is_used = 0U; + (void)USBD_LL_CloseEP(pdev, RNDISInEpAdd); + pdev->ep_in[RNDISInEpAdd & 0xFU].is_used = 0U; /* Close EP OUT */ - (void)USBD_LL_CloseEP(pdev, CDC_RNDIS_OUT_EP); - pdev->ep_out[CDC_RNDIS_OUT_EP & 0xFU].is_used = 0U; + (void)USBD_LL_CloseEP(pdev, RNDISOutEpAdd); + pdev->ep_out[RNDISOutEpAdd & 0xFU].is_used = 0U; /* Close Command IN EP */ - (void)USBD_LL_CloseEP(pdev, CDC_RNDIS_CMD_EP); - pdev->ep_in[CDC_RNDIS_CMD_EP & 0xFU].is_used = 0U; - pdev->ep_in[CDC_RNDIS_CMD_EP & 0xFU].bInterval = 0U; + (void)USBD_LL_CloseEP(pdev, RNDISCmdEpAdd); + pdev->ep_in[RNDISCmdEpAdd & 0xFU].is_used = 0U; + pdev->ep_in[RNDISCmdEpAdd & 0xFU].bInterval = 0U; /* DeInit physical Interface components */ - if (pdev->pClassData != NULL) + if (pdev->pClassDataCmsit[pdev->classId] != NULL) { - ((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData)->DeInit(); - USBD_free(pdev->pClassData); + ((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData[pdev->classId])->DeInit(); + USBD_free(pdev->pClassDataCmsit[pdev->classId]); + pdev->pClassDataCmsit[pdev->classId] = NULL; pdev->pClassData = NULL; } @@ -670,8 +495,8 @@ static uint8_t USBD_CDC_RNDIS_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) static uint8_t USBD_CDC_RNDIS_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; - USBD_CDC_RNDIS_CtrlMsgTypeDef *Msg = (USBD_CDC_RNDIS_CtrlMsgTypeDef *)(void *)hcdc->data; + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_CDC_RNDIS_CtrlMsgTypeDef *Msg; uint8_t ifalt = 0U; uint16_t status_info = 0U; USBD_StatusTypeDef ret = USBD_OK; @@ -681,6 +506,8 @@ static uint8_t USBD_CDC_RNDIS_Setup(USBD_HandleTypeDef *pdev, return (uint8_t)USBD_FAIL; } + Msg = (USBD_CDC_RNDIS_CtrlMsgTypeDef *)(void *)hcdc->data; + switch (req->bmRequest & USB_REQ_TYPE_MASK) { case USB_REQ_TYPE_CLASS : @@ -703,9 +530,9 @@ static uint8_t USBD_CDC_RNDIS_Setup(USBD_HandleTypeDef *pdev, } /* Allow application layer to pre-process data or add own processing before sending response */ - ((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData)->Control(req->bRequest, - (uint8_t *)hcdc->data, - req->wLength); + ((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData[pdev->classId])->Control(req->bRequest, + (uint8_t *)hcdc->data, + req->wLength); /* Check if Response is ready */ if (hcdc->ResponseRdy != 0U) { @@ -729,7 +556,7 @@ static uint8_t USBD_CDC_RNDIS_Setup(USBD_HandleTypeDef *pdev, else { hcdc->CmdOpCode = req->bRequest; - hcdc->CmdLength = (uint8_t)MIN(CDC_RNDIS_CMD_PACKET_SIZE, req->wLength); + hcdc->CmdLength = (uint8_t)MIN(CDC_RNDIS_MAX_INFO_BUFF_SZ, req->wLength); (void)USBD_CtlPrepareRx(pdev, (uint8_t *)hcdc->data, hcdc->CmdLength); } @@ -738,8 +565,8 @@ static uint8_t USBD_CDC_RNDIS_Setup(USBD_HandleTypeDef *pdev, so let application layer manage this case */ else { - ((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData)->Control(req->bRequest, - (uint8_t *)req, 0U); + ((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData[pdev->classId])->Control(req->bRequest, + (uint8_t *)req, 0U); } break; @@ -807,16 +634,21 @@ static uint8_t USBD_CDC_RNDIS_Setup(USBD_HandleTypeDef *pdev, static uint8_t USBD_CDC_RNDIS_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) { USBD_CDC_RNDIS_HandleTypeDef *hcdc; - PCD_HandleTypeDef *hpcd = pdev->pData; + PCD_HandleTypeDef *hpcd = (PCD_HandleTypeDef *)pdev->pData; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + RNDISInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ - if (pdev->pClassData == NULL) + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_FAIL; } - hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; + hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; - if (epnum == (CDC_RNDIS_IN_EP & 0x7FU)) + if (epnum == (RNDISInEpAdd & 0x7FU)) { if ((pdev->ep_in[epnum & 0xFU].total_length > 0U) && ((pdev->ep_in[epnum & 0xFU].total_length % hpcd->IN_ep[epnum & 0xFU].maxpacket) == 0U)) @@ -831,13 +663,14 @@ static uint8_t USBD_CDC_RNDIS_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) { hcdc->TxState = 0U; - if (((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData)->TransmitCplt != NULL) + if (((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData[pdev->classId])->TransmitCplt != NULL) { - ((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData)->TransmitCplt(hcdc->TxBuffer, &hcdc->TxLength, epnum); + ((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData[pdev->classId])->TransmitCplt(hcdc->TxBuffer, \ + &hcdc->TxLength, epnum); } } } - else if (epnum == (CDC_RNDIS_CMD_EP & 0x7FU)) + else if (epnum == (RNDISCmdEpAdd & 0x7FU)) { if (hcdc->NotificationStatus != 0U) { @@ -867,14 +700,18 @@ static uint8_t USBD_CDC_RNDIS_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) USBD_CDC_RNDIS_HandleTypeDef *hcdc; uint32_t CurrPcktLen; - if (pdev->pClassData == NULL) +#ifdef USE_USBD_COMPOSITE /* Get the Endpoints addresses allocated for this class instance */ + RNDISOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_FAIL; } - hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; + hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; - if (epnum == CDC_RNDIS_OUT_EP) + if (epnum == RNDISOutEpAdd) { /* Get the received data length */ CurrPcktLen = USBD_LL_GetRxDataSize(pdev, epnum); @@ -895,7 +732,7 @@ static uint8_t USBD_CDC_RNDIS_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) else { /* Prepare Out endpoint to receive next packet in current/new frame */ - (void)USBD_LL_PrepareReceive(pdev, CDC_RNDIS_OUT_EP, + (void)USBD_LL_PrepareReceive(pdev, RNDISOutEpAdd, (uint8_t *)(hcdc->RxBuffer + hcdc->RxLength), hcdc->MaxPcktLen); } @@ -916,14 +753,14 @@ static uint8_t USBD_CDC_RNDIS_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) */ static uint8_t USBD_CDC_RNDIS_EP0_RxReady(USBD_HandleTypeDef *pdev) { - USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hcdc == NULL) { return (uint8_t)USBD_FAIL; } - if ((pdev->pUserData != NULL) && (hcdc->CmdOpCode != 0xFFU)) + if ((pdev->pUserData[pdev->classId] != NULL) && (hcdc->CmdOpCode != 0xFFU)) { /* Check if the received command is SendEncapsulated command */ if (hcdc->CmdOpCode == CDC_RNDIS_SEND_ENCAPSULATED_COMMAND) @@ -946,47 +783,98 @@ static uint8_t USBD_CDC_RNDIS_EP0_RxReady(USBD_HandleTypeDef *pdev) return (uint8_t)USBD_OK; } - +#ifndef USE_USBD_COMPOSITE /** * @brief USBD_CDC_RNDIS_GetFSCfgDesc * Return configuration descriptor - * @param speed : current device speed * @param length : pointer data length * @retval pointer to descriptor buffer */ static uint8_t *USBD_CDC_RNDIS_GetFSCfgDesc(uint16_t *length) { - *length = (uint16_t)(sizeof(USBD_CDC_RNDIS_CfgFSDesc)); + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_CDC_RNDIS_CfgDesc, CDC_RNDIS_CMD_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CDC_RNDIS_CfgDesc, CDC_RNDIS_OUT_EP); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CDC_RNDIS_CfgDesc, CDC_RNDIS_IN_EP); - return USBD_CDC_RNDIS_CfgFSDesc; + if (pEpCmdDesc != NULL) + { + pEpCmdDesc->bInterval = CDC_RNDIS_FS_BINTERVAL; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = CDC_RNDIS_DATA_FS_MAX_PACKET_SIZE; + } + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = CDC_RNDIS_DATA_FS_MAX_PACKET_SIZE; + } + + *length = (uint16_t)(sizeof(USBD_CDC_RNDIS_CfgDesc)); + return USBD_CDC_RNDIS_CfgDesc; } /** * @brief USBD_CDC_RNDIS_GetHSCfgDesc * Return configuration descriptor - * @param speed : current device speed * @param length : pointer data length * @retval pointer to descriptor buffer */ static uint8_t *USBD_CDC_RNDIS_GetHSCfgDesc(uint16_t *length) { - *length = (uint16_t)(sizeof(USBD_CDC_RNDIS_CfgHSDesc)); + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_CDC_RNDIS_CfgDesc, CDC_RNDIS_CMD_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CDC_RNDIS_CfgDesc, CDC_RNDIS_OUT_EP); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CDC_RNDIS_CfgDesc, CDC_RNDIS_IN_EP); - return USBD_CDC_RNDIS_CfgHSDesc; + if (pEpCmdDesc != NULL) + { + pEpCmdDesc->bInterval = CDC_RNDIS_HS_BINTERVAL; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = CDC_RNDIS_DATA_HS_MAX_PACKET_SIZE; + } + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = CDC_RNDIS_DATA_HS_MAX_PACKET_SIZE; + } + + *length = (uint16_t)(sizeof(USBD_CDC_RNDIS_CfgDesc)); + return USBD_CDC_RNDIS_CfgDesc; } /** * @brief USBD_CDC_RNDIS_GetOtherSpeedCfgDesc * Return configuration descriptor - * @param speed : current device speed * @param length : pointer data length * @retval pointer to descriptor buffer */ static uint8_t *USBD_CDC_RNDIS_GetOtherSpeedCfgDesc(uint16_t *length) { - *length = (uint16_t)(sizeof(USBD_CDC_RNDIS_OtherSpeedCfgDesc)); + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_CDC_RNDIS_CfgDesc, CDC_RNDIS_CMD_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CDC_RNDIS_CfgDesc, CDC_RNDIS_OUT_EP); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CDC_RNDIS_CfgDesc, CDC_RNDIS_IN_EP); + + if (pEpCmdDesc != NULL) + { + pEpCmdDesc->bInterval = CDC_RNDIS_FS_BINTERVAL; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = CDC_RNDIS_DATA_FS_MAX_PACKET_SIZE; + } + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = CDC_RNDIS_DATA_FS_MAX_PACKET_SIZE; + } - return USBD_CDC_RNDIS_OtherSpeedCfgDesc; + *length = (uint16_t)(sizeof(USBD_CDC_RNDIS_CfgDesc)); + return USBD_CDC_RNDIS_CfgDesc; } /** @@ -1001,7 +889,7 @@ uint8_t *USBD_CDC_RNDIS_GetDeviceQualifierDescriptor(uint16_t *length) return USBD_CDC_RNDIS_DeviceQualifierDesc; } - +#endif /* USE_USBD_COMPOSITE */ /** * @brief USBD_CDC_RNDIS_RegisterInterface * @param pdev: device instance @@ -1016,7 +904,7 @@ uint8_t USBD_CDC_RNDIS_RegisterInterface(USBD_HandleTypeDef *pdev, return (uint8_t)USBD_FAIL; } - pdev->pUserData = fops; + pdev->pUserData[pdev->classId] = fops; return (uint8_t)USBD_OK; } @@ -1025,7 +913,7 @@ uint8_t USBD_CDC_RNDIS_RegisterInterface(USBD_HandleTypeDef *pdev, /** * @brief USBD_CDC_RNDIS_USRStringDescriptor * Manages the transfer of user string descriptors. - * @param speed : current device speed + * @param pdev: device instance * @param index: descriptor index * @param length : pointer data length * @retval pointer to the descriptor table or NULL if the descriptor is not supported. @@ -1038,7 +926,8 @@ static uint8_t *USBD_CDC_RNDIS_USRStringDescriptor(USBD_HandleTypeDef *pdev, uin /* Check if the requested string interface is supported */ if (index == CDC_RNDIS_MAC_STRING_INDEX) { - USBD_GetString((uint8_t *)((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData)->pStrDesc, USBD_StrDesc, length); + USBD_GetString((uint8_t *)((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData[pdev->classId])->pStrDesc, USBD_StrDesc, + length); return USBD_StrDesc; } /* Not supported Interface Descriptor index */ @@ -1049,15 +938,25 @@ static uint8_t *USBD_CDC_RNDIS_USRStringDescriptor(USBD_HandleTypeDef *pdev, uin } #endif /* USBD_SUPPORT_USER_STRING_DESC */ + /** * @brief USBD_CDC_RNDIS_SetTxBuffer * @param pdev: device instance * @param pbuff: Tx Buffer + * @param length: Tx Buffer length + * @param ClassId: The Class ID * @retval status */ +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CDC_RNDIS_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, uint32_t length, uint8_t ClassId) +{ + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[ClassId]; +#else uint8_t USBD_CDC_RNDIS_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, uint32_t length) { - USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; + + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#endif /* USE_USBD_COMPOSITE */ if (hcdc == NULL) { @@ -1079,7 +978,7 @@ uint8_t USBD_CDC_RNDIS_SetTxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff, uin */ uint8_t USBD_CDC_RNDIS_SetRxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff) { - USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hcdc == NULL) { @@ -1096,20 +995,32 @@ uint8_t USBD_CDC_RNDIS_SetRxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff) * @brief USBD_CDC_RNDIS_TransmitPacket * Transmit packet on IN endpoint * @param pdev: device instance + * @param ClassId: The Class ID * @retval status */ +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CDC_RNDIS_TransmitPacket(USBD_HandleTypeDef *pdev, uint8_t ClassId) +{ + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[ClassId]; +#else uint8_t USBD_CDC_RNDIS_TransmitPacket(USBD_HandleTypeDef *pdev) { - USBD_CDC_RNDIS_HandleTypeDef *hcdc; + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#endif /* USE_USBD_COMPOSITE */ + USBD_CDC_RNDIS_PacketMsgTypeDef *PacketMsg; USBD_StatusTypeDef ret = USBD_BUSY; - if (pdev->pClassData == NULL) +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + RNDISInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, ClassId); +#endif /* USE_USBD_COMPOSITE */ + + if (hcdc == NULL) { return (uint8_t)USBD_FAIL; } - hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; PacketMsg = (USBD_CDC_RNDIS_PacketMsgTypeDef *)(void *)hcdc->TxBuffer; if (hcdc->TxState == 0U) @@ -1131,10 +1042,10 @@ uint8_t USBD_CDC_RNDIS_TransmitPacket(USBD_HandleTypeDef *pdev) PacketMsg->Reserved = 0U; /* Update the packet total length */ - pdev->ep_in[CDC_RNDIS_IN_EP & 0xFU].total_length = hcdc->TxLength; + pdev->ep_in[RNDISInEpAdd & 0xFU].total_length = hcdc->TxLength; /* Transmit next packet */ - (void)USBD_LL_Transmit(pdev, CDC_RNDIS_IN_EP, hcdc->TxBuffer, hcdc->TxLength); + (void)USBD_LL_Transmit(pdev, RNDISInEpAdd, hcdc->TxBuffer, hcdc->TxLength); ret = USBD_OK; } @@ -1153,15 +1064,20 @@ uint8_t USBD_CDC_RNDIS_ReceivePacket(USBD_HandleTypeDef *pdev) { USBD_CDC_RNDIS_HandleTypeDef *hcdc; - if (pdev->pClassData == NULL) +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + RNDISOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_FAIL; } - hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; + hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; /* Prepare Out endpoint to receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, CDC_RNDIS_OUT_EP, + (void)USBD_LL_PrepareReceive(pdev, RNDISOutEpAdd, hcdc->RxBuffer, hcdc->MaxPcktLen); return (uint8_t)USBD_OK; @@ -1178,12 +1094,12 @@ uint8_t USBD_CDC_RNDIS_ReceivePacket(USBD_HandleTypeDef *pdev) * @retval status */ uint8_t USBD_CDC_RNDIS_SendNotification(USBD_HandleTypeDef *pdev, - USBD_CDC_RNDIS_NotifCodeTypeDef Notif, + USBD_CDC_NotifCodeTypeDef Notif, uint16_t bVal, uint8_t *pData) { uint32_t Idx; uint16_t ReqSize = 0U; - USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; USBD_StatusTypeDef ret = USBD_OK; UNUSED(bVal); @@ -1194,6 +1110,11 @@ uint8_t USBD_CDC_RNDIS_SendNotification(USBD_HandleTypeDef *pdev, return (uint8_t)USBD_FAIL; } +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + RNDISCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + /* Initialize the request fields */ (hcdc->Req).bmRequest = CDC_RNDIS_BMREQUEST_TYPE_RNDIS; (hcdc->Req).bRequest = (uint8_t)Notif; @@ -1221,7 +1142,7 @@ uint8_t USBD_CDC_RNDIS_SendNotification(USBD_HandleTypeDef *pdev, /* Transmit notification packet */ if (ReqSize != 0U) { - (void)USBD_LL_Transmit(pdev, CDC_RNDIS_CMD_EP, (uint8_t *) &(hcdc->Req), ReqSize); + (void)USBD_LL_Transmit(pdev, RNDISCmdEpAdd, (uint8_t *)&hcdc->Req, ReqSize); } return (uint8_t)ret; @@ -1295,7 +1216,7 @@ static uint8_t USBD_CDC_RNDIS_ProcessInitMsg(USBD_HandleTypeDef *pdev, USBD_CDC_RNDIS_InitMsgTypeDef *Msg) { /* Get the CDC_RNDIS handle pointer */ - USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; /* Get and format the Msg input */ USBD_CDC_RNDIS_InitMsgTypeDef *InitMessage = (USBD_CDC_RNDIS_InitMsgTypeDef *)Msg; @@ -1360,7 +1281,7 @@ static uint8_t USBD_CDC_RNDIS_ProcessHaltMsg(USBD_HandleTypeDef *pdev, USBD_CDC_RNDIS_HaltMsgTypeDef *Msg) { /* Get the CDC_RNDIS handle pointer */ - USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hcdc == NULL) { @@ -1389,7 +1310,7 @@ static uint8_t USBD_CDC_RNDIS_ProcessKeepAliveMsg(USBD_HandleTypeDef *pdev, USBD_CDC_RNDIS_KpAliveMsgTypeDef *Msg) { /* Get the CDC_RNDIS handle pointer */ - USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; /* Use same Msg input buffer as response buffer */ USBD_CDC_RNDIS_KpAliveCpltMsgTypeDef *InitResponse = (USBD_CDC_RNDIS_KpAliveCpltMsgTypeDef *)(void *)Msg; @@ -1439,7 +1360,7 @@ static uint8_t USBD_CDC_RNDIS_ProcessQueryMsg(USBD_HandleTypeDef *pdev, USBD_CDC_RNDIS_QueryMsgTypeDef *Msg) { /* Get the CDC_RNDIS handle pointer */ - USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; /* Use same Msg input buffer as response buffer */ USBD_CDC_RNDIS_QueryCpltMsgTypeDef *QueryResponse = (USBD_CDC_RNDIS_QueryCpltMsgTypeDef *)(void *)Msg; @@ -1584,7 +1505,7 @@ static uint8_t USBD_CDC_RNDIS_ProcessSetMsg(USBD_HandleTypeDef *pdev, USBD_CDC_RNDIS_SetMsgTypeDef *Msg) { /* Get the CDC_RNDIS handle pointer */ - USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; /* Get and format the Msg input */ USBD_CDC_RNDIS_SetMsgTypeDef *SetMessage = (USBD_CDC_RNDIS_SetMsgTypeDef *)Msg; @@ -1647,7 +1568,7 @@ static uint8_t USBD_CDC_RNDIS_ProcessResetMsg(USBD_HandleTypeDef *pdev, /* Get and format the Msg input */ USBD_CDC_RNDIS_ResetMsgTypeDef *ResetMessage = (USBD_CDC_RNDIS_ResetMsgTypeDef *)Msg; /* Get the CDC_RNDIS handle pointer */ - USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; /* Use same Msg input buffer as response buffer */ USBD_CDC_RNDIS_ResetCpltMsgTypeDef *ResetResponse = (USBD_CDC_RNDIS_ResetCpltMsgTypeDef *)(void *)Msg; @@ -1695,10 +1616,11 @@ static uint8_t USBD_CDC_RNDIS_ProcessResetMsg(USBD_HandleTypeDef *pdev, static uint8_t USBD_CDC_RNDIS_ProcessPacketMsg(USBD_HandleTypeDef *pdev, USBD_CDC_RNDIS_PacketMsgTypeDef *Msg) { - uint32_t tmp1, tmp2; + uint32_t tmp1; + uint32_t tmp2; /* Get the CDC_RNDIS handle pointer */ - USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; /* Get and format the Msg input */ USBD_CDC_RNDIS_PacketMsgTypeDef *PacketMsg = (USBD_CDC_RNDIS_PacketMsgTypeDef *)Msg; @@ -1723,7 +1645,7 @@ static uint8_t USBD_CDC_RNDIS_ProcessPacketMsg(USBD_HandleTypeDef *pdev, hcdc->RxLength = PacketMsg->DataLength; /* Process data by application */ - ((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData)->Receive(hcdc->RxBuffer, &hcdc->RxLength); + ((USBD_CDC_RNDIS_ItfTypeDef *)pdev->pUserData[pdev->classId])->Receive(hcdc->RxBuffer, &hcdc->RxLength); return (uint8_t)USBD_OK; } @@ -1740,7 +1662,7 @@ static uint8_t USBD_CDC_RNDIS_ProcessUnsupportedMsg(USBD_HandleTypeDef *pdev, USBD_CDC_RNDIS_CtrlMsgTypeDef *Msg) { /* Get the CDC_RNDIS handle pointer */ - USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassData; + USBD_CDC_RNDIS_HandleTypeDef *hcdc = (USBD_CDC_RNDIS_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; /* Use same Msg input buffer as response buffer */ USBD_CDC_RNDIS_StsChangeMsgTypeDef *Response = (USBD_CDC_RNDIS_StsChangeMsgTypeDef *)(void *)Msg; @@ -1780,5 +1702,3 @@ static uint8_t USBD_CDC_RNDIS_ProcessUnsupportedMsg(USBD_HandleTypeDef *pdev, /** * @} */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Src/usbd_cdc_rndis_if_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Src/usbd_cdc_rndis_if_template.c index ad88c320..6cffddd0 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Src/usbd_cdc_rndis_if_template.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CDC_RNDIS/Src/usbd_cdc_rndis_if_template.c @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2019 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2019 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -31,21 +30,23 @@ #include "ethernetif.h" */ -#include "main.h" +#include "usbd_cdc_rndis_if_template.h" /* Private typedef -----------------------------------------------------------*/ /* Private define ------------------------------------------------------------*/ /* Private macro -------------------------------------------------------------*/ /* Private variables ---------------------------------------------------------*/ +/* Received Data over USB are stored in this buffer */ #if defined ( __ICCARM__ ) /*!< IAR Compiler */ #pragma data_alignment=4 -#endif -__ALIGN_BEGIN uint8_t UserRxBuffer[CDC_RNDIS_ETH_MAX_SEGSZE + 100] __ALIGN_END; /* Received Data over USB are stored in this buffer */ +#endif /* __ICCARM__ */ +__ALIGN_BEGIN uint8_t UserRxBuffer[CDC_RNDIS_ETH_MAX_SEGSZE + 100] __ALIGN_END; +/* Transmitted Data over CDC_RNDIS (CDC_RNDIS interface) are stored in this buffer */ #if defined ( __ICCARM__ ) /*!< IAR Compiler */ #pragma data_alignment=4 -#endif -__ALIGN_BEGIN static uint8_t UserTxBuffer[CDC_RNDIS_ETH_MAX_SEGSZE + 100] __ALIGN_END; /* Received Data over CDC_RNDIS (CDC_RNDIS interface) are stored in this buffer */ +#endif /* __ICCARM__ */ +__ALIGN_BEGIN static uint8_t UserTxBuffer[CDC_RNDIS_ETH_MAX_SEGSZE + 100] __ALIGN_END; static uint8_t CDC_RNDISInitialized = 0U; @@ -94,7 +95,11 @@ static int8_t CDC_RNDIS_Itf_Init(void) } /* Set Application Buffers */ +#ifdef USE_USBD_COMPOSITE + (void)USBD_CDC_RNDIS_SetTxBuffer(&USBD_Device, UserTxBuffer, 0U, 0U); +#else (void)USBD_CDC_RNDIS_SetTxBuffer(&USBD_Device, UserTxBuffer, 0U); +#endif /* USE_USBD_COMPOSITE */ (void)USBD_CDC_RNDIS_SetRxBuffer(&USBD_Device, UserRxBuffer); return (0); @@ -108,7 +113,12 @@ static int8_t CDC_RNDIS_Itf_Init(void) */ static int8_t CDC_RNDIS_Itf_DeInit(void) { +#ifdef USE_USBD_COMPOSITE + USBD_CDC_RNDIS_HandleTypeDef *hcdc_cdc_rndis = (USBD_CDC_RNDIS_HandleTypeDef *) \ + (USBD_Device.pClassDataCmsit[USBD_Device.classId]); +#else USBD_CDC_RNDIS_HandleTypeDef *hcdc_cdc_rndis = (USBD_CDC_RNDIS_HandleTypeDef *)(USBD_Device.pClassData); +#endif /* USE_USBD_COMPOSITE */ /* Add your code here @@ -130,7 +140,12 @@ static int8_t CDC_RNDIS_Itf_DeInit(void) */ static int8_t CDC_RNDIS_Itf_Control(uint8_t cmd, uint8_t *pbuf, uint16_t length) { +#ifdef USE_USBD_COMPOSITE + USBD_CDC_RNDIS_HandleTypeDef *hcdc_cdc_rndis = (USBD_CDC_RNDIS_HandleTypeDef *) \ + (USBD_Device.pClassDataCmsit[USBD_Device.classId]); +#else USBD_CDC_RNDIS_HandleTypeDef *hcdc_cdc_rndis = (USBD_CDC_RNDIS_HandleTypeDef *)(USBD_Device.pClassData); +#endif /* USE_USBD_COMPOSITE */ switch (cmd) { @@ -173,7 +188,12 @@ static int8_t CDC_RNDIS_Itf_Control(uint8_t cmd, uint8_t *pbuf, uint16_t length) static int8_t CDC_RNDIS_Itf_Receive(uint8_t *Buf, uint32_t *Len) { /* Get the CDC_RNDIS handler pointer */ +#ifdef USE_USBD_COMPOSITE + USBD_CDC_RNDIS_HandleTypeDef *hcdc_cdc_rndis = (USBD_CDC_RNDIS_HandleTypeDef *) \ + (USBD_Device.pClassDataCmsit[USBD_Device.classId]); +#else USBD_CDC_RNDIS_HandleTypeDef *hcdc_cdc_rndis = (USBD_CDC_RNDIS_HandleTypeDef *)(USBD_Device.pClassData); +#endif /* USE_USBD_COMPOSITE */ /* Call Eth buffer processing */ hcdc_cdc_rndis->RxState = 1U; @@ -216,9 +236,18 @@ static int8_t CDC_RNDIS_Itf_TransmitCplt(uint8_t *Buf, uint32_t *Len, uint8_t ep static int8_t CDC_RNDIS_Itf_Process(USBD_HandleTypeDef *pdev) { /* Get the CDC_RNDIS handler pointer */ - USBD_CDC_RNDIS_HandleTypeDef *hcdc_cdc_rndis = (USBD_CDC_RNDIS_HandleTypeDef *)(pdev->pClassData); +#ifdef USE_USBD_COMPOSITE + USBD_CDC_RNDIS_HandleTypeDef *hcdc_cdc_rndis = (USBD_CDC_RNDIS_HandleTypeDef *)(pdev->pClassDataCmsit[pdev->classId]); +#else + USBD_CDC_RNDIS_HandleTypeDef *hcdc_cdc_rndis = (USBD_CDC_RNDIS_HandleTypeDef *)(pdev->pClassData); +#endif /* USE_USBD_COMPOSITE */ - if ((hcdc_cdc_rndis != NULL) && (hcdc_cdc_rndis->LinkStatus != 0U)) + if (hcdc_cdc_rndis == NULL) + { + return (-1); + } + + if (hcdc_cdc_rndis->LinkStatus != 0U) { /* Add your code here @@ -230,4 +259,3 @@ static int8_t CDC_RNDIS_Itf_Process(USBD_HandleTypeDef *pdev) return (0); } -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CompositeBuilder/Inc/usbd_composite_builder.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CompositeBuilder/Inc/usbd_composite_builder.h new file mode 100644 index 00000000..3a8aac3f --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CompositeBuilder/Inc/usbd_composite_builder.h @@ -0,0 +1,289 @@ +/** + ****************************************************************************** + * @file usbd_composite_builder.h + * @author MCD Application Team + * @brief Header for the usbd_composite_builder.c file + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Define to prevent recursive inclusion -------------------------------------*/ +#ifndef __USBD_COMPOSITE_BUILDER_H__ +#define __USBD_COMPOSITE_BUILDER_H__ + +#ifdef __cplusplus +extern "C" { +#endif + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_ioreq.h" + +#if USBD_CMPSIT_ACTIVATE_HID == 1U +#include "usbd_hid.h" +#endif /* USBD_CMPSIT_ACTIVATE_HID */ + +#if USBD_CMPSIT_ACTIVATE_MSC == 1U +#include "usbd_msc.h" +#endif /* USBD_CMPSIT_ACTIVATE_MSC */ + +#if USBD_CMPSIT_ACTIVATE_CDC == 1U +#include "usbd_cdc.h" +#endif /* USBD_CMPSIT_ACTIVATE_CDC */ + +#if USBD_CMPSIT_ACTIVATE_DFU == 1U +#include "usbd_dfu.h" +#endif /* USBD_CMPSIT_ACTIVATE_DFU */ + +#if USBD_CMPSIT_ACTIVATE_RNDIS == 1U +#include "usbd_cdc_rndis.h" +#endif /* USBD_CMPSIT_ACTIVATE_RNDIS */ + +#if USBD_CMPSIT_ACTIVATE_CDC_ECM == 1U +#include "usbd_cdc_ecm.h" + +#ifndef __USBD_CDC_ECM_IF_H +#include "usbd_cdc_ecm_if_template.h" +#endif /* __USBD_CDC_ECM_IF_H */ +#endif /* USBD_CMPSIT_ACTIVATE_CDC_ECM */ + +#if USBD_CMPSIT_ACTIVATE_AUDIO == 1 +#include "usbd_audio.h" +#endif /* USBD_CMPSIT_ACTIVATE_AUDIO */ + +#if USBD_CMPSIT_ACTIVATE_CUSTOMHID == 1 +#include "usbd_customhid.h" +#endif /* USBD_CMPSIT_ACTIVATE_CUSTOMHID */ + +#if USBD_CMPSIT_ACTIVATE_VIDEO == 1 +#include "usbd_video.h" +#endif /* USBD_CMPSIT_ACTIVATE_VIDEO */ + +#if USBD_CMPSIT_ACTIVATE_PRINTER == 1 +#include "usbd_printer.h" +#endif /* USBD_CMPSIT_ACTIVATE_PRINTER */ + +#if USBD_CMPSIT_ACTIVATE_CCID == 1U +#include "usbd_ccid.h" +#endif /* USBD_CMPSIT_ACTIVATE_CCID */ + +#if USBD_CMPSIT_ACTIVATE_MTP == 1U +#include "usbd_mtp.h" +#endif /* USBD_CMPSIT_ACTIVATE_MTP */ + +/* Private defines -----------------------------------------------------------*/ +/* By default all classes are deactivated, in order to activate a class + define its value to zero */ +#ifndef USBD_CMPSIT_ACTIVATE_HID +#define USBD_CMPSIT_ACTIVATE_HID 0U +#endif /* USBD_CMPSIT_ACTIVATE_HID */ + +#ifndef USBD_CMPSIT_ACTIVATE_MSC +#define USBD_CMPSIT_ACTIVATE_MSC 0U +#endif /* USBD_CMPSIT_ACTIVATE_MSC */ + +#ifndef USBD_CMPSIT_ACTIVATE_DFU +#define USBD_CMPSIT_ACTIVATE_DFU 0U +#endif /* USBD_CMPSIT_ACTIVATE_DFU */ + +#ifndef USBD_CMPSIT_ACTIVATE_CDC +#define USBD_CMPSIT_ACTIVATE_CDC 0U +#endif /* USBD_CMPSIT_ACTIVATE_CDC */ + +#ifndef USBD_CMPSIT_ACTIVATE_CDC_ECM +#define USBD_CMPSIT_ACTIVATE_CDC_ECM 0U +#endif /* USBD_CMPSIT_ACTIVATE_CDC_ECM */ + +#ifndef USBD_CMPSIT_ACTIVATE_RNDIS +#define USBD_CMPSIT_ACTIVATE_RNDIS 0U +#endif /* USBD_CMPSIT_ACTIVATE_RNDIS */ + +#ifndef USBD_CMPSIT_ACTIVATE_AUDIO +#define USBD_CMPSIT_ACTIVATE_AUDIO 0U +#endif /* USBD_CMPSIT_ACTIVATE_AUDIO */ + +#ifndef USBD_CMPSIT_ACTIVATE_CUSTOMHID +#define USBD_CMPSIT_ACTIVATE_CUSTOMHID 0U +#endif /* USBD_CMPSIT_ACTIVATE_CUSTOMHID */ + +#ifndef USBD_CMPSIT_ACTIVATE_VIDEO +#define USBD_CMPSIT_ACTIVATE_VIDEO 0U +#endif /* USBD_CMPSIT_ACTIVATE_VIDEO */ + +#ifndef USBD_CMPSIT_ACTIVATE_PRINTER +#define USBD_CMPSIT_ACTIVATE_PRINTER 0U +#endif /* USBD_CMPSIT_ACTIVATE_PRINTER */ + +#ifndef USBD_CMPSIT_ACTIVATE_CCID +#define USBD_CMPSIT_ACTIVATE_CCID 0U +#endif /* USBD_CMPSIT_ACTIVATE_CCID */ + +#ifndef USBD_CMPSIT_ACTIVATE_MTP +#define USBD_CMPSIT_ACTIVATE_MTP 0U +#endif /* USBD_CMPSIT_ACTIVATE_MTP */ + + +/* This is the maximum supported configuration descriptor size + User may define this value in usbd_conf.h in order to optimize footprint */ +#ifndef USBD_CMPST_MAX_CONFDESC_SZ +#define USBD_CMPST_MAX_CONFDESC_SZ 300U +#endif /* USBD_CMPST_MAX_CONFDESC_SZ */ + +#ifndef USBD_CONFIG_STR_DESC_IDX +#define USBD_CONFIG_STR_DESC_IDX 4U +#endif /* USBD_CONFIG_STR_DESC_IDX */ + +/* Exported types ------------------------------------------------------------*/ +/* USB Iad descriptors structure */ +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bFirstInterface; + uint8_t bInterfaceCount; + uint8_t bFunctionClass; + uint8_t bFunctionSubClass; + uint8_t bFunctionProtocol; + uint8_t iFunction; +} USBD_IadDescTypeDef; + +/* USB interface descriptors structure */ +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bInterfaceNumber; + uint8_t bAlternateSetting; + uint8_t bNumEndpoints; + uint8_t bInterfaceClass; + uint8_t bInterfaceSubClass; + uint8_t bInterfaceProtocol; + uint8_t iInterface; +} USBD_IfDescTypeDef; + +#if (USBD_CMPSIT_ACTIVATE_CDC == 1) || (USBD_CMPSIT_ACTIVATE_RNDIS == 1) || (USBD_CMPSIT_ACTIVATE_CDC_ECM == 1) +typedef struct +{ + /* + * CDC Class specification revision 1.2 + * Table 15: Class-Specific Descriptor Header Format + */ + /* Header Functional Descriptor */ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubtype; + uint16_t bcdCDC; +} __PACKED USBD_CDCHeaderFuncDescTypeDef; + +typedef struct +{ + /* Call Management Functional Descriptor */ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubtype; + uint8_t bmCapabilities; + uint8_t bDataInterface; +} USBD_CDCCallMgmFuncDescTypeDef; + +typedef struct +{ + /* ACM Functional Descriptor */ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubtype; + uint8_t bmCapabilities; +} USBD_CDCACMFuncDescTypeDef; + +typedef struct +{ + /* + * CDC Class specification revision 1.2 + * Table 16: Union Interface Functional Descriptor + */ + /* Union Functional Descriptor */ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubtype; + uint8_t bMasterInterface; + uint8_t bSlaveInterface; +} USBD_CDCUnionFuncDescTypeDef; + +#endif /* (USBD_CMPSIT_ACTIVATE_CDC == 1) || (USBD_CMPSIT_ACTIVATE_RNDIS == 1) || (USBD_CMPSIT_ACTIVATE_CDC_ECM == 1)*/ + +extern USBD_ClassTypeDef USBD_CMPSIT; + +/* Exported functions prototypes ---------------------------------------------*/ +uint8_t USBD_CMPSIT_AddToConfDesc(USBD_HandleTypeDef *pdev); + +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CMPSIT_AddClass(USBD_HandleTypeDef *pdev, + USBD_ClassTypeDef *pclass, + USBD_CompositeClassTypeDef class, + uint8_t cfgidx); + +uint32_t USBD_CMPSIT_SetClassID(USBD_HandleTypeDef *pdev, + USBD_CompositeClassTypeDef Class, + uint32_t Instance); + +uint32_t USBD_CMPSIT_GetClassID(USBD_HandleTypeDef *pdev, + USBD_CompositeClassTypeDef Class, + uint32_t Instance); +#endif /* USE_USBD_COMPOSITE */ + +uint8_t USBD_CMPST_ClearConfDesc(USBD_HandleTypeDef *pdev); + +/* Private macro -----------------------------------------------------------*/ +#define __USBD_CMPSIT_SET_EP(epadd, eptype, epsize, HSinterval, FSinterval) \ + do { \ + /* Append Endpoint descriptor to Configuration descriptor */ \ + pEpDesc = ((USBD_EpDescTypeDef*)((uint32_t)pConf + *Sze)); \ + pEpDesc->bLength = (uint8_t)sizeof(USBD_EpDescTypeDef); \ + pEpDesc->bDescriptorType = USB_DESC_TYPE_ENDPOINT; \ + pEpDesc->bEndpointAddress = (epadd); \ + pEpDesc->bmAttributes = (eptype); \ + pEpDesc->wMaxPacketSize = (uint16_t)(epsize); \ + if(speed == (uint8_t)USBD_SPEED_HIGH) \ + { \ + pEpDesc->bInterval = HSinterval; \ + } \ + else \ + { \ + pEpDesc->bInterval = FSinterval; \ + } \ + *Sze += (uint32_t)sizeof(USBD_EpDescTypeDef); \ + } while(0) + +#define __USBD_CMPSIT_SET_IF(ifnum, alt, eps, class, subclass, protocol, istring) \ + do { \ + /* Interface Descriptor */ \ + pIfDesc = ((USBD_IfDescTypeDef*)((uint32_t)pConf + *Sze)); \ + pIfDesc->bLength = (uint8_t)sizeof(USBD_IfDescTypeDef); \ + pIfDesc->bDescriptorType = USB_DESC_TYPE_INTERFACE; \ + pIfDesc->bInterfaceNumber = ifnum; \ + pIfDesc->bAlternateSetting = alt; \ + pIfDesc->bNumEndpoints = eps; \ + pIfDesc->bInterfaceClass = class; \ + pIfDesc->bInterfaceSubClass = subclass; \ + pIfDesc->bInterfaceProtocol = protocol; \ + pIfDesc->iInterface = istring; \ + *Sze += (uint32_t)sizeof(USBD_IfDescTypeDef); \ + } while(0) + +#ifdef __cplusplus +} +#endif + +#endif /* __USBD_COMPOSITE_BUILDER_H__ */ + +/** + * @} + */ + diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CompositeBuilder/Src/usbd_composite_builder.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CompositeBuilder/Src/usbd_composite_builder.c new file mode 100644 index 00000000..e14c5cd8 --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CompositeBuilder/Src/usbd_composite_builder.c @@ -0,0 +1,1879 @@ +/** + ****************************************************************************** + * @file usbd_composite_builder.c + * @author MCD Application Team + * @brief This file provides all the composite builder functions. + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + * @verbatim + * + * =================================================================== + * Composite Builder Description + * =================================================================== + * + * The composite builder builds the configuration descriptors based on + * the selection of classes by user. + * It includes all USB Device classes in order to instantiate their + * descriptors, but for better management, it is possible to optimize + * footprint by removing unused classes. It is possible to do so by + * commenting the relative define in usbd_conf.h. + * + * @endverbatim + * + ****************************************************************************** + */ + +/* BSPDependencies +- None +EndBSPDependencies */ + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_composite_builder.h" + +#ifdef USE_USBD_COMPOSITE + +/** @addtogroup STM32_USB_DEVICE_LIBRARY + * @{ + */ + + +/** @defgroup CMPSIT_CORE + * @brief Mass storage core module + * @{ + */ + +/** @defgroup CMPSIT_CORE_Private_TypesDefinitions + * @{ + */ +/** + * @} + */ + + +/** @defgroup CMPSIT_CORE_Private_Defines + * @{ + */ + +/** + * @} + */ + + +/** @defgroup CMPSIT_CORE_Private_Macros + * @{ + */ +/** + * @} + */ + + +/** @defgroup CMPSIT_CORE_Private_FunctionPrototypes + * @{ + */ +/* uint8_t USBD_CMPSIT_Init (USBD_HandleTypeDef *pdev, + uint8_t cfgidx); */ /* Function not used for the moment */ + +/* uint8_t USBD_CMPSIT_DeInit (USBD_HandleTypeDef *pdev, + uint8_t cfgidx); */ /* Function not used for the moment */ + +uint8_t *USBD_CMPSIT_GetFSCfgDesc(uint16_t *length); +#ifdef USE_USB_HS +uint8_t *USBD_CMPSIT_GetHSCfgDesc(uint16_t *length); +#endif /* USE_USB_HS */ + +uint8_t *USBD_CMPSIT_GetOtherSpeedCfgDesc(uint16_t *length); + +uint8_t *USBD_CMPSIT_GetDeviceQualifierDescriptor(uint16_t *length); + +static uint8_t USBD_CMPSIT_FindFreeIFNbr(USBD_HandleTypeDef *pdev); + +static void USBD_CMPSIT_AddConfDesc(uint32_t Conf, __IO uint32_t *pSze); + +static void USBD_CMPSIT_AssignEp(USBD_HandleTypeDef *pdev, uint8_t Add, uint8_t Type, uint32_t Sze); + + +#if USBD_CMPSIT_ACTIVATE_HID == 1U +static void USBD_CMPSIT_HIDMouseDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed); +#endif /* USBD_CMPSIT_ACTIVATE_HID == 1U */ + +#if USBD_CMPSIT_ACTIVATE_MSC == 1U +static void USBD_CMPSIT_MSCDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed); +#endif /* USBD_CMPSIT_ACTIVATE_MSC == 1U */ + +#if USBD_CMPSIT_ACTIVATE_CDC == 1U +static void USBD_CMPSIT_CDCDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed); +#endif /* USBD_CMPSIT_ACTIVATE_CDC == 1U */ + +#if USBD_CMPSIT_ACTIVATE_DFU == 1U +static void USBD_CMPSIT_DFUDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed); +#endif /* USBD_CMPSIT_ACTIVATE_DFU == 1U */ + +#if USBD_CMPSIT_ACTIVATE_RNDIS == 1U +static void USBD_CMPSIT_RNDISDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed); +#endif /* USBD_CMPSIT_ACTIVATE_RNDIS == 1U */ + +#if USBD_CMPSIT_ACTIVATE_CDC_ECM == 1U +static void USBD_CMPSIT_CDC_ECMDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed); +#endif /* USBD_CMPSIT_ACTIVATE_CDC_ECM == 1U */ + +#if USBD_CMPSIT_ACTIVATE_AUDIO == 1U +static void USBD_CMPSIT_AUDIODesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed); +#endif /* USBD_CMPSIT_ACTIVATE_AUDIO == 1U */ + +#if USBD_CMPSIT_ACTIVATE_CUSTOMHID == 1 +static void USBD_CMPSIT_CUSTOMHIDDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed); +#endif /* USBD_CMPSIT_ACTIVATE_CUSTOMHID == 1U */ + +#if USBD_CMPSIT_ACTIVATE_VIDEO == 1U +static void USBD_CMPSIT_VIDEODesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed); +#endif /* USBD_CMPSIT_ACTIVATE_VIDEO == 1U */ + +#if USBD_CMPSIT_ACTIVATE_PRINTER == 1U +static void USBD_CMPSIT_PRNTDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed); +#endif /* USBD_CMPSIT_ACTIVATE_PRINTER == 1U */ + +#if USBD_CMPSIT_ACTIVATE_CCID == 1U +static void USBD_CMPSIT_CCIDDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed); +#endif /* USBD_CMPSIT_ACTIVATE_CCID == 1U */ + +#if USBD_CMPSIT_ACTIVATE_MTP == 1U +static void USBD_CMPSIT_MTPDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed); +#endif /* USBD_CMPSIT_ACTIVATE_MTP == 1U */ + +/** + * @} + */ + + +/** @defgroup CMPSIT_CORE_Private_Variables + * @{ + */ +/* This structure is used only for the Configuration descriptors and Device Qualifier */ +USBD_ClassTypeDef USBD_CMPSIT = +{ + NULL, /* Init, */ + NULL, /* DeInit, */ + NULL, /* Setup, */ + NULL, /* EP0_TxSent, */ + NULL, /* EP0_RxReady, */ + NULL, /* DataIn, */ + NULL, /* DataOut, */ + NULL, /* SOF, */ + NULL, + NULL, +#ifdef USE_USB_HS + USBD_CMPSIT_GetHSCfgDesc, +#else + NULL, +#endif /* USE_USB_HS */ + USBD_CMPSIT_GetFSCfgDesc, + USBD_CMPSIT_GetOtherSpeedCfgDesc, + USBD_CMPSIT_GetDeviceQualifierDescriptor, +#if (USBD_SUPPORT_USER_STRING_DESC == 1U) + NULL, +#endif /* USBD_SUPPORT_USER_STRING_DESC */ +}; + +/* The generic configuration descriptor buffer that will be filled by builder + Size of the buffer is the maximum possible configuration descriptor size. */ +__ALIGN_BEGIN static uint8_t USBD_CMPSIT_FSCfgDesc[USBD_CMPST_MAX_CONFDESC_SZ] __ALIGN_END = {0}; +static uint8_t *pCmpstFSConfDesc = USBD_CMPSIT_FSCfgDesc; +/* Variable that dynamically holds the current size of the configuration descriptor */ +static __IO uint32_t CurrFSConfDescSz = 0U; + +#ifdef USE_USB_HS +__ALIGN_BEGIN static uint8_t USBD_CMPSIT_HSCfgDesc[USBD_CMPST_MAX_CONFDESC_SZ] __ALIGN_END = {0}; +static uint8_t *pCmpstHSConfDesc = USBD_CMPSIT_HSCfgDesc; +/* Variable that dynamically holds the current size of the configuration descriptor */ +static __IO uint32_t CurrHSConfDescSz = 0U; +#endif /* USE_USB_HS */ + +/* USB Standard Device Descriptor */ +__ALIGN_BEGIN static uint8_t USBD_CMPSIT_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_DESC] __ALIGN_END = +{ + USB_LEN_DEV_QUALIFIER_DESC, /* bLength */ + USB_DESC_TYPE_DEVICE_QUALIFIER, /* bDescriptorType */ + 0x00, /* bcdDevice low */ + 0x02, /* bcdDevice high */ + 0xEF, /* Class */ + 0x02, /* SubClass */ + 0x01, /* Protocol */ + 0x40, /* bMaxPacketSize0 */ + 0x01, /* bNumConfigurations */ + 0x00, /* bReserved */ +}; + +/** + * @} + */ + + +/** @defgroup CMPSIT_CORE_Private_Functions + * @{ + */ + +/** + * @brief USBD_CMPSIT_AddClass + * Register a class in the class builder + * @param pdev: device instance + * @param pclass: pointer to the class structure to be added + * @param class: type of the class to be added (from USBD_CompositeClassTypeDef) + * @param cfgidx: configuration index + * @retval status + */ +uint8_t USBD_CMPSIT_AddClass(USBD_HandleTypeDef *pdev, + USBD_ClassTypeDef *pclass, + USBD_CompositeClassTypeDef class, + uint8_t cfgidx) +{ + if ((pdev->classId < USBD_MAX_SUPPORTED_CLASS) && (pdev->tclasslist[pdev->classId].Active == 0U)) + { + /* Store the class parameters in the global tab */ + pdev->pClass[pdev->classId] = pclass; + pdev->tclasslist[pdev->classId].ClassId = pdev->classId; + pdev->tclasslist[pdev->classId].Active = 1U; + pdev->tclasslist[pdev->classId].ClassType = class; + + /* Call configuration descriptor builder and endpoint configuration builder */ + if (USBD_CMPSIT_AddToConfDesc(pdev) != (uint8_t)USBD_OK) + { + return (uint8_t)USBD_FAIL; + } + } + + UNUSED(cfgidx); + + return (uint8_t)USBD_OK; +} + + +/** + * @brief USBD_CMPSIT_AddToConfDesc + * Add a new class to the configuration descriptor + * @param pdev: device instance + * @retval status + */ +uint8_t USBD_CMPSIT_AddToConfDesc(USBD_HandleTypeDef *pdev) +{ + uint8_t idxIf = 0U; + uint8_t iEp = 0U; + + /* For the first class instance, start building the config descriptor common part */ + if (pdev->classId == 0U) + { + /* Add configuration and IAD descriptors */ + USBD_CMPSIT_AddConfDesc((uint32_t)pCmpstFSConfDesc, &CurrFSConfDescSz); +#ifdef USE_USB_HS + USBD_CMPSIT_AddConfDesc((uint32_t)pCmpstHSConfDesc, &CurrHSConfDescSz); +#endif /* USE_USB_HS */ + } + + switch (pdev->tclasslist[pdev->classId].ClassType) + { +#if USBD_CMPSIT_ACTIVATE_HID == 1 + case CLASS_TYPE_HID: + /* Setup Max packet sizes (for HID, no dependency on USB Speed, both HS/FS have same packet size) */ + pdev->tclasslist[pdev->classId].CurrPcktSze = HID_EPIN_SIZE; + + /* Find the first available interface slot and Assign number of interfaces */ + idxIf = USBD_CMPSIT_FindFreeIFNbr(pdev); + pdev->tclasslist[pdev->classId].NumIf = 1U; + pdev->tclasslist[pdev->classId].Ifs[0] = idxIf; + + /* Assign endpoint numbers */ + pdev->tclasslist[pdev->classId].NumEps = 1U; /* EP1_IN */ + + /* Set IN endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[0]; + + /* Assign IN Endpoint */ + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_INTR, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Configure and Append the Descriptor */ + USBD_CMPSIT_HIDMouseDesc(pdev, (uint32_t)pCmpstFSConfDesc, &CurrFSConfDescSz, (uint8_t)USBD_SPEED_FULL); + +#ifdef USE_USB_HS + USBD_CMPSIT_HIDMouseDesc(pdev, (uint32_t)pCmpstHSConfDesc, &CurrHSConfDescSz, (uint8_t)USBD_SPEED_HIGH); +#endif /* USE_USB_HS */ + + break; +#endif /* USBD_CMPSIT_ACTIVATE_HID */ + +#if USBD_CMPSIT_ACTIVATE_MSC == 1 + case CLASS_TYPE_MSC: + /* Setup default Max packet size */ + pdev->tclasslist[pdev->classId].CurrPcktSze = MSC_MAX_FS_PACKET; + + /* Find the first available interface slot and Assign number of interfaces */ + idxIf = USBD_CMPSIT_FindFreeIFNbr(pdev); + pdev->tclasslist[pdev->classId].NumIf = 1U; + pdev->tclasslist[pdev->classId].Ifs[0] = idxIf; + + /* Assign endpoint numbers */ + pdev->tclasslist[pdev->classId].NumEps = 2U; /* EP1_IN, EP1_OUT */ + + /* Set IN endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[0]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_BULK, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Set OUT endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[1]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_BULK, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Configure and Append the Descriptor */ + USBD_CMPSIT_MSCDesc(pdev, (uint32_t)pCmpstFSConfDesc, &CurrFSConfDescSz, (uint8_t)USBD_SPEED_FULL); + +#ifdef USE_USB_HS + USBD_CMPSIT_MSCDesc(pdev, (uint32_t)pCmpstHSConfDesc, &CurrHSConfDescSz, (uint8_t)USBD_SPEED_HIGH); +#endif /* USE_USB_HS */ + + break; +#endif /* USBD_CMPSIT_ACTIVATE_MSC */ + +#if USBD_CMPSIT_ACTIVATE_CDC == 1 + case CLASS_TYPE_CDC: + /* Setup default Max packet size for FS device */ + pdev->tclasslist[pdev->classId].CurrPcktSze = CDC_DATA_FS_MAX_PACKET_SIZE; + + /* Find the first available interface slot and Assign number of interfaces */ + idxIf = USBD_CMPSIT_FindFreeIFNbr(pdev); + pdev->tclasslist[pdev->classId].NumIf = 2U; + pdev->tclasslist[pdev->classId].Ifs[0] = idxIf; + pdev->tclasslist[pdev->classId].Ifs[1] = (uint8_t)(idxIf + 1U); + + /* Assign endpoint numbers */ + pdev->tclasslist[pdev->classId].NumEps = 3U; + + /* Set IN endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[0]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_BULK, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Set OUT endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[1]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_BULK, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Set the second IN endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[2]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_INTR, CDC_CMD_PACKET_SIZE); + + /* Configure and Append the Descriptor */ + USBD_CMPSIT_CDCDesc(pdev, (uint32_t)pCmpstFSConfDesc, &CurrFSConfDescSz, (uint8_t)USBD_SPEED_FULL); + +#ifdef USE_USB_HS + USBD_CMPSIT_CDCDesc(pdev, (uint32_t)pCmpstHSConfDesc, &CurrHSConfDescSz, (uint8_t)USBD_SPEED_HIGH); +#endif /* USE_USB_HS */ + + break; +#endif /* USBD_CMPSIT_ACTIVATE_CDC */ + +#if USBD_CMPSIT_ACTIVATE_DFU == 1 + case CLASS_TYPE_DFU: + /* Setup Max packet sizes (for DFU, no dependency on USB Speed, both HS/FS have same packet size) */ + pdev->tclasslist[pdev->classId].CurrPcktSze = 64U; + + /* Find the first available interface slot and Assign number of interfaces */ + idxIf = USBD_CMPSIT_FindFreeIFNbr(pdev); + pdev->tclasslist[pdev->classId].NumIf = 1U; + pdev->tclasslist[pdev->classId].Ifs[0] = idxIf; + + /* Assign endpoint numbers */ + pdev->tclasslist[pdev->classId].NumEps = 0U; /* only EP0 is used */ + + /* Configure and Append the Descriptor */ + USBD_CMPSIT_DFUDesc(pdev, (uint32_t)pCmpstFSConfDesc, &CurrFSConfDescSz, (uint8_t)USBD_SPEED_FULL); + +#ifdef USE_USB_HS + USBD_CMPSIT_DFUDesc(pdev, (uint32_t)pCmpstHSConfDesc, &CurrHSConfDescSz, (uint8_t)USBD_SPEED_HIGH); +#endif /* USE_USB_HS */ + + break; +#endif /* USBD_CMPSIT_ACTIVATE_DFU */ + +#if USBD_CMPSIT_ACTIVATE_RNDIS == 1 + case CLASS_TYPE_RNDIS: + /* Setup default Max packet size */ + pdev->tclasslist[pdev->classId].CurrPcktSze = CDC_RNDIS_DATA_FS_MAX_PACKET_SIZE; + + /* Find the first available interface slot and Assign number of interfaces */ + idxIf = USBD_CMPSIT_FindFreeIFNbr(pdev); + pdev->tclasslist[pdev->classId].NumIf = 2U; + pdev->tclasslist[pdev->classId].Ifs[0] = idxIf; + pdev->tclasslist[pdev->classId].Ifs[1] = (uint8_t)(idxIf + 1U); + + /* Assign endpoint numbers */ + pdev->tclasslist[pdev->classId].NumEps = 3U; + + /* Set IN endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[0]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_BULK, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Set OUT endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[1]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_BULK, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Set the second IN endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[2]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_INTR, CDC_RNDIS_CMD_PACKET_SIZE); + + /* Configure and Append the Descriptor */ + USBD_CMPSIT_RNDISDesc(pdev, (uint32_t)pCmpstFSConfDesc, &CurrFSConfDescSz, (uint8_t)USBD_SPEED_FULL); + +#ifdef USE_USB_HS + USBD_CMPSIT_RNDISDesc(pdev, (uint32_t)pCmpstHSConfDesc, &CurrHSConfDescSz, (uint8_t)USBD_SPEED_HIGH); +#endif /* USE_USB_HS */ + + break; +#endif /* USBD_CMPSIT_ACTIVATE_RNDIS */ + +#if USBD_CMPSIT_ACTIVATE_CDC_ECM == 1 + case CLASS_TYPE_ECM: + /* Setup default Max packet size */ + pdev->tclasslist[pdev->classId].CurrPcktSze = CDC_ECM_DATA_FS_MAX_PACKET_SIZE; + + /* Find the first available interface slot and Assign number of interfaces */ + idxIf = USBD_CMPSIT_FindFreeIFNbr(pdev); + pdev->tclasslist[pdev->classId].NumIf = 2U; + pdev->tclasslist[pdev->classId].Ifs[0] = idxIf; + pdev->tclasslist[pdev->classId].Ifs[1] = (uint8_t)(idxIf + 1U); + + /* Assign endpoint numbers */ + pdev->tclasslist[pdev->classId].NumEps = 3U; /* EP1_IN, EP1_OUT,CMD_EP2 */ + + /* Set IN endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[0]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_BULK, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Set OUT endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[1]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_BULK, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Set the second IN endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[2]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_INTR, CDC_ECM_CMD_PACKET_SIZE); + + /* Configure and Append the Descriptor */ + USBD_CMPSIT_CDC_ECMDesc(pdev, (uint32_t)pCmpstFSConfDesc, &CurrFSConfDescSz, (uint8_t)USBD_SPEED_FULL); + +#ifdef USE_USB_HS + USBD_CMPSIT_CDC_ECMDesc(pdev, (uint32_t)pCmpstHSConfDesc, &CurrHSConfDescSz, (uint8_t)USBD_SPEED_HIGH); +#endif /* USE_USB_HS */ + + break; +#endif /* USBD_CMPSIT_ACTIVATE_CDC_ECM */ + +#if USBD_CMPSIT_ACTIVATE_AUDIO == 1 + case CLASS_TYPE_AUDIO: + /* Setup Max packet sizes*/ + pdev->tclasslist[pdev->classId].CurrPcktSze = USBD_AUDIO_GetEpPcktSze(pdev, 0U, 0U); + + /* Find the first available interface slot and Assign number of interfaces */ + idxIf = USBD_CMPSIT_FindFreeIFNbr(pdev); + pdev->tclasslist[pdev->classId].NumIf = 2U; + pdev->tclasslist[pdev->classId].Ifs[0] = idxIf; + pdev->tclasslist[pdev->classId].Ifs[1] = (uint8_t)(idxIf + 1U); + + /* Assign endpoint numbers */ + pdev->tclasslist[pdev->classId].NumEps = 1U; /* EP1_OUT*/ + + /* Set OUT endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[0]; + + /* Assign OUT Endpoint */ + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_ISOC, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Configure and Append the Descriptor (only FS mode supported) */ + USBD_CMPSIT_AUDIODesc(pdev, (uint32_t)pCmpstFSConfDesc, &CurrFSConfDescSz, (uint8_t)USBD_SPEED_FULL); + + break; +#endif /* USBD_CMPSIT_ACTIVATE_AUDIO */ + +#if USBD_CMPSIT_ACTIVATE_CUSTOMHID == 1 + case CLASS_TYPE_CHID: + /* Setup Max packet sizes */ + pdev->tclasslist[pdev->classId].CurrPcktSze = CUSTOM_HID_EPOUT_SIZE; + + /* Find the first available interface slot and Assign number of interfaces */ + idxIf = USBD_CMPSIT_FindFreeIFNbr(pdev); + pdev->tclasslist[pdev->classId].NumIf = 1U; + pdev->tclasslist[pdev->classId].Ifs[0] = idxIf; + + /* Assign endpoint numbers */ + pdev->tclasslist[pdev->classId].NumEps = 2U; /* EP1_IN, EP1_OUT */ + + /* Set IN endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[0]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_INTR, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Set OUT endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[1]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_INTR, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Configure and Append the Descriptor */ + USBD_CMPSIT_CUSTOMHIDDesc(pdev, (uint32_t)pCmpstFSConfDesc, &CurrFSConfDescSz, (uint8_t)USBD_SPEED_FULL); + +#ifdef USE_USB_HS + USBD_CMPSIT_CUSTOMHIDDesc(pdev, (uint32_t)pCmpstHSConfDesc, &CurrHSConfDescSz, (uint8_t)USBD_SPEED_HIGH); +#endif /* USE_USB_HS */ + + break; +#endif /* USBD_CMPSIT_ACTIVATE_CUSTOMHID */ + +#if USBD_CMPSIT_ACTIVATE_VIDEO == 1 + case CLASS_TYPE_VIDEO: + /* Setup default Max packet size */ + pdev->tclasslist[pdev->classId].CurrPcktSze = UVC_ISO_FS_MPS; + + /* Find the first available interface slot and Assign number of interfaces */ + idxIf = USBD_CMPSIT_FindFreeIFNbr(pdev); + pdev->tclasslist[pdev->classId].NumIf = 2U; + pdev->tclasslist[pdev->classId].Ifs[0] = idxIf; + pdev->tclasslist[pdev->classId].Ifs[1] = (uint8_t)(idxIf + 1U); + + /* Assign endpoint numbers */ + pdev->tclasslist[pdev->classId].NumEps = 1U; /* EP1_IN */ + + /* Set IN endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[0]; + + /* Assign IN Endpoint */ + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_ISOC, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Configure and Append the Descriptor */ + USBD_CMPSIT_VIDEODesc(pdev, (uint32_t)pCmpstFSConfDesc, &CurrFSConfDescSz, (uint8_t)USBD_SPEED_FULL); + +#ifdef USE_USB_HS + USBD_CMPSIT_VIDEODesc(pdev, (uint32_t)pCmpstHSConfDesc, &CurrHSConfDescSz, (uint8_t)USBD_SPEED_HIGH); +#endif /* USE_USB_HS */ + + break; +#endif /* USBD_CMPSIT_ACTIVATE_VIDEO */ + +#if USBD_CMPSIT_ACTIVATE_PRINTER == 1 + case CLASS_TYPE_PRINTER: + + /* Setup default Max packet size */ + pdev->tclasslist[pdev->classId].CurrPcktSze = PRNT_DATA_FS_MAX_PACKET_SIZE; + + /* Find the first available interface slot and Assign number of interfaces */ + idxIf = USBD_CMPSIT_FindFreeIFNbr(pdev); + pdev->tclasslist[pdev->classId].NumIf = 1U; + pdev->tclasslist[pdev->classId].Ifs[0] = idxIf; + + /* Assign endpoint numbers */ + pdev->tclasslist[pdev->classId].NumEps = 2U; + + /* Set IN endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[0]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_BULK, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Set OUT endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[1]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_BULK, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Configure and Append the Descriptor */ + USBD_CMPSIT_PRNTDesc(pdev, (uint32_t)pCmpstFSConfDesc, &CurrFSConfDescSz, (uint8_t)USBD_SPEED_FULL); + +#ifdef USE_USB_HS + USBD_CMPSIT_PRNTDesc(pdev, (uint32_t)pCmpstHSConfDesc, &CurrHSConfDescSz, (uint8_t)USBD_SPEED_HIGH); +#endif /* USE_USB_HS */ + + break; +#endif /* USBD_CMPSIT_ACTIVATE_PRINTER */ + +#if USBD_CMPSIT_ACTIVATE_CCID == 1 + + case CLASS_TYPE_CCID: + /* Setup default Max packet size */ + pdev->tclasslist[pdev->classId].CurrPcktSze = CCID_DATA_FS_MAX_PACKET_SIZE; + + /* Find the first available interface slot and Assign number of interfaces */ + idxIf = USBD_CMPSIT_FindFreeIFNbr(pdev); + pdev->tclasslist[pdev->classId].NumIf = 1U; + pdev->tclasslist[pdev->classId].Ifs[0] = idxIf; + + /* Assign endpoint numbers */ + pdev->tclasslist[pdev->classId].NumEps = 3U; + + /* Set IN endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[0]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_BULK, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Set OUT endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[1]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_BULK, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Set the second IN endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[2]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_INTR, CCID_CMD_PACKET_SIZE); + + /* Configure and Append the Descriptor */ + USBD_CMPSIT_CCIDDesc(pdev, (uint32_t)pCmpstFSConfDesc, &CurrFSConfDescSz, (uint8_t)USBD_SPEED_FULL); + +#ifdef USE_USB_HS + USBD_CMPSIT_CCIDDesc(pdev, (uint32_t)pCmpstHSConfDesc, &CurrHSConfDescSz, (uint8_t)USBD_SPEED_HIGH); +#endif /* USE_USB_HS */ + + break; +#endif /* USBD_CMPSIT_ACTIVATE_CCID */ + +#if USBD_CMPSIT_ACTIVATE_MTP == 1 + + case CLASS_TYPE_MTP: + /* Setup default Max packet sizes */ + pdev->tclasslist[pdev->classId].CurrPcktSze = MTP_DATA_MAX_FS_PACKET_SIZE; + + /* Find the first available interface slot and Assign number of interfaces */ + idxIf = USBD_CMPSIT_FindFreeIFNbr(pdev); + pdev->tclasslist[pdev->classId].NumIf = 1U; + pdev->tclasslist[pdev->classId].Ifs[0] = idxIf; + + /* Assign endpoint numbers */ + pdev->tclasslist[pdev->classId].NumEps = 3U; + + /* Set IN endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[0]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_BULK, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Set OUT endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[1]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_BULK, pdev->tclasslist[pdev->classId].CurrPcktSze); + + /* Set the second IN endpoint slot */ + iEp = pdev->tclasslist[pdev->classId].EpAdd[2]; + USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_INTR, MTP_CMD_PACKET_SIZE); + + /* Configure and Append the Descriptor */ + USBD_CMPSIT_MTPDesc(pdev, (uint32_t)pCmpstFSConfDesc, &CurrFSConfDescSz, (uint8_t)USBD_SPEED_FULL); + +#ifdef USE_USB_HS + USBD_CMPSIT_MTPDesc(pdev, (uint32_t)pCmpstHSConfDesc, &CurrHSConfDescSz, (uint8_t)USBD_SPEED_HIGH); +#endif /* USE_USB_HS */ + + break; +#endif /* USBD_CMPSIT_ACTIVATE_MTP */ + + default: + UNUSED(idxIf); + UNUSED(iEp); + UNUSED(USBD_CMPSIT_FindFreeIFNbr); + UNUSED(USBD_CMPSIT_AssignEp); + break; + } + + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_CMPSIT_GetFSCfgDesc + * return configuration descriptor for both FS and HS modes + * @param length : pointer data length + * @retval pointer to descriptor buffer + */ +uint8_t *USBD_CMPSIT_GetFSCfgDesc(uint16_t *length) +{ + *length = (uint16_t)CurrFSConfDescSz; + + return USBD_CMPSIT_FSCfgDesc; +} + +#ifdef USE_USB_HS +/** + * @brief USBD_CMPSIT_GetHSCfgDesc + * return configuration descriptor for both FS and HS modes + * @param length : pointer data length + * @retval pointer to descriptor buffer + */ +uint8_t *USBD_CMPSIT_GetHSCfgDesc(uint16_t *length) +{ + *length = (uint16_t)CurrHSConfDescSz; + + return USBD_CMPSIT_HSCfgDesc; +} +#endif /* USE_USB_HS */ + +/** + * @brief USBD_CMPSIT_GetOtherSpeedCfgDesc + * return other speed configuration descriptor + * @param length : pointer data length + * @retval pointer to descriptor buffer + */ +uint8_t *USBD_CMPSIT_GetOtherSpeedCfgDesc(uint16_t *length) +{ + *length = (uint16_t)CurrFSConfDescSz; + + return USBD_CMPSIT_FSCfgDesc; +} + +/** + * @brief DeviceQualifierDescriptor + * return Device Qualifier descriptor + * @param length : pointer data length + * @retval pointer to descriptor buffer + */ +uint8_t *USBD_CMPSIT_GetDeviceQualifierDescriptor(uint16_t *length) +{ + *length = (uint16_t)(sizeof(USBD_CMPSIT_DeviceQualifierDesc)); + return USBD_CMPSIT_DeviceQualifierDesc; +} + +/** + * @brief USBD_CMPSIT_FindFreeIFNbr + * Find the first interface available slot + * @param pdev: device instance + * @retval The interface number to be used + */ +static uint8_t USBD_CMPSIT_FindFreeIFNbr(USBD_HandleTypeDef *pdev) +{ + uint32_t idx = 0U; + + /* Unroll all already activated classes */ + for (uint32_t i = 0U; i < pdev->NumClasses; i++) + { + /* Unroll each class interfaces */ + for (uint32_t j = 0U; j < pdev->tclasslist[i].NumIf; j++) + { + /* Increment the interface counter index */ + idx++; + } + } + + /* Return the first available interface slot */ + return (uint8_t)idx; +} + +/** + * @brief USBD_CMPSIT_AddToConfDesc + * Add a new class to the configuration descriptor + * @param pdev: device instance + * @retval none + */ +static void USBD_CMPSIT_AddConfDesc(uint32_t Conf, __IO uint32_t *pSze) +{ + /* Intermediate variable to comply with MISRA-C Rule 11.3 */ + USBD_ConfigDescTypeDef *ptr = (USBD_ConfigDescTypeDef *)Conf; + + ptr->bLength = (uint8_t)sizeof(USBD_ConfigDescTypeDef); + ptr->bDescriptorType = USB_DESC_TYPE_CONFIGURATION; + ptr->wTotalLength = 0U; + ptr->bNumInterfaces = 0U; + ptr->bConfigurationValue = 1U; + ptr->iConfiguration = USBD_CONFIG_STR_DESC_IDX; + +#if (USBD_SELF_POWERED == 1U) + ptr->bmAttributes = 0xC0U; /* bmAttributes: Self Powered according to user configuration */ +#else + ptr->bmAttributes = 0x80U; /* bmAttributes: Bus Powered according to user configuration */ +#endif /* USBD_SELF_POWERED */ + + ptr->bMaxPower = USBD_MAX_POWER; + + *pSze += sizeof(USBD_ConfigDescTypeDef); +} + +/** + * @brief USBD_CMPSIT_AssignEp + * Assign and endpoint + * @param pdev: device instance + * @param Add: Endpoint address + * @param Type: Endpoint type + * @param Sze: Endpoint max packet size + * @retval none + */ +static void USBD_CMPSIT_AssignEp(USBD_HandleTypeDef *pdev, uint8_t Add, uint8_t Type, uint32_t Sze) +{ + uint32_t idx = 0U; + + /* Find the first available endpoint slot */ + while (((idx < (pdev->tclasslist[pdev->classId]).NumEps) && \ + ((pdev->tclasslist[pdev->classId].Eps[idx].is_used) != 0U))) + { + /* Increment the index */ + idx++; + } + + /* Configure the endpoint */ + pdev->tclasslist[pdev->classId].Eps[idx].add = Add; + pdev->tclasslist[pdev->classId].Eps[idx].type = Type; + pdev->tclasslist[pdev->classId].Eps[idx].size = (uint8_t)Sze; + pdev->tclasslist[pdev->classId].Eps[idx].is_used = 1U; +} + +#if USBD_CMPSIT_ACTIVATE_HID == 1 +/** + * @brief USBD_CMPSIT_HIDMouseDesc + * Configure and Append the HID Mouse Descriptor + * @param pdev: device instance + * @param pConf: Configuration descriptor pointer + * @param Sze: pointer to the current configuration descriptor size + * @retval None + */ +static void USBD_CMPSIT_HIDMouseDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, + __IO uint32_t *Sze, uint8_t speed) +{ + static USBD_IfDescTypeDef *pIfDesc; + static USBD_EpDescTypeDef *pEpDesc; + static USBD_HIDDescTypeDef *pHidMouseDesc; + + /* Append HID Interface descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[0], 0U, \ + (uint8_t)(pdev->tclasslist[pdev->classId].NumEps), 0x03U, 0x01U, 0x02U, 0U); + + /* Append HID Functional descriptor to Configuration descriptor */ + pHidMouseDesc = ((USBD_HIDDescTypeDef *)(pConf + *Sze)); + pHidMouseDesc->bLength = (uint8_t)sizeof(USBD_HIDDescTypeDef); + pHidMouseDesc->bDescriptorType = HID_DESCRIPTOR_TYPE; + pHidMouseDesc->bcdHID = 0x0111U; + pHidMouseDesc->bCountryCode = 0x00U; + pHidMouseDesc->bNumDescriptors = 0x01U; + pHidMouseDesc->bHIDDescriptorType = 0x22U; + pHidMouseDesc->wItemLength = HID_MOUSE_REPORT_DESC_SIZE; + *Sze += (uint32_t)sizeof(USBD_HIDDescTypeDef); + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP(pdev->tclasslist[pdev->classId].Eps[0].add, USBD_EP_TYPE_INTR, HID_EPIN_SIZE, \ + HID_HS_BINTERVAL, HID_FS_BINTERVAL); + + /* Update Config Descriptor and IAD descriptor */ + ((USBD_ConfigDescTypeDef *)pConf)->bNumInterfaces += 1U; + ((USBD_ConfigDescTypeDef *)pConf)->wTotalLength = (uint16_t)(*Sze); +} +#endif /* USBD_CMPSIT_ACTIVATE_HID == 1 */ + +#if USBD_CMPSIT_ACTIVATE_MSC == 1 +/** + * @brief USBD_CMPSIT_MSCDesc + * Configure and Append the MSC Descriptor + * @param pdev: device instance + * @param pConf: Configuration descriptor pointer + * @param Sze: pointer to the current configuration descriptor size + * @retval None + */ +static void USBD_CMPSIT_MSCDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed) +{ + USBD_IfDescTypeDef *pIfDesc; + USBD_EpDescTypeDef *pEpDesc; + + /* Append MSC Interface descriptor */ + __USBD_CMPSIT_SET_IF((pdev->tclasslist[pdev->classId].Ifs[0]), (0U), \ + (uint8_t)(pdev->tclasslist[pdev->classId].NumEps), (0x08U), (0x06U), (0x50U), (0U)); + + if (speed == (uint8_t)USBD_SPEED_HIGH) + { + pdev->tclasslist[pdev->classId].CurrPcktSze = MSC_MAX_HS_PACKET; + } + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP((pdev->tclasslist[pdev->classId].Eps[0].add), (USBD_EP_TYPE_BULK), \ + (pdev->tclasslist[pdev->classId].CurrPcktSze), (0U), (0U)); + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP((pdev->tclasslist[pdev->classId].Eps[1].add), (USBD_EP_TYPE_BULK), \ + (pdev->tclasslist[pdev->classId].CurrPcktSze), (0U), (0U)); + + /* Update Config Descriptor and IAD descriptor */ + ((USBD_ConfigDescTypeDef *)pConf)->bNumInterfaces += 1U; + ((USBD_ConfigDescTypeDef *)pConf)->wTotalLength = (uint16_t)(*Sze); +} +#endif /* USBD_CMPSIT_ACTIVATE_MSC == 1 */ + +#if USBD_CMPSIT_ACTIVATE_CDC == 1 +/** + * @brief USBD_CMPSIT_MSCDesc + * Configure and Append the HID Mouse Descriptor + * @param pdev: device instance + * @param pConf: Configuration descriptor pointer + * @param Sze: pointer to the current configuration descriptor size + * @retval None + */ +static void USBD_CMPSIT_CDCDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed) +{ + static USBD_IfDescTypeDef *pIfDesc; + static USBD_EpDescTypeDef *pEpDesc; + static USBD_CDCHeaderFuncDescTypeDef *pHeadDesc; + static USBD_CDCCallMgmFuncDescTypeDef *pCallMgmDesc; + static USBD_CDCACMFuncDescTypeDef *pACMDesc; + static USBD_CDCUnionFuncDescTypeDef *pUnionDesc; +#if USBD_COMPOSITE_USE_IAD == 1 + static USBD_IadDescTypeDef *pIadDesc; +#endif /* USBD_COMPOSITE_USE_IAD == 1 */ + +#if USBD_COMPOSITE_USE_IAD == 1 + pIadDesc = ((USBD_IadDescTypeDef *)(pConf + *Sze)); + pIadDesc->bLength = (uint8_t)sizeof(USBD_IadDescTypeDef); + pIadDesc->bDescriptorType = USB_DESC_TYPE_IAD; /* IAD descriptor */ + pIadDesc->bFirstInterface = pdev->tclasslist[pdev->classId].Ifs[0]; + pIadDesc->bInterfaceCount = 2U; /* 2 interfaces */ + pIadDesc->bFunctionClass = 0x02U; + pIadDesc->bFunctionSubClass = 0x02U; + pIadDesc->bFunctionProtocol = 0x01U; + pIadDesc->iFunction = 0U; /* String Index */ + *Sze += (uint32_t)sizeof(USBD_IadDescTypeDef); +#endif /* USBD_COMPOSITE_USE_IAD == 1 */ + + /* Control Interface Descriptor */ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[0], 0U, 1U, 0x02, 0x02U, 0x01U, 0U); + + /* Control interface headers */ + pHeadDesc = ((USBD_CDCHeaderFuncDescTypeDef *)((uint32_t)pConf + *Sze)); + /* Header Functional Descriptor*/ + pHeadDesc->bLength = 0x05U; + pHeadDesc->bDescriptorType = 0x24U; + pHeadDesc->bDescriptorSubtype = 0x00U; + pHeadDesc->bcdCDC = 0x0110U; + *Sze += (uint32_t)sizeof(USBD_CDCHeaderFuncDescTypeDef); + + /* Call Management Functional Descriptor */ + pCallMgmDesc = ((USBD_CDCCallMgmFuncDescTypeDef *)((uint32_t)pConf + *Sze)); + pCallMgmDesc->bLength = 0x05U; + pCallMgmDesc->bDescriptorType = 0x24U; + pCallMgmDesc->bDescriptorSubtype = 0x01U; + pCallMgmDesc->bmCapabilities = 0x00U; + pCallMgmDesc->bDataInterface = pdev->tclasslist[pdev->classId].Ifs[1]; + *Sze += (uint32_t)sizeof(USBD_CDCCallMgmFuncDescTypeDef); + + /* ACM Functional Descriptor*/ + pACMDesc = ((USBD_CDCACMFuncDescTypeDef *)((uint32_t)pConf + *Sze)); + pACMDesc->bLength = 0x04U; + pACMDesc->bDescriptorType = 0x24U; + pACMDesc->bDescriptorSubtype = 0x02U; + pACMDesc->bmCapabilities = 0x02U; + *Sze += (uint32_t)sizeof(USBD_CDCACMFuncDescTypeDef); + + /* Union Functional Descriptor*/ + pUnionDesc = ((USBD_CDCUnionFuncDescTypeDef *)((uint32_t)pConf + *Sze)); + pUnionDesc->bLength = 0x05U; + pUnionDesc->bDescriptorType = 0x24U; + pUnionDesc->bDescriptorSubtype = 0x06U; + pUnionDesc->bMasterInterface = pdev->tclasslist[pdev->classId].Ifs[0]; + pUnionDesc->bSlaveInterface = pdev->tclasslist[pdev->classId].Ifs[1]; + *Sze += (uint32_t)sizeof(USBD_CDCUnionFuncDescTypeDef); + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP(pdev->tclasslist[pdev->classId].Eps[2].add, \ + USBD_EP_TYPE_INTR, CDC_CMD_PACKET_SIZE, CDC_HS_BINTERVAL, CDC_FS_BINTERVAL); + + /* Data Interface Descriptor */ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[1], 0U, 2U, 0x0A, 0U, 0U, 0U); + + if (speed == (uint8_t)USBD_SPEED_HIGH) + { + pdev->tclasslist[pdev->classId].CurrPcktSze = CDC_DATA_HS_MAX_PACKET_SIZE; + } + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP((pdev->tclasslist[pdev->classId].Eps[0].add), \ + (USBD_EP_TYPE_BULK), (pdev->tclasslist[pdev->classId].CurrPcktSze), (0U), (0U)); + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP((pdev->tclasslist[pdev->classId].Eps[1].add), \ + (USBD_EP_TYPE_BULK), (pdev->tclasslist[pdev->classId].CurrPcktSze), (0U), (0U)); + + /* Update Config Descriptor and IAD descriptor */ + ((USBD_ConfigDescTypeDef *)pConf)->bNumInterfaces += 2U; + ((USBD_ConfigDescTypeDef *)pConf)->wTotalLength = (uint16_t)(*Sze); +} +#endif /* USBD_CMPSIT_ACTIVATE_CDC == 1 */ + +#if USBD_CMPSIT_ACTIVATE_DFU == 1 +/** + * @brief USBD_CMPSIT_DFUDesc + * Configure and Append the DFU Descriptor + * @param pdev: device instance + * @param pConf: Configuration descriptor pointer + * @param Sze: pointer to the current configuration descriptor size + * @retval None + */ +static void USBD_CMPSIT_DFUDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed) +{ + static USBD_IfDescTypeDef *pIfDesc; + static USBD_DFUFuncDescTypeDef *pDFUFuncDesc; + uint32_t idx; + UNUSED(speed); + + for (idx = 0U; idx < USBD_DFU_MAX_ITF_NUM; idx++) + { + /* Append DFU Interface descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[0], (uint8_t)idx, 0U, 0xFEU, 0x01U, 0x02U, \ + (uint8_t)USBD_IDX_INTERFACE_STR + 1U + (uint8_t)idx); + } + + /* Append DFU Functional descriptor to Configuration descriptor */ + pDFUFuncDesc = ((USBD_DFUFuncDescTypeDef *)(pConf + *Sze)); + pDFUFuncDesc->bLength = (uint8_t)sizeof(USBD_DFUFuncDescTypeDef); + pDFUFuncDesc->bDescriptorType = DFU_DESCRIPTOR_TYPE; + pDFUFuncDesc->bmAttributes = USBD_DFU_BM_ATTRIBUTES; + pDFUFuncDesc->wDetachTimeout = USBD_DFU_DETACH_TIMEOUT; + pDFUFuncDesc->wTransferSze = USBD_DFU_XFER_SIZE; + pDFUFuncDesc->bcdDFUVersion = 0x011AU; + *Sze += (uint32_t)sizeof(USBD_DFUFuncDescTypeDef); + + /* Update Config Descriptor and IAD descriptor */ + ((USBD_ConfigDescTypeDef *)pConf)->bNumInterfaces += 1U; + ((USBD_ConfigDescTypeDef *)pConf)->wTotalLength = (uint16_t)(*Sze); + + UNUSED(idx); +} +#endif /* USBD_CMPSIT_ACTIVATE_DFU == 1 */ + +#if USBD_CMPSIT_ACTIVATE_CDC_ECM == 1 +/** + * @brief USBD_CMPSIT_CDC_ECMDesc + * Configure and Append the CDC_ECM Descriptor + * @param pdev: device instance + * @param pConf: Configuration descriptor pointer + * @param Sze: pointer to the current configuration descriptor size + * @retval None + */ +static void USBD_CMPSIT_CDC_ECMDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed) +{ + static USBD_IfDescTypeDef *pIfDesc; + static USBD_EpDescTypeDef *pEpDesc; + static USBD_ECMFuncDescTypeDef *pFuncDesc; + static USBD_IadDescTypeDef *pIadDesc; + + static USBD_CDCHeaderFuncDescTypeDef *pHeadDesc; + static USBD_CDCUnionFuncDescTypeDef *pUnionDesc; + +#if USBD_COMPOSITE_USE_IAD == 1 + pIadDesc = ((USBD_IadDescTypeDef *)(pConf + *Sze)); + pIadDesc->bLength = (uint8_t)sizeof(USBD_IadDescTypeDef); + pIadDesc->bDescriptorType = USB_DESC_TYPE_IAD; /* IAD descriptor */ + pIadDesc->bFirstInterface = pdev->tclasslist[pdev->classId].Ifs[0]; + pIadDesc->bInterfaceCount = 2U; /* 2 interfaces */ + pIadDesc->bFunctionClass = 0x02U; + pIadDesc->bFunctionSubClass = 0x06U; + pIadDesc->bFunctionProtocol = 0x00U; + pIadDesc->iFunction = 0U; /* String Index */ + *Sze += (uint32_t)sizeof(USBD_IadDescTypeDef); +#endif /* USBD_COMPOSITE_USE_IAD == 1 */ + + /* Append ECM Interface descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[0], 0U, 1U, 0x02U, 0x06U, 0U, 0U); + + /* Append ECM header functional descriptor to Configuration descriptor */ + pHeadDesc = ((USBD_CDCHeaderFuncDescTypeDef *)(pConf + *Sze)); + pHeadDesc->bLength = (uint8_t)sizeof(USBD_CDCHeaderFuncDescTypeDef); + pHeadDesc->bDescriptorType = USBD_FUNC_DESCRIPTOR_TYPE; + pHeadDesc->bDescriptorSubtype = 0x00U; + pHeadDesc->bcdCDC = 0x1000U; + *Sze += (uint32_t)sizeof(USBD_CDCHeaderFuncDescTypeDef); + + /* Append ECM functional descriptor to Configuration descriptor */ + pFuncDesc = ((USBD_ECMFuncDescTypeDef *)(pConf + *Sze)); + pFuncDesc->bFunctionLength = (uint8_t)sizeof(USBD_ECMFuncDescTypeDef); + pFuncDesc->bDescriptorType = USBD_FUNC_DESCRIPTOR_TYPE; + pFuncDesc->bDescriptorSubType = USBD_DESC_SUBTYPE_ACM; + pFuncDesc->iMacAddress = CDC_ECM_MAC_STRING_INDEX; + pFuncDesc->bEthernetStatistics3 = CDC_ECM_ETH_STATS_BYTE3; + pFuncDesc->bEthernetStatistics2 = CDC_ECM_ETH_STATS_BYTE2; + pFuncDesc->bEthernetStatistics1 = CDC_ECM_ETH_STATS_BYTE1; + pFuncDesc->bEthernetStatistics0 = CDC_ECM_ETH_STATS_BYTE0; + pFuncDesc->wMaxSegmentSize = CDC_ECM_ETH_MAX_SEGSZE; + pFuncDesc->bNumberMCFiltes = CDC_ECM_ETH_NBR_MACFILTERS; + pFuncDesc->bNumberPowerFiltes = CDC_ECM_ETH_NBR_PWRFILTERS; + *Sze += (uint32_t)sizeof(USBD_ECMFuncDescTypeDef); + + /* Append ECM Union functional descriptor to Configuration descriptor */ + pUnionDesc = ((USBD_CDCUnionFuncDescTypeDef *)(pConf + *Sze)); + pUnionDesc->bLength = (uint8_t)sizeof(USBD_CDCUnionFuncDescTypeDef); + pUnionDesc->bDescriptorType = 0x24U; + pUnionDesc->bDescriptorSubtype = 0x06U; + pUnionDesc->bMasterInterface = pdev->tclasslist[pdev->classId].Ifs[0]; + pUnionDesc->bSlaveInterface = pdev->tclasslist[pdev->classId].Ifs[1]; + *Sze += (uint32_t)sizeof(USBD_CDCUnionFuncDescTypeDef); + + /* Append ECM Communication IN Endpoint Descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP(pdev->tclasslist[pdev->classId].Eps[2].add, USBD_EP_TYPE_INTR, CDC_ECM_CMD_PACKET_SIZE, \ + CDC_ECM_HS_BINTERVAL, CDC_ECM_FS_BINTERVAL); + + /* Append ECM Data class interface descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[1], 0U, 2U, 0x0AU, 0U, 0U, 0U); + + if (speed == (uint8_t)USBD_SPEED_HIGH) + { + pdev->tclasslist[pdev->classId].CurrPcktSze = CDC_ECM_DATA_HS_MAX_PACKET_SIZE; + } + + /* Append ECM OUT Endpoint Descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP((pdev->tclasslist[pdev->classId].Eps[0].add), (USBD_EP_TYPE_BULK), \ + (pdev->tclasslist[pdev->classId].CurrPcktSze), (CDC_ECM_HS_BINTERVAL), (CDC_ECM_FS_BINTERVAL)); + + /* Append ECM IN Endpoint Descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP((pdev->tclasslist[pdev->classId].Eps[1].add), (USBD_EP_TYPE_BULK), \ + (pdev->tclasslist[pdev->classId].CurrPcktSze), (CDC_ECM_HS_BINTERVAL), (CDC_ECM_FS_BINTERVAL)); + + /* Update Config Descriptor and IAD descriptor */ + ((USBD_ConfigDescTypeDef *)pConf)->bNumInterfaces += 2U; + ((USBD_ConfigDescTypeDef *)pConf)->wTotalLength = (uint16_t)(*Sze); +} +#endif /* USBD_CMPSIT_ACTIVATE_CDC_ECM */ + +#if USBD_CMPSIT_ACTIVATE_AUDIO == 1 +/** + * @brief USBD_CMPSIT_AUDIODesc + * Configure and Append the AUDIO Descriptor + * @param pdev: device instance + * @param pConf: Configuration descriptor pointer + * @param Sze: pointer to the current configuration descriptor size + * @retval None + */ +static void USBD_CMPSIT_AUDIODesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed) +{ + static USBD_IfDescTypeDef *pIfDesc; + static USBD_IadDescTypeDef *pIadDesc; + UNUSED(speed); + + /* Append AUDIO Interface descriptor to Configuration descriptor */ + USBD_SpeakerIfDescTypeDef *pSpIfDesc; + USBD_SpeakerInDescTypeDef *pSpInDesc; + USBD_SpeakerFeatureDescTypeDef *pSpFDesc; + USBD_SpeakerOutDescTypeDef *pSpOutDesc; + USBD_SpeakerStreamIfDescTypeDef *pSpStrDesc; + USBD_SpeakerIIIFormatIfDescTypeDef *pSpIIIDesc; + USBD_SpeakerEndDescTypeDef *pSpEpDesc; + USBD_SpeakerEndStDescTypeDef *pSpEpStDesc; + +#if USBD_COMPOSITE_USE_IAD == 1 + pIadDesc = ((USBD_IadDescTypeDef *)(pConf + *Sze)); + pIadDesc->bLength = (uint8_t)sizeof(USBD_IadDescTypeDef); + pIadDesc->bDescriptorType = USB_DESC_TYPE_IAD; /* IAD descriptor */ + pIadDesc->bFirstInterface = pdev->tclasslist[pdev->classId].Ifs[0]; + pIadDesc->bInterfaceCount = 2U; /* 2 interfaces */ + pIadDesc->bFunctionClass = USB_DEVICE_CLASS_AUDIO; + pIadDesc->bFunctionSubClass = AUDIO_SUBCLASS_AUDIOCONTROL; + pIadDesc->bFunctionProtocol = AUDIO_PROTOCOL_UNDEFINED; + pIadDesc->iFunction = 0U; /* String Index */ + *Sze += (uint32_t)sizeof(USBD_IadDescTypeDef); +#endif /* USBD_COMPOSITE_USE_IAD == 1 */ + + /* Append AUDIO Interface descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[0], 0U, 0U, USB_DEVICE_CLASS_AUDIO, \ + AUDIO_SUBCLASS_AUDIOCONTROL, AUDIO_PROTOCOL_UNDEFINED, 0U); + + /* Append AUDIO USB Speaker Class-specific AC Interface descriptor to Configuration descriptor */ + pSpIfDesc = ((USBD_SpeakerIfDescTypeDef *)(pConf + *Sze)); + pSpIfDesc->bLength = (uint8_t)sizeof(USBD_IfDescTypeDef); + pSpIfDesc->bDescriptorType = AUDIO_INTERFACE_DESCRIPTOR_TYPE; + pSpIfDesc->bDescriptorSubtype = AUDIO_CONTROL_HEADER; + pSpIfDesc->bcdADC = 0x0100U; + pSpIfDesc->wTotalLength = 0x0027U; + pSpIfDesc->bInCollection = 0x01U; + pSpIfDesc->baInterfaceNr = 0x01U; + *Sze += (uint32_t)sizeof(USBD_IfDescTypeDef); + + /* Append USB Speaker Input Terminal Descriptor to Configuration descriptor*/ + pSpInDesc = ((USBD_SpeakerInDescTypeDef *)(pConf + *Sze)); + pSpInDesc->bLength = (uint8_t)sizeof(USBD_SpeakerInDescTypeDef); + pSpInDesc->bDescriptorType = AUDIO_INTERFACE_DESCRIPTOR_TYPE; + pSpInDesc->bDescriptorSubtype = AUDIO_CONTROL_INPUT_TERMINAL; + pSpInDesc->bTerminalID = 0x01U; + pSpInDesc->wTerminalType = 0x0101U; + pSpInDesc->bAssocTerminal = 0x00U; + pSpInDesc->bNrChannels = 0x01U; + pSpInDesc->wChannelConfig = 0x0000U; + pSpInDesc->iChannelNames = 0x00U; + pSpInDesc->iTerminal = 0x00U; + *Sze += (uint32_t)sizeof(USBD_SpeakerInDescTypeDef); + + /*Append USB Speaker Audio Feature Unit Descriptor to Configuration descriptor */ + pSpFDesc = ((USBD_SpeakerFeatureDescTypeDef *)(pConf + *Sze)); + pSpFDesc->bLength = (uint8_t)sizeof(USBD_SpeakerFeatureDescTypeDef); + pSpFDesc->bDescriptorType = AUDIO_INTERFACE_DESCRIPTOR_TYPE; + pSpFDesc->bDescriptorSubtype = AUDIO_CONTROL_FEATURE_UNIT; + pSpFDesc->bUnitID = AUDIO_OUT_STREAMING_CTRL; + pSpFDesc->bSourceID = 0x01U; + pSpFDesc->bControlSize = 0x01U; + pSpFDesc->bmaControls = AUDIO_CONTROL_MUTE; + pSpFDesc->iTerminal = 0x00U; + *Sze += (uint32_t)sizeof(USBD_SpeakerFeatureDescTypeDef); + + /*Append USB Speaker Output Terminal Descriptor to Configuration descriptor*/ + pSpOutDesc = ((USBD_SpeakerOutDescTypeDef *)(pConf + *Sze)); + pSpOutDesc->bLength = (uint8_t)sizeof(USBD_SpeakerOutDescTypeDef); + pSpOutDesc->bDescriptorType = AUDIO_INTERFACE_DESCRIPTOR_TYPE; + pSpOutDesc->bDescriptorSubtype = AUDIO_CONTROL_OUTPUT_TERMINAL; + pSpOutDesc->bTerminalID = 0x03U; + pSpOutDesc->wTerminalType = 0x0301U; + pSpOutDesc->bAssocTerminal = 0x00U; + pSpOutDesc->bSourceID = 0x02U; + pSpOutDesc->iTerminal = 0x00U; + *Sze += (uint32_t)sizeof(USBD_SpeakerOutDescTypeDef); + + /* USB Speaker Standard AS Interface Descriptor - Audio Streaming Zero Bandwidth */ + /* Interface 1, Alternate Setting 0*/ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[1], 0U, 0U, USB_DEVICE_CLASS_AUDIO, \ + AUDIO_SUBCLASS_AUDIOSTREAMING, AUDIO_PROTOCOL_UNDEFINED, 0U); + + /* USB Speaker Standard AS Interface Descriptor -Audio Streaming Operational */ + /* Interface 1, Alternate Setting 1*/ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[1], 0x01U, 0x01U, USB_DEVICE_CLASS_AUDIO, \ + AUDIO_SUBCLASS_AUDIOSTREAMING, AUDIO_PROTOCOL_UNDEFINED, 0U); + + /* USB Speaker Audio Streaming Interface Descriptor */ + pSpStrDesc = ((USBD_SpeakerStreamIfDescTypeDef *)(pConf + *Sze)); + pSpStrDesc->bLength = (uint8_t)sizeof(USBD_SpeakerStreamIfDescTypeDef); + pSpStrDesc->bDescriptorType = AUDIO_INTERFACE_DESCRIPTOR_TYPE; + pSpStrDesc->bDescriptorSubtype = AUDIO_STREAMING_GENERAL; + pSpStrDesc->bTerminalLink = 0x01U; + pSpStrDesc->bDelay = 0x01U; + pSpStrDesc->wFormatTag = 0x0001U; + *Sze += (uint32_t)sizeof(USBD_SpeakerStreamIfDescTypeDef); + + /* USB Speaker Audio Type III Format Interface Descriptor */ + pSpIIIDesc = ((USBD_SpeakerIIIFormatIfDescTypeDef *)(pConf + *Sze)); + pSpIIIDesc->bLength = (uint8_t)sizeof(USBD_SpeakerIIIFormatIfDescTypeDef); + pSpIIIDesc->bDescriptorType = AUDIO_INTERFACE_DESCRIPTOR_TYPE; + pSpIIIDesc->bDescriptorSubtype = AUDIO_STREAMING_FORMAT_TYPE; + pSpIIIDesc->bFormatType = AUDIO_FORMAT_TYPE_I; + pSpIIIDesc->bNrChannels = 0x02U; + pSpIIIDesc->bSubFrameSize = 0x02U; + pSpIIIDesc->bBitResolution = 16U; + pSpIIIDesc->bSamFreqType = 1U; + pSpIIIDesc->tSamFreq2 = 0x80U; + pSpIIIDesc->tSamFreq1 = 0xBBU; + pSpIIIDesc->tSamFreq0 = 0x00U; + *Sze += (uint32_t)sizeof(USBD_SpeakerIIIFormatIfDescTypeDef); + + /* Endpoint 1 - Standard Descriptor */ + pSpEpDesc = ((USBD_SpeakerEndDescTypeDef *)(pConf + *Sze)); + pSpEpDesc->bLength = 0x09U; + pSpEpDesc->bDescriptorType = USB_DESC_TYPE_ENDPOINT; + pSpEpDesc->bEndpointAddress = pdev->tclasslist[pdev->classId].Eps[0].add; + pSpEpDesc->bmAttributes = USBD_EP_TYPE_ISOC; + pSpEpDesc->wMaxPacketSize = (uint16_t)USBD_AUDIO_GetEpPcktSze(pdev, 0U, 0U); + pSpEpDesc->bInterval = 0x01U; + pSpEpDesc->bRefresh = 0x00U; + pSpEpDesc->bSynchAddress = 0x00U; + *Sze += 0x09U; + + /* Endpoint - Audio Streaming Descriptor*/ + pSpEpStDesc = ((USBD_SpeakerEndStDescTypeDef *)(pConf + *Sze)); + pSpEpStDesc->bLength = (uint8_t)sizeof(USBD_SpeakerEndStDescTypeDef); + pSpEpStDesc->bDescriptorType = AUDIO_ENDPOINT_DESCRIPTOR_TYPE; + pSpEpStDesc->bDescriptor = AUDIO_ENDPOINT_GENERAL; + pSpEpStDesc->bmAttributes = 0x00U; + pSpEpStDesc->bLockDelayUnits = 0x00U; + pSpEpStDesc->wLockDelay = 0x0000U; + *Sze += (uint32_t)sizeof(USBD_SpeakerEndStDescTypeDef); + + /* Update Config Descriptor and IAD descriptor */ + ((USBD_ConfigDescTypeDef *)pConf)->bNumInterfaces += 2U; + ((USBD_ConfigDescTypeDef *)pConf)->wTotalLength = (uint16_t)(*Sze); +} +#endif /* USBD_CMPSIT_ACTIVATE_AUDIO */ + +#if USBD_CMPSIT_ACTIVATE_RNDIS == 1 +/** + * @brief USBD_CMPSIT_MSCDesc + * Configure and Append the CDC_RNDIS Descriptor + * @param pdev: device instance + * @param pConf: Configuration descriptor pointer + * @param Sze: pointer to the current configuration descriptor size + * @retval None + */ +static void USBD_CMPSIT_RNDISDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed) +{ + static USBD_IfDescTypeDef *pIfDesc; + static USBD_EpDescTypeDef *pEpDesc; + static USBD_CDCHeaderFuncDescTypeDef *pHeadDesc; + static USBD_CDCCallMgmFuncDescTypeDef *pCallMgmDesc; + static USBD_CDCACMFuncDescTypeDef *pACMDesc; + static USBD_CDCUnionFuncDescTypeDef *pUnionDesc; + static USBD_IadDescTypeDef *pIadDesc; + +#if USBD_COMPOSITE_USE_IAD == 1 + pIadDesc = ((USBD_IadDescTypeDef *)(pConf + *Sze)); + pIadDesc->bLength = (uint8_t)sizeof(USBD_IadDescTypeDef); + pIadDesc->bDescriptorType = USB_DESC_TYPE_IAD; /* IAD descriptor */ + pIadDesc->bFirstInterface = pdev->tclasslist[pdev->classId].Ifs[0]; + pIadDesc->bInterfaceCount = 2U; /* 2 interfaces */ + pIadDesc->bFunctionClass = 0xE0U; + pIadDesc->bFunctionSubClass = 0x01U; + pIadDesc->bFunctionProtocol = 0x03U; + pIadDesc->iFunction = 0U; /* String Index */ + *Sze += (uint32_t)sizeof(USBD_IadDescTypeDef); +#endif /* USBD_COMPOSITE_USE_IAD == 1 */ + + /* Control Interface Descriptor */ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[0], 0U, 1U, 0x02, 0x02, 0xFF, 0U); + + /* Control interface headers */ + pHeadDesc = ((USBD_CDCHeaderFuncDescTypeDef *)(pConf + *Sze)); + /* Header Functional Descriptor*/ + pHeadDesc->bLength = (uint8_t)sizeof(USBD_CDCHeaderFuncDescTypeDef); + pHeadDesc->bDescriptorType = 0x24U; + pHeadDesc->bDescriptorSubtype = 0x00U; + pHeadDesc->bcdCDC = 0x0110U; + *Sze += (uint32_t)sizeof(USBD_CDCHeaderFuncDescTypeDef); + + /* Call Management Functional Descriptor*/ + pCallMgmDesc = ((USBD_CDCCallMgmFuncDescTypeDef *)(pConf + *Sze)); + pCallMgmDesc->bLength = (uint8_t)sizeof(USBD_CDCCallMgmFuncDescTypeDef); + pCallMgmDesc->bDescriptorType = 0x24U; + pCallMgmDesc->bDescriptorSubtype = 0x01U; + pCallMgmDesc->bmCapabilities = 0x00U; + pCallMgmDesc->bDataInterface = pdev->tclasslist[pdev->classId].Ifs[1]; + *Sze += (uint32_t)sizeof(USBD_CDCCallMgmFuncDescTypeDef); + + /* ACM Functional Descriptor*/ + pACMDesc = ((USBD_CDCACMFuncDescTypeDef *)(pConf + *Sze)); + pACMDesc->bLength = (uint8_t)sizeof(USBD_CDCACMFuncDescTypeDef); + pACMDesc->bDescriptorType = 0x24U; + pACMDesc->bDescriptorSubtype = 0x02U; + pACMDesc->bmCapabilities = 0x00U; + *Sze += (uint32_t)sizeof(USBD_CDCACMFuncDescTypeDef); + + /* Union Functional Descriptor*/ + pUnionDesc = ((USBD_CDCUnionFuncDescTypeDef *)(pConf + *Sze)); + pUnionDesc->bLength = (uint8_t)sizeof(USBD_CDCUnionFuncDescTypeDef); + pUnionDesc->bDescriptorType = 0x24U; + pUnionDesc->bDescriptorSubtype = 0x06U; + pUnionDesc->bMasterInterface = pdev->tclasslist[pdev->classId].Ifs[0]; + pUnionDesc->bSlaveInterface = pdev->tclasslist[pdev->classId].Ifs[1]; + *Sze += (uint32_t)sizeof(USBD_CDCUnionFuncDescTypeDef); + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP(pdev->tclasslist[pdev->classId].Eps[2].add, USBD_EP_TYPE_INTR, \ + CDC_RNDIS_CMD_PACKET_SIZE, CDC_RNDIS_HS_BINTERVAL, CDC_RNDIS_FS_BINTERVAL); + + /* Data Interface Descriptor */ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[1], 0U, 2U, 0x0AU, 0x00U, 0x00U, 0U); + + if (speed == (uint8_t)USBD_SPEED_HIGH) + { + pdev->tclasslist[pdev->classId].CurrPcktSze = CDC_RNDIS_DATA_HS_MAX_PACKET_SIZE; + } + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP((pdev->tclasslist[pdev->classId].Eps[0].add), (USBD_EP_TYPE_BULK), \ + (pdev->tclasslist[pdev->classId].CurrPcktSze), (0U), (0U)); + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP((pdev->tclasslist[pdev->classId].Eps[1].add), (USBD_EP_TYPE_BULK), \ + (pdev->tclasslist[pdev->classId].CurrPcktSze), (0U), (0U)); + + /* Update Config Descriptor and IAD descriptor */ + ((USBD_ConfigDescTypeDef *)pConf)->bNumInterfaces += 2U; + ((USBD_ConfigDescTypeDef *)pConf)->wTotalLength = (uint16_t)(*Sze); +} +#endif /* USBD_CMPSIT_ACTIVATE_RNDIS == 1 */ + +#if USBD_CMPSIT_ACTIVATE_CUSTOMHID == 1 +/** + * @brief USBD_CMPSIT_CUSTOMHIDDesc + * Configure and Append the MSC Descriptor + * @param pdev: device instance + * @param pConf: Configuration descriptor pointer + * @param Sze: pointer to the current configuration descriptor size + * @retval None + */ +static void USBD_CMPSIT_CUSTOMHIDDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed) +{ + static USBD_IfDescTypeDef *pIfDesc; + static USBD_EpDescTypeDef *pEpDesc; + static USBD_DescTypeDef *pDesc; + + /* Control Interface Descriptor */ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[0], 0U, 2U, 3U, 0U, 0U, 0U); + + /* Descriptor of CUSTOM_HID */ + pDesc = ((USBD_DescTypeDef *)((uint32_t)pConf + *Sze)); + pDesc->bLength = 0x09U; + pDesc->bDescriptorTypeCHID = CUSTOM_HID_DESCRIPTOR_TYPE; + pDesc->bcdCUSTOM_HID = 0x0111U; + pDesc->bCountryCode = 0x00U; + pDesc->bNumDescriptors = 0x01U; + pDesc->bDescriptorType = 0x22U; + pDesc->wItemLength = USBD_CUSTOM_HID_REPORT_DESC_SIZE; + *Sze += (uint32_t)sizeof(USBD_DescTypeDef); + + /* Descriptor of Custom HID endpoints */ + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP(pdev->tclasslist[pdev->classId].Eps[0].add, \ + USBD_EP_TYPE_INTR, CUSTOM_HID_EPIN_SIZE, CUSTOM_HID_HS_BINTERVAL, CUSTOM_HID_FS_BINTERVAL); + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP(pdev->tclasslist[pdev->classId].Eps[1].add, \ + USBD_EP_TYPE_INTR, CUSTOM_HID_EPIN_SIZE, CUSTOM_HID_HS_BINTERVAL, CUSTOM_HID_FS_BINTERVAL); + + /* Update Config Descriptor and IAD descriptor */ + ((USBD_ConfigDescTypeDef *)pConf)->bNumInterfaces += 1U; + ((USBD_ConfigDescTypeDef *)pConf)->wTotalLength = (uint16_t)(*Sze); +} +#endif /* USBD_CMPSIT_ACTIVATE_CUSTOMHID == 1U */ + +#if USBD_CMPSIT_ACTIVATE_VIDEO == 1 +/** + * @brief USBD_CMPSIT_VIDEODesc + * Configure and Append the VIDEO Descriptor + * @param pdev: device instance + * @param pConf: Configuration descriptor pointer + * @param Sze: pointer to the current configuration descriptor size + * @retval None + */ +static void USBD_CMPSIT_VIDEODesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed) +{ +#ifdef USBD_UVC_FORMAT_UNCOMPRESSED + __ALIGN_BEGIN static uint8_t usbd_uvc_guid[16] __ALIGN_END = {DBVAL(UVC_UNCOMPRESSED_GUID), 0x00, 0x00, 0x10, + 0x00, 0x80, 0x00, 0x00, 0xAA, 0x00, 0x38, 0x9B, 0x71 + }; +#endif /* USBD_UVC_FORMAT_UNCOMPRESSED */ + static USBD_IfDescTypeDef *pIfDesc; + static USBD_IadDescTypeDef *pIadDesc; + + /* Append AUDIO Interface descriptor to Configuration descriptor */ + USBD_specificVCInDescTypeDef *pSVCInDesc; + USBD_InputTerminalDescTypeDef *pInTerDesc; + USBD_OutputTerminalDescTypeDef *pOuTerDesc; + USBD_ClassSpecificVsHeaderDescTypeDef *pSpHeaDesc; + USBD_PayloadFormatDescTypeDef *pPayForDesc; +#ifdef USBD_UVC_FORMAT_UNCOMPRESSED + USBD_ColorMatchingDescTypeDef *pColMaDesc; +#endif /* USBD_UVC_FORMAT_UNCOMPRESSED */ + USBD_StandardVCDataEPDescTypeDef *pSVCDEP; + USBD_VIDEO_VSFrameDescTypeDef *pClassSpecVS; + +#if USBD_COMPOSITE_USE_IAD == 1 + pIadDesc = ((USBD_IadDescTypeDef *)(pConf + *Sze)); + pIadDesc->bLength = (uint8_t)sizeof(USBD_IadDescTypeDef); + pIadDesc->bDescriptorType = USB_DESC_TYPE_IAD; /* IAD descriptor */ + pIadDesc->bFirstInterface = pdev->tclasslist[pdev->classId].Ifs[0]; + pIadDesc->bInterfaceCount = 2U; /* 2 interfaces */ + pIadDesc->bFunctionClass = UVC_CC_VIDEO; + pIadDesc->bFunctionSubClass = SC_VIDEO_INTERFACE_COLLECTION; + pIadDesc->bFunctionProtocol = PC_PROTOCOL_UNDEFINED; + pIadDesc->iFunction = 0U; /* String Index */ + *Sze += (uint32_t)sizeof(USBD_IadDescTypeDef); +#endif /* USBD_COMPOSITE_USE_IAD == 1 */ + + /* Append VIDEO Interface descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[0], 0U, 0U, UVC_CC_VIDEO, 1U, PC_PROTOCOL_UNDEFINED, 0U); + + /* Append Class-specific VC Interface Descriptor to Configuration descriptor*/ + pSVCInDesc = ((USBD_specificVCInDescTypeDef *)(pConf + *Sze)); + pSVCInDesc->bLength = (uint8_t)sizeof(USBD_specificVCInDescTypeDef); + pSVCInDesc->bDescriptorType = CS_INTERFACE; + pSVCInDesc->bDescriptorSubtype = VC_HEADER; + pSVCInDesc->bcdUVC = UVC_VERSION; + pSVCInDesc->wTotalLength = 0x001EU; + pSVCInDesc->dwClockFrequency = 0x02DC6C00U; + pSVCInDesc->baInterfaceNr = 0x01U; + pSVCInDesc->iTerminal = 0x01U; + *Sze += (uint32_t)sizeof(USBD_specificVCInDescTypeDef); + + /*Append Input Terminal Descriptor to Configuration descriptor */ + pInTerDesc = ((USBD_InputTerminalDescTypeDef *)(pConf + *Sze)); + pInTerDesc->bLength = (uint8_t)sizeof(USBD_InputTerminalDescTypeDef); + pInTerDesc->bDescriptorType = CS_INTERFACE; + pInTerDesc->bDescriptorSubtype = VC_INPUT_TERMINAL; + pInTerDesc->bTerminalID = 0x01U; + pInTerDesc->wTerminalType = ITT_VENDOR_SPECIFIC; + pInTerDesc->bAssocTerminal = 0x00U; + pInTerDesc->iTerminal = 0x00U; + *Sze += (uint32_t)sizeof(USBD_InputTerminalDescTypeDef); + + /* Append Output Terminal Descriptor to Configuration descriptor */ + pOuTerDesc = ((USBD_OutputTerminalDescTypeDef *)(pConf + *Sze)); + pOuTerDesc->bLength = (uint8_t)sizeof(USBD_OutputTerminalDescTypeDef); + pOuTerDesc->bDescriptorType = CS_INTERFACE; + pOuTerDesc->bDescriptorSubtype = VC_OUTPUT_TERMINAL; + pOuTerDesc->bTerminalID = 0x02U; + pOuTerDesc->wTerminalType = TT_STREAMING; + pOuTerDesc->bAssocTerminal = 0x00U; + pOuTerDesc->bSourceID = 0x01U; + pOuTerDesc->iTerminal = 0x00U; + *Sze += (uint32_t)sizeof(USBD_OutputTerminalDescTypeDef); + + /* Standard VS (Video Streaming) Interface Descriptor */ + /* Interface 1, Alternate Setting 0 = Zero Bandwidth*/ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[1], 0U, 0U, UVC_CC_VIDEO, \ + SC_VIDEOSTREAMING, PC_PROTOCOL_UNDEFINED, 0U); + + /* Append Class-specific VS Header Descriptor (Input) to Configuration descriptor */ + pSpHeaDesc = ((USBD_ClassSpecificVsHeaderDescTypeDef *)(pConf + *Sze)); + pSpHeaDesc->bLength = (uint8_t)sizeof(USBD_ClassSpecificVsHeaderDescTypeDef); + pSpHeaDesc->bDescriptorType = CS_INTERFACE; + pSpHeaDesc->bDescriptorSubtype = VS_INPUT_HEADER; + pSpHeaDesc->bNumFormats = 0x4D01U; + pSpHeaDesc->bVideoControlSize = 0x00U; + pSpHeaDesc->bEndPointAddress = UVC_IN_EP; + pSpHeaDesc->bmInfo = 0x00U; + pSpHeaDesc->bTerminalLink = 0x02U; + pSpHeaDesc->bStillCaptureMethod = 0x00U; + pSpHeaDesc->bTriggerSupport = 0x00U; + pSpHeaDesc->bTriggerUsage = 0x00U; + pSpHeaDesc->bControlSize = 0x01U; + pSpHeaDesc->bmaControls = 0x00U; + *Sze += (uint32_t)sizeof(USBD_ClassSpecificVsHeaderDescTypeDef); + + /* Append Payload Format Descriptor to Configuration descriptor */ + pPayForDesc = ((USBD_PayloadFormatDescTypeDef *)(pConf + *Sze)); + pPayForDesc->bLength = (uint8_t)sizeof(USBD_PayloadFormatDescTypeDef); + pPayForDesc->bDescriptorType = CS_INTERFACE; + pPayForDesc->bDescriptorSubType = VS_FORMAT_SUBTYPE; + pPayForDesc->bFormatIndex = 0x01U; + pPayForDesc->bNumFrameDescriptor = 0x01U; +#ifdef USBD_UVC_FORMAT_UNCOMPRESSED + (void)USBD_memcpy(pPayForDesc->pGiudFormat, usbd_uvc_guid, 16); + pPayForDesc->bBitsPerPixel = UVC_BITS_PER_PIXEL; +#else + pPayForDesc->bmFlags = 0x01U; +#endif /* USBD_UVC_FORMAT_UNCOMPRESSED */ + pPayForDesc->bDefaultFrameIndex = 0x01U; + pPayForDesc->bAspectRatioX = 0x00U; + pPayForDesc->bAspectRatioY = 0x00U; + pPayForDesc->bInterlaceFlags = 0x00U; + pPayForDesc->bCopyProtect = 0x00U; + *Sze += (uint32_t)sizeof(USBD_PayloadFormatDescTypeDef); + + /* Append Class-specific VS (Video Streaming) Frame Descriptor to Configuration descriptor */ + pClassSpecVS = ((USBD_VIDEO_VSFrameDescTypeDef *)(pConf + *Sze)); + pClassSpecVS->bLength = (uint8_t)sizeof(USBD_VIDEO_VSFrameDescTypeDef); + pClassSpecVS->bDescriptorType = CS_INTERFACE; + pClassSpecVS->bDescriptorSubType = VS_FRAME_SUBTYPE; + pClassSpecVS->bFrameIndex = 0x01U; + +#ifdef USBD_UVC_FORMAT_UNCOMPRESSED + pClassSpecVS->bmCapabilities = 0x00U; +#else + pClassSpecVS->bmCapabilities = 0x02U; +#endif /* USBD_UVC_FORMAT_UNCOMPRESSED */ + + pClassSpecVS->wWidth = UVC_WIDTH; + pClassSpecVS->wHeight = UVC_HEIGHT; + + if (speed == (uint8_t)USBD_SPEED_HIGH) + { + pClassSpecVS->dwMinBitRate = UVC_MIN_BIT_RATE(UVC_CAM_FPS_HS); + pClassSpecVS->dwMaxBitRate = UVC_MAX_BIT_RATE(UVC_CAM_FPS_HS); + pClassSpecVS->dwDefaultFrameInterval = UVC_INTERVAL(UVC_CAM_FPS_HS); + pClassSpecVS->dwMinFrameInterval = UVC_INTERVAL(UVC_CAM_FPS_HS); + } + else + { + pClassSpecVS->dwMinBitRate = UVC_MIN_BIT_RATE(UVC_CAM_FPS_FS); + pClassSpecVS->dwMaxBitRate = UVC_MAX_BIT_RATE(UVC_CAM_FPS_FS); + pClassSpecVS->dwDefaultFrameInterval = UVC_INTERVAL(UVC_CAM_FPS_FS); + pClassSpecVS->dwMinFrameInterval = UVC_INTERVAL(UVC_CAM_FPS_FS); + } + + pClassSpecVS->dwMaxVideoFrameBufSize = UVC_MAX_FRAME_SIZE; + pClassSpecVS->bFrameIntervalType = 0x01U; + + *Sze += (uint32_t)sizeof(USBD_VIDEO_VSFrameDescTypeDef); + +#ifdef USBD_UVC_FORMAT_UNCOMPRESSED + /* Append Color Matching Descriptor to Configuration descriptor */ + pColMaDesc = ((USBD_ColorMatchingDescTypeDef *)(pConf + *Sze)); + pColMaDesc->bLength = (uint8_t)sizeof(USBD_ColorMatchingDescTypeDef); + pColMaDesc->bDescriptorType = CS_INTERFACE; + pColMaDesc->bDescriptorSubType = VS_COLORFORMAT; + pColMaDesc->bColorPrimarie = UVC_COLOR_PRIMARIE; + pColMaDesc->bTransferCharacteristics = UVC_TFR_CHARACTERISTICS; + pColMaDesc->bMatrixCoefficients = UVC_MATRIX_COEFFICIENTS; + *Sze += (uint32_t)sizeof(USBD_ColorMatchingDescTypeDef); +#endif /* USBD_UVC_FORMAT_UNCOMPRESSED */ + + /* USB Standard VS Interface Descriptor - data transfer mode */ + /* Interface 1, Alternate Setting 1*/ + __USBD_CMPSIT_SET_IF(1U, 1U, 1U, UVC_CC_VIDEO, SC_VIDEOSTREAMING, PC_PROTOCOL_UNDEFINED, 0U); + + /* Standard VS (Video Streaming) data Endpoint */ + pSVCDEP = ((USBD_StandardVCDataEPDescTypeDef *)(pConf + *Sze)); + pSVCDEP->bLength = (uint8_t)sizeof(USBD_StandardVCDataEPDescTypeDef); + pSVCDEP->bDescriptorType = USB_DESC_TYPE_ENDPOINT; + pSVCDEP->bEndpointAddress = UVC_IN_EP; + pSVCDEP->bmAttributes = 0x05U; + pSVCDEP->bInterval = 0x01U; + + if (speed == (uint8_t)USBD_SPEED_HIGH) + { + pSVCDEP->wMaxPacketSize = UVC_ISO_HS_MPS; + } + else + { + pSVCDEP->wMaxPacketSize = UVC_ISO_FS_MPS; + } + + *Sze += (uint32_t)sizeof(USBD_StandardVCDataEPDescTypeDef); + + /* Update Config Descriptor and IAD descriptor */ + ((USBD_ConfigDescTypeDef *)pConf)->bNumInterfaces += 2U; + ((USBD_ConfigDescTypeDef *)pConf)->wTotalLength = (uint16_t)(*Sze); +} +#endif /* USBD_CMPSIT_ACTIVATE_VIDEO == 1 */ + +#if USBD_CMPSIT_ACTIVATE_PRINTER == 1 +/** + * @brief USBD_CMPSIT_PRINTERDesc + * Configure and Append the PRINTER Descriptor + * @param pdev: device instance + * @param pConf: Configuration descriptor pointer + * @param Sze: pointer to the current configuration descriptor size + * @retval None + */ +static void USBD_CMPSIT_PRNTDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed) +{ + static USBD_IfDescTypeDef *pIfDesc; + static USBD_EpDescTypeDef *pEpDesc; + + /* Control Interface Descriptor */ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[0], 0U, 0x02, 0x07, 0x01U, USB_PRNT_BIDIRECTIONAL, 0U); + + if (speed == (uint8_t)USBD_SPEED_HIGH) + { + pdev->tclasslist[pdev->classId].CurrPcktSze = PRNT_DATA_HS_MAX_PACKET_SIZE; + } + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP((pdev->tclasslist[pdev->classId].Eps[1].add), \ + (USBD_EP_TYPE_BULK), (pdev->tclasslist[pdev->classId].CurrPcktSze), (0U), (0U)); + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP((pdev->tclasslist[pdev->classId].Eps[0].add), \ + (USBD_EP_TYPE_BULK), (pdev->tclasslist[pdev->classId].CurrPcktSze), (0U), (0U)); + + /* Update Config Descriptor and IAD descriptor */ + ((USBD_ConfigDescTypeDef *)pConf)->bNumInterfaces += 1U; + ((USBD_ConfigDescTypeDef *)pConf)->wTotalLength = (uint16_t)(*Sze); +} +#endif /* USBD_CMPSIT_ACTIVATE_PRINTER == 1 */ + +#if USBD_CMPSIT_ACTIVATE_CCID == 1 +/** + * @brief USBD_CMPSIT_CCIDDesc + * Configure and Append the CCID Descriptor + * @param pdev: device instance + * @param pConf: Configuration descriptor pointer + * @param Sze: pointer to the current configuration descriptor size + * @retval None + */ +static void USBD_CMPSIT_CCIDDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed) +{ + static USBD_IfDescTypeDef *pIfDesc; + static USBD_EpDescTypeDef *pEpDesc; + static USBD_CCID_DescTypeDef *pDesc; + + /* Control Interface Descriptor */ + __USBD_CMPSIT_SET_IF(pdev->tclasslist[pdev->classId].Ifs[0], 0U, 0x03, 0x0BU, 0U, 0U, 0U); + + /* Control interface headers */ + pDesc = ((USBD_CCID_DescTypeDef *)((uint32_t)pConf + *Sze)); + + /* Device Descriptor */ + pDesc->bLength = 0x36U; + pDesc->bDescriptorType = 0x21U; + pDesc->bcdCCID = 0x0110U; + pDesc->bMaxSlotIndex = 0x00U; + pDesc->bVoltageSupport = CCID_VOLTAGE_SUPP; + pDesc->dwProtocols = USBD_CCID_PROTOCOL; + pDesc->dwDefaultClock = USBD_CCID_DEFAULT_CLOCK_FREQ; + pDesc->dwMaximumClock = USBD_CCID_MAX_CLOCK_FREQ; + pDesc->bNumClockSupported = 0x00U; + pDesc->dwDataRate = USBD_CCID_DEFAULT_DATA_RATE; + pDesc->dwMaxDataRate = USBD_CCID_MAX_DATA_RATE; + pDesc->bNumDataRatesSupported = 0x35U; + pDesc->dwMaxIFSD = USBD_CCID_MAX_INF_FIELD_SIZE; + pDesc->dwSynchProtocols = 0U; + pDesc->dwMechanical = 0U; + pDesc->dwFeatures = 0x000104BAU; + pDesc->dwMaxCCIDMessageLength = CCID_MAX_BLOCK_SIZE_HEADER; + pDesc->bClassGetResponse = 0U; + pDesc->bClassEnvelope = 0U; + pDesc->wLcdLayout = 0U; + pDesc->bPINSupport = 0x03U; + pDesc->bMaxCCIDBusySlots = 0x01U; + + *Sze += (uint32_t)sizeof(USBD_CCID_DescTypeDef); + + if (speed == (uint8_t)USBD_SPEED_HIGH) + { + pdev->tclasslist[pdev->classId].CurrPcktSze = CCID_DATA_HS_MAX_PACKET_SIZE; + } + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP((pdev->tclasslist[pdev->classId].Eps[0].add), \ + (USBD_EP_TYPE_BULK), (pdev->tclasslist[pdev->classId].CurrPcktSze), (0U), (0U)); + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP((pdev->tclasslist[pdev->classId].Eps[1].add), \ + (USBD_EP_TYPE_BULK), (pdev->tclasslist[pdev->classId].CurrPcktSze), (0U), (0U)); + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP(pdev->tclasslist[pdev->classId].Eps[2].add, \ + USBD_EP_TYPE_INTR, CCID_CMD_PACKET_SIZE, CCID_CMD_HS_BINTERVAL, CCID_CMD_FS_BINTERVAL); + + /* Update Config Descriptor and IAD descriptor */ + ((USBD_ConfigDescTypeDef *)pConf)->bNumInterfaces += 1U; + ((USBD_ConfigDescTypeDef *)pConf)->wTotalLength = (uint16_t)(*Sze); +} +#endif /* USBD_CMPSIT_ACTIVATE_CCID == 1 */ + +#if USBD_CMPSIT_ACTIVATE_MTP == 1 +/** + * @brief USBD_CMPSIT_MTPDesc + * Configure and Append the MTP Descriptor + * @param pdev: device instance + * @param pConf: Configuration descriptor pointer + * @param Sze: pointer to the current configuration descriptor size + * @retval None + */ +static void USBD_CMPSIT_MTPDesc(USBD_HandleTypeDef *pdev, uint32_t pConf, __IO uint32_t *Sze, uint8_t speed) +{ + USBD_IfDescTypeDef *pIfDesc; + USBD_EpDescTypeDef *pEpDesc; + + /* Append MTP Interface descriptor */ + __USBD_CMPSIT_SET_IF((pdev->tclasslist[pdev->classId].Ifs[0]), (0U), \ + (uint8_t)(pdev->tclasslist[pdev->classId].NumEps), USB_MTP_INTRERFACE_CLASS, \ + USB_MTP_INTRERFACE_SUB_CLASS, USB_MTP_INTRERFACE_PROTOCOL, (0U)); + + if (speed == (uint8_t)USBD_SPEED_HIGH) + { + pdev->tclasslist[pdev->classId].CurrPcktSze = MTP_DATA_MAX_HS_PACKET_SIZE; + } + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP((pdev->tclasslist[pdev->classId].Eps[0].add), (USBD_EP_TYPE_BULK), \ + (pdev->tclasslist[pdev->classId].CurrPcktSze), (0U), (0U)); + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP((pdev->tclasslist[pdev->classId].Eps[1].add), (USBD_EP_TYPE_BULK), \ + (pdev->tclasslist[pdev->classId].CurrPcktSze), (0U), (0U)); + + /* Append Endpoint descriptor to Configuration descriptor */ + __USBD_CMPSIT_SET_EP(pdev->tclasslist[pdev->classId].Eps[2].add, \ + USBD_EP_TYPE_INTR, MTP_CMD_PACKET_SIZE, MTP_HS_BINTERVAL, MTP_FS_BINTERVAL); + + /* Update Config Descriptor and IAD descriptor */ + ((USBD_ConfigDescTypeDef *)pConf)->bNumInterfaces += 1U; + ((USBD_ConfigDescTypeDef *)pConf)->wTotalLength = (uint16_t)(*Sze); +} +#endif /* USBD_CMPSIT_ACTIVATE_MTP == 1 */ + +/** + * @brief USBD_CMPSIT_SetClassID + * Find and set the class ID relative to selected class type and instance + * @param pdev: device instance + * @param Class: Class type, can be CLASS_TYPE_NONE if requested to find class from setup request + * @param Instance: Instance number of the class (0 if first/unique instance, >0 otherwise) + * @retval The Class ID, The pdev->classId is set with the value of the selected class ID. + */ +uint32_t USBD_CMPSIT_SetClassID(USBD_HandleTypeDef *pdev, USBD_CompositeClassTypeDef Class, uint32_t Instance) +{ + uint32_t idx; + uint32_t inst = 0U; + + /* Unroll all already activated classes */ + for (idx = 0U; idx < pdev->NumClasses; idx++) + { + /* Check if the class correspond to the requested type and if it is active */ + if (((USBD_CompositeClassTypeDef)(pdev->tclasslist[idx].ClassType) == Class) && + ((pdev->tclasslist[idx].Active) == 1U)) + { + if (inst == Instance) + { + /* Set the new class ID */ + pdev->classId = idx; + + /* Return the class ID value */ + return (idx); + } + else + { + /* Increment instance index and look for next instance */ + inst++; + } + } + } + + /* No class found, return 0xFF */ + return 0xFFU; +} + +/** + * @brief USBD_CMPSIT_GetClassID + * Returns the class ID relative to selected class type and instance + * @param pdev: device instance + * @param Class: Class type, can be CLASS_TYPE_NONE if requested to find class from setup request + * @param Instance: Instance number of the class (0 if first/unique instance, >0 otherwise) + * @retval The Class ID (this function does not set the pdev->classId field. + */ +uint32_t USBD_CMPSIT_GetClassID(USBD_HandleTypeDef *pdev, USBD_CompositeClassTypeDef Class, uint32_t Instance) +{ + uint32_t idx; + uint32_t inst = 0U; + + /* Unroll all already activated classes */ + for (idx = 0U; idx < pdev->NumClasses; idx++) + { + /* Check if the class correspond to the requested type and if it is active */ + if (((USBD_CompositeClassTypeDef)(pdev->tclasslist[idx].ClassType) == Class) && + ((pdev->tclasslist[idx].Active) == 1U)) + { + if (inst == Instance) + { + /* Return the class ID value */ + return (idx); + } + else + { + /* Increment instance index and look for next instance */ + inst++; + } + } + } + + /* No class found, return 0xFF */ + return 0xFFU; +} + +/** + * @brief USBD_CMPST_ClearConfDesc + * Reset the configuration descriptor + * @param pdev: device instance (reserved for future use) + * @retval Status. + */ +uint8_t USBD_CMPST_ClearConfDesc(USBD_HandleTypeDef *pdev) +{ + UNUSED(pdev); + + /* Reset the configuration descriptor pointer to default value and its size to zero */ + pCmpstFSConfDesc = USBD_CMPSIT_FSCfgDesc; + CurrFSConfDescSz = 0U; + +#ifdef USE_USB_HS + pCmpstHSConfDesc = USBD_CMPSIT_HSCfgDesc; + CurrHSConfDescSz = 0U; +#endif /* USE_USB_HS */ + + /* All done, can't fail */ + return (uint8_t)USBD_OK; +} + +#endif /* USE_USBD_COMPOSITE */ + +/** + * @} + */ + + +/** + * @} + */ + + +/** + * @} + */ + + diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Inc/usbd_customhid.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Inc/usbd_customhid.h index da016850..2f4c6344 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Inc/usbd_customhid.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Inc/usbd_customhid.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -41,17 +40,21 @@ extern "C" { /** @defgroup USBD_CUSTOM_HID_Exported_Defines * @{ */ +#ifndef CUSTOM_HID_EPIN_ADDR #define CUSTOM_HID_EPIN_ADDR 0x81U +#endif /* CUSTOM_HID_EPIN_ADDR */ #ifndef CUSTOM_HID_EPIN_SIZE #define CUSTOM_HID_EPIN_SIZE 0x02U -#endif +#endif /* CUSTOM_HID_EPIN_SIZE */ +#ifndef CUSTOM_HID_EPOUT_ADDR #define CUSTOM_HID_EPOUT_ADDR 0x01U +#endif /* CUSTOM_HID_EPOUT_ADDR */ #ifndef CUSTOM_HID_EPOUT_SIZE #define CUSTOM_HID_EPOUT_SIZE 0x02U -#endif +#endif /* CUSTOM_HID_EPOUT_SIZE*/ #define USB_CUSTOM_HID_CONFIG_DESC_SIZ 41U #define USB_CUSTOM_HID_DESC_SIZ 9U @@ -103,7 +106,12 @@ typedef struct _USBD_CUSTOM_HID_Itf int8_t (* Init)(void); int8_t (* DeInit)(void); int8_t (* OutEvent)(uint8_t event_idx, uint8_t state); - +#ifdef USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED + int8_t (* CtrlReqComplete)(uint8_t request, uint16_t wLength); +#endif /* USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED */ +#ifdef USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED + uint8_t *(* GetReport)(uint16_t *ReportLength); +#endif /* USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED */ } USBD_CUSTOM_HID_ItfTypeDef; typedef struct @@ -115,6 +123,23 @@ typedef struct uint32_t IsReportAvailable; CUSTOM_HID_StateTypeDef state; } USBD_CUSTOM_HID_HandleTypeDef; + +/* + * HID Class specification version 1.1 + * 6.2.1 HID Descriptor + */ + +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorTypeCHID; + uint16_t bcdCUSTOM_HID; + uint8_t bCountryCode; + uint8_t bNumDescriptors; + uint8_t bDescriptorType; + uint16_t wItemLength; +} __PACKED USBD_DescTypeDef; + /** * @} */ @@ -142,9 +167,13 @@ extern USBD_ClassTypeDef USBD_CUSTOM_HID; /** @defgroup USB_CORE_Exported_Functions * @{ */ +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CUSTOM_HID_SendReport(USBD_HandleTypeDef *pdev, + uint8_t *report, uint16_t len, uint8_t ClassId); +#else uint8_t USBD_CUSTOM_HID_SendReport(USBD_HandleTypeDef *pdev, uint8_t *report, uint16_t len); - +#endif /* USE_USBD_COMPOSITE */ uint8_t USBD_CUSTOM_HID_ReceivePacket(USBD_HandleTypeDef *pdev); uint8_t USBD_CUSTOM_HID_RegisterInterface(USBD_HandleTypeDef *pdev, @@ -167,4 +196,3 @@ uint8_t USBD_CUSTOM_HID_RegisterInterface(USBD_HandleTypeDef *pdev, * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Inc/usbd_customhid_if_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Inc/usbd_customhid_if_template.h index e3906938..e2a04af9 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Inc/usbd_customhid_if_template.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Inc/usbd_customhid_if_template.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -40,4 +39,3 @@ extern USBD_CUSTOM_HID_ItfTypeDef USBD_CustomHID_template_fops; #endif /* __USBD_CUSTOMHID_IF_TEMPLATE_H */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Src/usbd_customhid.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Src/usbd_customhid.c index 2fbc0570..6add68a8 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Src/usbd_customhid.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Src/usbd_customhid.c @@ -4,6 +4,17 @@ * @author MCD Application Team * @brief This file provides the CUSTOM_HID core functions. * + ****************************************************************************** + * @attention + * + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** * @verbatim * * =================================================================== @@ -24,17 +35,6 @@ * @endverbatim * ****************************************************************************** - * @attention - * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> - * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** */ /* BSPDependencies @@ -45,6 +45,7 @@ EndBSPDependencies */ /* Includes ------------------------------------------------------------------*/ #include "usbd_customhid.h" #include "usbd_ctlreq.h" +#include "usbd_def.h" /** @addtogroup STM32_USB_DEVICE_LIBRARY @@ -91,12 +92,12 @@ static uint8_t USBD_CUSTOM_HID_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqType static uint8_t USBD_CUSTOM_HID_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum); static uint8_t USBD_CUSTOM_HID_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum); static uint8_t USBD_CUSTOM_HID_EP0_RxReady(USBD_HandleTypeDef *pdev); - +#ifndef USE_USBD_COMPOSITE static uint8_t *USBD_CUSTOM_HID_GetFSCfgDesc(uint16_t *length); static uint8_t *USBD_CUSTOM_HID_GetHSCfgDesc(uint16_t *length); static uint8_t *USBD_CUSTOM_HID_GetOtherSpeedCfgDesc(uint16_t *length); static uint8_t *USBD_CUSTOM_HID_GetDeviceQualifierDesc(uint16_t *length); - +#endif /* USE_USBD_COMPOSITE */ /** * @} */ @@ -117,27 +118,36 @@ USBD_ClassTypeDef USBD_CUSTOM_HID = NULL, /*SOF */ NULL, NULL, +#ifdef USE_USBD_COMPOSITE + NULL, + NULL, + NULL, + NULL, +#else USBD_CUSTOM_HID_GetHSCfgDesc, USBD_CUSTOM_HID_GetFSCfgDesc, USBD_CUSTOM_HID_GetOtherSpeedCfgDesc, USBD_CUSTOM_HID_GetDeviceQualifierDesc, +#endif /* USE_USBD_COMPOSITE */ }; +#ifndef USE_USBD_COMPOSITE /* USB CUSTOM_HID device FS Configuration Descriptor */ -__ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_CfgFSDesc[USB_CUSTOM_HID_CONFIG_DESC_SIZ] __ALIGN_END = +__ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_CfgDesc[USB_CUSTOM_HID_CONFIG_DESC_SIZ] __ALIGN_END = { 0x09, /* bLength: Configuration Descriptor size */ USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ - USB_CUSTOM_HID_CONFIG_DESC_SIZ, /* wTotalLength: Bytes returned */ - 0x00, + LOBYTE(USB_CUSTOM_HID_CONFIG_DESC_SIZ), /* wTotalLength: Bytes returned */ + HIBYTE(USB_CUSTOM_HID_CONFIG_DESC_SIZ), 0x01, /* bNumInterfaces: 1 interface */ 0x01, /* bConfigurationValue: Configuration value */ - 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ + 0x00, /* iConfiguration: Index of string descriptor + describing the configuration */ #if (USBD_SELF_POWERED == 1U) 0xC0, /* bmAttributes: Bus Powered according to user configuration */ #else 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif +#endif /* USBD_SELF_POWERED */ USBD_MAX_POWER, /* MaxPower (mA) */ /************** Descriptor of CUSTOM HID interface ****************/ @@ -160,8 +170,8 @@ __ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_CfgFSDesc[USB_CUSTOM_HID_CONFIG_DES 0x00, /* bCountryCode: Hardware target country */ 0x01, /* bNumDescriptors: Number of CUSTOM_HID class descriptors to follow */ 0x22, /* bDescriptorType */ - USBD_CUSTOM_HID_REPORT_DESC_SIZE, /* wItemLength: Total length of Report descriptor */ - 0x00, + LOBYTE(USBD_CUSTOM_HID_REPORT_DESC_SIZE), /* wItemLength: Total length of Report descriptor */ + HIBYTE(USBD_CUSTOM_HID_REPORT_DESC_SIZE), /******************** Descriptor of Custom HID endpoints ********************/ /* 27 */ 0x07, /* bLength: Endpoint Descriptor size */ @@ -169,8 +179,8 @@ __ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_CfgFSDesc[USB_CUSTOM_HID_CONFIG_DES CUSTOM_HID_EPIN_ADDR, /* bEndpointAddress: Endpoint Address (IN) */ 0x03, /* bmAttributes: Interrupt endpoint */ - CUSTOM_HID_EPIN_SIZE, /* wMaxPacketSize: 2 Bytes max */ - 0x00, + LOBYTE(CUSTOM_HID_EPIN_SIZE), /* wMaxPacketSize: 2 Bytes max */ + HIBYTE(CUSTOM_HID_EPIN_SIZE), CUSTOM_HID_FS_BINTERVAL, /* bInterval: Polling Interval */ /* 34 */ @@ -178,8 +188,8 @@ __ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_CfgFSDesc[USB_CUSTOM_HID_CONFIG_DES USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: */ CUSTOM_HID_EPOUT_ADDR, /* bEndpointAddress: Endpoint Address (OUT) */ 0x03, /* bmAttributes: Interrupt endpoint */ - CUSTOM_HID_EPOUT_SIZE, /* wMaxPacketSize: 2 Bytes max */ - 0x00, + LOBYTE(CUSTOM_HID_EPOUT_SIZE), /* wMaxPacketSize: 2 Bytes max */ + HIBYTE(CUSTOM_HID_EPOUT_SIZE), CUSTOM_HID_FS_BINTERVAL, /* bInterval: Polling Interval */ /* 41 */ }; @@ -189,8 +199,8 @@ __ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_CfgHSDesc[USB_CUSTOM_HID_CONFIG_DES { 0x09, /* bLength: Configuration Descriptor size */ USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ - USB_CUSTOM_HID_CONFIG_DESC_SIZ, /* wTotalLength: Bytes returned */ - 0x00, + LOBYTE(USB_CUSTOM_HID_CONFIG_DESC_SIZ), /* wTotalLength: Bytes returned */ + HIBYTE(USB_CUSTOM_HID_CONFIG_DESC_SIZ), 0x01, /* bNumInterfaces: 1 interface */ 0x01, /* bConfigurationValue: Configuration value */ 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ @@ -221,8 +231,8 @@ __ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_CfgHSDesc[USB_CUSTOM_HID_CONFIG_DES 0x00, /* bCountryCode: Hardware target country */ 0x01, /* bNumDescriptors: Number of CUSTOM_HID class descriptors to follow */ 0x22, /* bDescriptorType */ - USBD_CUSTOM_HID_REPORT_DESC_SIZE, /* wItemLength: Total length of Report descriptor */ - 0x00, + LOBYTE(USBD_CUSTOM_HID_REPORT_DESC_SIZE), /* wItemLength: Total length of Report descriptor */ + HIBYTE(USBD_CUSTOM_HID_REPORT_DESC_SIZE), /******************** Descriptor of Custom HID endpoints ********************/ /* 27 */ 0x07, /* bLength: Endpoint Descriptor size */ @@ -230,8 +240,8 @@ __ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_CfgHSDesc[USB_CUSTOM_HID_CONFIG_DES CUSTOM_HID_EPIN_ADDR, /* bEndpointAddress: Endpoint Address (IN) */ 0x03, /* bmAttributes: Interrupt endpoint */ - CUSTOM_HID_EPIN_SIZE, /* wMaxPacketSize: 2 Bytes max */ - 0x00, + LOBYTE(CUSTOM_HID_EPIN_SIZE), /* wMaxPacketSize: 2 Bytes max */ + HIBYTE(CUSTOM_HID_EPIN_SIZE), CUSTOM_HID_HS_BINTERVAL, /* bInterval: Polling Interval */ /* 34 */ @@ -239,8 +249,8 @@ __ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_CfgHSDesc[USB_CUSTOM_HID_CONFIG_DES USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: */ CUSTOM_HID_EPOUT_ADDR, /* bEndpointAddress: Endpoint Address (OUT) */ 0x03, /* bmAttributes: Interrupt endpoint */ - CUSTOM_HID_EPOUT_SIZE, /* wMaxPacketSize: 2 Bytes max */ - 0x00, + LOBYTE(CUSTOM_HID_EPOUT_SIZE), /* wMaxPacketSize: 2 Bytes max */ + HIBYTE(CUSTOM_HID_EPOUT_SIZE), CUSTOM_HID_HS_BINTERVAL, /* bInterval: Polling Interval */ /* 41 */ }; @@ -250,8 +260,8 @@ __ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_OtherSpeedCfgDesc[USB_CUSTOM_HID_CO { 0x09, /* bLength: Configuration Descriptor size */ USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ - USB_CUSTOM_HID_CONFIG_DESC_SIZ, /* wTotalLength: Bytes returned */ - 0x00, + LOBYTE(USB_CUSTOM_HID_CONFIG_DESC_SIZ), /* wTotalLength: Bytes returned */ + HIBYTE(USB_CUSTOM_HID_CONFIG_DESC_SIZ), 0x01, /* bNumInterfaces: 1 interface */ 0x01, /* bConfigurationValue: Configuration value */ 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ @@ -282,8 +292,8 @@ __ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_OtherSpeedCfgDesc[USB_CUSTOM_HID_CO 0x00, /* bCountryCode: Hardware target country */ 0x01, /* bNumDescriptors: Number of CUSTOM_HID class descriptors to follow */ 0x22, /* bDescriptorType */ - USBD_CUSTOM_HID_REPORT_DESC_SIZE, /* wItemLength: Total length of Report descriptor */ - 0x00, + LOBYTE(USBD_CUSTOM_HID_REPORT_DESC_SIZE), /* wItemLength: Total length of Report descriptor */ + HIBYTE(USBD_CUSTOM_HID_REPORT_DESC_SIZE), /******************** Descriptor of Custom HID endpoints ********************/ /* 27 */ 0x07, /* bLength: Endpoint Descriptor size */ @@ -291,8 +301,8 @@ __ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_OtherSpeedCfgDesc[USB_CUSTOM_HID_CO CUSTOM_HID_EPIN_ADDR, /* bEndpointAddress: Endpoint Address (IN) */ 0x03, /* bmAttributes: Interrupt endpoint */ - CUSTOM_HID_EPIN_SIZE, /* wMaxPacketSize: 2 Bytes max */ - 0x00, + LOBYTE(CUSTOM_HID_EPIN_SIZE), /* wMaxPacketSize: 2 Bytes max */ + HIBYTE(CUSTOM_HID_EPIN_SIZE), CUSTOM_HID_FS_BINTERVAL, /* bInterval: Polling Interval */ /* 34 */ @@ -300,11 +310,12 @@ __ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_OtherSpeedCfgDesc[USB_CUSTOM_HID_CO USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: */ CUSTOM_HID_EPOUT_ADDR, /* bEndpointAddress: Endpoint Address (OUT) */ 0x03, /* bmAttributes: Interrupt endpoint */ - CUSTOM_HID_EPOUT_SIZE, /* wMaxPacketSize: 2 Bytes max */ - 0x00, + LOBYTE(CUSTOM_HID_EPOUT_SIZE), /* wMaxPacketSize: 2 Bytes max */ + HIBYTE(CUSTOM_HID_EPOUT_SIZE), CUSTOM_HID_FS_BINTERVAL, /* bInterval: Polling Interval */ /* 41 */ }; +#endif /* USE_USBD_COMPOSITE */ /* USB CUSTOM_HID device Configuration Descriptor */ __ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_Desc[USB_CUSTOM_HID_DESC_SIZ] __ALIGN_END = @@ -315,12 +326,14 @@ __ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_Desc[USB_CUSTOM_HID_DESC_SIZ] __ALI 0x11, /* bCUSTOM_HIDUSTOM_HID: CUSTOM_HID Class Spec release number */ 0x01, 0x00, /* bCountryCode: Hardware target country */ - 0x01, /* bNumDescriptors: Number of CUSTOM_HID class descriptors to follow */ + 0x01, /* bNumDescriptors: Number of CUSTOM_HID class descriptors + to follow */ 0x22, /* bDescriptorType */ - USBD_CUSTOM_HID_REPORT_DESC_SIZE, /* wItemLength: Total length of Report descriptor */ - 0x00, + LOBYTE(USBD_CUSTOM_HID_REPORT_DESC_SIZE), /* wItemLength: Total length of Report descriptor */ + HIBYTE(USBD_CUSTOM_HID_REPORT_DESC_SIZE), }; +#ifndef USE_USBD_COMPOSITE /* USB Standard Device Descriptor */ __ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_DESC] __ALIGN_END = { @@ -335,7 +348,10 @@ __ALIGN_BEGIN static uint8_t USBD_CUSTOM_HID_DeviceQualifierDesc[USB_LEN_DEV_QUA 0x01, 0x00, }; +#endif /* USE_USBD_COMPOSITE */ +static uint8_t CUSTOMHIDInEpAdd = CUSTOM_HID_EPIN_ADDR; +static uint8_t CUSTOMHIDOutEpAdd = CUSTOM_HID_EPOUT_ADDR; /** * @} */ @@ -360,42 +376,51 @@ static uint8_t USBD_CUSTOM_HID_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) if (hhid == NULL) { - pdev->pClassData = NULL; + pdev->pClassDataCmsit[pdev->classId] = NULL; return (uint8_t)USBD_EMEM; } - pdev->pClassData = (void *)hhid; + pdev->pClassDataCmsit[pdev->classId] = (void *)hhid; + pdev->pClassData = pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + CUSTOMHIDInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); + CUSTOMHIDOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (pdev->dev_speed == USBD_SPEED_HIGH) { - pdev->ep_in[CUSTOM_HID_EPIN_ADDR & 0xFU].bInterval = CUSTOM_HID_HS_BINTERVAL; - pdev->ep_out[CUSTOM_HID_EPOUT_ADDR & 0xFU].bInterval = CUSTOM_HID_HS_BINTERVAL; + pdev->ep_in[CUSTOMHIDInEpAdd & 0xFU].bInterval = CUSTOM_HID_HS_BINTERVAL; + pdev->ep_out[CUSTOMHIDOutEpAdd & 0xFU].bInterval = CUSTOM_HID_HS_BINTERVAL; } else /* LOW and FULL-speed endpoints */ { - pdev->ep_in[CUSTOM_HID_EPIN_ADDR & 0xFU].bInterval = CUSTOM_HID_FS_BINTERVAL; - pdev->ep_out[CUSTOM_HID_EPOUT_ADDR & 0xFU].bInterval = CUSTOM_HID_FS_BINTERVAL; + pdev->ep_in[CUSTOMHIDInEpAdd & 0xFU].bInterval = CUSTOM_HID_FS_BINTERVAL; + pdev->ep_out[CUSTOMHIDOutEpAdd & 0xFU].bInterval = CUSTOM_HID_FS_BINTERVAL; } /* Open EP IN */ - (void)USBD_LL_OpenEP(pdev, CUSTOM_HID_EPIN_ADDR, USBD_EP_TYPE_INTR, + (void)USBD_LL_OpenEP(pdev, CUSTOMHIDInEpAdd, USBD_EP_TYPE_INTR, CUSTOM_HID_EPIN_SIZE); - pdev->ep_in[CUSTOM_HID_EPIN_ADDR & 0xFU].is_used = 1U; + pdev->ep_in[CUSTOMHIDInEpAdd & 0xFU].is_used = 1U; /* Open EP OUT */ - (void)USBD_LL_OpenEP(pdev, CUSTOM_HID_EPOUT_ADDR, USBD_EP_TYPE_INTR, + (void)USBD_LL_OpenEP(pdev, CUSTOMHIDOutEpAdd, USBD_EP_TYPE_INTR, CUSTOM_HID_EPOUT_SIZE); - pdev->ep_out[CUSTOM_HID_EPOUT_ADDR & 0xFU].is_used = 1U; + pdev->ep_out[CUSTOMHIDOutEpAdd & 0xFU].is_used = 1U; hhid->state = CUSTOM_HID_IDLE; - ((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData)->Init(); + ((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->Init(); +#ifndef USBD_CUSTOMHID_OUT_PREPARE_RECEIVE_DISABLED /* Prepare Out endpoint to receive 1st packet */ - (void)USBD_LL_PrepareReceive(pdev, CUSTOM_HID_EPOUT_ADDR, hhid->Report_buf, + (void)USBD_LL_PrepareReceive(pdev, CUSTOMHIDOutEpAdd, hhid->Report_buf, USBD_CUSTOMHID_OUTREPORT_BUF_SIZE); +#endif /* USBD_CUSTOMHID_OUT_PREPARE_RECEIVE_DISABLED */ return (uint8_t)USBD_OK; } @@ -411,21 +436,28 @@ static uint8_t USBD_CUSTOM_HID_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) { UNUSED(cfgidx); +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + CUSTOMHIDInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); + CUSTOMHIDOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + /* Close CUSTOM_HID EP IN */ - (void)USBD_LL_CloseEP(pdev, CUSTOM_HID_EPIN_ADDR); - pdev->ep_in[CUSTOM_HID_EPIN_ADDR & 0xFU].is_used = 0U; - pdev->ep_in[CUSTOM_HID_EPIN_ADDR & 0xFU].bInterval = 0U; + (void)USBD_LL_CloseEP(pdev, CUSTOMHIDInEpAdd); + pdev->ep_in[CUSTOMHIDInEpAdd & 0xFU].is_used = 0U; + pdev->ep_in[CUSTOMHIDInEpAdd & 0xFU].bInterval = 0U; /* Close CUSTOM_HID EP OUT */ - (void)USBD_LL_CloseEP(pdev, CUSTOM_HID_EPOUT_ADDR); - pdev->ep_out[CUSTOM_HID_EPOUT_ADDR & 0xFU].is_used = 0U; - pdev->ep_out[CUSTOM_HID_EPOUT_ADDR & 0xFU].bInterval = 0U; + (void)USBD_LL_CloseEP(pdev, CUSTOMHIDOutEpAdd); + pdev->ep_out[CUSTOMHIDOutEpAdd & 0xFU].is_used = 0U; + pdev->ep_out[CUSTOMHIDOutEpAdd & 0xFU].bInterval = 0U; /* Free allocated memory */ - if (pdev->pClassData != NULL) + if (pdev->pClassDataCmsit[pdev->classId] != NULL) { - ((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData)->DeInit(); - USBD_free(pdev->pClassData); + ((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->DeInit(); + USBD_free(pdev->pClassDataCmsit[pdev->classId]); + pdev->pClassDataCmsit[pdev->classId] = NULL; pdev->pClassData = NULL; } @@ -442,8 +474,11 @@ static uint8_t USBD_CUSTOM_HID_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) static uint8_t USBD_CUSTOM_HID_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_CUSTOM_HID_HandleTypeDef *hhid = (USBD_CUSTOM_HID_HandleTypeDef *)pdev->pClassData; + USBD_CUSTOM_HID_HandleTypeDef *hhid = (USBD_CUSTOM_HID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; uint16_t len = 0U; +#ifdef USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED + uint16_t ReportLength = 0U; +#endif /* USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED */ uint8_t *pbuf = NULL; uint16_t status_info = 0U; USBD_StatusTypeDef ret = USBD_OK; @@ -475,10 +510,59 @@ static uint8_t USBD_CUSTOM_HID_Setup(USBD_HandleTypeDef *pdev, break; case CUSTOM_HID_REQ_SET_REPORT: +#ifdef USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED + if (((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->CtrlReqComplete != NULL) + { + /* Let the application decide when to enable EP0 to receive the next report */ + ((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->CtrlReqComplete(req->bRequest, + req->wLength); + } +#endif /* USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED */ +#ifndef USBD_CUSTOMHID_EP0_OUT_PREPARE_RECEIVE_DISABLED hhid->IsReportAvailable = 1U; (void)USBD_CtlPrepareRx(pdev, hhid->Report_buf, MIN(req->wLength, USBD_CUSTOMHID_OUTREPORT_BUF_SIZE)); +#endif /* USBD_CUSTOMHID_EP0_OUT_PREPARE_RECEIVE_DISABLED */ + break; +#ifdef USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED + case CUSTOM_HID_REQ_GET_REPORT: + if (((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->GetReport != NULL) + { + ReportLength = req->wLength; + + /* Get report data buffer */ + pbuf = ((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->GetReport(&ReportLength); + } + + if ((pbuf != NULL) && (ReportLength != 0U)) + { + len = MIN(ReportLength, req->wLength); + + /* Send the report data over EP0 */ + (void)USBD_CtlSendData(pdev, pbuf, len); + } + else + { +#ifdef USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED + if (((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->CtrlReqComplete != NULL) + { + /* Let the application decide what to do, keep EP0 data phase in NAK state and + use USBD_CtlSendData() when data become available or stall the EP0 data phase */ + ((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->CtrlReqComplete(req->bRequest, + req->wLength); + } + else + { + /* Stall EP0 if no data available */ + USBD_CtlError(pdev, req); + } +#else + /* Stall EP0 if no data available */ + USBD_CtlError(pdev, req); +#endif /* USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED */ + } break; +#endif /* USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED */ default: USBD_CtlError(pdev, req); @@ -506,7 +590,7 @@ static uint8_t USBD_CUSTOM_HID_Setup(USBD_HandleTypeDef *pdev, if ((req->wValue >> 8) == CUSTOM_HID_REPORT_DESC) { len = MIN(USBD_CUSTOM_HID_REPORT_DESC_SIZE, req->wLength); - pbuf = ((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData)->pReport; + pbuf = ((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->pReport; } else { @@ -567,26 +651,37 @@ static uint8_t USBD_CUSTOM_HID_Setup(USBD_HandleTypeDef *pdev, * Send CUSTOM_HID Report * @param pdev: device instance * @param buff: pointer to report + * @param ClassId: The Class ID * @retval status */ +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_CUSTOM_HID_SendReport(USBD_HandleTypeDef *pdev, + uint8_t *report, uint16_t len, uint8_t ClassId) +{ + USBD_CUSTOM_HID_HandleTypeDef *hhid = (USBD_CUSTOM_HID_HandleTypeDef *)pdev->pClassDataCmsit[ClassId]; +#else uint8_t USBD_CUSTOM_HID_SendReport(USBD_HandleTypeDef *pdev, uint8_t *report, uint16_t len) { - USBD_CUSTOM_HID_HandleTypeDef *hhid; + USBD_CUSTOM_HID_HandleTypeDef *hhid = (USBD_CUSTOM_HID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#endif /* USE_USBD_COMPOSITE */ - if (pdev->pClassData == NULL) + if (hhid == NULL) { return (uint8_t)USBD_FAIL; } - hhid = (USBD_CUSTOM_HID_HandleTypeDef *)pdev->pClassData; +#ifdef USE_USBD_COMPOSITE + /* Get Endpoint IN address allocated for this class instance */ + CUSTOMHIDInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, ClassId); +#endif /* USE_USBD_COMPOSITE */ if (pdev->dev_state == USBD_STATE_CONFIGURED) { if (hhid->state == CUSTOM_HID_IDLE) { hhid->state = CUSTOM_HID_BUSY; - (void)USBD_LL_Transmit(pdev, CUSTOM_HID_EPIN_ADDR, report, len); + (void)USBD_LL_Transmit(pdev, CUSTOMHIDInEpAdd, report, len); } else { @@ -595,7 +690,7 @@ uint8_t USBD_CUSTOM_HID_SendReport(USBD_HandleTypeDef *pdev, } return (uint8_t)USBD_OK; } - +#ifndef USE_USBD_COMPOSITE /** * @brief USBD_CUSTOM_HID_GetFSCfgDesc * return FS configuration descriptor @@ -605,9 +700,23 @@ uint8_t USBD_CUSTOM_HID_SendReport(USBD_HandleTypeDef *pdev, */ static uint8_t *USBD_CUSTOM_HID_GetFSCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_CUSTOM_HID_CfgFSDesc); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CUSTOM_HID_CfgDesc, CUSTOM_HID_EPIN_ADDR); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CUSTOM_HID_CfgDesc, CUSTOM_HID_EPOUT_ADDR); - return USBD_CUSTOM_HID_CfgFSDesc; + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = CUSTOM_HID_EPIN_SIZE; + pEpInDesc->bInterval = CUSTOM_HID_FS_BINTERVAL; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = CUSTOM_HID_EPOUT_SIZE; + pEpOutDesc->bInterval = CUSTOM_HID_FS_BINTERVAL; + } + + *length = (uint16_t)sizeof(USBD_CUSTOM_HID_CfgDesc); + return USBD_CUSTOM_HID_CfgDesc; } /** @@ -619,9 +728,23 @@ static uint8_t *USBD_CUSTOM_HID_GetFSCfgDesc(uint16_t *length) */ static uint8_t *USBD_CUSTOM_HID_GetHSCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_CUSTOM_HID_CfgHSDesc); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CUSTOM_HID_CfgDesc, CUSTOM_HID_EPIN_ADDR); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CUSTOM_HID_CfgDesc, CUSTOM_HID_EPOUT_ADDR); + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = CUSTOM_HID_EPIN_SIZE; + pEpInDesc->bInterval = CUSTOM_HID_HS_BINTERVAL; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = CUSTOM_HID_EPOUT_SIZE; + pEpOutDesc->bInterval = CUSTOM_HID_HS_BINTERVAL; + } - return USBD_CUSTOM_HID_CfgHSDesc; + *length = (uint16_t)sizeof(USBD_CUSTOM_HID_CfgDesc); + return USBD_CUSTOM_HID_CfgDesc; } /** @@ -633,10 +756,25 @@ static uint8_t *USBD_CUSTOM_HID_GetHSCfgDesc(uint16_t *length) */ static uint8_t *USBD_CUSTOM_HID_GetOtherSpeedCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_CUSTOM_HID_OtherSpeedCfgDesc); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_CUSTOM_HID_CfgDesc, CUSTOM_HID_EPIN_ADDR); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_CUSTOM_HID_CfgDesc, CUSTOM_HID_EPOUT_ADDR); + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = CUSTOM_HID_EPIN_SIZE; + pEpInDesc->bInterval = CUSTOM_HID_FS_BINTERVAL; + } - return USBD_CUSTOM_HID_OtherSpeedCfgDesc; + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = CUSTOM_HID_EPOUT_SIZE; + pEpOutDesc->bInterval = CUSTOM_HID_FS_BINTERVAL; + } + + *length = (uint16_t)sizeof(USBD_CUSTOM_HID_CfgDesc); + return USBD_CUSTOM_HID_CfgDesc; } +#endif /* USE_USBD_COMPOSITE */ /** * @brief USBD_CUSTOM_HID_DataIn @@ -651,7 +789,7 @@ static uint8_t USBD_CUSTOM_HID_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) /* Ensure that the FIFO is empty before a new transfer, this condition could be caused by a new transfer before the end of the previous transfer */ - ((USBD_CUSTOM_HID_HandleTypeDef *)pdev->pClassData)->state = CUSTOM_HID_IDLE; + ((USBD_CUSTOM_HID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId])->state = CUSTOM_HID_IDLE; return (uint8_t)USBD_OK; } @@ -668,17 +806,17 @@ static uint8_t USBD_CUSTOM_HID_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) UNUSED(epnum); USBD_CUSTOM_HID_HandleTypeDef *hhid; - if (pdev->pClassData == NULL) + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_FAIL; } - hhid = (USBD_CUSTOM_HID_HandleTypeDef *)pdev->pClassData; + hhid = (USBD_CUSTOM_HID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; /* USB data will be immediately processed, this allow next USB traffic being NAKed till the end of the application processing */ - ((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData)->OutEvent(hhid->Report_buf[0], - hhid->Report_buf[1]); + ((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->OutEvent(hhid->Report_buf[0], + hhid->Report_buf[1]); return (uint8_t)USBD_OK; } @@ -694,15 +832,20 @@ uint8_t USBD_CUSTOM_HID_ReceivePacket(USBD_HandleTypeDef *pdev) { USBD_CUSTOM_HID_HandleTypeDef *hhid; - if (pdev->pClassData == NULL) + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_FAIL; } - hhid = (USBD_CUSTOM_HID_HandleTypeDef *)pdev->pClassData; +#ifdef USE_USBD_COMPOSITE + /* Get OUT Endpoint address allocated for this class instance */ + CUSTOMHIDOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + hhid = (USBD_CUSTOM_HID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; /* Resume USB Out process */ - (void)USBD_LL_PrepareReceive(pdev, CUSTOM_HID_EPOUT_ADDR, hhid->Report_buf, + (void)USBD_LL_PrepareReceive(pdev, CUSTOMHIDOutEpAdd, hhid->Report_buf, USBD_CUSTOMHID_OUTREPORT_BUF_SIZE); return (uint8_t)USBD_OK; @@ -717,7 +860,7 @@ uint8_t USBD_CUSTOM_HID_ReceivePacket(USBD_HandleTypeDef *pdev) */ static uint8_t USBD_CUSTOM_HID_EP0_RxReady(USBD_HandleTypeDef *pdev) { - USBD_CUSTOM_HID_HandleTypeDef *hhid = (USBD_CUSTOM_HID_HandleTypeDef *)pdev->pClassData; + USBD_CUSTOM_HID_HandleTypeDef *hhid = (USBD_CUSTOM_HID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hhid == NULL) { @@ -726,14 +869,15 @@ static uint8_t USBD_CUSTOM_HID_EP0_RxReady(USBD_HandleTypeDef *pdev) if (hhid->IsReportAvailable == 1U) { - ((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData)->OutEvent(hhid->Report_buf[0], - hhid->Report_buf[1]); + ((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->OutEvent(hhid->Report_buf[0], + hhid->Report_buf[1]); hhid->IsReportAvailable = 0U; } return (uint8_t)USBD_OK; } +#ifndef USE_USBD_COMPOSITE /** * @brief DeviceQualifierDescriptor * return Device Qualifier descriptor @@ -746,7 +890,7 @@ static uint8_t *USBD_CUSTOM_HID_GetDeviceQualifierDesc(uint16_t *length) return USBD_CUSTOM_HID_DeviceQualifierDesc; } - +#endif /* USE_USBD_COMPOSITE */ /** * @brief USBD_CUSTOM_HID_RegisterInterface * @param pdev: device instance @@ -761,7 +905,7 @@ uint8_t USBD_CUSTOM_HID_RegisterInterface(USBD_HandleTypeDef *pdev, return (uint8_t)USBD_FAIL; } - pdev->pUserData = fops; + pdev->pUserData[pdev->classId] = fops; return (uint8_t)USBD_OK; } @@ -779,4 +923,3 @@ uint8_t USBD_CUSTOM_HID_RegisterInterface(USBD_HandleTypeDef *pdev, * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Src/usbd_customhid_if_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Src/usbd_customhid_if_template.c index 97e43379..85881bdc 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Src/usbd_customhid_if_template.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/CustomHID/Src/usbd_customhid_if_template.c @@ -8,13 +8,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -34,13 +33,31 @@ EndBSPDependencies */ static int8_t TEMPLATE_CUSTOM_HID_Init(void); static int8_t TEMPLATE_CUSTOM_HID_DeInit(void); static int8_t TEMPLATE_CUSTOM_HID_OutEvent(uint8_t event_idx, uint8_t state); + +#ifdef USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED +static int8_t TEMPLATE_CUSTOM_HID_CtrlReqComplete(uint8_t request, uint16_t wLength); +#endif /* USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED */ + +#ifdef USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED +static uint8_t *TEMPLATE_CUSTOM_HID_GetReport(uint16_t *ReportLength); +#endif /* USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED */ /* Private variables ---------------------------------------------------------*/ +extern USBD_HandleTypeDef USBD_Device; + +__ALIGN_BEGIN static uint8_t TEMPLATE_CUSTOM_HID_ReportDesc[USBD_CUSTOM_HID_REPORT_DESC_SIZE] __ALIGN_END = {0}; + USBD_CUSTOM_HID_ItfTypeDef USBD_CustomHID_template_fops = { TEMPLATE_CUSTOM_HID_ReportDesc, TEMPLATE_CUSTOM_HID_Init, TEMPLATE_CUSTOM_HID_DeInit, TEMPLATE_CUSTOM_HID_OutEvent, +#ifdef USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED + TEMPLATE_CUSTOM_HID_CtrlReqComplete, +#endif /* USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED */ +#ifdef USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED + TEMPLATE_CUSTOM_HID_GetReport, +#endif /* USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED */ }; /* Private functions ---------------------------------------------------------*/ @@ -84,8 +101,58 @@ static int8_t TEMPLATE_CUSTOM_HID_OutEvent(uint8_t event_idx, uint8_t state) UNUSED(state); /* Start next USB packet transfer once data processing is completed */ - USBD_CUSTOM_HID_ReceivePacket(&USBD_Device); + if (USBD_CUSTOM_HID_ReceivePacket(&USBD_Device) != (uint8_t)USBD_OK) + { + return -1; + } + + return (0); +} + +#ifdef USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED +/** + * @brief TEMPLATE_CUSTOM_HID_CtrlReqComplete + * Manage the CUSTOM HID control request complete + * @param request: control request + * @param wLength: request wLength + * @retval Result of the operation: USBD_OK if all operations are OK else USBD_FAIL + */ +static int8_t TEMPLATE_CUSTOM_HID_CtrlReqComplete(uint8_t request, uint16_t wLength) +{ + UNUSED(wLength); + + switch (request) + { + case CUSTOM_HID_REQ_SET_REPORT: + + break; + + case CUSTOM_HID_REQ_GET_REPORT: + + break; + + default: + break; + } return (0); } -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ +#endif /* USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED */ + + +#ifdef USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED +/** + * @brief TEMPLATE_CUSTOM_HID_GetReport + * Manage the CUSTOM HID control Get Report request + * @param event_idx: event index + * @param state: event state + * @retval return pointer to HID report + */ +static uint8_t *TEMPLATE_CUSTOM_HID_GetReport(uint16_t *ReportLength) +{ + UNUSED(ReportLength); + uint8_t *pbuff; + + return (pbuff); +} +#endif /* USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED */ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Inc/usbd_dfu.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Inc/usbd_dfu.h index 307c96d9..7858394a 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Inc/usbd_dfu.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Inc/usbd_dfu.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -53,6 +52,14 @@ extern "C" { #define USBD_DFU_APP_DEFAULT_ADD 0x08008000U /* The first sector (32 KB) is reserved for DFU code */ #endif /* USBD_DFU_APP_DEFAULT_ADD */ +#ifndef USBD_DFU_BM_ATTRIBUTES +#define USBD_DFU_BM_ATTRIBUTES 0x0BU +#endif /* USBD_DFU_BM_ATTRIBUTES */ + +#ifndef USBD_DFU_DETACH_TIMEOUT +#define USBD_DFU_DETACH_TIMEOUT 0xFFU +#endif /* USBD_DFU_DETACH_TIMEOUT */ + #define USB_DFU_CONFIG_DESC_SIZ (18U + (9U * USBD_DFU_MAX_ITF_NUM)) #define USB_DFU_DESC_SIZ 9U @@ -115,7 +122,8 @@ extern "C" { /* Other defines */ /**************************************************/ /* Bit Detach capable = bit 3 in bmAttributes field */ -#define DFU_DETACH_MASK (1U << 4) +#define DFU_DETACH_MASK (1U << 3) +#define DFU_MANIFEST_MASK (1U << 2) #define DFU_STATUS_DEPTH 6U typedef enum @@ -133,20 +141,20 @@ typedef void (*pFunction)(void); /********** Descriptor of DFU interface 0 Alternate setting n ****************/ -#define USBD_DFU_IF_DESC(n) 0x09, /* bLength: Interface Descriptor size */ \ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType */ \ - 0x00, /* bInterfaceNumber: Number of Interface */ \ - (n), /* bAlternateSetting: Alternate setting */ \ - 0x00, /* bNumEndpoints*/ \ - 0xFE, /* bInterfaceClass: Application Specific Class Code */ \ - 0x01, /* bInterfaceSubClass : Device Firmware Upgrade Code */ \ - 0x02, /* nInterfaceProtocol: DFU mode protocol */ \ - USBD_IDX_INTERFACE_STR + (n) + 1U /* iInterface: Index of string descriptor */ \ +#define USBD_DFU_IF_DESC(n) \ + 0x09, /* bLength: Interface Descriptor size */ \ + USB_DESC_TYPE_INTERFACE, /* bDescriptorType */ \ + 0x00, /* bInterfaceNumber: Number of Interface */ \ + (n), /* bAlternateSetting: Alternate setting */ \ + 0x00, /* bNumEndpoints*/ \ + 0xFE, /* bInterfaceClass: Application Specific Class Code */ \ + 0x01, /* bInterfaceSubClass : Device Firmware Upgrade Code */ \ + 0x02, /* nInterfaceProtocol: DFU mode protocol */ \ + USBD_IDX_INTERFACE_STR + (n) + 1U /* iInterface: Index of string descriptor */ -#define TRANSFER_SIZE_BYTES(size) ((uint8_t)(size)), /* XFERSIZEB0 */\ - ((uint8_t)((size) >> 8)) /* XFERSIZEB1 */ +#define TRANSFER_SIZE_BYTES(size) ((uint8_t)(size)), ((uint8_t)((size) >> 8)) -#define IS_PROTECTED_AREA(add) (uint8_t)((((add) >= 0x08000000) && ((add) < (APP_DEFAULT_ADD)))? 1:0) +#define IS_PROTECTED_AREA(add) (uint8_t)((((add) >= 0x08000000) && ((add) < (APP_DEFAULT_ADD))) ? 1 : 0) /** * @} @@ -186,6 +194,17 @@ typedef struct uint8_t *(* Read)(uint8_t *src, uint8_t *dest, uint32_t Len); uint16_t (* GetStatus)(uint32_t Add, uint8_t cmd, uint8_t *buff); } USBD_DFU_MediaTypeDef; + +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bmAttributes; + uint16_t wDetachTimeout; + uint16_t wTransferSze; + uint16_t bcdDFUVersion; +} __PACKED USBD_DFUFuncDescTypeDef; + /** * @} */ @@ -231,5 +250,3 @@ uint8_t USBD_DFU_RegisterMedia(USBD_HandleTypeDef *pdev, /** * @} */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Inc/usbd_dfu_media_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Inc/usbd_dfu_media_template.h index 25efe526..7be25d48 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Inc/usbd_dfu_media_template.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Inc/usbd_dfu_media_template.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -94,4 +93,3 @@ extern USBD_DFU_MediaTypeDef USBD_DFU_MEDIA_Template_fops; /** * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Src/usbd_dfu.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Src/usbd_dfu.c index fc942a76..58954970 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Src/usbd_dfu.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Src/usbd_dfu.c @@ -4,6 +4,17 @@ * @author MCD Application Team * @brief This file provides the DFU core functions. * + ****************************************************************************** + * @attention + * + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** * @verbatim * * =================================================================== @@ -34,17 +45,6 @@ * @endverbatim * ****************************************************************************** - * @attention - * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> - * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** */ /* BSPDependencies @@ -104,13 +104,15 @@ static uint8_t USBD_DFU_EP0_RxReady(USBD_HandleTypeDef *pdev); static uint8_t USBD_DFU_EP0_TxReady(USBD_HandleTypeDef *pdev); static uint8_t USBD_DFU_SOF(USBD_HandleTypeDef *pdev); +#ifndef USE_USBD_COMPOSITE static uint8_t *USBD_DFU_GetCfgDesc(uint16_t *length); static uint8_t *USBD_DFU_GetDeviceQualifierDesc(uint16_t *length); +#endif /* USE_USBD_COMPOSITE */ #if (USBD_SUPPORT_USER_STRING_DESC == 1U) static uint8_t *USBD_DFU_GetUsrStringDesc(USBD_HandleTypeDef *pdev, uint8_t index, uint16_t *length); -#endif +#endif /* USBD_SUPPORT_USER_STRING_DESC */ static void DFU_Detach(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); static void DFU_Download(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); @@ -120,6 +122,7 @@ static void DFU_ClearStatus(USBD_HandleTypeDef *pdev); static void DFU_GetState(USBD_HandleTypeDef *pdev); static void DFU_Abort(USBD_HandleTypeDef *pdev); static void DFU_Leave(USBD_HandleTypeDef *pdev); +static void *USBD_DFU_GetDfuFuncDesc(uint8_t *pConfDesc); /** * @} @@ -141,15 +144,23 @@ USBD_ClassTypeDef USBD_DFU = USBD_DFU_SOF, NULL, NULL, +#ifdef USE_USBD_COMPOSITE + NULL, + NULL, + NULL, + NULL, +#else USBD_DFU_GetCfgDesc, USBD_DFU_GetCfgDesc, USBD_DFU_GetCfgDesc, USBD_DFU_GetDeviceQualifierDesc, +#endif /* USE_USBD_COMPOSITE */ #if (USBD_SUPPORT_USER_STRING_DESC == 1U) USBD_DFU_GetUsrStringDesc -#endif +#endif /* USBD_SUPPORT_USER_STRING_DESC */ }; +#ifndef USE_USBD_COMPOSITE /* USB DFU device Configuration Descriptor */ __ALIGN_BEGIN static uint8_t USBD_DFU_CfgDesc[USB_DFU_CONFIG_DESC_SIZ] __ALIGN_END = { @@ -159,12 +170,13 @@ __ALIGN_BEGIN static uint8_t USBD_DFU_CfgDesc[USB_DFU_CONFIG_DESC_SIZ] __ALIGN_E 0x00, 0x01, /* bNumInterfaces: 1 interface */ 0x01, /* bConfigurationValue: Configuration value */ - 0x02, /* iConfiguration: Index of string descriptor describing the configuration */ + 0x02, /* iConfiguration: Index of string descriptor + describing the configuration */ #if (USBD_SELF_POWERED == 1U) 0xC0, /* bmAttributes: Bus Powered according to user configuration */ #else 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif +#endif /* USBD_SELF_POWERED */ USBD_MAX_POWER, /* MaxPower (mA) */ /* 09 */ @@ -235,6 +247,7 @@ __ALIGN_BEGIN static uint8_t USBD_DFU_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_ 0x01, 0x00, }; +#endif /* USE_USBD_COMPOSITE */ /** * @} @@ -262,11 +275,12 @@ static uint8_t USBD_DFU_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) if (hdfu == NULL) { - pdev->pClassData = NULL; + pdev->pClassDataCmsit[pdev->classId] = NULL; return (uint8_t)USBD_EMEM; } - pdev->pClassData = (void *)hdfu; + pdev->pClassDataCmsit[pdev->classId] = (void *)hdfu; + pdev->pClassData = pdev->pClassDataCmsit[pdev->classId]; hdfu->alt_setting = 0U; hdfu->data_ptr = USBD_DFU_APP_DEFAULT_ADD; @@ -284,7 +298,7 @@ static uint8_t USBD_DFU_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) hdfu->dev_status[5] = 0U; /* Initialize Hardware layer */ - if (((USBD_DFU_MediaTypeDef *)pdev->pUserData)->Init() != USBD_OK) + if (((USBD_DFU_MediaTypeDef *)pdev->pUserData[pdev->classId])->Init() != USBD_OK) { return (uint8_t)USBD_FAIL; } @@ -293,7 +307,7 @@ static uint8_t USBD_DFU_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) } /** - * @brief USBD_DFU_Init + * @brief USBD_DFU_DeInit * De-Initialize the DFU layer * @param pdev: device instance * @param cfgidx: Configuration index @@ -304,12 +318,12 @@ static uint8_t USBD_DFU_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) UNUSED(cfgidx); USBD_DFU_HandleTypeDef *hdfu; - if (pdev->pClassData == NULL) + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_EMEM; } - hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassData; + hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; hdfu->wblock_num = 0U; hdfu->wlength = 0U; @@ -318,8 +332,9 @@ static uint8_t USBD_DFU_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) hdfu->dev_status[4] = DFU_STATE_IDLE; /* DeInit physical Interface components and Hardware Layer */ - ((USBD_DFU_MediaTypeDef *)pdev->pUserData)->DeInit(); - USBD_free(pdev->pClassData); + ((USBD_DFU_MediaTypeDef *)pdev->pUserData[pdev->classId])->DeInit(); + USBD_free(pdev->pClassDataCmsit[pdev->classId]); + pdev->pClassDataCmsit[pdev->classId] = NULL; pdev->pClassData = NULL; return (uint8_t)USBD_OK; @@ -334,10 +349,10 @@ static uint8_t USBD_DFU_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) */ static uint8_t USBD_DFU_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassData; + USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; USBD_StatusTypeDef ret = USBD_OK; - uint8_t *pbuf = NULL; - uint16_t len = 0U; + uint8_t *pbuf; + uint16_t len; uint16_t status_info = 0U; if (hdfu == NULL) @@ -403,11 +418,19 @@ static uint8_t USBD_DFU_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *re case USB_REQ_GET_DESCRIPTOR: if ((req->wValue >> 8) == DFU_DESCRIPTOR_TYPE) { - pbuf = USBD_DFU_CfgDesc + (9U * (USBD_DFU_MAX_ITF_NUM + 1U)); - len = MIN(USB_DFU_DESC_SIZ, req->wLength); - } + pbuf = (uint8_t *)USBD_DFU_GetDfuFuncDesc(pdev->pConfDesc); - (void)USBD_CtlSendData(pdev, pbuf, len); + if (pbuf != NULL) + { + len = MIN(USB_DFU_DESC_SIZ, req->wLength); + (void)USBD_CtlSendData(pdev, pbuf, len); + } + else + { + USBD_CtlError(pdev, req); + ret = USBD_FAIL; + } + } break; case USB_REQ_GET_INTERFACE: @@ -462,11 +485,10 @@ static uint8_t USBD_DFU_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *re return (uint8_t)ret; } - +#ifndef USE_USBD_COMPOSITE /** * @brief USBD_DFU_GetCfgDesc * return configuration descriptor - * @param speed : current device speed * @param length : pointer data length * @retval pointer to descriptor buffer */ @@ -476,7 +498,7 @@ static uint8_t *USBD_DFU_GetCfgDesc(uint16_t *length) return USBD_DFU_CfgDesc; } - +#endif /* USE_USBD_COMPOSITE */ /** * @brief USBD_DFU_EP0_RxReady @@ -500,8 +522,8 @@ static uint8_t USBD_DFU_EP0_TxReady(USBD_HandleTypeDef *pdev) { USBD_SetupReqTypedef req; uint32_t addr; - USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassData; - USBD_DFU_MediaTypeDef *DfuInterface = (USBD_DFU_MediaTypeDef *)pdev->pUserData; + USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_DFU_MediaTypeDef *DfuInterface = (USBD_DFU_MediaTypeDef *)pdev->pUserData[pdev->classId]; if (hdfu == NULL) { @@ -610,7 +632,7 @@ static uint8_t USBD_DFU_SOF(USBD_HandleTypeDef *pdev) return (uint8_t)USBD_OK; } - +#ifndef USE_USBD_COMPOSITE /** * @brief DeviceQualifierDescriptor * return Device Qualifier descriptor @@ -623,11 +645,12 @@ static uint8_t *USBD_DFU_GetDeviceQualifierDesc(uint16_t *length) return USBD_DFU_DeviceQualifierDesc; } +#endif /* USE_USBD_COMPOSITE */ /** * @brief USBD_DFU_GetUsrStringDesc * Manages the transfer of memory interfaces string descriptors. - * @param speed : current device speed + * @param pdev: device instance * @param index: descriptor index * @param length : pointer data length * @retval pointer to the descriptor table or NULL if the descriptor is not supported. @@ -636,7 +659,7 @@ static uint8_t *USBD_DFU_GetDeviceQualifierDesc(uint16_t *length) static uint8_t *USBD_DFU_GetUsrStringDesc(USBD_HandleTypeDef *pdev, uint8_t index, uint16_t *length) { static uint8_t USBD_StrDesc[255]; - USBD_DFU_MediaTypeDef *DfuInterface = (USBD_DFU_MediaTypeDef *)pdev->pUserData; + USBD_DFU_MediaTypeDef *DfuInterface = (USBD_DFU_MediaTypeDef *)pdev->pUserData[pdev->classId]; /* Check if the requested string interface is supported */ if (index <= (USBD_IDX_INTERFACE_STR + USBD_DFU_MAX_ITF_NUM)) @@ -647,13 +670,15 @@ static uint8_t *USBD_DFU_GetUsrStringDesc(USBD_HandleTypeDef *pdev, uint8_t inde else { /* Not supported Interface Descriptor index */ + length = 0U; return NULL; } } -#endif +#endif /* USBD_SUPPORT_USER_STRING_DESC */ /** * @brief USBD_MSC_RegisterStorage + * @param pdev: device instance * @param fops: storage callback * @retval status */ @@ -665,7 +690,7 @@ uint8_t USBD_DFU_RegisterMedia(USBD_HandleTypeDef *pdev, return (uint8_t)USBD_FAIL; } - pdev->pUserData = fops; + pdev->pUserData[pdev->classId] = fops; return (uint8_t)USBD_OK; } @@ -682,9 +707,10 @@ uint8_t USBD_DFU_RegisterMedia(USBD_HandleTypeDef *pdev, */ static void DFU_Detach(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassData; + USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_DFUFuncDescTypeDef *pDfuFunc = (USBD_DFUFuncDescTypeDef *)USBD_DFU_GetDfuFuncDesc(pdev->pConfDesc); - if (hdfu == NULL) + if ((hdfu == NULL) || (pDfuFunc == NULL)) { return; } @@ -708,7 +734,7 @@ static void DFU_Detach(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) } /* Check the detach capability in the DFU functional descriptor */ - if (((USBD_DFU_CfgDesc[12U + (9U * USBD_DFU_MAX_ITF_NUM)]) & DFU_DETACH_MASK) != 0U) + if ((pDfuFunc->bmAttributes & DFU_DETACH_MASK) != 0U) { /* Perform an Attach-Detach operation on USB bus */ (void)USBD_Stop(pdev); @@ -730,7 +756,7 @@ static void DFU_Detach(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) */ static void DFU_Download(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassData; + USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hdfu == NULL) { @@ -790,8 +816,8 @@ static void DFU_Download(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) */ static void DFU_Upload(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassData; - USBD_DFU_MediaTypeDef *DfuInterface = (USBD_DFU_MediaTypeDef *)pdev->pUserData; + USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_DFU_MediaTypeDef *DfuInterface = (USBD_DFU_MediaTypeDef *)pdev->pUserData[pdev->classId]; uint8_t *phaddr; uint32_t addr; @@ -888,10 +914,11 @@ static void DFU_Upload(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) */ static void DFU_GetStatus(USBD_HandleTypeDef *pdev) { - USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassData; - USBD_DFU_MediaTypeDef *DfuInterface = (USBD_DFU_MediaTypeDef *)pdev->pUserData; + USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_DFU_MediaTypeDef *DfuInterface = (USBD_DFU_MediaTypeDef *)pdev->pUserData[pdev->classId]; + USBD_DFUFuncDescTypeDef *pDfuFunc = (USBD_DFUFuncDescTypeDef *)USBD_DFU_GetDfuFuncDesc(pdev->pConfDesc); - if (hdfu == NULL) + if ((hdfu == NULL) || (DfuInterface == NULL) || (pDfuFunc == NULL)) { return; } @@ -941,7 +968,7 @@ static void DFU_GetStatus(USBD_HandleTypeDef *pdev) else { if ((hdfu->manif_state == DFU_MANIFEST_COMPLETE) && - (((USBD_DFU_CfgDesc[(11U + (9U * USBD_DFU_MAX_ITF_NUM))]) & 0x04U) != 0U)) + ((pDfuFunc->bmAttributes & DFU_MANIFEST_MASK) != 0U)) { hdfu->dev_state = DFU_STATE_IDLE; @@ -969,7 +996,7 @@ static void DFU_GetStatus(USBD_HandleTypeDef *pdev) */ static void DFU_ClearStatus(USBD_HandleTypeDef *pdev) { - USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassData; + USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hdfu == NULL) { @@ -1007,7 +1034,7 @@ static void DFU_ClearStatus(USBD_HandleTypeDef *pdev) */ static void DFU_GetState(USBD_HandleTypeDef *pdev) { - USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassData; + USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hdfu == NULL) { @@ -1026,7 +1053,7 @@ static void DFU_GetState(USBD_HandleTypeDef *pdev) */ static void DFU_Abort(USBD_HandleTypeDef *pdev) { - USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassData; + USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hdfu == NULL) { @@ -1060,16 +1087,17 @@ static void DFU_Abort(USBD_HandleTypeDef *pdev) */ static void DFU_Leave(USBD_HandleTypeDef *pdev) { - USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassData; + USBD_DFU_HandleTypeDef *hdfu = (USBD_DFU_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_DFUFuncDescTypeDef *pDfuFunc = (USBD_DFUFuncDescTypeDef *)USBD_DFU_GetDfuFuncDesc(pdev->pConfDesc); - if (hdfu == NULL) + if ((hdfu == NULL) || (pDfuFunc == NULL)) { return; } hdfu->manif_state = DFU_MANIFEST_COMPLETE; - if (((USBD_DFU_CfgDesc[(11U + (9U * USBD_DFU_MAX_ITF_NUM))]) & 0x04U) != 0U) + if ((pDfuFunc->bmAttributes & DFU_MANIFEST_MASK) != 0U) { hdfu->dev_state = DFU_STATE_MANIFEST_SYNC; @@ -1098,6 +1126,38 @@ static void DFU_Leave(USBD_HandleTypeDef *pdev) } } +/** + * @brief USBD_DFU_GetDfuFuncDesc + * This function return the DFU descriptor + * @param pdev: device instance + * @param pConfDesc: pointer to Bos descriptor + * @retval pointer to the DFU descriptor + */ +static void *USBD_DFU_GetDfuFuncDesc(uint8_t *pConfDesc) +{ + USBD_ConfigDescTypeDef *desc = (USBD_ConfigDescTypeDef *)(void *)pConfDesc; + USBD_DescHeaderTypeDef *pdesc = (USBD_DescHeaderTypeDef *)(void *)pConfDesc; + uint8_t *pDfuDesc = NULL; + uint16_t ptr; + + if (desc->wTotalLength > desc->bLength) + { + ptr = desc->bLength; + + while (ptr < desc->wTotalLength) + { + pdesc = USBD_GetNextDesc((uint8_t *)pdesc, &ptr); + + if (pdesc->bDescriptorType == DFU_DESCRIPTOR_TYPE) + { + pDfuDesc = (uint8_t *)pdesc; + break; + } + } + } + return pDfuDesc; +} + /** * @} */ @@ -1112,4 +1172,3 @@ static void DFU_Leave(USBD_HandleTypeDef *pdev) * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Src/usbd_dfu_media_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Src/usbd_dfu_media_template.c index b40a10cb..679d3264 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Src/usbd_dfu_media_template.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/DFU/Src/usbd_dfu_media_template.c @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -81,6 +80,8 @@ uint16_t MEM_If_DeInit(void) */ uint16_t MEM_If_Erase(uint32_t Add) { + UNUSED(Add); + return 0; } @@ -93,6 +94,10 @@ uint16_t MEM_If_Erase(uint32_t Add) */ uint16_t MEM_If_Write(uint8_t *src, uint8_t *dest, uint32_t Len) { + UNUSED(src); + UNUSED(dest); + UNUSED(Len); + return 0; } @@ -105,6 +110,10 @@ uint16_t MEM_If_Write(uint8_t *src, uint8_t *dest, uint32_t Len) */ uint8_t *MEM_If_Read(uint8_t *src, uint8_t *dest, uint32_t Len) { + UNUSED(src); + UNUSED(dest); + UNUSED(Len); + /* Return a valid address to avoid HardFault */ return (uint8_t *)(0); } @@ -118,6 +127,9 @@ uint8_t *MEM_If_Read(uint8_t *src, uint8_t *dest, uint32_t Len) */ uint16_t MEM_If_GetStatus(uint32_t Add, uint8_t Cmd, uint8_t *buffer) { + UNUSED(Add); + UNUSED(buffer); + switch (Cmd) { case DFU_MEDIA_PROGRAM: @@ -131,5 +143,4 @@ uint16_t MEM_If_GetStatus(uint32_t Add, uint8_t Cmd, uint8_t *buffer) } return (0); } -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/HID/Inc/usbd_hid.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/HID/Inc/usbd_hid.h index 76fcdc7c..8b22a8df 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/HID/Inc/usbd_hid.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/HID/Inc/usbd_hid.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -41,7 +40,9 @@ extern "C" { /** @defgroup USBD_HID_Exported_Defines * @{ */ +#ifndef HID_EPIN_ADDR #define HID_EPIN_ADDR 0x81U +#endif /* HID_EPIN_ADDR */ #define HID_EPIN_SIZE 0x04U #define USB_HID_CONFIG_DESC_SIZ 34U @@ -59,14 +60,14 @@ extern "C" { #define HID_FS_BINTERVAL 0x0AU #endif /* HID_FS_BINTERVAL */ -#define HID_REQ_SET_PROTOCOL 0x0BU -#define HID_REQ_GET_PROTOCOL 0x03U +#define USBD_HID_REQ_SET_PROTOCOL 0x0BU +#define USBD_HID_REQ_GET_PROTOCOL 0x03U -#define HID_REQ_SET_IDLE 0x0AU -#define HID_REQ_GET_IDLE 0x02U +#define USBD_HID_REQ_SET_IDLE 0x0AU +#define USBD_HID_REQ_GET_IDLE 0x02U -#define HID_REQ_SET_REPORT 0x09U -#define HID_REQ_GET_REPORT 0x01U +#define USBD_HID_REQ_SET_REPORT 0x09U +#define USBD_HID_REQ_GET_REPORT 0x01U /** * @} */ @@ -77,9 +78,9 @@ extern "C" { */ typedef enum { - HID_IDLE = 0, - HID_BUSY, -} HID_StateTypeDef; + USBD_HID_IDLE = 0, + USBD_HID_BUSY, +} USBD_HID_StateTypeDef; typedef struct @@ -87,8 +88,25 @@ typedef struct uint32_t Protocol; uint32_t IdleState; uint32_t AltSetting; - HID_StateTypeDef state; + USBD_HID_StateTypeDef state; } USBD_HID_HandleTypeDef; + +/* + * HID Class specification version 1.1 + * 6.2.1 HID Descriptor + */ + +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint16_t bcdHID; + uint8_t bCountryCode; + uint8_t bNumDescriptors; + uint8_t bHIDDescriptorType; + uint16_t wItemLength; +} __PACKED USBD_HIDDescTypeDef; + /** * @} */ @@ -116,7 +134,11 @@ extern USBD_ClassTypeDef USBD_HID; /** @defgroup USB_CORE_Exported_Functions * @{ */ +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_HID_SendReport(USBD_HandleTypeDef *pdev, uint8_t *report, uint16_t len, uint8_t ClassId); +#else uint8_t USBD_HID_SendReport(USBD_HandleTypeDef *pdev, uint8_t *report, uint16_t len); +#endif /* USE_USBD_COMPOSITE */ uint32_t USBD_HID_GetPollingInterval(USBD_HandleTypeDef *pdev); /** @@ -136,4 +158,3 @@ uint32_t USBD_HID_GetPollingInterval(USBD_HandleTypeDef *pdev); * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/HID/Src/usbd_hid.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/HID/Src/usbd_hid.c index 75256d20..fdf75781 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/HID/Src/usbd_hid.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/HID/Src/usbd_hid.c @@ -4,6 +4,17 @@ * @author MCD Application Team * @brief This file provides the HID core functions. * + ****************************************************************************** + * @attention + * + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** * @verbatim * * =================================================================== @@ -25,17 +36,6 @@ * @endverbatim * ****************************************************************************** - * @attention - * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> - * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** */ /* BSPDependencies @@ -91,12 +91,12 @@ static uint8_t USBD_HID_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx); static uint8_t USBD_HID_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx); static uint8_t USBD_HID_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); static uint8_t USBD_HID_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum); - +#ifndef USE_USBD_COMPOSITE static uint8_t *USBD_HID_GetFSCfgDesc(uint16_t *length); static uint8_t *USBD_HID_GetHSCfgDesc(uint16_t *length); static uint8_t *USBD_HID_GetOtherSpeedCfgDesc(uint16_t *length); static uint8_t *USBD_HID_GetDeviceQualifierDesc(uint16_t *length); - +#endif /* USE_USBD_COMPOSITE */ /** * @} */ @@ -117,14 +117,22 @@ USBD_ClassTypeDef USBD_HID = NULL, /* SOF */ NULL, NULL, +#ifdef USE_USBD_COMPOSITE + NULL, + NULL, + NULL, + NULL, +#else USBD_HID_GetHSCfgDesc, USBD_HID_GetFSCfgDesc, USBD_HID_GetOtherSpeedCfgDesc, USBD_HID_GetDeviceQualifierDesc, +#endif /* USE_USBD_COMPOSITE */ }; +#ifndef USE_USBD_COMPOSITE /* USB HID device FS Configuration Descriptor */ -__ALIGN_BEGIN static uint8_t USBD_HID_CfgFSDesc[USB_HID_CONFIG_DESC_SIZ] __ALIGN_END = +__ALIGN_BEGIN static uint8_t USBD_HID_CfgDesc[USB_HID_CONFIG_DESC_SIZ] __ALIGN_END = { 0x09, /* bLength: Configuration Descriptor size */ USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ @@ -132,12 +140,13 @@ __ALIGN_BEGIN static uint8_t USBD_HID_CfgFSDesc[USB_HID_CONFIG_DESC_SIZ] __ALIGN 0x00, 0x01, /* bNumInterfaces: 1 interface */ 0x01, /* bConfigurationValue: Configuration value */ - 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ + 0x00, /* iConfiguration: Index of string descriptor + describing the configuration */ #if (USBD_SELF_POWERED == 1U) 0xE0, /* bmAttributes: Bus Powered according to user configuration */ #else 0xA0, /* bmAttributes: Bus Powered according to user configuration */ -#endif +#endif /* USBD_SELF_POWERED */ USBD_MAX_POWER, /* MaxPower (mA) */ /************** Descriptor of Joystick Mouse interface ****************/ @@ -174,111 +183,7 @@ __ALIGN_BEGIN static uint8_t USBD_HID_CfgFSDesc[USB_HID_CONFIG_DESC_SIZ] __ALIGN HID_FS_BINTERVAL, /* bInterval: Polling Interval */ /* 34 */ }; - -/* USB HID device HS Configuration Descriptor */ -__ALIGN_BEGIN static uint8_t USBD_HID_CfgHSDesc[USB_HID_CONFIG_DESC_SIZ] __ALIGN_END = -{ - 0x09, /* bLength: Configuration Descriptor size */ - USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ - USB_HID_CONFIG_DESC_SIZ, /* wTotalLength: Bytes returned */ - 0x00, - 0x01, /* bNumInterfaces: 1 interface */ - 0x01, /* bConfigurationValue: Configuration value */ - 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ -#if (USBD_SELF_POWERED == 1U) - 0xE0, /* bmAttributes: Bus Powered according to user configuration */ -#else - 0xA0, /* bmAttributes: Bus Powered according to user configuration */ -#endif - USBD_MAX_POWER, /* MaxPower (mA) */ - - /************** Descriptor of Joystick Mouse interface ****************/ - /* 09 */ - 0x09, /* bLength: Interface Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: Interface descriptor type */ - 0x00, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x01, /* bNumEndpoints */ - 0x03, /* bInterfaceClass: HID */ - 0x01, /* bInterfaceSubClass : 1=BOOT, 0=no boot */ - 0x02, /* nInterfaceProtocol : 0=none, 1=keyboard, 2=mouse */ - 0, /* iInterface: Index of string descriptor */ - /******************** Descriptor of Joystick Mouse HID ********************/ - /* 18 */ - 0x09, /* bLength: HID Descriptor size */ - HID_DESCRIPTOR_TYPE, /* bDescriptorType: HID */ - 0x11, /* bcdHID: HID Class Spec release number */ - 0x01, - 0x00, /* bCountryCode: Hardware target country */ - 0x01, /* bNumDescriptors: Number of HID class descriptors to follow */ - 0x22, /* bDescriptorType */ - HID_MOUSE_REPORT_DESC_SIZE, /* wItemLength: Total length of Report descriptor */ - 0x00, - /******************** Descriptor of Mouse endpoint ********************/ - /* 27 */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: */ - - HID_EPIN_ADDR, /* bEndpointAddress: Endpoint Address (IN) */ - 0x03, /* bmAttributes: Interrupt endpoint */ - HID_EPIN_SIZE, /* wMaxPacketSize: 4 Bytes max */ - 0x00, - HID_HS_BINTERVAL, /* bInterval: Polling Interval */ - /* 34 */ -}; - -/* USB HID device Other Speed Configuration Descriptor */ -__ALIGN_BEGIN static uint8_t USBD_HID_OtherSpeedCfgDesc[USB_HID_CONFIG_DESC_SIZ] __ALIGN_END = -{ - 0x09, /* bLength: Configuration Descriptor size */ - USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ - USB_HID_CONFIG_DESC_SIZ, /* wTotalLength: Bytes returned */ - 0x00, - 0x01, /* bNumInterfaces: 1 interface */ - 0x01, /* bConfigurationValue: Configuration value */ - 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ -#if (USBD_SELF_POWERED == 1U) - 0xE0, /* bmAttributes: Bus Powered according to user configuration */ -#else - 0xA0, /* bmAttributes: Bus Powered according to user configuration */ -#endif - USBD_MAX_POWER, /* MaxPower (mA) */ - - /************** Descriptor of Joystick Mouse interface ****************/ - /* 09 */ - 0x09, /* bLength: Interface Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: Interface descriptor type */ - 0x00, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x01, /* bNumEndpoints */ - 0x03, /* bInterfaceClass: HID */ - 0x01, /* bInterfaceSubClass : 1=BOOT, 0=no boot */ - 0x02, /* nInterfaceProtocol : 0=none, 1=keyboard, 2=mouse */ - 0, /* iInterface: Index of string descriptor */ - /******************** Descriptor of Joystick Mouse HID ********************/ - /* 18 */ - 0x09, /* bLength: HID Descriptor size */ - HID_DESCRIPTOR_TYPE, /* bDescriptorType: HID */ - 0x11, /* bcdHID: HID Class Spec release number */ - 0x01, - 0x00, /* bCountryCode: Hardware target country */ - 0x01, /* bNumDescriptors: Number of HID class descriptors to follow */ - 0x22, /* bDescriptorType */ - HID_MOUSE_REPORT_DESC_SIZE, /* wItemLength: Total length of Report descriptor */ - 0x00, - /******************** Descriptor of Mouse endpoint ********************/ - /* 27 */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: */ - - HID_EPIN_ADDR, /* bEndpointAddress: Endpoint Address (IN) */ - 0x03, /* bmAttributes: Interrupt endpoint */ - HID_EPIN_SIZE, /* wMaxPacketSize: 4 Bytes max */ - 0x00, - HID_FS_BINTERVAL, /* bInterval: Polling Interval */ - /* 34 */ -}; - +#endif /* USE_USBD_COMPOSITE */ /* USB HID device Configuration Descriptor */ __ALIGN_BEGIN static uint8_t USBD_HID_Desc[USB_HID_DESC_SIZ] __ALIGN_END = @@ -295,6 +200,7 @@ __ALIGN_BEGIN static uint8_t USBD_HID_Desc[USB_HID_DESC_SIZ] __ALIGN_END = 0x00, }; +#ifndef USE_USBD_COMPOSITE /* USB Standard Device Descriptor */ __ALIGN_BEGIN static uint8_t USBD_HID_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_DESC] __ALIGN_END = { @@ -309,6 +215,7 @@ __ALIGN_BEGIN static uint8_t USBD_HID_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_ 0x01, 0x00, }; +#endif /* USE_USBD_COMPOSITE */ __ALIGN_BEGIN static uint8_t HID_MOUSE_ReportDesc[HID_MOUSE_REPORT_DESC_SIZE] __ALIGN_END = { @@ -352,6 +259,8 @@ __ALIGN_BEGIN static uint8_t HID_MOUSE_ReportDesc[HID_MOUSE_REPORT_DESC_SIZE] __ 0xC0 /* End Collection */ }; +static uint8_t HIDInEpAdd = HID_EPIN_ADDR; + /** * @} */ @@ -377,26 +286,32 @@ static uint8_t USBD_HID_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) if (hhid == NULL) { - pdev->pClassData = NULL; + pdev->pClassDataCmsit[pdev->classId] = NULL; return (uint8_t)USBD_EMEM; } - pdev->pClassData = (void *)hhid; + pdev->pClassDataCmsit[pdev->classId] = (void *)hhid; + pdev->pClassData = pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + HIDInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (pdev->dev_speed == USBD_SPEED_HIGH) { - pdev->ep_in[HID_EPIN_ADDR & 0xFU].bInterval = HID_HS_BINTERVAL; + pdev->ep_in[HIDInEpAdd & 0xFU].bInterval = HID_HS_BINTERVAL; } else /* LOW and FULL-speed endpoints */ { - pdev->ep_in[HID_EPIN_ADDR & 0xFU].bInterval = HID_FS_BINTERVAL; + pdev->ep_in[HIDInEpAdd & 0xFU].bInterval = HID_FS_BINTERVAL; } /* Open EP IN */ - (void)USBD_LL_OpenEP(pdev, HID_EPIN_ADDR, USBD_EP_TYPE_INTR, HID_EPIN_SIZE); - pdev->ep_in[HID_EPIN_ADDR & 0xFU].is_used = 1U; + (void)USBD_LL_OpenEP(pdev, HIDInEpAdd, USBD_EP_TYPE_INTR, HID_EPIN_SIZE); + pdev->ep_in[HIDInEpAdd & 0xFU].is_used = 1U; - hhid->state = HID_IDLE; + hhid->state = USBD_HID_IDLE; return (uint8_t)USBD_OK; } @@ -412,16 +327,21 @@ static uint8_t USBD_HID_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) { UNUSED(cfgidx); +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + HIDInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + /* Close HID EPs */ - (void)USBD_LL_CloseEP(pdev, HID_EPIN_ADDR); - pdev->ep_in[HID_EPIN_ADDR & 0xFU].is_used = 0U; - pdev->ep_in[HID_EPIN_ADDR & 0xFU].bInterval = 0U; + (void)USBD_LL_CloseEP(pdev, HIDInEpAdd); + pdev->ep_in[HIDInEpAdd & 0xFU].is_used = 0U; + pdev->ep_in[HIDInEpAdd & 0xFU].bInterval = 0U; /* Free allocated memory */ - if (pdev->pClassData != NULL) + if (pdev->pClassDataCmsit[pdev->classId] != NULL) { - (void)USBD_free(pdev->pClassData); - pdev->pClassData = NULL; + (void)USBD_free(pdev->pClassDataCmsit[pdev->classId]); + pdev->pClassDataCmsit[pdev->classId] = NULL; } return (uint8_t)USBD_OK; @@ -436,7 +356,7 @@ static uint8_t USBD_HID_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) */ static uint8_t USBD_HID_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_HID_HandleTypeDef *hhid = (USBD_HID_HandleTypeDef *)pdev->pClassData; + USBD_HID_HandleTypeDef *hhid = (USBD_HID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; USBD_StatusTypeDef ret = USBD_OK; uint16_t len; uint8_t *pbuf; @@ -452,19 +372,19 @@ static uint8_t USBD_HID_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *re case USB_REQ_TYPE_CLASS : switch (req->bRequest) { - case HID_REQ_SET_PROTOCOL: + case USBD_HID_REQ_SET_PROTOCOL: hhid->Protocol = (uint8_t)(req->wValue); break; - case HID_REQ_GET_PROTOCOL: + case USBD_HID_REQ_GET_PROTOCOL: (void)USBD_CtlSendData(pdev, (uint8_t *)&hhid->Protocol, 1U); break; - case HID_REQ_SET_IDLE: + case USBD_HID_REQ_SET_IDLE: hhid->IdleState = (uint8_t)(req->wValue >> 8); break; - case HID_REQ_GET_IDLE: + case USBD_HID_REQ_GET_IDLE: (void)USBD_CtlSendData(pdev, (uint8_t *)&hhid->IdleState, 1U); break; @@ -552,28 +472,41 @@ static uint8_t USBD_HID_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *re return (uint8_t)ret; } + /** * @brief USBD_HID_SendReport * Send HID Report * @param pdev: device instance * @param buff: pointer to report + * @param ClassId: The Class ID * @retval status */ +#ifdef USE_USBD_COMPOSITE +uint8_t USBD_HID_SendReport(USBD_HandleTypeDef *pdev, uint8_t *report, uint16_t len, uint8_t ClassId) +{ + USBD_HID_HandleTypeDef *hhid = (USBD_HID_HandleTypeDef *)pdev->pClassDataCmsit[ClassId]; +#else uint8_t USBD_HID_SendReport(USBD_HandleTypeDef *pdev, uint8_t *report, uint16_t len) { - USBD_HID_HandleTypeDef *hhid = (USBD_HID_HandleTypeDef *)pdev->pClassData; + USBD_HID_HandleTypeDef *hhid = (USBD_HID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#endif /* USE_USBD_COMPOSITE */ if (hhid == NULL) { return (uint8_t)USBD_FAIL; } +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + HIDInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, ClassId); +#endif /* USE_USBD_COMPOSITE */ + if (pdev->dev_state == USBD_STATE_CONFIGURED) { - if (hhid->state == HID_IDLE) + if (hhid->state == USBD_HID_IDLE) { - hhid->state = HID_BUSY; - (void)USBD_LL_Transmit(pdev, HID_EPIN_ADDR, report, len); + hhid->state = USBD_HID_BUSY; + (void)USBD_LL_Transmit(pdev, HIDInEpAdd, report, len); } } @@ -608,6 +541,7 @@ uint32_t USBD_HID_GetPollingInterval(USBD_HandleTypeDef *pdev) return ((uint32_t)(polling_interval)); } +#ifndef USE_USBD_COMPOSITE /** * @brief USBD_HID_GetCfgFSDesc * return FS configuration descriptor @@ -617,9 +551,15 @@ uint32_t USBD_HID_GetPollingInterval(USBD_HandleTypeDef *pdev) */ static uint8_t *USBD_HID_GetFSCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_HID_CfgFSDesc); + USBD_EpDescTypeDef *pEpDesc = USBD_GetEpDesc(USBD_HID_CfgDesc, HID_EPIN_ADDR); + + if (pEpDesc != NULL) + { + pEpDesc->bInterval = HID_FS_BINTERVAL; + } - return USBD_HID_CfgFSDesc; + *length = (uint16_t)sizeof(USBD_HID_CfgDesc); + return USBD_HID_CfgDesc; } /** @@ -631,9 +571,15 @@ static uint8_t *USBD_HID_GetFSCfgDesc(uint16_t *length) */ static uint8_t *USBD_HID_GetHSCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_HID_CfgHSDesc); + USBD_EpDescTypeDef *pEpDesc = USBD_GetEpDesc(USBD_HID_CfgDesc, HID_EPIN_ADDR); - return USBD_HID_CfgHSDesc; + if (pEpDesc != NULL) + { + pEpDesc->bInterval = HID_HS_BINTERVAL; + } + + *length = (uint16_t)sizeof(USBD_HID_CfgDesc); + return USBD_HID_CfgDesc; } /** @@ -645,10 +591,17 @@ static uint8_t *USBD_HID_GetHSCfgDesc(uint16_t *length) */ static uint8_t *USBD_HID_GetOtherSpeedCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_HID_OtherSpeedCfgDesc); + USBD_EpDescTypeDef *pEpDesc = USBD_GetEpDesc(USBD_HID_CfgDesc, HID_EPIN_ADDR); + + if (pEpDesc != NULL) + { + pEpDesc->bInterval = HID_FS_BINTERVAL; + } - return USBD_HID_OtherSpeedCfgDesc; + *length = (uint16_t)sizeof(USBD_HID_CfgDesc); + return USBD_HID_CfgDesc; } +#endif /* USE_USBD_COMPOSITE */ /** * @brief USBD_HID_DataIn @@ -662,12 +615,12 @@ static uint8_t USBD_HID_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) UNUSED(epnum); /* Ensure that the FIFO is empty before a new transfer, this condition could be caused by a new transfer before the end of the previous transfer */ - ((USBD_HID_HandleTypeDef *)pdev->pClassData)->state = HID_IDLE; + ((USBD_HID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId])->state = USBD_HID_IDLE; return (uint8_t)USBD_OK; } - +#ifndef USE_USBD_COMPOSITE /** * @brief DeviceQualifierDescriptor * return Device Qualifier descriptor @@ -680,7 +633,7 @@ static uint8_t *USBD_HID_GetDeviceQualifierDesc(uint16_t *length) return USBD_HID_DeviceQualifierDesc; } - +#endif /* USE_USBD_COMPOSITE */ /** * @} */ @@ -695,4 +648,3 @@ static uint8_t *USBD_HID_GetDeviceQualifierDesc(uint16_t *length) * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc.h index a0bbab90..e55fef48 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -55,9 +54,13 @@ extern "C" { #define BOT_RESET 0xFF #define USB_MSC_CONFIG_DESC_SIZ 32 - +#ifndef MSC_EPIN_ADDR #define MSC_EPIN_ADDR 0x81U +#endif /* MSC_EPIN_ADDR */ + +#ifndef MSC_EPOUT_ADDR #define MSC_EPOUT_ADDR 0x01U +#endif /* MSC_EPOUT_ADDR */ /** * @} @@ -101,8 +104,7 @@ typedef struct uint32_t scsi_blk_addr; uint32_t scsi_blk_len; -} -USBD_MSC_BOT_HandleTypeDef; +} USBD_MSC_BOT_HandleTypeDef; /* Structure for MSC process */ extern USBD_ClassTypeDef USBD_MSC; @@ -126,5 +128,3 @@ uint8_t USBD_MSC_RegisterStorage(USBD_HandleTypeDef *pdev, /** * @} */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_bot.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_bot.h index fb99095c..8550a390 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_bot.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_bot.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -87,8 +86,7 @@ typedef struct uint8_t bCBLength; uint8_t CB[16]; uint8_t ReservedForAlign; -} -USBD_MSC_BOT_CBWTypeDef; +} USBD_MSC_BOT_CBWTypeDef; typedef struct @@ -98,8 +96,7 @@ typedef struct uint32_t dDataResidue; uint8_t bStatus; uint8_t ReservedForAlign[3]; -} -USBD_MSC_BOT_CSWTypeDef; +} USBD_MSC_BOT_CSWTypeDef; /** * @} @@ -146,5 +143,4 @@ void MSC_BOT_CplClrFeature(USBD_HandleTypeDef *pdev, /** * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_data.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_data.h index 3aacf04a..f946b957 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_data.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_data.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -101,5 +100,3 @@ extern uint8_t MSC_Mode_Sense10_data[MODE_SENSE10_LEN]; /** * @} */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_scsi.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_scsi.h index 9bf10a9e..477affbb 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_scsi.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_scsi.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -83,7 +82,7 @@ extern "C" { #define UNIT_ATTENTION 6U #define DATA_PROTECT 7U #define BLANK_CHECK 8U -#define VENDOR_SPECIFIC 9U +#define MSC_VENDOR_SPECIFIC 9U #define COPY_ABORTED 10U #define ABORTED_COMMAND 11U #define VOLUME_OVERFLOW 13U @@ -181,5 +180,3 @@ void SCSI_SenseCode(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t sKey, * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ - diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_storage_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_storage_template.h index 9591a3ba..cb8b89ad 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_storage_template.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Inc/usbd_msc_storage_template.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -94,4 +93,4 @@ extern USBD_StorageTypeDef USBD_MSC_Template_fops; /** * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ + diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc.c index 09a61a6d..7f2152f3 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc.c @@ -4,6 +4,17 @@ * @author MCD Application Team * @brief This file provides all the MSC core functions. * + ****************************************************************************** + * @attention + * + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** * @verbatim * * =================================================================== @@ -19,17 +30,6 @@ * @endverbatim * ****************************************************************************** - * @attention - * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> - * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** */ /* BSPDependencies @@ -86,11 +86,12 @@ uint8_t USBD_MSC_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); uint8_t USBD_MSC_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum); uint8_t USBD_MSC_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum); +#ifndef USE_USBD_COMPOSITE uint8_t *USBD_MSC_GetHSCfgDesc(uint16_t *length); uint8_t *USBD_MSC_GetFSCfgDesc(uint16_t *length); uint8_t *USBD_MSC_GetOtherSpeedCfgDesc(uint16_t *length); uint8_t *USBD_MSC_GetDeviceQualifierDescriptor(uint16_t *length); - +#endif /* USE_USBD_COMPOSITE */ /** * @} */ @@ -113,62 +114,24 @@ USBD_ClassTypeDef USBD_MSC = NULL, /*SOF */ NULL, NULL, +#ifdef USE_USBD_COMPOSITE + NULL, + NULL, + NULL, + NULL, +#else USBD_MSC_GetHSCfgDesc, USBD_MSC_GetFSCfgDesc, USBD_MSC_GetOtherSpeedCfgDesc, USBD_MSC_GetDeviceQualifierDescriptor, +#endif /* USE_USBD_COMPOSITE */ }; /* USB Mass storage device Configuration Descriptor */ -/* All Descriptors (Configuration, Interface, Endpoint, Class, Vendor */ -__ALIGN_BEGIN static uint8_t USBD_MSC_CfgHSDesc[USB_MSC_CONFIG_DESC_SIZ] __ALIGN_END = -{ - 0x09, /* bLength: Configuration Descriptor size */ - USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ - USB_MSC_CONFIG_DESC_SIZ, - - 0x00, - 0x01, /* bNumInterfaces: 1 interface */ - 0x01, /* bConfigurationValue */ - 0x04, /* iConfiguration */ -#if (USBD_SELF_POWERED == 1U) - 0xC0, /* bmAttributes: Bus Powered according to user configuration */ -#else - 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif - USBD_MAX_POWER, /* MaxPower (mA) */ - - /******************** Mass Storage interface ********************/ - 0x09, /* bLength: Interface Descriptor size */ - 0x04, /* bDescriptorType: */ - 0x00, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x02, /* bNumEndpoints */ - 0x08, /* bInterfaceClass: MSC Class */ - 0x06, /* bInterfaceSubClass : SCSI transparent */ - 0x50, /* nInterfaceProtocol */ - 0x05, /* iInterface */ - /******************** Mass Storage Endpoints ********************/ - 0x07, /* Endpoint descriptor length = 7 */ - 0x05, /* Endpoint descriptor type */ - MSC_EPIN_ADDR, /* Endpoint address (IN, address 1) */ - 0x02, /* Bulk endpoint type */ - LOBYTE(MSC_MAX_HS_PACKET), - HIBYTE(MSC_MAX_HS_PACKET), - 0x00, /* Polling interval in milliseconds */ - - 0x07, /* Endpoint descriptor length = 7 */ - 0x05, /* Endpoint descriptor type */ - MSC_EPOUT_ADDR, /* Endpoint address (OUT, address 1) */ - 0x02, /* Bulk endpoint type */ - LOBYTE(MSC_MAX_HS_PACKET), - HIBYTE(MSC_MAX_HS_PACKET), - 0x00 /* Polling interval in milliseconds */ -}; - +#ifndef USE_USBD_COMPOSITE /* USB Mass storage device Configuration Descriptor */ /* All Descriptors (Configuration, Interface, Endpoint, Class, Vendor */ -__ALIGN_BEGIN static uint8_t USBD_MSC_CfgFSDesc[USB_MSC_CONFIG_DESC_SIZ] __ALIGN_END = +__ALIGN_BEGIN static uint8_t USBD_MSC_CfgDesc[USB_MSC_CONFIG_DESC_SIZ] __ALIGN_END = { 0x09, /* bLength: Configuration Descriptor size */ USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ @@ -182,7 +145,7 @@ __ALIGN_BEGIN static uint8_t USBD_MSC_CfgFSDesc[USB_MSC_CONFIG_DESC_SIZ] __ALIG 0xC0, /* bmAttributes: Bus Powered according to user configuration */ #else 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif +#endif /* USBD_SELF_POWERED */ USBD_MAX_POWER, /* MaxPower (mA) */ /******************** Mass Storage interface ********************/ @@ -213,51 +176,6 @@ __ALIGN_BEGIN static uint8_t USBD_MSC_CfgFSDesc[USB_MSC_CONFIG_DESC_SIZ] __ALIG 0x00 /* Polling interval in milliseconds */ }; -__ALIGN_BEGIN static uint8_t USBD_MSC_OtherSpeedCfgDesc[USB_MSC_CONFIG_DESC_SIZ] __ALIGN_END = -{ - 0x09, /* bLength: Configuration Descriptor size */ - USB_DESC_TYPE_OTHER_SPEED_CONFIGURATION, - USB_MSC_CONFIG_DESC_SIZ, - - 0x00, - 0x01, /* bNumInterfaces: 1 interface */ - 0x01, /* bConfigurationValue */ - 0x04, /* iConfiguration */ -#if (USBD_SELF_POWERED == 1U) - 0xC0, /* bmAttributes: Bus Powered according to user configuration */ -#else - 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif - USBD_MAX_POWER, /* MaxPower (mA) */ - - /******************** Mass Storage interface ********************/ - 0x09, /* bLength: Interface Descriptor size */ - 0x04, /* bDescriptorType */ - 0x00, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x02, /* bNumEndpoints */ - 0x08, /* bInterfaceClass: MSC Class */ - 0x06, /* bInterfaceSubClass : SCSI transparent command set */ - 0x50, /* nInterfaceProtocol */ - 0x05, /* iInterface */ - /******************** Mass Storage Endpoints ********************/ - 0x07, /* Endpoint descriptor length = 7 */ - 0x05, /* Endpoint descriptor type */ - MSC_EPIN_ADDR, /* Endpoint address (IN, address 1) */ - 0x02, /* Bulk endpoint type */ - 0x40, - 0x00, - 0x00, /* Polling interval in milliseconds */ - - 0x07, /* Endpoint descriptor length = 7 */ - 0x05, /* Endpoint descriptor type */ - MSC_EPOUT_ADDR, /* Endpoint address (OUT, address 1) */ - 0x02, /* Bulk endpoint type */ - 0x40, - 0x00, - 0x00 /* Polling interval in milliseconds */ -}; - /* USB Standard Device Descriptor */ __ALIGN_BEGIN static uint8_t USBD_MSC_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_DESC] __ALIGN_END = { @@ -272,6 +190,11 @@ __ALIGN_BEGIN static uint8_t USBD_MSC_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_ 0x01, 0x00, }; +#endif /* USE_USBD_COMPOSITE */ + +uint8_t MSCInEpAdd = MSC_EPIN_ADDR; +uint8_t MSCOutEpAdd = MSC_EPOUT_ADDR; + /** * @} */ @@ -297,31 +220,38 @@ uint8_t USBD_MSC_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) if (hmsc == NULL) { - pdev->pClassData = NULL; + pdev->pClassDataCmsit[pdev->classId] = NULL; return (uint8_t)USBD_EMEM; } - pdev->pClassData = (void *)hmsc; + pdev->pClassDataCmsit[pdev->classId] = (void *)hmsc; + pdev->pClassData = pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MSCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + MSCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (pdev->dev_speed == USBD_SPEED_HIGH) { /* Open EP OUT */ - (void)USBD_LL_OpenEP(pdev, MSC_EPOUT_ADDR, USBD_EP_TYPE_BULK, MSC_MAX_HS_PACKET); - pdev->ep_out[MSC_EPOUT_ADDR & 0xFU].is_used = 1U; + (void)USBD_LL_OpenEP(pdev, MSCOutEpAdd, USBD_EP_TYPE_BULK, MSC_MAX_HS_PACKET); + pdev->ep_out[MSCOutEpAdd & 0xFU].is_used = 1U; /* Open EP IN */ - (void)USBD_LL_OpenEP(pdev, MSC_EPIN_ADDR, USBD_EP_TYPE_BULK, MSC_MAX_HS_PACKET); - pdev->ep_in[MSC_EPIN_ADDR & 0xFU].is_used = 1U; + (void)USBD_LL_OpenEP(pdev, MSCInEpAdd, USBD_EP_TYPE_BULK, MSC_MAX_HS_PACKET); + pdev->ep_in[MSCInEpAdd & 0xFU].is_used = 1U; } else { /* Open EP OUT */ - (void)USBD_LL_OpenEP(pdev, MSC_EPOUT_ADDR, USBD_EP_TYPE_BULK, MSC_MAX_FS_PACKET); - pdev->ep_out[MSC_EPOUT_ADDR & 0xFU].is_used = 1U; + (void)USBD_LL_OpenEP(pdev, MSCOutEpAdd, USBD_EP_TYPE_BULK, MSC_MAX_FS_PACKET); + pdev->ep_out[MSCOutEpAdd & 0xFU].is_used = 1U; /* Open EP IN */ - (void)USBD_LL_OpenEP(pdev, MSC_EPIN_ADDR, USBD_EP_TYPE_BULK, MSC_MAX_FS_PACKET); - pdev->ep_in[MSC_EPIN_ADDR & 0xFU].is_used = 1U; + (void)USBD_LL_OpenEP(pdev, MSCInEpAdd, USBD_EP_TYPE_BULK, MSC_MAX_FS_PACKET); + pdev->ep_in[MSCInEpAdd & 0xFU].is_used = 1U; } /* Init the BOT layer */ @@ -341,21 +271,28 @@ uint8_t USBD_MSC_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) { UNUSED(cfgidx); +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MSCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + MSCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + /* Close MSC EPs */ - (void)USBD_LL_CloseEP(pdev, MSC_EPOUT_ADDR); - pdev->ep_out[MSC_EPOUT_ADDR & 0xFU].is_used = 0U; + (void)USBD_LL_CloseEP(pdev, MSCOutEpAdd); + pdev->ep_out[MSCOutEpAdd & 0xFU].is_used = 0U; /* Close EP IN */ - (void)USBD_LL_CloseEP(pdev, MSC_EPIN_ADDR); - pdev->ep_in[MSC_EPIN_ADDR & 0xFU].is_used = 0U; + (void)USBD_LL_CloseEP(pdev, MSCInEpAdd); + pdev->ep_in[MSCInEpAdd & 0xFU].is_used = 0U; /* Free MSC Class Resources */ - if (pdev->pClassData != NULL) + if (pdev->pClassDataCmsit[pdev->classId] != NULL) { /* De-Init the BOT layer */ MSC_BOT_DeInit(pdev); - (void)USBD_free(pdev->pClassData); + (void)USBD_free(pdev->pClassDataCmsit[pdev->classId]); + pdev->pClassDataCmsit[pdev->classId] = NULL; pdev->pClassData = NULL; } @@ -370,10 +307,16 @@ uint8_t USBD_MSC_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) */ uint8_t USBD_MSC_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; USBD_StatusTypeDef ret = USBD_OK; uint16_t status_info = 0U; +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MSCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + MSCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + if (hmsc == NULL) { return (uint8_t)USBD_FAIL; @@ -389,7 +332,7 @@ uint8_t USBD_MSC_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) if ((req->wValue == 0U) && (req->wLength == 1U) && ((req->bmRequest & 0x80U) == 0x80U)) { - hmsc->max_lun = (uint32_t)((USBD_StorageTypeDef *)pdev->pUserData)->GetMaxLun(); + hmsc->max_lun = (uint32_t)((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->GetMaxLun(); (void)USBD_CtlSendData(pdev, (uint8_t *)&hmsc->max_lun, 1U); } else @@ -515,7 +458,7 @@ uint8_t USBD_MSC_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) return (uint8_t)USBD_OK; } - +#ifndef USE_USBD_COMPOSITE /** * @brief USBD_MSC_GetHSCfgDesc * return configuration descriptor @@ -524,9 +467,21 @@ uint8_t USBD_MSC_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) */ uint8_t *USBD_MSC_GetHSCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_MSC_CfgHSDesc); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_MSC_CfgDesc, MSC_EPIN_ADDR); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_MSC_CfgDesc, MSC_EPOUT_ADDR); + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = MSC_MAX_HS_PACKET; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = MSC_MAX_HS_PACKET; + } - return USBD_MSC_CfgHSDesc; + *length = (uint16_t)sizeof(USBD_MSC_CfgDesc); + return USBD_MSC_CfgDesc; } /** @@ -537,9 +492,21 @@ uint8_t *USBD_MSC_GetHSCfgDesc(uint16_t *length) */ uint8_t *USBD_MSC_GetFSCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_MSC_CfgFSDesc); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_MSC_CfgDesc, MSC_EPIN_ADDR); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_MSC_CfgDesc, MSC_EPOUT_ADDR); - return USBD_MSC_CfgFSDesc; + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = MSC_MAX_FS_PACKET; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = MSC_MAX_FS_PACKET; + } + + *length = (uint16_t)sizeof(USBD_MSC_CfgDesc); + return USBD_MSC_CfgDesc; } /** @@ -550,9 +517,21 @@ uint8_t *USBD_MSC_GetFSCfgDesc(uint16_t *length) */ uint8_t *USBD_MSC_GetOtherSpeedCfgDesc(uint16_t *length) { - *length = (uint16_t)sizeof(USBD_MSC_OtherSpeedCfgDesc); + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_MSC_CfgDesc, MSC_EPIN_ADDR); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_MSC_CfgDesc, MSC_EPOUT_ADDR); - return USBD_MSC_OtherSpeedCfgDesc; + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = MSC_MAX_FS_PACKET; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = MSC_MAX_FS_PACKET; + } + + *length = (uint16_t)sizeof(USBD_MSC_CfgDesc); + return USBD_MSC_CfgDesc; } /** * @brief DeviceQualifierDescriptor @@ -566,7 +545,7 @@ uint8_t *USBD_MSC_GetDeviceQualifierDescriptor(uint16_t *length) return USBD_MSC_DeviceQualifierDesc; } - +#endif /* USE_USBD_COMPOSITE */ /** * @brief USBD_MSC_RegisterStorage * @param fops: storage callback @@ -579,7 +558,7 @@ uint8_t USBD_MSC_RegisterStorage(USBD_HandleTypeDef *pdev, USBD_StorageTypeDef * return (uint8_t)USBD_FAIL; } - pdev->pUserData = fops; + pdev->pUserData[pdev->classId] = fops; return (uint8_t)USBD_OK; } @@ -598,4 +577,3 @@ uint8_t USBD_MSC_RegisterStorage(USBD_HandleTypeDef *pdev, USBD_StorageTypeDef * * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_bot.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_bot.c index bbe94303..c51b0136 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_bot.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_bot.c @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -67,7 +66,8 @@ EndBSPDependencies */ /** @defgroup MSC_BOT_Private_Variables * @{ */ - +extern uint8_t MSCInEpAdd; +extern uint8_t MSCOutEpAdd; /** * @} */ @@ -97,7 +97,13 @@ static void MSC_BOT_Abort(USBD_HandleTypeDef *pdev); */ void MSC_BOT_Init(USBD_HandleTypeDef *pdev) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MSCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + MSCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (hmsc == NULL) { @@ -111,13 +117,13 @@ void MSC_BOT_Init(USBD_HandleTypeDef *pdev) hmsc->scsi_sense_head = 0U; hmsc->scsi_medium_state = SCSI_MEDIUM_UNLOCKED; - ((USBD_StorageTypeDef *)pdev->pUserData)->Init(0U); + ((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->Init(0U); - (void)USBD_LL_FlushEP(pdev, MSC_EPOUT_ADDR); - (void)USBD_LL_FlushEP(pdev, MSC_EPIN_ADDR); + (void)USBD_LL_FlushEP(pdev, MSCOutEpAdd); + (void)USBD_LL_FlushEP(pdev, MSCInEpAdd); /* Prepare EP to Receive First BOT Cmd */ - (void)USBD_LL_PrepareReceive(pdev, MSC_EPOUT_ADDR, (uint8_t *)&hmsc->cbw, + (void)USBD_LL_PrepareReceive(pdev, MSCOutEpAdd, (uint8_t *)&hmsc->cbw, USBD_BOT_CBW_LENGTH); } @@ -129,7 +135,13 @@ void MSC_BOT_Init(USBD_HandleTypeDef *pdev) */ void MSC_BOT_Reset(USBD_HandleTypeDef *pdev) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MSCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + MSCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (hmsc == NULL) { @@ -139,11 +151,11 @@ void MSC_BOT_Reset(USBD_HandleTypeDef *pdev) hmsc->bot_state = USBD_BOT_IDLE; hmsc->bot_status = USBD_BOT_STATUS_RECOVERY; - (void)USBD_LL_ClearStallEP(pdev, MSC_EPIN_ADDR); - (void)USBD_LL_ClearStallEP(pdev, MSC_EPOUT_ADDR); + (void)USBD_LL_ClearStallEP(pdev, MSCInEpAdd); + (void)USBD_LL_ClearStallEP(pdev, MSCOutEpAdd); /* Prepare EP to Receive First BOT Cmd */ - (void)USBD_LL_PrepareReceive(pdev, MSC_EPOUT_ADDR, (uint8_t *)&hmsc->cbw, + (void)USBD_LL_PrepareReceive(pdev, MSCOutEpAdd, (uint8_t *)&hmsc->cbw, USBD_BOT_CBW_LENGTH); } @@ -155,7 +167,7 @@ void MSC_BOT_Reset(USBD_HandleTypeDef *pdev) */ void MSC_BOT_DeInit(USBD_HandleTypeDef *pdev) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc != NULL) { @@ -174,7 +186,7 @@ void MSC_BOT_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) { UNUSED(epnum); - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { @@ -210,7 +222,7 @@ void MSC_BOT_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) { UNUSED(epnum); - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { @@ -243,7 +255,13 @@ void MSC_BOT_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) */ static void MSC_BOT_CBW_Decode(USBD_HandleTypeDef *pdev) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MSCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + MSCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (hmsc == NULL) { @@ -253,7 +271,7 @@ static void MSC_BOT_CBW_Decode(USBD_HandleTypeDef *pdev) hmsc->csw.dTag = hmsc->cbw.dTag; hmsc->csw.dDataResidue = hmsc->cbw.dDataLength; - if ((USBD_LL_GetRxDataSize(pdev, MSC_EPOUT_ADDR) != USBD_BOT_CBW_LENGTH) || + if ((USBD_LL_GetRxDataSize(pdev, MSCOutEpAdd) != USBD_BOT_CBW_LENGTH) || (hmsc->cbw.dSignature != USBD_BOT_CBW_SIGNATURE) || (hmsc->cbw.bLUN > 1U) || (hmsc->cbw.bCBLength < 1U) || (hmsc->cbw.bCBLength > 16U)) @@ -311,20 +329,28 @@ static void MSC_BOT_CBW_Decode(USBD_HandleTypeDef *pdev) */ static void MSC_BOT_SendData(USBD_HandleTypeDef *pdev, uint8_t *pbuf, uint32_t len) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + + uint32_t length; - uint32_t length = MIN(hmsc->cbw.dDataLength, len); +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MSCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + MSCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (hmsc == NULL) { return; } + length = MIN(hmsc->cbw.dDataLength, len); + hmsc->csw.dDataResidue -= len; hmsc->csw.bStatus = USBD_CSW_CMD_PASSED; hmsc->bot_state = USBD_BOT_SEND_DATA; - (void)USBD_LL_Transmit(pdev, MSC_EPIN_ADDR, pbuf, length); + (void)USBD_LL_Transmit(pdev, MSCInEpAdd, pbuf, length); } /** @@ -336,7 +362,13 @@ static void MSC_BOT_SendData(USBD_HandleTypeDef *pdev, uint8_t *pbuf, uint32_t */ void MSC_BOT_SendCSW(USBD_HandleTypeDef *pdev, uint8_t CSW_Status) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MSCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + MSCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (hmsc == NULL) { @@ -347,11 +379,11 @@ void MSC_BOT_SendCSW(USBD_HandleTypeDef *pdev, uint8_t CSW_Status) hmsc->csw.bStatus = CSW_Status; hmsc->bot_state = USBD_BOT_IDLE; - (void)USBD_LL_Transmit(pdev, MSC_EPIN_ADDR, (uint8_t *)&hmsc->csw, + (void)USBD_LL_Transmit(pdev, MSCInEpAdd, (uint8_t *)&hmsc->csw, USBD_BOT_CSW_LENGTH); /* Prepare EP to Receive next Cmd */ - (void)USBD_LL_PrepareReceive(pdev, MSC_EPOUT_ADDR, (uint8_t *)&hmsc->cbw, + (void)USBD_LL_PrepareReceive(pdev, MSCOutEpAdd, (uint8_t *)&hmsc->cbw, USBD_BOT_CBW_LENGTH); } @@ -364,7 +396,13 @@ void MSC_BOT_SendCSW(USBD_HandleTypeDef *pdev, uint8_t CSW_Status) static void MSC_BOT_Abort(USBD_HandleTypeDef *pdev) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MSCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + MSCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (hmsc == NULL) { @@ -375,15 +413,15 @@ static void MSC_BOT_Abort(USBD_HandleTypeDef *pdev) (hmsc->cbw.dDataLength != 0U) && (hmsc->bot_status == USBD_BOT_STATUS_NORMAL)) { - (void)USBD_LL_StallEP(pdev, MSC_EPOUT_ADDR); + (void)USBD_LL_StallEP(pdev, MSCOutEpAdd); } - (void)USBD_LL_StallEP(pdev, MSC_EPIN_ADDR); + (void)USBD_LL_StallEP(pdev, MSCInEpAdd); if (hmsc->bot_status == USBD_BOT_STATUS_ERROR) { - (void)USBD_LL_StallEP(pdev, MSC_EPIN_ADDR); - (void)USBD_LL_StallEP(pdev, MSC_EPOUT_ADDR); + (void)USBD_LL_StallEP(pdev, MSCInEpAdd); + (void)USBD_LL_StallEP(pdev, MSCOutEpAdd); } } @@ -397,7 +435,13 @@ static void MSC_BOT_Abort(USBD_HandleTypeDef *pdev) void MSC_BOT_CplClrFeature(USBD_HandleTypeDef *pdev, uint8_t epnum) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MSCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + MSCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (hmsc == NULL) { @@ -406,8 +450,8 @@ void MSC_BOT_CplClrFeature(USBD_HandleTypeDef *pdev, uint8_t epnum) if (hmsc->bot_status == USBD_BOT_STATUS_ERROR) /* Bad CBW Signature */ { - (void)USBD_LL_StallEP(pdev, MSC_EPIN_ADDR); - (void)USBD_LL_StallEP(pdev, MSC_EPOUT_ADDR); + (void)USBD_LL_StallEP(pdev, MSCInEpAdd); + (void)USBD_LL_StallEP(pdev, MSCOutEpAdd); } else if (((epnum & 0x80U) == 0x80U) && (hmsc->bot_status != USBD_BOT_STATUS_RECOVERY)) { @@ -432,4 +476,3 @@ void MSC_BOT_CplClrFeature(USBD_HandleTypeDef *pdev, uint8_t epnum) * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_data.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_data.c index 34ef4438..fabd835c 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_data.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_data.c @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -180,4 +179,3 @@ uint8_t MSC_Mode_Sense10_data[MODE_SENSE10_LEN] = * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_scsi.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_scsi.c index 02fdc833..efa85a40 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_scsi.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_scsi.c @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -68,7 +67,8 @@ EndBSPDependencies */ /** @defgroup MSC_SCSI_Private_Variables * @{ */ - +extern uint8_t MSCInEpAdd; +extern uint8_t MSCOutEpAdd; /** * @} */ @@ -121,7 +121,7 @@ static int8_t SCSI_UpdateBotData(USBD_MSC_BOT_HandleTypeDef *hmsc, int8_t SCSI_ProcessCmd(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *cmd) { int8_t ret; - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { @@ -211,7 +211,7 @@ int8_t SCSI_ProcessCmd(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *cmd) static int8_t SCSI_TestUnitReady(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params) { UNUSED(params); - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { @@ -233,7 +233,7 @@ static int8_t SCSI_TestUnitReady(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t return -1; } - if (((USBD_StorageTypeDef *)pdev->pUserData)->IsReady(lun) != 0) + if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->IsReady(lun) != 0) { SCSI_SenseCode(pdev, lun, NOT_READY, MEDIUM_NOT_PRESENT); hmsc->bot_state = USBD_BOT_NO_DATA; @@ -257,7 +257,7 @@ static int8_t SCSI_Inquiry(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param { uint8_t *pPage; uint16_t len; - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { @@ -290,7 +290,9 @@ static int8_t SCSI_Inquiry(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param } else { - pPage = (uint8_t *) &((USBD_StorageTypeDef *)pdev->pUserData)->pInquiry[lun * STANDARD_INQUIRY_DATA_LEN]; + + pPage = (uint8_t *) & ((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId]) \ + ->pInquiry[lun * STANDARD_INQUIRY_DATA_LEN]; len = (uint16_t)pPage[4] + 5U; if (params[4] <= len) @@ -316,14 +318,15 @@ static int8_t SCSI_ReadCapacity10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t { UNUSED(params); int8_t ret; - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { return -1; } - ret = ((USBD_StorageTypeDef *)pdev->pUserData)->GetCapacity(lun, &hmsc->scsi_blk_nbr, &hmsc->scsi_blk_size); + ret = ((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->GetCapacity(lun, &hmsc->scsi_blk_nbr, + &hmsc->scsi_blk_size); if ((ret != 0) || (hmsc->scsi_medium_state == SCSI_MEDIUM_EJECTED)) { @@ -358,16 +361,17 @@ static int8_t SCSI_ReadCapacity10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t static int8_t SCSI_ReadCapacity16(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params) { UNUSED(params); - uint8_t idx; + uint32_t idx; int8_t ret; - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { return -1; } - ret = ((USBD_StorageTypeDef *)pdev->pUserData)->GetCapacity(lun, &hmsc->scsi_blk_nbr, &hmsc->scsi_blk_size); + ret = ((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->GetCapacity(lun, &hmsc->scsi_blk_nbr, + &hmsc->scsi_blk_size); if ((ret != 0) || (hmsc->scsi_medium_state == SCSI_MEDIUM_EJECTED)) { @@ -418,14 +422,14 @@ static int8_t SCSI_ReadFormatCapacity(USBD_HandleTypeDef *pdev, uint8_t lun, uin uint32_t blk_nbr; uint16_t i; int8_t ret; - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { return -1; } - ret = ((USBD_StorageTypeDef *)pdev->pUserData)->GetCapacity(lun, &blk_nbr, &blk_size); + ret = ((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->GetCapacity(lun, &blk_nbr, &blk_size); if ((ret != 0) || (hmsc->scsi_medium_state == SCSI_MEDIUM_EJECTED)) { @@ -465,7 +469,7 @@ static int8_t SCSI_ReadFormatCapacity(USBD_HandleTypeDef *pdev, uint8_t lun, uin static int8_t SCSI_ModeSense6(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params) { UNUSED(lun); - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; uint16_t len = MODE_SENSE6_LEN; if (hmsc == NULL) @@ -494,7 +498,7 @@ static int8_t SCSI_ModeSense6(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *pa static int8_t SCSI_ModeSense10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params) { UNUSED(lun); - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; uint16_t len = MODE_SENSE10_LEN; if (hmsc == NULL) @@ -524,7 +528,7 @@ static int8_t SCSI_RequestSense(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t * { UNUSED(lun); uint8_t i; - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { @@ -581,7 +585,7 @@ static int8_t SCSI_RequestSense(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t * void SCSI_SenseCode(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t sKey, uint8_t ASC) { UNUSED(lun); - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { @@ -610,7 +614,7 @@ void SCSI_SenseCode(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t sKey, uint8_t static int8_t SCSI_StartStopUnit(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params) { UNUSED(lun); - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { @@ -656,7 +660,7 @@ static int8_t SCSI_StartStopUnit(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t static int8_t SCSI_AllowPreventRemovable(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params) { UNUSED(lun); - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { @@ -687,7 +691,7 @@ static int8_t SCSI_AllowPreventRemovable(USBD_HandleTypeDef *pdev, uint8_t lun, */ static int8_t SCSI_Read10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { @@ -710,7 +714,7 @@ static int8_t SCSI_Read10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params return -1; } - if (((USBD_StorageTypeDef *)pdev->pUserData)->IsReady(lun) != 0) + if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->IsReady(lun) != 0) { SCSI_SenseCode(pdev, lun, NOT_READY, MEDIUM_NOT_PRESENT); return -1; @@ -753,7 +757,7 @@ static int8_t SCSI_Read10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params */ static int8_t SCSI_Read12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { @@ -775,7 +779,7 @@ static int8_t SCSI_Read12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params return -1; } - if (((USBD_StorageTypeDef *)pdev->pUserData)->IsReady(lun) != 0) + if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->IsReady(lun) != 0) { SCSI_SenseCode(pdev, lun, NOT_READY, MEDIUM_NOT_PRESENT); return -1; @@ -821,7 +825,7 @@ static int8_t SCSI_Read12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params */ static int8_t SCSI_Write10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; uint32_t len; if (hmsc == NULL) @@ -829,6 +833,11 @@ static int8_t SCSI_Write10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param return -1; } +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MSCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + if (hmsc->bot_state == USBD_BOT_IDLE) /* Idle */ { if (hmsc->cbw.dDataLength == 0U) @@ -845,14 +854,14 @@ static int8_t SCSI_Write10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param } /* Check whether Media is ready */ - if (((USBD_StorageTypeDef *)pdev->pUserData)->IsReady(lun) != 0) + if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->IsReady(lun) != 0) { SCSI_SenseCode(pdev, lun, NOT_READY, MEDIUM_NOT_PRESENT); return -1; } /* Check If media is write-protected */ - if (((USBD_StorageTypeDef *)pdev->pUserData)->IsWriteProtected(lun) != 0) + if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->IsWriteProtected(lun) != 0) { SCSI_SenseCode(pdev, lun, NOT_READY, WRITE_PROTECTED); return -1; @@ -886,7 +895,7 @@ static int8_t SCSI_Write10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param /* Prepare EP to receive first data packet */ hmsc->bot_state = USBD_BOT_DATA_OUT; - (void)USBD_LL_PrepareReceive(pdev, MSC_EPOUT_ADDR, hmsc->bot_data, len); + (void)USBD_LL_PrepareReceive(pdev, MSCOutEpAdd, hmsc->bot_data, len); } else /* Write Process ongoing */ { @@ -906,13 +915,17 @@ static int8_t SCSI_Write10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param */ static int8_t SCSI_Write12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; uint32_t len; if (hmsc == NULL) { return -1; } +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MSCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (hmsc->bot_state == USBD_BOT_IDLE) /* Idle */ { @@ -930,7 +943,7 @@ static int8_t SCSI_Write12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param } /* Check whether Media is ready */ - if (((USBD_StorageTypeDef *)pdev->pUserData)->IsReady(lun) != 0) + if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->IsReady(lun) != 0) { SCSI_SenseCode(pdev, lun, NOT_READY, MEDIUM_NOT_PRESENT); hmsc->bot_state = USBD_BOT_NO_DATA; @@ -938,7 +951,7 @@ static int8_t SCSI_Write12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param } /* Check If media is write-protected */ - if (((USBD_StorageTypeDef *)pdev->pUserData)->IsWriteProtected(lun) != 0) + if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->IsWriteProtected(lun) != 0) { SCSI_SenseCode(pdev, lun, NOT_READY, WRITE_PROTECTED); hmsc->bot_state = USBD_BOT_NO_DATA; @@ -975,7 +988,7 @@ static int8_t SCSI_Write12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param /* Prepare EP to receive first data packet */ hmsc->bot_state = USBD_BOT_DATA_OUT; - (void)USBD_LL_PrepareReceive(pdev, MSC_EPOUT_ADDR, hmsc->bot_data, len); + (void)USBD_LL_PrepareReceive(pdev, MSCOutEpAdd, hmsc->bot_data, len); } else /* Write Process ongoing */ { @@ -995,7 +1008,7 @@ static int8_t SCSI_Write12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param */ static int8_t SCSI_Verify10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { @@ -1029,7 +1042,7 @@ static int8_t SCSI_Verify10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *para static int8_t SCSI_CheckAddressRange(USBD_HandleTypeDef *pdev, uint8_t lun, uint32_t blk_offset, uint32_t blk_nbr) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hmsc == NULL) { @@ -1053,25 +1066,32 @@ static int8_t SCSI_CheckAddressRange(USBD_HandleTypeDef *pdev, uint8_t lun, */ static int8_t SCSI_ProcessRead(USBD_HandleTypeDef *pdev, uint8_t lun) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; - uint32_t len = hmsc->scsi_blk_len * hmsc->scsi_blk_size; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint32_t len; if (hmsc == NULL) { return -1; } + len = hmsc->scsi_blk_len * hmsc->scsi_blk_size; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MSCInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + len = MIN(len, MSC_MEDIA_PACKET); - if (((USBD_StorageTypeDef *)pdev->pUserData)->Read(lun, hmsc->bot_data, - hmsc->scsi_blk_addr, - (len / hmsc->scsi_blk_size)) < 0) + if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->Read(lun, hmsc->bot_data, + hmsc->scsi_blk_addr, + (len / hmsc->scsi_blk_size)) < 0) { SCSI_SenseCode(pdev, lun, HARDWARE_ERROR, UNRECOVERED_READ_ERROR); return -1; } - (void)USBD_LL_Transmit(pdev, MSC_EPIN_ADDR, hmsc->bot_data, len); + (void)USBD_LL_Transmit(pdev, MSCInEpAdd, hmsc->bot_data, len); hmsc->scsi_blk_addr += (len / hmsc->scsi_blk_size); hmsc->scsi_blk_len -= (len / hmsc->scsi_blk_size); @@ -1095,19 +1115,26 @@ static int8_t SCSI_ProcessRead(USBD_HandleTypeDef *pdev, uint8_t lun) */ static int8_t SCSI_ProcessWrite(USBD_HandleTypeDef *pdev, uint8_t lun) { - USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassData; - uint32_t len = hmsc->scsi_blk_len * hmsc->scsi_blk_size; + USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint32_t len; if (hmsc == NULL) { return -1; } + len = hmsc->scsi_blk_len * hmsc->scsi_blk_size; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MSCOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + len = MIN(len, MSC_MEDIA_PACKET); - if (((USBD_StorageTypeDef *)pdev->pUserData)->Write(lun, hmsc->bot_data, - hmsc->scsi_blk_addr, - (len / hmsc->scsi_blk_size)) < 0) + if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->Write(lun, hmsc->bot_data, + hmsc->scsi_blk_addr, + (len / hmsc->scsi_blk_size)) < 0) { SCSI_SenseCode(pdev, lun, HARDWARE_ERROR, WRITE_FAULT); return -1; @@ -1128,7 +1155,7 @@ static int8_t SCSI_ProcessWrite(USBD_HandleTypeDef *pdev, uint8_t lun) len = MIN((hmsc->scsi_blk_len * hmsc->scsi_blk_size), MSC_MEDIA_PACKET); /* Prepare EP to Receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, MSC_EPOUT_ADDR, hmsc->bot_data, len); + (void)USBD_LL_PrepareReceive(pdev, MSCOutEpAdd, hmsc->bot_data, len); } return 0; @@ -1177,4 +1204,3 @@ static int8_t SCSI_UpdateBotData(USBD_MSC_BOT_HandleTypeDef *hmsc, * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_storage_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_storage_template.c index 875c1f9d..2163943d 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_storage_template.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MSC/Src/usbd_msc_storage_template.c @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -87,91 +86,106 @@ USBD_StorageTypeDef USBD_MSC_Template_fops = STORAGE_Inquirydata, }; -/******************************************************************************* - * Function Name : Read_Memory - * Description : Handle the Read operation from the microSD card. - * Input : None. - * Output : None. - * Return : None. - *******************************************************************************/ + +/** + * @brief Initializes the storage unit (medium) + * @param lun: Logical unit number + * @retval Status (0 : OK / -1 : Error) + */ int8_t STORAGE_Init(uint8_t lun) { + UNUSED(lun); + return (0); } -/******************************************************************************* - * Function Name : Read_Memory - * Description : Handle the Read operation from the STORAGE card. - * Input : None. - * Output : None. - * Return : None. - *******************************************************************************/ +/** + * @brief Returns the medium capacity. + * @param lun: Logical unit number + * @param block_num: Number of total block number + * @param block_size: Block size + * @retval Status (0: OK / -1: Error) + */ int8_t STORAGE_GetCapacity(uint8_t lun, uint32_t *block_num, uint16_t *block_size) { + UNUSED(lun); + *block_num = STORAGE_BLK_NBR; *block_size = STORAGE_BLK_SIZ; return (0); } -/******************************************************************************* - * Function Name : Read_Memory - * Description : Handle the Read operation from the STORAGE card. - * Input : None. - * Output : None. - * Return : None. - *******************************************************************************/ + +/** + * @brief Checks whether the medium is ready. + * @param lun: Logical unit number + * @retval Status (0: OK / -1: Error) + */ int8_t STORAGE_IsReady(uint8_t lun) { + UNUSED(lun); + return (0); } -/******************************************************************************* - * Function Name : Read_Memory - * Description : Handle the Read operation from the STORAGE card. - * Input : None. - * Output : None. - * Return : None. - *******************************************************************************/ +/** + * @brief Checks whether the medium is write protected. + * @param lun: Logical unit number + * @retval Status (0: write enabled / -1: otherwise) + */ int8_t STORAGE_IsWriteProtected(uint8_t lun) { + UNUSED(lun); + return 0; } -/******************************************************************************* - * Function Name : Read_Memory - * Description : Handle the Read operation from the STORAGE card. - * Input : None. - * Output : None. - * Return : None. - *******************************************************************************/ +/** + * @brief Reads data from the medium. + * @param lun: Logical unit number + * @param buf: data buffer + * @param blk_addr: Logical block address + * @param blk_len: Blocks number + * @retval Status (0: OK / -1: Error) + */ int8_t STORAGE_Read(uint8_t lun, uint8_t *buf, uint32_t blk_addr, uint16_t blk_len) { + UNUSED(lun); + UNUSED(buf); + UNUSED(blk_addr); + UNUSED(blk_len); + return 0; } -/******************************************************************************* - * Function Name : Write_Memory - * Description : Handle the Write operation to the STORAGE card. - * Input : None. - * Output : None. - * Return : None. - *******************************************************************************/ + +/** + * @brief Writes data into the medium. + * @param lun: Logical unit number + * @param buf: data buffer + * @param blk_addr: Logical block address + * @param blk_len: Blocks number + * @retval Status (0 : OK / -1 : Error) + */ int8_t STORAGE_Write(uint8_t lun, uint8_t *buf, uint32_t blk_addr, uint16_t blk_len) { + UNUSED(lun); + UNUSED(buf); + UNUSED(blk_addr); + UNUSED(blk_len); + return (0); } -/******************************************************************************* - * Function Name : Write_Memory - * Description : Handle the Write operation to the STORAGE card. - * Input : None. - * Output : None. - * Return : None. - *******************************************************************************/ + +/** + * @brief Returns the Max Supported LUNs. + * @param None + * @retval Lun(s) number + */ int8_t STORAGE_GetMaxLun(void) { return (STORAGE_LUN_NBR - 1); } -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp.h new file mode 100644 index 00000000..df2644fd --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp.h @@ -0,0 +1,358 @@ +/** + ****************************************************************************** + * @file usbd_mtp.h + * @author MCD Application Team + * @brief header file for the usbd_mtp.c file. + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Define to prevent recursive inclusion -------------------------------------*/ +#ifndef __USB_MTP_H +#define __USB_MTP_H + +#ifdef __cplusplus +extern "C" { +#endif + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_ioreq.h" +#include "usbd_ctlreq.h" + +/** @addtogroup STM32_USB_DEVICE_LIBRARY + * @{ + */ + +/** @defgroup USBD_MTP + * @brief This file is the header file for usbd_mtp.c + * @{ + */ + + +/** @defgroup USBD_MTP_Exported_Defines + * @{ + */ +#ifndef MTP_IN_EP +#define MTP_IN_EP 0x81U /* EP1 for data IN */ +#endif /* MTP_IN_EP */ +#ifndef MTP_OUT_EP +#define MTP_OUT_EP 0x01U /* EP1 for data OUT */ +#endif /* MTP_OUT_EP */ +#ifndef MTP_CMD_EP +#define MTP_CMD_EP 0x82U /* EP2 for MTP commands */ +#endif /* MTP_CMD_EP */ + +#ifndef MTP_CMD_ITF_NBR +#define MTP_CMD_ITF_NBR 0x00U /* Command Interface Number 0 */ +#endif /* MTP_CMD_ITF_NBR */ + +#ifndef MTP_COM_ITF_NBR +#define MTP_COM_ITF_NBR 0x01U /* Communication Interface Number 0 */ +#endif /* MTP_CMD_ITF_NBR */ + +#ifndef MTP_HS_BINTERVAL +#define MTP_HS_BINTERVAL 0x10U +#endif /* MTP_HS_BINTERVAL */ + +#ifndef MTP_FS_BINTERVAL +#define MTP_FS_BINTERVAL 0x10U +#endif /* MTP_FS_BINTERVAL */ + +#define MTP_DATA_MAX_HS_PACKET_SIZE 512U +#define MTP_DATA_MAX_FS_PACKET_SIZE 64U /* Endpoint IN & OUT Packet size */ +#define MTP_CMD_PACKET_SIZE 8U /* Control Endpoint Packet size */ + +#define MTP_MEDIA_PACKET 512U +#define MTP_CONT_HEADER_SIZE 12U + +#define MTP_CONFIG_DESC_SIZ 39U +#define MTP_INTERFACE_DESC_SIZE 0x09U +#define USB_MTP_INTRERFACE_CLASS 0x06U +#define USB_MTP_INTRERFACE_SUB_CLASS 0x01U +#define USB_MTP_INTRERFACE_PROTOCOL 0x01U +#define MTP_ENDPOINT_DESC_SIZE 0x07U + +/*---------------------------------------------------------------------*/ +/* MTP definitions */ +/*---------------------------------------------------------------------*/ + +/* MTP class requests */ +#define MTP_REQ_CANCEL 0x64U +#define MTP_REQ_GET_EXT_EVENT_DATA 0x65U +#define MTP_REQ_RESET 0x66U +#define MTP_REQ_GET_DEVICE_STATUS 0x67U + + +/* Max info items size */ +#define MTP_SUPPORTED_OPERATIONS_NBR 100U +#define MTP_SUPPORTED_EVENTS_NBR 100U +#define MTP_SUPPORTED_PROPRIETIES_NBR 100U +#define MTP_CAPTURE_FORMATS_NBR 100U +#define MTP_IMAGE_FORMATS_NBR 100U +#define MTP_MAX_STR_SIZE 255U + +/* MTP response code */ +#define MTP_RESPONSE_OK 0x2001U +#define MTP_RESPONSE_GENERAL_ERROR 0x2002U +#define MTP_RESPONSE_PARAMETER_NOT_SUPPORTED 0x2006U +#define MTP_RESPONSE_INCOMPLETE_TRANSFER 0x2007U +#define MTP_RESPONSE_INVALID_STORAGE_ID 0x2008U +#define MTP_RESPONSE_INVALID_OBJECT_HANDLE 0x2009U +#define MTP_RESPONSE_DEVICEPROP_NOT_SUPPORTED 0x200AU +#define MTP_RESPONSE_STORE_FULL 0x200CU +#define MTP_RESPONSE_ACCESS_DENIED 0x200FU +#define MTP_RESPONSE_STORE_NOT_AVAILABLE 0x2013U +#define MTP_RESPONSE_SPECIFICATION_BY_FORMAT_NOT_SUPPORTED 0x2014U +#define MTP_RESPONSE_NO_VALID_OBJECT_INFO 0x2015U +#define MTP_RESPONSE_DEVICE_BUSY 0x2019U +#define MTP_RESPONSE_INVALID_PARENT_OBJECT 0x201AU +#define MTP_RESPONSE_INVALID_PARAMETER 0x201DU +#define MTP_RESPONSE_SESSION_ALREADY_OPEN 0x201EU +#define MTP_RESPONSE_TRANSACTION_CANCELLED 0x201FU +#define MTP_RESPONSE_INVALID_OBJECT_PROP_CODE 0xA801U +#define MTP_RESPONSE_SPECIFICATION_BY_GROUP_UNSUPPORTED 0xA807U +#define MTP_RESPONSE_OBJECT_PROP_NOT_SUPPORTED 0xA80AU + +/* + * MTP Class specification Revision 1.1 + * Appendix A. Object Formats + */ + +/* MTP Object format codes */ +#define MTP_OBJ_FORMAT_UNDEFINED 0x3000U +#define MTP_OBJ_FORMAT_ASSOCIATION 0x3001U +#define MTP_OBJ_FORMAT_SCRIPT 0x3002U +#define MTP_OBJ_FORMAT_EXECUTABLE 0x3003U +#define MTP_OBJ_FORMAT_TEXT 0x3004U +#define MTP_OBJ_FORMAT_HTML 0x3005U +#define MTP_OBJ_FORMAT_DPOF 0x3006U +#define MTP_OBJ_FORMAT_AIFF 0x3007U +#define MTP_OBJ_FORMAT_WAV 0x3008U +#define MTP_OBJ_FORMAT_MP3 0x3009U +#define MTP_OBJ_FORMAT_AVI 0x300AU +#define MTP_OBJ_FORMAT_MPEG 0x300BU +#define MTP_OBJ_FORMAT_ASF 0x300CU +#define MTP_OBJ_FORMAT_DEFINED 0x3800U +#define MTP_OBJ_FORMAT_EXIF_JPEG 0x3801U +#define MTP_OBJ_FORMAT_TIFF_EP 0x3802U +#define MTP_OBJ_FORMAT_FLASHPIX 0x3803U +#define MTP_OBJ_FORMAT_BMP 0x3804U +#define MTP_OBJ_FORMAT_CIFF 0x3805U +#define MTP_OBJ_FORMAT_UNDEFINED_RESERVED0 0x3806U +#define MTP_OBJ_FORMAT_GIF 0x3807U +#define MTP_OBJ_FORMAT_JFIF 0x3808U +#define MTP_OBJ_FORMAT_CD 0x3809U +#define MTP_OBJ_FORMAT_PICT 0x380AU +#define MTP_OBJ_FORMAT_PNG 0x380BU +#define MTP_OBJ_FORMAT_UNDEFINED_RESERVED1 0x380CU +#define MTP_OBJ_FORMAT_TIFF 0x380DU +#define MTP_OBJ_FORMAT_TIFF_IT 0x380EU +#define MTP_OBJ_FORMAT_JP2 0x380FU +#define MTP_OBJ_FORMAT_JPX 0x3810U +#define MTP_OBJ_FORMAT_UNDEFINED_FIRMWARE 0xB802U +#define MTP_OBJ_FORMAT_WINDOWS_IMAGE_FORMAT 0xB881U +#define MTP_OBJ_FORMAT_UNDEFINED_AUDIO 0xB900U +#define MTP_OBJ_FORMAT_WMA 0xB901U +#define MTP_OBJ_FORMAT_OGG 0xB902U +#define MTP_OBJ_FORMAT_AAC 0xB903U +#define MTP_OBJ_FORMAT_AUDIBLE 0xB904U +#define MTP_OBJ_FORMAT_FLAC 0xB906U +#define MTP_OBJ_FORMAT_UNDEFINED_VIDEO 0xB980U +#define MTP_OBJ_FORMAT_WMV 0xB981U +#define MTP_OBJ_FORMAT_MP4_CONTAINER 0xB982U +#define MTP_OBJ_FORMAT_MP2 0xB983U +#define MTP_OBJ_FORMAT_3GP_CONTAINER 0xB984U +/** + * @} + */ + + +/** @defgroup USBD_CORE_Exported_TypesDefinitions + * @{ + */ + +/** + * @} + */ + + +/* MTP Session state */ +typedef enum +{ + MTP_SESSION_NOT_OPENED = 0x00, + MTP_SESSION_OPENED = 0x01, +} MTP_SessionStateTypeDef; + +/* MTP response phases */ +typedef enum +{ + MTP_PHASE_IDLE = 0x00, + MTP_RESPONSE_PHASE = 0x01, + MTP_READ_DATA = 0x02, + MTP_RECEIVE_DATA = 0x03, +} MTP_ResponsePhaseTypeDef; + +typedef struct +{ + uint32_t temp_length; + uint32_t prv_len; + uint32_t totallen; + uint32_t rx_length; + uint32_t readbytes; /* File write/read counts */ +} MTP_DataLengthTypeDef; + +typedef enum +{ + RECEIVE_IDLE_STATE = 0x00U, + RECEIVE_COMMAND_DATA = 0x01U, + RECEIVE_FIRST_DATA = 0x02U, + RECEIVE_REST_OF_DATA = 0x03U, + SEND_RESPONSE = 0x04U, +} MTP_RECEIVE_DATA_STATUS; + +typedef struct +{ + uint32_t length; + uint16_t type; + uint16_t code; + uint32_t trans_id; + uint32_t Param1; + uint32_t Param2; + uint32_t Param3; + uint32_t Param4; + uint32_t Param5; +} MTP_OperationsTypeDef; + +typedef struct +{ + uint32_t length; + uint16_t type; + uint16_t code; + uint32_t trans_id; + uint8_t data[MTP_MEDIA_PACKET]; +} MTP_GenericContainerTypeDef; + +#if defined ( __GNUC__ ) +typedef __PACKED_STRUCT +#else +__packed typedef struct +#endif /* __GNUC__ */ +{ + uint32_t Storage_id; + uint16_t ObjectFormat; + uint16_t ProtectionStatus; + uint32_t ObjectCompressedSize; + uint16_t ThumbFormat; + uint32_t ThumbCompressedSize; + uint32_t ThumbPixWidth; + uint32_t ThumbPixHeight; + uint32_t ImagePixWidth; + uint32_t ImagePixHeight; + uint32_t ImageBitDepth; + uint32_t ParentObject; + uint16_t AssociationType; + uint32_t AssociationDesc; + uint32_t SequenceNumber; + uint8_t Filename_len; + uint16_t Filename[255]; + uint32_t CaptureDate; + uint32_t ModificationDate; + uint8_t Keywords; +} MTP_ObjectInfoTypeDef; + +typedef struct +{ + uint32_t alt_setting; + uint32_t dev_status; + uint32_t ResponseLength; + uint32_t ResponseCode; + __IO uint16_t MaxPcktLen; + uint32_t rx_buff[MTP_MEDIA_PACKET / 4U]; /* Force 32-bit alignment */ + MTP_ResponsePhaseTypeDef MTP_ResponsePhase; + MTP_SessionStateTypeDef MTP_SessionState; + MTP_RECEIVE_DATA_STATUS RECEIVE_DATA_STATUS; + MTP_OperationsTypeDef OperationsContainer; + MTP_GenericContainerTypeDef GenericContainer; +} USBD_MTP_HandleTypeDef; + +/** @defgroup USBD_CORE_Exported_Macros + * @{ + */ + +/** + * @} + */ + +/** @defgroup USBD_CORE_Exported_Variables + * @{ + */ + +extern USBD_ClassTypeDef USBD_MTP; +#define USBD_MTP_CLASS &USBD_MTP + +/** + * @} + */ + +/** @defgroup USB_CORE_Exported_Functions + * @{ + */ + +typedef struct _USBD_MTP_ItfTypeDef +{ + uint8_t (*Init)(void); + uint8_t (*DeInit)(void); + uint32_t (*ReadData)(uint32_t Param1, uint8_t *buff, MTP_DataLengthTypeDef *data_length); + uint16_t (*Create_NewObject)(MTP_ObjectInfoTypeDef ObjectInfo, uint32_t objhandle); + + uint32_t (*GetIdx)(uint32_t Param3, uint32_t *obj_handle); + uint32_t (*GetParentObject)(uint32_t Param); + uint16_t (*GetObjectFormat)(uint32_t Param); + uint8_t (*GetObjectName_len)(uint32_t Param); + void (*GetObjectName)(uint32_t Param, uint8_t obj_len, uint16_t *buf); + uint32_t (*GetObjectSize)(uint32_t Param); + uint64_t (*GetMaxCapability)(void); + uint64_t (*GetFreeSpaceInBytes)(void); + uint32_t (*GetNewIndex)(uint16_t objformat); + void (*WriteData)(uint16_t len, uint8_t buff[]); + uint32_t (*GetContainerLength)(uint32_t Param1); + uint16_t (*DeleteObject)(uint32_t Param1); + void (*Cancel)(uint32_t Phase); + uint32_t *ScratchBuff; + uint32_t ScratchBuffSze; +} USBD_MTP_ItfTypeDef; + +/** + * @} + */ +/** @defgroup USB_CORE_Exported_Functions + * @{ + */ +uint8_t USBD_MTP_RegisterInterface(USBD_HandleTypeDef *pdev, USBD_MTP_ItfTypeDef *fops); + + +/** + * @} + */ + +#ifdef __cplusplus +} +#endif + +#endif /* __USB_MTP_H */ +/** + * @} + */ + +/** + * @} + */ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp_if_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp_if_template.h new file mode 100644 index 00000000..d86e2505 --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp_if_template.h @@ -0,0 +1,96 @@ +/** + ****************************************************************************** + * @file usbd_mtp_if_template.h + * @author MCD Application Team + * @brief Header file for the usbd_mtp_if_template.c file. + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Define to prevent recursive inclusion -------------------------------------*/ +#ifndef __USBD_MTP_IF_TEMPLATE_H +#define __USBD_MTP_IF_TEMPLATE_H + +#ifdef __cplusplus +extern "C" { +#endif + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_mtp.h" + +/* Exported Define -----------------------------------------------------------*/ +#define USBD_MTP_DEVICE_PROP_SUPPORTED 1U +#define USBD_MTP_CAPTURE_FORMAT_SUPPORTED 1U +#define USBD_MTP_VEND_EXT_DESC_SUPPORTED 1U +#define USBD_MTP_EVENTS_SUPPORTED 1U + +#if USBD_MTP_EVENTS_SUPPORTED == 1 +#define SUPP_EVENTS_LEN (uint8_t)((uint8_t)sizeof(SuppEvents) / 2U) +#else +#define SUPP_EVENTS_LEN 0U +#endif /* USBD_MTP_EVENTS_SUPPORTED */ + +#if USBD_MTP_VEND_EXT_DESC_SUPPORTED == 1 +#define VEND_EXT_DESC_LEN (sizeof(VendExtDesc) / 2U) +#else +#define VEND_EXT_DESC_LEN 0U +#endif /* USBD_MTP_VEND_EXT_DESC_SUPPORTED */ + +#if USBD_MTP_CAPTURE_FORMAT_SUPPORTED == 1 +#define SUPP_CAPT_FORMAT_LEN (uint8_t)((uint8_t)sizeof(SuppCaptFormat) / 2U) +#else +#define SUPP_CAPT_FORMAT_LEN 0U +#endif /* USBD_MTP_CAPTURE_FORMAT_SUPPORTED */ + +#if USBD_MTP_DEVICE_PROP_SUPPORTED == 1 +#define SUPP_DEVICE_PROP_LEN (uint8_t)((uint8_t)sizeof(DevicePropSupp) / 2U) +#else +#define SUPP_DEVICE_PROP_LEN 0U +#endif /* USBD_MTP_DEVICE_PROP_SUPPORTED */ + +#define MTP_IF_SCRATCH_BUFF_SZE 1024U + +/* Exported types ------------------------------------------------------------*/ +extern USBD_MTP_ItfTypeDef USBD_MTP_fops; + +/* Exported macros -----------------------------------------------------------*/ +/* Exported variables --------------------------------------------------------*/ +/* Exported functions ------------------------------------------------------- */ + +static const uint16_t Manuf[] = {'S', 'T', 'M', 0}; /* last 2 bytes must be 0*/ +static const uint16_t Model[] = {'S', 'T', 'M', '3', '2', 0}; /* last 2 bytes must be 0*/ +static const uint16_t VendExtDesc[] = {'m', 'i', 'c', 'r', 'o', 's', 'o', 'f', 't', '.', + 'c', 'o', 'm', ':', ' ', '1', '.', '0', ';', ' ', 0 + }; /* last 2 bytes must be 0*/ +/*SerialNbr shall be 32 character hexadecimal string for legacy compatibility reasons */ +static const uint16_t SerialNbr[] = {'0', '0', '0', '0', '1', '0', '0', '0', '0', '1', '0', '0', '0', '0', + '1', '0', '0', '0', '0', '1', '0', '0', '0', '0', '1', '0', '0', '0', + '0', '1', '0', '0', 0 + }; /* last 2 bytes must be 0*/ +static const uint16_t DeviceVers[] = {'V', '1', '.', '0', '0', 0}; /* last 2 bytes must be 0*/ + +static const uint16_t DefaultFileName[] = {'N', 'e', 'w', ' ', 'F', 'o', 'l', 'd', 'e', 'r', 0}; + +static const uint16_t DevicePropDefVal[] = {'S', 'T', 'M', '3', '2', 0}; /* last 2 bytes must be 0*/ +static const uint16_t DevicePropCurDefVal[] = {'S', 'T', 'M', '3', '2', ' ', 'V', '1', '.', '0', 0}; + + +/** + * @} + */ + +#ifdef __cplusplus +} +#endif + +#endif /* __USBD_MTP_IF_TEMPLATE_H */ + diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp_opt.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp_opt.h new file mode 100644 index 00000000..4dee923d --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp_opt.h @@ -0,0 +1,681 @@ +/** + ****************************************************************************** + * @file usbd_mtp_opt.h + * @author MCD Application Team + * @brief header file for the usbd_mtp_opt.c file. + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Define to prevent recursive inclusion -------------------------------------*/ +#ifndef __USBD_MTP_OPT_H__ +#define __USBD_MTP_OPT_H__ + +#ifdef __cplusplus +extern "C" { +#endif + +/* Includes ------------------------------------------------------------------*/ +#ifndef __USBD_MTP_IF_H +#include "usbd_mtp_if_template.h" +#endif /* __USBD_MTP_IF_H */ +#include "usbd_mtp.h" + +/** @addtogroup STM32_USB_DEVICE_LIBRARY + * @{ + */ + +/** @defgroup USBD_MTP_OPT + * @brief This file is the header file for usbd_mtp_opt.c + * @{ + */ + + +/** @defgroup USBD_MTP_OPT_Exported_Defines + * @{ + */ + +/* + * MTP Class specification Revision 1.1 + * Appendix B. Object Properties + */ + +/* MTP OBJECT PROPERTIES supported*/ +#define MTP_OB_PROP_STORAGE_ID 0xDC01U +#define MTP_OB_PROP_OBJECT_FORMAT 0xDC02U +#define MTP_OB_PROP_PROTECTION_STATUS 0xDC03U +#define MTP_OB_PROP_OBJECT_SIZE 0xDC04U +#define MTP_OB_PROP_ASSOC_TYPE 0xDC05U +#define MTP_OB_PROP_ASSOC_DESC 0xDC06U +#define MTP_OB_PROP_OBJ_FILE_NAME 0xDC07U +#define MTP_OB_PROP_DATE_CREATED 0xDC08U +#define MTP_OB_PROP_DATE_MODIFIED 0xDC09U +#define MTP_OB_PROP_KEYWORDS 0xDC0AU +#define MTP_OB_PROP_PARENT_OBJECT 0xDC0BU +#define MTP_OB_PROP_ALLOWED_FOLD_CONTENTS 0xDC0CU +#define MTP_OB_PROP_HIDDEN 0xDC0DU +#define MTP_OB_PROP_SYSTEM_OBJECT 0xDC0EU +#define MTP_OB_PROP_PERS_UNIQ_OBJ_IDEN 0xDC41U +#define MTP_OB_PROP_SYNCID 0xDC42U +#define MTP_OB_PROP_PROPERTY_BAG 0xDC43U +#define MTP_OB_PROP_NAME 0xDC44U +#define MTP_OB_PROP_CREATED_BY 0xDC45U +#define MTP_OB_PROP_ARTIST 0xDC46U +#define MTP_OB_PROP_DATE_AUTHORED 0xDC47U +#define MTP_OB_PROP_DESCRIPTION 0xDC48U +#define MTP_OB_PROP_URL_REFERENCE 0xDC49U +#define MTP_OB_PROP_LANGUAGELOCALE 0xDC4AU +#define MTP_OB_PROP_COPYRIGHT_INFORMATION 0xDC4BU +#define MTP_OB_PROP_SOURCE 0xDC4CU +#define MTP_OB_PROP_ORIGIN_LOCATION 0xDC4DU +#define MTP_OB_PROP_DATE_ADDED 0xDC4EU +#define MTP_OB_PROP_NON_CONSUMABLE 0xDC4FU +#define MTP_OB_PROP_CORRUPTUNPLAYABLE 0xDC50U +#define MTP_OB_PROP_PRODUCERSERIALNUMBER 0xDC51U +#define MTP_OB_PROP_REPRESENTATIVE_SAMPLE_FORMAT 0xDC81U +#define MTP_OB_PROP_REPRESENTATIVE_SAMPLE_SIZE 0xDC82U +#define MTP_OB_PROP_REPRESENTATIVE_SAMPLE_HEIGHT 0xDC83U +#define MTP_OB_PROP_REPRESENTATIVE_SAMPLE_WIDTH 0xDC84U +#define MTP_OB_PROP_REPRESENTATIVE_SAMPLE_DURATION 0xDC85U +#define MTP_OB_PROP_REPRESENTATIVE_SAMPLE_DATA 0xDC86U +#define MTP_OB_PROP_WIDTH 0xDC87U +#define MTP_OB_PROP_HEIGHT 0xDC88U +#define MTP_OB_PROP_DURATION 0xDC89U +#define MTP_OB_PROP_RATING 0xDC8AU +#define MTP_OB_PROP_TRACK 0xDC8BU +#define MTP_OB_PROP_GENRE 0xDC8CU +#define MTP_OB_PROP_CREDITS 0xDC8DU +#define MTP_OB_PROP_LYRICS 0xDC8EU +#define MTP_OB_PROP_SUBSCRIPTION_CONTENT_ID 0xDC8FU +#define MTP_OB_PROP_PRODUCED_BY 0xDC90U +#define MTP_OB_PROP_USE_COUNT 0xDC91U +#define MTP_OB_PROP_SKIP_COUNT 0xDC92U +#define MTP_OB_PROP_LAST_ACCESSED 0xDC93U +#define MTP_OB_PROP_PARENTAL_RATING 0xDC94U +#define MTP_OB_PROP_META_GENRE 0xDC95U +#define MTP_OB_PROP_COMPOSER 0xDC96U +#define MTP_OB_PROP_EFFECTIVE_RATING 0xDC97U +#define MTP_OB_PROP_SUBTITLE 0xDC98U +#define MTP_OB_PROP_ORIGINAL_RELEASE_DATE 0xDC99U +#define MTP_OB_PROP_ALBUM_NAME 0xDC9AU +#define MTP_OB_PROP_ALBUM_ARTIST 0xDC9BU +#define MTP_OB_PROP_MOOD 0xDC9CU +#define MTP_OB_PROP_DRM_STATUS 0xDC9DU +#define MTP_OB_PROP_SUB_DESCRIPTION 0xDC9EU +#define MTP_OB_PROP_IS_CROPPED 0xDCD1U +#define MTP_OB_PROP_IS_COLOUR_CORRECTED 0xDCD2U +#define MTP_OB_PROP_IMAGE_BIT_DEPTH 0xDCD3U +#define MTP_OB_PROP_FNUMBER 0xDCD4U +#define MTP_OB_PROP_EXPOSURE_TIME 0xDCD5U +#define MTP_OB_PROP_EXPOSURE_INDEX 0xDCD6U +#define MTP_OB_PROP_TOTAL_BITRATE 0xDE91U +#define MTP_OB_PROP_BITRATE_TYPE 0xDE92U +#define MTP_OB_PROP_SAMPLE_RATE 0xDE93U +#define MTP_OB_PROP_NUMBER_OF_CHANNELS 0xDE94U +#define MTP_OB_PROP_AUDIO_BITDEPTH 0xDE95U +#define MTP_OB_PROP_SCAN_TYPE 0xDE97U +#define MTP_OB_PROP_AUDIO_WAVE_CODEC 0xDE99U +#define MTP_OB_PROP_AUDIO_BITRATE 0xDE9AU +#define MTP_OB_PROP_VIDEO_FOURCC_CODEC 0xDE9BU +#define MTP_OB_PROP_VIDEO_BITRATE 0xDE9CU +#define MTP_OB_PROP_FRAMES_PER_THOUSAND_SECONDS 0xDE9DU +#define MTP_OB_PROP_KEYFRAME_DISTANCE 0xDE9EU +#define MTP_OB_PROP_BUFFER_SIZE 0xDE9FU +#define MTP_OB_PROP_ENCODING_QUALITY 0xDEA0U +#define MTP_OB_PROP_ENCODING_PROFILE 0xDEA1U +#define MTP_OB_PROP_DISPLAY_NAME 0xDCE0U +#define MTP_OB_PROP_BODY_TEXT 0xDCE1U +#define MTP_OB_PROP_SUBJECT 0xDCE2U +#define MTP_OB_PROP_PRIORITY 0xDCE3U +#define MTP_OB_PROP_GIVEN_NAME 0xDD00U +#define MTP_OB_PROP_MIDDLE_NAMES 0xDD01U +#define MTP_OB_PROP_FAMILY_NAME 0xDD02U +#define MTP_OB_PROP_PREFIX 0xDD03U +#define MTP_OB_PROP_SUFFIX 0xDD04U +#define MTP_OB_PROP_PHONETIC_GIVEN_NAME 0xDD05U +#define MTP_OB_PROP_PHONETIC_FAMILY_NAME 0xDD06U +#define MTP_OB_PROP_EMAIL_PRIMARY 0xDD07U +#define MTP_OB_PROP_EMAIL_PERSONAL_1 0xDD08U +#define MTP_OB_PROP_EMAIL_PERSONAL_2 0xDD09U +#define MTP_OB_PROP_EMAIL_BUSINESS_1 0xDD0AU +#define MTP_OB_PROP_EMAIL_BUSINESS_2 0xDD0BU +#define MTP_OB_PROP_EMAIL_OTHERS 0xDD0CU +#define MTP_OB_PROP_PHONE_NUMBER_PRIMARY 0xDD0DU +#define MTP_OB_PROP_PHONE_NUMBER_PERSONAL 0xDD0EU +#define MTP_OB_PROP_PHONE_NUMBER_PERSONAL_2 0xDD0FU +#define MTP_OB_PROP_PHONE_NUMBER_BUSINESS 0xDD10U +#define MTP_OB_PROP_PHONE_NUMBER_BUSINESS_2 0xDD11U +#define MTP_OB_PROP_PHONE_NUMBER_MOBILE 0xDD12U +#define MTP_OB_PROP_PHONE_NUMBER_MOBILE_2 0xDD13U +#define MTP_OB_PROP_FAX_NUMBER_PRIMARY 0xDD14U +#define MTP_OB_PROP_FAX_NUMBER_PERSONAL 0xDD15U +#define MTP_OB_PROP_FAX_NUMBER_BUSINESS 0xDD16U +#define MTP_OB_PROP_PAGER_NUMBER 0xDD17U +#define MTP_OB_PROP_PHONE_NUMBER_OTHERS 0xDD18U +#define MTP_OB_PROP_PRIMARY_WEB_ADDRESS 0xDD19U +#define MTP_OB_PROP_PERSONAL_WEB_ADDRESS 0xDD1AU +#define MTP_OB_PROP_BUSINESS_WEB_ADDRESS 0xDD1BU +#define MTP_OB_PROP_INSTANT_MESSENGER_ADDRESS 0xDD1CU +#define MTP_OB_PROP_INSTANT_MESSENGER_ADDRESS_2 0xDD1DU +#define MTP_OB_PROP_INSTANT_MESSENGER_ADDRESS_3 0xDD1EU +#define MTP_OB_PROP_POSTAL_ADDRESS_PERSONAL_FULL 0xDD1FU +#define MTP_OB_PROP_POSTAL_ADDRESS_PERSONAL_LINE_1 0xDD20U +#define MTP_OB_PROP_POSTAL_ADDRESS_PERSONAL_LINE_2 0xDD21U +#define MTP_OB_PROP_POSTAL_ADDRESS_PERSONAL_CITY 0xDD22U +#define MTP_OB_PROP_POSTAL_ADDRESS_PERSONAL_REGION 0xDD23U +#define MTP_OB_PROP_POSTAL_ADDRESS_PERSONAL_POSTAL_CODE 0xDD24U +#define MTP_OB_PROP_POSTAL_ADDRESS_PERSONAL_COUNTRY 0xDD25U +#define MTP_OB_PROP_POSTAL_ADDRESS_BUSINESS_FULL 0xDD26U +#define MTP_OB_PROP_POSTAL_ADDRESS_BUSINESS_LINE_1 0xDD27U +#define MTP_OB_PROP_POSTAL_ADDRESS_BUSINESS_LINE_2 0xDD28U +#define MTP_OB_PROP_POSTAL_ADDRESS_BUSINESS_CITY 0xDD29U +#define MTP_OB_PROP_POSTAL_ADDRESS_BUSINESS_REGION 0xDD2AU +#define MTP_OB_PROP_POSTAL_ADDRESS_BUSINESS_POSTAL_CODE 0xDD2BU +#define MTP_OB_PROP_POSTAL_ADDRESS_BUSINESS_COUNTRY 0xDD2CU +#define MTP_OB_PROP_POSTAL_ADDRESS_OTHER_FULL 0xDD2DU +#define MTP_OB_PROP_POSTAL_ADDRESS_OTHER_LINE_1 0xDD2EU +#define MTP_OB_PROP_POSTAL_ADDRESS_OTHER_LINE_2 0xDD2FU +#define MTP_OB_PROP_POSTAL_ADDRESS_OTHER_CITY 0xDD30U +#define MTP_OB_PROP_POSTAL_ADDRESS_OTHER_REGION 0xDD31U +#define MTP_OB_PROP_POSTAL_ADDRESS_OTHER_POSTAL_CODE 0xDD32U +#define MTP_OB_PROP_POSTAL_ADDRESS_OTHER_COUNTRY 0xDD33U +#define MTP_OB_PROP_ORGANIZATION_NAME 0xDD34U +#define MTP_OB_PROP_PHONETIC_ORGANIZATION_NAME 0xDD35U +#define MTP_OB_PROP_ROLE 0xDD36U +#define MTP_OB_PROP_BIRTHDATE 0xDD37U +#define MTP_OB_PROP_MESSAGE_TO 0xDD40U +#define MTP_OB_PROP_MESSAGE_CC 0xDD41U +#define MTP_OB_PROP_MESSAGE_BCC 0xDD42U +#define MTP_OB_PROP_MESSAGE_READ 0xDD43U +#define MTP_OB_PROP_MESSAGE_RECEIVED_TIME 0xDD44U +#define MTP_OB_PROP_MESSAGE_SENDER 0xDD45U +#define MTP_OB_PROP_ACT_BEGIN_TIME 0xDD50U +#define MTP_OB_PROP_ACT_END_TIME 0xDD51U +#define MTP_OB_PROP_ACT_LOCATION 0xDD52U +#define MTP_OB_PROP_ACT_REQUIRED_ATTENDEES 0xDD54U +#define MTP_OB_PROP_ACT_OPTIONAL_ATTENDEES 0xDD55U +#define MTP_OB_PROP_ACT_RESOURCES 0xDD56U +#define MTP_OB_PROP_ACT_ACCEPTED 0xDD57U +#define MTP_OB_PROP_OWNER 0xDD5DU +#define MTP_OB_PROP_EDITOR 0xDD5EU +#define MTP_OB_PROP_WEBMASTER 0xDD5FU +#define MTP_OB_PROP_URL_SOURCE 0xDD60U +#define MTP_OB_PROP_URL_DESTINATION 0xDD61U +#define MTP_OB_PROP_TIME_BOOKMARK 0xDD62U +#define MTP_OB_PROP_OBJECT_BOOKMARK 0xDD63U +#define MTP_OB_PROP_BYTE_BOOKMARK 0xDD64U +#define MTP_OB_PROP_LAST_BUILD_DATE 0xDD70U +#define MTP_OB_PROP_TIME_TO_LIVE 0xDD71U +#define MTP_OB_PROP_MEDIA_GUID 0xDD72U + +/* MTP event codes*/ +#define MTP_EVENT_UNDEFINED 0x4000U +#define MTP_EVENT_CANCELTRANSACTION 0x4001U +#define MTP_EVENT_OBJECTADDED 0x4002U +#define MTP_EVENT_OBJECTREMOVED 0x4003U +#define MTP_EVENT_STOREADDED 0x4004U +#define MTP_EVENT_STOREREMOVED 0x4005U +#define MTP_EVENT_DEVICEPROPCHANGED 0x4006U +#define MTP_EVENT_OBJECTINFOCHANGED 0x4007U +#define MTP_EVENT_DEVICEINFOCHANGED 0x4008U +#define MTP_EVENT_REQUESTOBJECTTRANSFER 0x4009U +#define MTP_EVENT_STOREFULL 0x400AU +#define MTP_EVENT_DEVICERESET 0x400BU +#define MTP_EVENT_STORAGEINFOCHANGED 0x400CU +#define MTP_EVENT_CAPTURECOMPLETE 0x400DU +#define MTP_EVENT_UNREPORTEDSTATUS 0x400EU +#define MTP_EVENT_OBJECTPROPCHANGED 0xC801U +#define MTP_EVENT_OBJECTPROPDESCCHANGED 0xC802U +#define MTP_EVENT_OBJECTREFERENCESCHANGED 0xC803U + +/* + * MTP Class specification Revision 1.1 + * Appendix D. Operations + */ + +/* Operations code */ +#define MTP_OP_GET_DEVICE_INFO 0x1001U +#define MTP_OP_OPEN_SESSION 0x1002U +#define MTP_OP_CLOSE_SESSION 0x1003U +#define MTP_OP_GET_STORAGE_IDS 0x1004U +#define MTP_OP_GET_STORAGE_INFO 0x1005U +#define MTP_OP_GET_NUM_OBJECTS 0x1006U +#define MTP_OP_GET_OBJECT_HANDLES 0x1007U +#define MTP_OP_GET_OBJECT_INFO 0x1008U +#define MTP_OP_GET_OBJECT 0x1009U +#define MTP_OP_GET_THUMB 0x100AU +#define MTP_OP_DELETE_OBJECT 0x100BU +#define MTP_OP_SEND_OBJECT_INFO 0x100CU +#define MTP_OP_SEND_OBJECT 0x100DU +#define MTP_OP_FORMAT_STORE 0x100FU +#define MTP_OP_RESET_DEVICE 0x1010U +#define MTP_OP_GET_DEVICE_PROP_DESC 0x1014U +#define MTP_OP_GET_DEVICE_PROP_VALUE 0x1015U +#define MTP_OP_SET_DEVICE_PROP_VALUE 0x1016U +#define MTP_OP_RESET_DEVICE_PROP_VALUE 0x1017U +#define MTP_OP_TERMINATE_OPEN_CAPTURE 0x1018U +#define MTP_OP_MOVE_OBJECT 0x1019U +#define MTP_OP_COPY_OBJECT 0x101AU +#define MTP_OP_GET_PARTIAL_OBJECT 0x101BU +#define MTP_OP_INITIATE_OPEN_CAPTURE 0x101CU +#define MTP_OP_GET_OBJECT_PROPS_SUPPORTED 0x9801U +#define MTP_OP_GET_OBJECT_PROP_DESC 0x9802U +#define MTP_OP_GET_OBJECT_PROP_VALUE 0x9803U +#define MTP_OP_SET_OBJECT_PROP_VALUE 0x9804U +#define MTP_OP_GET_OBJECT_PROPLIST 0x9805U +#define MTP_OP_GET_OBJECT_PROP_REFERENCES 0x9810U +#define MTP_OP_GETSERVICEIDS 0x9301U +#define MTP_OP_GETSERVICEINFO 0x9302U +#define MTP_OP_GETSERVICECAPABILITIES 0x9303U +#define MTP_OP_GETSERVICEPROPDESC 0x9304U + +/* + * MTP Class specification Revision 1.1 + * Appendix C. Device Properties + */ + +/* MTP device properties code*/ +#define MTP_DEV_PROP_UNDEFINED 0x5000U +#define MTP_DEV_PROP_BATTERY_LEVEL 0x5001U +#define MTP_DEV_PROP_FUNCTIONAL_MODE 0x5002U +#define MTP_DEV_PROP_IMAGE_SIZE 0x5003U +#define MTP_DEV_PROP_COMPRESSION_SETTING 0x5004U +#define MTP_DEV_PROP_WHITE_BALANCE 0x5005U +#define MTP_DEV_PROP_RGB_GAIN 0x5006U +#define MTP_DEV_PROP_F_NUMBER 0x5007U +#define MTP_DEV_PROP_FOCAL_LENGTH 0x5008U +#define MTP_DEV_PROP_FOCUS_DISTANCE 0x5009U +#define MTP_DEV_PROP_FOCUS_MODE 0x500AU +#define MTP_DEV_PROP_EXPOSURE_METERING_MODE 0x500BU +#define MTP_DEV_PROP_FLASH_MODE 0x500CU +#define MTP_DEV_PROP_EXPOSURE_TIME 0x500DU +#define MTP_DEV_PROP_EXPOSURE_PROGRAM_MODE 0x500EU +#define MTP_DEV_PROP_EXPOSURE_INDEX 0x500FU +#define MTP_DEV_PROP_EXPOSURE_BIAS_COMPENSATION 0x5010U +#define MTP_DEV_PROP_DATETIME 0x5011U +#define MTP_DEV_PROP_CAPTURE_DELAY 0x5012U +#define MTP_DEV_PROP_STILL_CAPTURE_MODE 0x5013U +#define MTP_DEV_PROP_CONTRAST 0x5014U +#define MTP_DEV_PROP_SHARPNESS 0x5015U +#define MTP_DEV_PROP_DIGITAL_ZOOM 0x5016U +#define MTP_DEV_PROP_EFFECT_MODE 0x5017U +#define MTP_DEV_PROP_BURST_NUMBER 0x5018U +#define MTP_DEV_PROP_BURST_INTERVAL 0x5019U +#define MTP_DEV_PROP_TIMELAPSE_NUMBER 0x501AU +#define MTP_DEV_PROP_TIMELAPSE_INTERVAL 0x501BU +#define MTP_DEV_PROP_FOCUS_METERING_MODE 0x501CU +#define MTP_DEV_PROP_UPLOAD_URL 0x501DU +#define MTP_DEV_PROP_ARTIST 0x501EU +#define MTP_DEV_PROP_COPYRIGHT_INFO 0x501FU +#define MTP_DEV_PROP_SYNCHRONIZATION_PARTNER 0xD401U +#define MTP_DEV_PROP_DEVICE_FRIENDLY_NAME 0xD402U +#define MTP_DEV_PROP_VOLUME 0xD403U +#define MTP_DEV_PROP_SUPPORTEDFORMATSORDERED 0xD404U +#define MTP_DEV_PROP_DEVICEICON 0xD405U +#define MTP_DEV_PROP_PLAYBACK_RATE 0xD410U +#define MTP_DEV_PROP_PLAYBACK_OBJECT 0xD411U +#define MTP_DEV_PROP_PLAYBACK_CONTAINER 0xD412U +#define MTP_DEV_PROP_SESSION_INITIATOR_VERSION_INFO 0xD406U +#define MTP_DEV_PROP_PERCEIVED_DEVICE_TYPE 0xD407U + + +/* Container Types */ +#define MTP_CONT_TYPE_UNDEFINED 0U +#define MTP_CONT_TYPE_COMMAND 1U +#define MTP_CONT_TYPE_DATA 2U +#define MTP_CONT_TYPE_RESPONSE 3U +#define MTP_CONT_TYPE_EVENT 4U + +#ifndef MTP_STORAGE_ID +#define MTP_STORAGE_ID 0x00010001U /* SD card is inserted*/ +#endif /* MTP_STORAGE_ID */ + +#define MTP_NBR_STORAGE_ID 1U +#define FREE_SPACE_IN_OBJ_NOT_USED 0xFFFFFFFFU + +/* MTP storage type */ +#define MTP_STORAGE_UNDEFINED 0U +#define MTP_STORAGE_FIXED_ROM 0x0001U +#define MTP_STORAGE_REMOVABLE_ROM 0x0002U +#define MTP_STORAGE_FIXED_RAM 0x0003U +#define MTP_STORAGE_REMOVABLE_RAM 0x0004U + +/* MTP file system type */ +#define MTP_FILESYSTEM_UNDEFINED 0U +#define MTP_FILESYSTEM_GENERIC_FLAT 0x0001U +#define MTP_FILESYSTEM_GENERIC_HIERARCH 0x0002U +#define MTP_FILESYSTEM_DCF 0x0003U + +/* MTP access capability */ +#define MTP_ACCESS_CAP_RW 0U /* read write */ +#define MTP_ACCESS_CAP_RO_WITHOUT_DEL 0x0001U +#define MTP_ACCESS_CAP_RO_WITH_DEL 0x0002U + +/* MTP standard data types supported */ +#define MTP_DATATYPE_INT8 0x0001U +#define MTP_DATATYPE_UINT8 0x0002U +#define MTP_DATATYPE_INT16 0x0003U +#define MTP_DATATYPE_UINT16 0x0004U +#define MTP_DATATYPE_INT32 0x0005U +#define MTP_DATATYPE_UINT32 0x0006U +#define MTP_DATATYPE_INT64 0x0007U +#define MTP_DATATYPE_UINT64 0x0008U +#define MTP_DATATYPE_UINT128 0x000AU +#define MTP_DATATYPE_STR 0xFFFFU + +/* MTP reading only or reading/writing */ +#define MTP_PROP_GET 0x00U +#define MTP_PROP_GET_SET 0x01U + + +/* MTP functional mode */ +#define STANDARD_MODE 0U +#define SLEEP_STATE 1U +#define FUNCTIONAL_MODE STANDARD_MODE + +/* MTP device info */ +#define STANDARD_VERSION 100U +#define VEND_EXT_ID 0x06U +#define VEND_EXT_VERSION 100U +#define MANUF_LEN (sizeof(Manuf) / 2U) +#define MODEL_LEN (sizeof(Model) / 2U) +#define SUPP_OP_LEN (sizeof(SuppOP) / 2U) +#define SERIAL_NBR_LEN (sizeof(SerialNbr) / 2U) +#define DEVICE_VERSION_LEN (sizeof(DeviceVers) / 2U) +#define SUPP_IMG_FORMAT_LEN (sizeof(SuppImgFormat) / 2U) +#define SUPP_OBJ_PROP_LEN (sizeof(ObjectPropSupp) / 2U) + +#ifndef MAX_FILE_NAME +#define MAX_FILE_NAME 255U +#endif /* MAX_FILE_NAME */ + +#ifndef MAX_OBJECT_HANDLE_LEN +#define MAX_OBJECT_HANDLE_LEN 100U +#endif /* MAX_OBJECT_HANDLE_LEN */ + +#ifndef DEVICE_PROP_DESC_DEF_LEN +#define DEVICE_PROP_DESC_DEF_LEN (uint8_t)(sizeof(DevicePropDefVal) / 2U) +#endif /* DEVICE_PROP_DESC_DEF_LEN */ + +#ifndef DEVICE_PROP_DESC_CUR_LEN +#define DEVICE_PROP_DESC_CUR_LEN (uint8_t)(sizeof(DevicePropCurDefVal) / 2U) +#endif /* DEVICE_PROP_DESC_CUR_LEN */ + +#ifndef DEFAULT_FILE_NAME_LEN +#define DEFAULT_FILE_NAME_LEN (uint8_t)(sizeof(DefaultFileName) / 2U) +#endif /* DEFAULT_FILE_NAME_LEN */ + +/** + * @} + */ + + +/** @defgroup USBD_MTP_OPT_Exported_TypesDefinitions + * @{ + */ + +/** + * @} + */ + +static const uint16_t SuppOP[] = { MTP_OP_GET_DEVICE_INFO, MTP_OP_OPEN_SESSION, MTP_OP_CLOSE_SESSION, + MTP_OP_GET_STORAGE_IDS, MTP_OP_GET_STORAGE_INFO, MTP_OP_GET_NUM_OBJECTS, + MTP_OP_GET_OBJECT_HANDLES, MTP_OP_GET_OBJECT_INFO, MTP_OP_GET_OBJECT, + MTP_OP_DELETE_OBJECT, MTP_OP_SEND_OBJECT_INFO, MTP_OP_SEND_OBJECT, + MTP_OP_GET_DEVICE_PROP_DESC, MTP_OP_GET_DEVICE_PROP_VALUE, + MTP_OP_SET_OBJECT_PROP_VALUE, MTP_OP_GET_OBJECT_PROP_VALUE, + MTP_OP_GET_OBJECT_PROPS_SUPPORTED, MTP_OP_GET_OBJECT_PROPLIST, + MTP_OP_GET_OBJECT_PROP_DESC, MTP_OP_GET_OBJECT_PROP_REFERENCES + }; + +static const uint16_t SuppEvents[] = {MTP_EVENT_OBJECTADDED}; +static const uint16_t SuppImgFormat[] = {MTP_OBJ_FORMAT_UNDEFINED, MTP_OBJ_FORMAT_TEXT, MTP_OBJ_FORMAT_ASSOCIATION, + MTP_OBJ_FORMAT_EXECUTABLE, MTP_OBJ_FORMAT_WAV, MTP_OBJ_FORMAT_MP3, + MTP_OBJ_FORMAT_EXIF_JPEG, MTP_OBJ_FORMAT_MPEG, MTP_OBJ_FORMAT_MP4_CONTAINER, + MTP_OBJ_FORMAT_WINDOWS_IMAGE_FORMAT, MTP_OBJ_FORMAT_PNG, MTP_OBJ_FORMAT_WMA, + MTP_OBJ_FORMAT_WMV + }; + +static const uint16_t SuppCaptFormat[] = {MTP_OBJ_FORMAT_UNDEFINED, MTP_OBJ_FORMAT_ASSOCIATION, MTP_OBJ_FORMAT_TEXT}; + +/* required for all object format : storageID, objectFormat, ObjectCompressedSize, +persistent unique object identifier, name*/ +static const uint16_t ObjectPropSupp[] = {MTP_OB_PROP_STORAGE_ID, MTP_OB_PROP_OBJECT_FORMAT, MTP_OB_PROP_OBJECT_SIZE, + MTP_OB_PROP_OBJ_FILE_NAME, MTP_OB_PROP_PARENT_OBJECT, MTP_OB_PROP_NAME, + MTP_OB_PROP_PERS_UNIQ_OBJ_IDEN, MTP_OB_PROP_PROTECTION_STATUS + }; + +static const uint16_t DevicePropSupp[] = {MTP_DEV_PROP_DEVICE_FRIENDLY_NAME, MTP_DEV_PROP_BATTERY_LEVEL}; + +/* for all mtp struct */ +typedef struct +{ + uint32_t StorageIDS_len; + uint32_t StorageIDS[MTP_NBR_STORAGE_ID]; +} MTP_StorageIDSTypeDef; + +#if defined ( __GNUC__ ) +typedef __PACKED_STRUCT +#else +__packed typedef struct +#endif /* __GNUC__ */ +{ + uint8_t FileName_len; + uint16_t FileName[MAX_FILE_NAME]; +} MTP_FileNameTypeDef; + + +typedef struct +{ + uint32_t ObjectHandle_len; + uint32_t ObjectHandle[MAX_OBJECT_HANDLE_LEN]; +} MTP_ObjectHandleTypeDef; + +#if defined ( __GNUC__ ) +typedef __PACKED_STRUCT +#else +__packed typedef struct +#endif /* __GNUC__ */ +{ + uint32_t ObjectPropSupp_len; + uint16_t ObjectPropSupp[SUPP_OBJ_PROP_LEN]; +} MTP_ObjectPropSuppTypeDef; + + +#if defined ( __GNUC__ ) +typedef __PACKED_STRUCT +#else +__packed typedef struct +#endif /* __GNUC__ */ +{ + uint16_t StorageType; + uint16_t FilesystemType; + uint16_t AccessCapability; + uint64_t MaxCapability; + uint64_t FreeSpaceInBytes; + uint32_t FreeSpaceInObjects; + uint8_t StorageDescription; + uint8_t VolumeLabel; +} MTP_StorageInfoTypedef; + +typedef union +{ + uint16_t str[100]; + uint8_t u8; + int8_t i8; + uint16_t u16; + int16_t i16; + uint32_t u32; + int32_t i32; + uint64_t u64; + int64_t i64; +} MTP_PropertyValueTypedef; + +#if defined ( __GNUC__ ) +typedef __PACKED_STRUCT +#else +__packed typedef struct +#endif /* __GNUC__ */ +{ + uint16_t ObjectPropertyCode; + uint16_t DataType; + uint8_t GetSet; + uint8_t *DefValue; + uint32_t GroupCode; + uint8_t FormFlag; +} MTP_ObjectPropDescTypeDef; + + +#if defined ( __GNUC__ ) +typedef __PACKED_STRUCT +#else +__packed typedef struct +#endif /* __GNUC__ */ +{ + uint32_t ObjectHandle; + uint16_t PropertyCode; + uint16_t Datatype; + uint8_t *propval; +} MTP_PropertiesTypedef; + +#if defined ( __GNUC__ ) +typedef __PACKED_STRUCT +#else +__packed typedef struct +#endif /* __GNUC__ */ +{ + uint32_t MTP_Properties_len; + MTP_PropertiesTypedef MTP_Properties[SUPP_OBJ_PROP_LEN]; +} MTP_PropertiesListTypedef; + + +#if defined ( __GNUC__ ) +typedef __PACKED_STRUCT +#else +__packed typedef struct +#endif /* __GNUC__ */ +{ + uint32_t ref_len; + uint32_t ref[1]; +} MTP_RefTypeDef; + +#if defined ( __GNUC__ ) +typedef __PACKED_STRUCT +#else +__packed typedef struct +#endif /* __GNUC__ */ +{ + uint16_t DevicePropertyCode; + uint16_t DataType; + uint8_t GetSet; + uint8_t DefValue_len; + uint16_t DefValue[DEVICE_PROP_DESC_DEF_LEN]; + uint8_t curDefValue_len; + uint16_t curDefValue[DEVICE_PROP_DESC_CUR_LEN]; + uint8_t FormFlag; +} MTP_DevicePropDescTypeDef; + +/* MTP device info structure */ +#if defined ( __GNUC__ ) +typedef __PACKED_STRUCT +#else +__packed typedef struct +#endif /* __GNUC__ */ +{ + uint16_t StandardVersion; + uint32_t VendorExtensionID; + uint16_t VendorExtensionVersion; + uint8_t VendorExtensionDesc_len; +#if USBD_MTP_VEND_EXT_DESC_SUPPORTED == 1 + uint16_t VendorExtensionDesc[VEND_EXT_DESC_LEN]; +#endif /* USBD_MTP_VEND_EXT_DESC_SUPPORTED */ + uint16_t FunctionalMode; + uint32_t OperationsSupported_len; + uint16_t OperationsSupported[SUPP_OP_LEN]; + uint32_t EventsSupported_len; +#if USBD_MTP_EVENTS_SUPPORTED == 1 + uint16_t EventsSupported[SUPP_EVENTS_LEN]; +#endif /* USBD_MTP_EVENTS_SUPPORTED */ + uint32_t DevicePropertiesSupported_len; +#if USBD_MTP_DEVICE_PROP_SUPPORTED == 1 + uint16_t DevicePropertiesSupported[SUPP_DEVICE_PROP_LEN]; +#endif /* USBD_MTP_DEVICE_PROP_SUPPORTED */ + uint32_t CaptureFormats_len; +#if USBD_MTP_CAPTURE_FORMAT_SUPPORTED == 1 + uint16_t CaptureFormats[SUPP_CAPT_FORMAT_LEN]; +#endif /* USBD_MTP_CAPTURE_FORMAT_SUPPORTED */ + uint32_t ImageFormats_len; + uint16_t ImageFormats[SUPP_IMG_FORMAT_LEN]; + uint8_t Manufacturer_len; + uint16_t Manufacturer[MANUF_LEN]; + uint8_t Model_len; + uint16_t Model[MODEL_LEN]; + uint8_t DeviceVersion_len; + uint16_t DeviceVersion[DEVICE_VERSION_LEN]; + uint8_t SerialNumber_len; + uint16_t SerialNumber[SERIAL_NBR_LEN]; +} MTP_DeviceInfoTypedef; + +/** @defgroup USBD_MTP_OPT_Exported_Macros + * @{ + */ + +/** + * @} + */ + +/** @defgroup USBD_MTP_OPT_Exported_Variables + * @{ + */ + +/** + * @} + */ + +/** @defgroup USBD_MTP_OPT_Exported_Functions + * @{ + */ + +void USBD_MTP_OPT_CreateObjectHandle(USBD_HandleTypeDef *pdev); +void USBD_MTP_OPT_GetDeviceInfo(USBD_HandleTypeDef *pdev); +void USBD_MTP_OPT_GetStorageIDS(USBD_HandleTypeDef *pdev); +void USBD_MTP_OPT_GetStorageInfo(USBD_HandleTypeDef *pdev); +void USBD_MTP_OPT_GetObjectHandle(USBD_HandleTypeDef *pdev); +void USBD_MTP_OPT_GetObjectInfo(USBD_HandleTypeDef *pdev); +void USBD_MTP_OPT_GetObjectReferences(USBD_HandleTypeDef *pdev); +void USBD_MTP_OPT_GetObjectPropSupp(USBD_HandleTypeDef *pdev); +void USBD_MTP_OPT_GetObjectPropDesc(USBD_HandleTypeDef *pdev); +void USBD_MTP_OPT_GetObjectPropValue(USBD_HandleTypeDef *pdev); +void USBD_MTP_OPT_GetObjectPropList(USBD_HandleTypeDef *pdev); +void USBD_MTP_OPT_GetDevicePropDesc(USBD_HandleTypeDef *pdev); +void USBD_MTP_OPT_SendObjectInfo(USBD_HandleTypeDef *pdev, uint8_t *buff, uint32_t len); +void USBD_MTP_OPT_SendObject(USBD_HandleTypeDef *pdev, uint8_t *buff, uint32_t len); +void USBD_MTP_OPT_GetObject(USBD_HandleTypeDef *pdev); +void USBD_MTP_OPT_DeleteObject(USBD_HandleTypeDef *pdev); + + +/** + * @} + */ + +#ifdef __cplusplus +} +#endif + +#endif /* __USBD_MTP_OPT_H__ */ +/** + * @} + */ + +/** + * @} + */ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp_storage.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp_storage.h new file mode 100644 index 00000000..ba01901e --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Inc/usbd_mtp_storage.h @@ -0,0 +1,118 @@ +/** + ****************************************************************************** + * @file usbd_mtp_storage.h + * @author MCD Application Team + * @brief header file for the usbd_mtp_storage.c file. + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Define to prevent recursive inclusion -------------------------------------*/ +#ifndef __USBD_MTP_STORAGE_H__ +#define __USBD_MTP_STORAGE_H__ + +#ifdef __cplusplus +extern "C" { +#endif + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_ctlreq.h" +#include "usbd_mtp_opt.h" + +/** @addtogroup STM32_USB_DEVICE_LIBRARY + * @{ + */ + +/** @defgroup USBD_MTP_STORAGE + * @brief This file is the header file for usbd_template_core.c + * @{ + */ + + +/** @defgroup USBD_MTP_STORAGE_Exported_Defines + * @{ + */ + +/** + * @} + */ + + +/** @defgroup USBD_MTP_STORAGE_Exported_TypesDefinitions + * @{ + */ + +typedef enum +{ + DATA_TYPE = 0x00, + REP_TYPE = 0x01, +} MTP_CONTAINER_TYPE; + + +typedef enum +{ + READ_FIRST_DATA = 0x00, + READ_REST_OF_DATA = 0x01, +} MTP_READ_DATA_STATUS; + + +/** + * @} + */ + + + +/** @defgroup USBD_MTP_STORAGE_Exported_Macros + * @{ + */ + +/** + * @} + */ + +/** @defgroup USBD_MTP_STORAGE_Exported_Variables + * @{ + */ + +/** + * @} + */ + +/** @defgroup USBD_MTP_STORAGE_Exported_Functions + * @{ + */ + +uint8_t USBD_MTP_STORAGE_Init(USBD_HandleTypeDef *pdev); +uint8_t USBD_MTP_STORAGE_DeInit(USBD_HandleTypeDef *pdev); +void USBD_MTP_STORAGE_Cancel(USBD_HandleTypeDef *pdev, MTP_ResponsePhaseTypeDef MTP_ResponsePhase); +uint8_t USBD_MTP_STORAGE_ReadData(USBD_HandleTypeDef *pdev); +uint8_t USBD_MTP_STORAGE_SendContainer(USBD_HandleTypeDef *pdev, MTP_CONTAINER_TYPE CONT_TYPE); +uint8_t USBD_MTP_STORAGE_ReceiveOpt(USBD_HandleTypeDef *pdev); +uint8_t USBD_MTP_STORAGE_ReceiveData(USBD_HandleTypeDef *pdev); + + +/** + * @} + */ + +#ifdef __cplusplus +} +#endif + +#endif /* __USBD_MTP_STORAGE_H */ +/** + * @} + */ + +/** + * @} + */ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp.c new file mode 100644 index 00000000..b0950e8e --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp.c @@ -0,0 +1,701 @@ +/** + ****************************************************************************** + * @file usbd_mtp.c + * @author MCD Application Team + * @brief This file provides the high layer firmware functions to manage the + * following functionalities of the USB MTP Class: + * - Initialization and Configuration of high and low layer + * - Enumeration as MTP Device (and enumeration for each implemented memory interface) + * - OUT/IN data transfer + * - Command IN transfer (class requests management) + * - Error management + * + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + * @verbatim + * + * =================================================================== + * MTP Class Driver Description + * =================================================================== + * This driver manages the "Universal Serial Bus Class Definitions for Media Transfer Protocol + * Revision 1.1 April 6, 2011" + * This driver implements the following aspects of the specification: + * - Device descriptor management + * - Configuration descriptor management + * - Enumeration as MTP device with 2 data endpoints (IN and OUT) and 1 command endpoint (IN) + * - Requests management + * + * + * @endverbatim + * + ****************************************************************************** + */ + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_mtp.h" +#include "usbd_mtp_storage.h" + +/** @addtogroup STM32_USB_DEVICE_LIBRARY + * @{ + */ + + +/** @defgroup USBD_MTP + * @brief usbd core module + * @{ + */ + +/** @defgroup USBD_MTP_Private_TypesDefinitions + * @{ + */ +/** + * @} + */ + + +/** @defgroup USBD_MTP_Private_Defines + * @{ + */ +/** + * @} + */ + + +/** @defgroup USBD_MTP_Private_Macros + * @{ + */ + +/** + * @} + */ + + +/** @defgroup USBD_MTP_Private_FunctionPrototypes + * @{ + */ +static uint8_t USBD_MTP_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx); +static uint8_t USBD_MTP_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx); +static uint8_t USBD_MTP_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); +static uint8_t USBD_MTP_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum); +static uint8_t USBD_MTP_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum); + +#ifndef USE_USBD_COMPOSITE +static uint8_t *USBD_MTP_GetHSCfgDesc(uint16_t *length); +static uint8_t *USBD_MTP_GetFSCfgDesc(uint16_t *length); +static uint8_t *USBD_MTP_GetOtherSpeedCfgDesc(uint16_t *length); +static uint8_t *USBD_MTP_GetDeviceQualifierDescriptor(uint16_t *length); +#endif /* USE_USBD_COMPOSITE */ + +/** + * @} + */ + +/** @defgroup USBD_MTP_Private_Variables + * @{ + */ + + +/* MTP interface class callbacks structure */ +USBD_ClassTypeDef USBD_MTP = +{ + USBD_MTP_Init, + USBD_MTP_DeInit, + USBD_MTP_Setup, + NULL, /*EP0_TxSent*/ + NULL, /*EP0_RxReady*/ + USBD_MTP_DataIn, + USBD_MTP_DataOut, + NULL, /*SOF */ + NULL, /*ISOIn*/ + NULL, /*ISOOut*/ +#ifdef USE_USBD_COMPOSITE + NULL, + NULL, + NULL, + NULL, +#else + USBD_MTP_GetHSCfgDesc, + USBD_MTP_GetFSCfgDesc, + USBD_MTP_GetOtherSpeedCfgDesc, + USBD_MTP_GetDeviceQualifierDescriptor, +#endif /* USE_USBD_COMPOSITE */ +}; + +#ifndef USE_USBD_COMPOSITE + +/* USB MTP device Configuration Descriptor */ +__ALIGN_BEGIN static uint8_t USBD_MTP_CfgDesc[MTP_CONFIG_DESC_SIZ] __ALIGN_END = +{ + /* Configuration Descriptor */ + 0x09, /* bLength: Configuration Descriptor size */ + USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ + LOBYTE(MTP_CONFIG_DESC_SIZ), /* wTotalLength: Total size of the Config descriptor */ + HIBYTE(MTP_CONFIG_DESC_SIZ), + 0x01, /* bNumInterfaces: 1 interface */ + 0x01, /* bConfigurationValue: Configuration value */ + 0x00, /* iConfiguration: Index of string descriptor + describing the configuration */ +#if (USBD_SELF_POWERED == 1U) + 0xC0, /* bmAttributes: Bus Powered according to user configuration */ +#else + 0x80, /* bmAttributes: Bus Powered according to user configuration */ +#endif /* USBD_SELF_POWERED */ + USBD_MAX_POWER, /* MaxPower (mA) */ + + /******************** MTP **** interface ********************/ + MTP_INTERFACE_DESC_SIZE, /* bLength: Interface Descriptor size */ + USB_DESC_TYPE_INTERFACE, /* bDescriptorType: Interface descriptor type */ + MTP_CMD_ITF_NBR, /* bInterfaceNumber: Number of Interface */ + 0x00, /* bAlternateSetting: Alternate setting */ + 0x03, /* bNumEndpoints: */ + USB_MTP_INTRERFACE_CLASS, /* bInterfaceClass: bInterfaceClass: user's interface for MTP */ + USB_MTP_INTRERFACE_SUB_CLASS, /* bInterfaceSubClass:Abstract Control Model */ + USB_MTP_INTRERFACE_PROTOCOL, /* bInterfaceProtocol: Common AT commands */ + 0x00, /* iInterface: */ + + /******************** MTP Endpoints ********************/ + MTP_ENDPOINT_DESC_SIZE, /* Endpoint descriptor length = 7 */ + USB_DESC_TYPE_ENDPOINT, /* Endpoint descriptor type */ + MTP_IN_EP, /* Endpoint address (IN, address 1) */ + USBD_EP_TYPE_BULK, /* Bulk endpoint type */ + LOBYTE(MTP_DATA_MAX_FS_PACKET_SIZE), + HIBYTE(MTP_DATA_MAX_FS_PACKET_SIZE), + 0x00, /* Polling interval in milliseconds */ + + MTP_ENDPOINT_DESC_SIZE, /* Endpoint descriptor length = 7 */ + USB_DESC_TYPE_ENDPOINT, /* Endpoint descriptor type */ + MTP_OUT_EP, /* Endpoint address (OUT, address 1) */ + USBD_EP_TYPE_BULK, /* Bulk endpoint type */ + LOBYTE(MTP_DATA_MAX_FS_PACKET_SIZE), + HIBYTE(MTP_DATA_MAX_FS_PACKET_SIZE), + 0x00, /* Polling interval in milliseconds */ + + MTP_ENDPOINT_DESC_SIZE, /* bLength: Endpoint Descriptor size */ + USB_DESC_TYPE_ENDPOINT, /* bDescriptorType:*/ + MTP_CMD_EP, /* bEndpointAddress: Endpoint Address (IN) */ + USBD_EP_TYPE_INTR, /* bmAttributes: Interrupt endpoint */ + LOBYTE(MTP_CMD_PACKET_SIZE), + HIBYTE(MTP_CMD_PACKET_SIZE), + MTP_FS_BINTERVAL /* Polling interval in milliseconds */ +}; + +/* USB Standard Device Descriptor */ +__ALIGN_BEGIN static uint8_t USBD_MTP_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_DESC] __ALIGN_END = +{ + USB_LEN_DEV_QUALIFIER_DESC, + USB_DESC_TYPE_DEVICE_QUALIFIER, + 0x00, + 0x02, + 0x00, + 0x00, + 0x00, + 0x40, + 0x01, + 0x00, +}; +#endif /* USE_USBD_COMPOSITE */ + +uint8_t MTPInEpAdd = MTP_IN_EP; +uint8_t MTPOutEpAdd = MTP_OUT_EP; +uint8_t MTPCmdEpAdd = MTP_CMD_EP; + +/** + * @} + */ + +/** @defgroup USBD_MTP_Private_Functions + * @{ + */ + +/** + * @brief USBD_MTP_Init + * Initialize the MTP interface + * @param pdev: device instance + * @param cfgidx: Configuration index + * @retval status + */ +static uint8_t USBD_MTP_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) +{ + UNUSED(cfgidx); + USBD_MTP_HandleTypeDef *hmtp; + + hmtp = (USBD_MTP_HandleTypeDef *)USBD_malloc(sizeof(USBD_MTP_HandleTypeDef)); + + if (hmtp == NULL) + { + pdev->pClassDataCmsit[pdev->classId] = NULL; + return (uint8_t)USBD_EMEM; + } + + /* Setup the pClassData pointer */ + pdev->pClassDataCmsit[pdev->classId] = (void *)hmtp; + pdev->pClassData = pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MTPInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + MTPOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + MTPCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + + /* Initialize all variables */ + (void)USBD_memset(hmtp, 0, sizeof(USBD_MTP_HandleTypeDef)); + + /* Setup the max packet size according to selected speed */ + if (pdev->dev_speed == USBD_SPEED_HIGH) + { + hmtp->MaxPcktLen = MTP_DATA_MAX_HS_PACKET_SIZE; + } + else + { + hmtp->MaxPcktLen = MTP_DATA_MAX_FS_PACKET_SIZE; + } + + /* Open EP IN */ + (void)USBD_LL_OpenEP(pdev, MTPInEpAdd, USBD_EP_TYPE_BULK, hmtp->MaxPcktLen); + pdev->ep_in[MTPInEpAdd & 0xFU].is_used = 1U; + + /* Open EP OUT */ + (void)USBD_LL_OpenEP(pdev, MTPOutEpAdd, USBD_EP_TYPE_BULK, hmtp->MaxPcktLen); + pdev->ep_out[MTPOutEpAdd & 0xFU].is_used = 1U; + + /* Open INTR EP IN */ + (void)USBD_LL_OpenEP(pdev, MTPCmdEpAdd, USBD_EP_TYPE_INTR, MTP_CMD_PACKET_SIZE); + pdev->ep_in[MTPCmdEpAdd & 0xFU].is_used = 1U; + + /* Init the MTP layer */ + (void)USBD_MTP_STORAGE_Init(pdev); + + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_MTP_DeInit + * DeInitialize the MTP layer + * @param pdev: device instance + * @param cfgidx: Configuration index + * @retval status + */ +static uint8_t USBD_MTP_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) +{ + UNUSED(cfgidx); + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this MTP class instance */ + MTPInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + MTPOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + MTPCmdEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_INTR, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + /* Close EP IN */ + (void)USBD_LL_CloseEP(pdev, MTPInEpAdd); + pdev->ep_in[MTPInEpAdd & 0xFU].is_used = 0U; + + /* Close EP OUT */ + (void)USBD_LL_CloseEP(pdev, MTPOutEpAdd); + pdev->ep_out[MTPOutEpAdd & 0xFU].is_used = 0U; + + /* Close EP Command */ + (void)USBD_LL_CloseEP(pdev, MTPCmdEpAdd); + pdev->ep_in[MTPCmdEpAdd & 0xFU].is_used = 0U; + + /* Free MTP Class Resources */ + if (pdev->pClassDataCmsit[pdev->classId] != NULL) + { + /* De-Init the MTP layer */ + (void)USBD_MTP_STORAGE_DeInit(pdev); + + (void)USBD_free(pdev->pClassDataCmsit[pdev->classId]); + pdev->pClassDataCmsit[pdev->classId] = NULL; + pdev->pClassData = NULL; + } + + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_MTP_Setup + * Handle the MTP specific requests + * @param pdev: instance + * @param req: usb requests + * @retval status + */ +static uint8_t USBD_MTP_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_StatusTypeDef ret = USBD_OK; + uint16_t len = 0U; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this MTP class instance */ + MTPOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + if (hmtp == NULL) + { + return (uint8_t)USBD_FAIL; + } + + switch (req->bmRequest & USB_REQ_TYPE_MASK) + { + /* Class request */ + case USB_REQ_TYPE_CLASS : + switch (req->bRequest) + { + case MTP_REQ_CANCEL: + len = MIN(hmtp->MaxPcktLen, req->wLength); + (void)USBD_CtlPrepareRx(pdev, (uint8_t *)(hmtp->rx_buff), len); + break; + + case MTP_REQ_GET_EXT_EVENT_DATA: + break; + + case MTP_REQ_RESET: + /* Stop low layer file system operations if any */ + USBD_MTP_STORAGE_Cancel(pdev, MTP_PHASE_IDLE); + + (void)USBD_LL_PrepareReceive(pdev, MTPOutEpAdd, (uint8_t *)&hmtp->rx_buff, hmtp->MaxPcktLen); + break; + + case MTP_REQ_GET_DEVICE_STATUS: + switch (hmtp->MTP_ResponsePhase) + { + case MTP_READ_DATA : + len = 4U; + hmtp->dev_status = ((uint32_t)MTP_RESPONSE_DEVICE_BUSY << 16) | len; + break; + + case MTP_RECEIVE_DATA : + len = 4U; + hmtp->dev_status = ((uint32_t)MTP_RESPONSE_TRANSACTION_CANCELLED << 16) | len; + break; + + case MTP_PHASE_IDLE : + len = 4U; + hmtp->dev_status = ((uint32_t)MTP_RESPONSE_OK << 16) | len; + break; + + default: + break; + } + (void)USBD_CtlSendData(pdev, (uint8_t *)&hmtp->dev_status, len); + break; + + default: + USBD_CtlError(pdev, req); + ret = USBD_FAIL; + break; + } + break; + + /* Interface & Endpoint request */ + case USB_REQ_TYPE_STANDARD: + switch (req->bRequest) + { + case USB_REQ_GET_INTERFACE : + + if (pdev->dev_state == USBD_STATE_CONFIGURED) + { + hmtp->alt_setting = 0U; + (void)USBD_CtlSendData(pdev, (uint8_t *)&hmtp->alt_setting, 1U); + } + break; + + case USB_REQ_SET_INTERFACE : + if (pdev->dev_state != USBD_STATE_CONFIGURED) + { + USBD_CtlError(pdev, req); + ret = USBD_FAIL; + } + break; + + case USB_REQ_CLEAR_FEATURE: + + /* Re-activate the EP */ + (void)USBD_LL_CloseEP(pdev, (uint8_t)req->wIndex); + + if ((((uint8_t)req->wIndex) & 0x80U) == 0x80U) + { + (void)USBD_LL_OpenEP(pdev, ((uint8_t)req->wIndex), USBD_EP_TYPE_BULK, hmtp->MaxPcktLen); + } + else + { + (void)USBD_LL_OpenEP(pdev, ((uint8_t)req->wIndex), USBD_EP_TYPE_BULK, hmtp->MaxPcktLen); + } + break; + + default: + break; + } + break; + + default: + break; + } + return (uint8_t)ret; +} + +/** + * @brief USBD_MTP_DataIn + * Data sent on non-control IN endpoint + * @param pdev: device instance + * @param epnum: endpoint number + * @retval status + */ +static uint8_t USBD_MTP_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) +{ + UNUSED(epnum); + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint16_t len; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this MTP class instance */ + MTPInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + MTPOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + if (epnum == (MTPInEpAdd & 0x7FU)) + { + switch (hmtp->MTP_ResponsePhase) + { + case MTP_RESPONSE_PHASE : + (void)USBD_MTP_STORAGE_SendContainer(pdev, REP_TYPE); + + /* prepare to receive next operation */ + len = MIN(hmtp->MaxPcktLen, pdev->request.wLength); + + (void)USBD_LL_PrepareReceive(pdev, MTPOutEpAdd, (uint8_t *)&hmtp->rx_buff, len); + hmtp->MTP_ResponsePhase = MTP_PHASE_IDLE; + break; + + case MTP_READ_DATA : + (void)USBD_MTP_STORAGE_ReadData(pdev); + + /* prepare to receive next operation */ + len = MIN(hmtp->MaxPcktLen, pdev->request.wLength); + + (void)USBD_LL_PrepareReceive(pdev, MTPInEpAdd, (uint8_t *)&hmtp->rx_buff, len); + break; + + case MTP_PHASE_IDLE : + /* prepare to receive next operation */ + len = MIN(hmtp->MaxPcktLen, pdev->request.wLength); + + (void)USBD_LL_PrepareReceive(pdev, MTPOutEpAdd, (uint8_t *)&hmtp->rx_buff, len); + + break; + default: + break; + } + } + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_MTP_DataOut + * Data received on non-control Out endpoint + * @param pdev: device instance + * @param epnum: endpoint number + * @retval status + */ +static uint8_t USBD_MTP_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) +{ + UNUSED(epnum); + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint16_t len; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this MTP class instance */ + MTPOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + (void)USBD_MTP_STORAGE_ReceiveOpt(pdev); + + switch (hmtp->MTP_ResponsePhase) + { + case MTP_RESPONSE_PHASE : + + if (hmtp->ResponseLength == MTP_CONT_HEADER_SIZE) + { + (void)USBD_MTP_STORAGE_SendContainer(pdev, REP_TYPE); + hmtp->MTP_ResponsePhase = MTP_PHASE_IDLE; + } + else + { + (void)USBD_MTP_STORAGE_SendContainer(pdev, DATA_TYPE); + } + break; + + case MTP_READ_DATA : + (void)USBD_MTP_STORAGE_ReadData(pdev); + break; + + case MTP_RECEIVE_DATA : + (void)USBD_MTP_STORAGE_ReceiveData(pdev); + + /* prepare endpoint to receive operations */ + len = MIN(hmtp->MaxPcktLen, pdev->request.wLength); + + (void)USBD_LL_PrepareReceive(pdev, MTPOutEpAdd, (uint8_t *)&hmtp->rx_buff, len); + break; + + case MTP_PHASE_IDLE : + /* prepare to receive next operation */ + len = MIN(hmtp->MaxPcktLen, pdev->request.wLength); + + (void)USBD_LL_PrepareReceive(pdev, MTPOutEpAdd, (uint8_t *)&hmtp->rx_buff, len); + break; + + default: + break; + } + + return (uint8_t)USBD_OK; +} + +#ifndef USE_USBD_COMPOSITE +/** + * @brief USBD_MTP_GetHSCfgDesc + * Return configuration descriptor + * @param speed : current device speed + * @param length : pointer data length + * @retval pointer to descriptor buffer + */ +static uint8_t *USBD_MTP_GetHSCfgDesc(uint16_t *length) +{ + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_MTP_CfgDesc, MTP_IN_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_MTP_CfgDesc, MTP_OUT_EP); + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_MTP_CfgDesc, MTP_CMD_EP); + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = MTP_DATA_MAX_HS_PACKET_SIZE; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = MTP_DATA_MAX_HS_PACKET_SIZE; + } + + if (pEpCmdDesc != NULL) + { + pEpCmdDesc->bInterval = MTP_HS_BINTERVAL; + } + + *length = (uint16_t)sizeof(USBD_MTP_CfgDesc); + return USBD_MTP_CfgDesc; +} + +/** + * @brief USBD_MTP_GetFSCfgDesc + * Return configuration descriptor + * @param speed : current device speed + * @param length : pointer data length + * @retval pointer to descriptor buffer + */ +static uint8_t *USBD_MTP_GetFSCfgDesc(uint16_t *length) +{ + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_MTP_CfgDesc, MTP_IN_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_MTP_CfgDesc, MTP_OUT_EP); + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_MTP_CfgDesc, MTP_CMD_EP); + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = MTP_DATA_MAX_FS_PACKET_SIZE; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = MTP_DATA_MAX_FS_PACKET_SIZE; + } + + if (pEpCmdDesc != NULL) + { + pEpCmdDesc->bInterval = MTP_FS_BINTERVAL; + } + + *length = (uint16_t)sizeof(USBD_MTP_CfgDesc); + return USBD_MTP_CfgDesc; +} + +/** + * @brief USBD_MTP_GetOtherSpeedCfgDesc + * Return configuration descriptor + * @param speed : current device speed + * @param length : pointer data length + * @retval pointer to descriptor buffer + */ +static uint8_t *USBD_MTP_GetOtherSpeedCfgDesc(uint16_t *length) +{ + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_MTP_CfgDesc, MTP_IN_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_MTP_CfgDesc, MTP_OUT_EP); + USBD_EpDescTypeDef *pEpCmdDesc = USBD_GetEpDesc(USBD_MTP_CfgDesc, MTP_CMD_EP); + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = MTP_DATA_MAX_FS_PACKET_SIZE; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = MTP_DATA_MAX_FS_PACKET_SIZE; + } + + if (pEpCmdDesc != NULL) + { + pEpCmdDesc->bInterval = MTP_FS_BINTERVAL; + } + + *length = (uint16_t)sizeof(USBD_MTP_CfgDesc); + return USBD_MTP_CfgDesc; +} + +/** + * @brief USBD_MTP_GetDeviceQualifierDescriptor + * return Device Qualifier descriptor + * @param length : pointer data length + * @retval pointer to descriptor buffer + */ +static uint8_t *USBD_MTP_GetDeviceQualifierDescriptor(uint16_t *length) +{ + *length = (uint16_t)(sizeof(USBD_MTP_DeviceQualifierDesc)); + return USBD_MTP_DeviceQualifierDesc; +} +#endif /* USE_USBD_COMPOSITE */ + +/** + * @brief USBD_MTP_RegisterInterface + * @param pdev: device instance + * @param fops: CD Interface callback + * @retval status + */ +uint8_t USBD_MTP_RegisterInterface(USBD_HandleTypeDef *pdev, USBD_MTP_ItfTypeDef *fops) +{ + if (fops == NULL) + { + return (uint8_t)USBD_FAIL; + } + + pdev->pUserData[pdev->classId] = fops; + + return (uint8_t)USBD_OK; +} + +/** + * @} + */ + +/** + * @} + */ + diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp_if_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp_if_template.c new file mode 100644 index 00000000..baeb9ca1 --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp_if_template.c @@ -0,0 +1,347 @@ +/** + ****************************************************************************** + * @file usbd_mtp_if.c + * @author MCD Application Team + * @brief Source file for USBD MTP file list_files. + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_mtp_if_template.h" + +/* Private typedef -----------------------------------------------------------*/ +/* Private variables ---------------------------------------------------------*/ +/* +static FILE MyFile; +static FATFS SDFatFs; +static char SDPath[4]; +static FolderLevel Fold_Lvl; +static FOLD_INFTypeDef FoldStruct; +static FILE_INFTypeDef FileStruct; +static SD_Object_TypeDef sd_object; +*/ +extern USBD_HandleTypeDef USBD_Device; + +uint32_t idx[200]; +uint32_t parent; +/* static char path[255]; */ +uint32_t sc_buff[MTP_IF_SCRATCH_BUFF_SZE / 4U]; +uint32_t sc_len = 0U; +uint32_t pckt_cnt = 1U; +uint32_t foldsize; + +/* Private define ------------------------------------------------------------*/ +/* Private macro -------------------------------------------------------------*/ +/* Private function prototypes -----------------------------------------------*/ +static uint8_t USBD_MTP_Itf_Init(void); +static uint8_t USBD_MTP_Itf_DeInit(void); +static uint32_t USBD_MTP_Itf_ReadData(uint32_t Param1, uint8_t *buff, MTP_DataLengthTypeDef *data_length); +static uint16_t USBD_MTP_Itf_Create_NewObject(MTP_ObjectInfoTypeDef ObjectInfo, uint32_t objhandle); + +static uint32_t USBD_MTP_Itf_GetIdx(uint32_t Param3, uint32_t *obj_handle); +static uint32_t USBD_MTP_Itf_GetParentObject(uint32_t Param); +static uint16_t USBD_MTP_Itf_GetObjectFormat(uint32_t Param); +static uint8_t USBD_MTP_Itf_GetObjectName_len(uint32_t Param); +static void USBD_MTP_Itf_GetObjectName(uint32_t Param, uint8_t obj_len, uint16_t *buf); +static uint32_t USBD_MTP_Itf_GetObjectSize(uint32_t Param); +static uint64_t USBD_MTP_Itf_GetMaxCapability(void); +static uint64_t USBD_MTP_Itf_GetFreeSpaceInBytes(void); +static uint32_t USBD_MTP_Itf_GetNewIndex(uint16_t objformat); +static void USBD_MTP_Itf_WriteData(uint16_t len, uint8_t *buff); +static uint32_t USBD_MTP_Itf_GetContainerLength(uint32_t Param1); +static uint16_t USBD_MTP_Itf_DeleteObject(uint32_t Param1); + +static void USBD_MTP_Itf_Cancel(uint32_t Phase); +/* static uint32_t USBD_MTP_Get_idx_to_delete(uint32_t Param, uint8_t *tab); */ + +USBD_MTP_ItfTypeDef USBD_MTP_fops = +{ + USBD_MTP_Itf_Init, + USBD_MTP_Itf_DeInit, + USBD_MTP_Itf_ReadData, + USBD_MTP_Itf_Create_NewObject, + USBD_MTP_Itf_GetIdx, + USBD_MTP_Itf_GetParentObject, + USBD_MTP_Itf_GetObjectFormat, + USBD_MTP_Itf_GetObjectName_len, + USBD_MTP_Itf_GetObjectName, + USBD_MTP_Itf_GetObjectSize, + USBD_MTP_Itf_GetMaxCapability, + USBD_MTP_Itf_GetFreeSpaceInBytes, + USBD_MTP_Itf_GetNewIndex, + USBD_MTP_Itf_WriteData, + USBD_MTP_Itf_GetContainerLength, + USBD_MTP_Itf_DeleteObject, + USBD_MTP_Itf_Cancel, + sc_buff, + MTP_IF_SCRATCH_BUFF_SZE, +}; + +/* Private functions ---------------------------------------------------------*/ + +/** + * @brief USBD_MTP_Itf_Init + * Initialize the file system Layer + * @param None + * @retval status value + */ +static uint8_t USBD_MTP_Itf_Init(void) +{ + return 0; +} + +/** + * @brief USBD_MTP_Itf_DeInit + * Uninitialize the file system Layer + * @param None + * @retval status value + */ +static uint8_t USBD_MTP_Itf_DeInit(void) +{ + return 0; +} + +/** + * @brief USBD_MTP_Itf_GetIdx + * Get all object handle + * @param Param3: current object handle + * @param obj_handle: all objects handle (subfolders/files) in current object + * @retval number of object handle in current object + */ +static uint32_t USBD_MTP_Itf_GetIdx(uint32_t Param3, uint32_t *obj_handle) +{ + uint32_t count = 0U; + UNUSED(Param3); + UNUSED(obj_handle); + + return count; +} + +/** + * @brief USBD_MTP_Itf_GetParentObject + * Get parent object + * @param Param: object handle (object index) + * @retval parent object + */ +static uint32_t USBD_MTP_Itf_GetParentObject(uint32_t Param) +{ + uint32_t parentobj = 0U; + UNUSED(Param); + + return parentobj; +} + +/** + * @brief USBD_MTP_Itf_GetObjectFormat + * Get object format + * @param Param: object handle (object index) + * @retval object format + */ +static uint16_t USBD_MTP_Itf_GetObjectFormat(uint32_t Param) +{ + uint16_t objformat = 0U; + UNUSED(Param); + + return objformat; +} + +/** + * @brief USBD_MTP_Itf_GetObjectName_len + * Get object name length + * @param Param: object handle (object index) + * @retval object name length + */ +static uint8_t USBD_MTP_Itf_GetObjectName_len(uint32_t Param) +{ + uint8_t obj_len = 0U; + UNUSED(Param); + + return obj_len; +} + +/** + * @brief USBD_MTP_Itf_GetObjectName + * Get object name + * @param Param: object handle (object index) + * @param obj_len: length of object name + * @param buf: pointer to object name + * @retval object size in SD card + */ +static void USBD_MTP_Itf_GetObjectName(uint32_t Param, uint8_t obj_len, uint16_t *buf) +{ + UNUSED(Param); + UNUSED(obj_len); + UNUSED(buf); + + return; +} + +/** + * @brief USBD_MTP_Itf_GetObjectSize + * Get size of current object + * @param Param: object handle (object index) + * @retval object size in SD card + */ +static uint32_t USBD_MTP_Itf_GetObjectSize(uint32_t Param) +{ + uint32_t ObjCompSize = 0U; + UNUSED(Param); + + return ObjCompSize; +} + +/** + * @brief USBD_MTP_Itf_Create_NewObject + * Create new object in SD card and store necessary information for future use + * @param ObjectInfo: object information to use + * @param objhandle: object handle (object index) + * @retval None + */ +static uint16_t USBD_MTP_Itf_Create_NewObject(MTP_ObjectInfoTypeDef ObjectInfo, uint32_t objhandle) +{ + uint16_t rep_code = 0U; + UNUSED(ObjectInfo); + UNUSED(objhandle); + + return rep_code; +} + +/** + * @brief USBD_MTP_Itf_GetMaxCapability + * Get max capability in SD card + * @param None + * @retval max capability + */ +static uint64_t USBD_MTP_Itf_GetMaxCapability(void) +{ + uint64_t max_cap = 0U; + + return max_cap; +} + +/** + * @brief USBD_MTP_Itf_GetFreeSpaceInBytes + * Get free space in bytes in SD card + * @param None + * @retval free space in bytes + */ +static uint64_t USBD_MTP_Itf_GetFreeSpaceInBytes(void) +{ + uint64_t f_space_inbytes = 0U; + + return f_space_inbytes; +} + +/** + * @brief USBD_MTP_Itf_GetNewIndex + * Create new object handle + * @param objformat: object format + * @retval object handle + */ +static uint32_t USBD_MTP_Itf_GetNewIndex(uint16_t objformat) +{ + uint32_t n_index = 0U; + UNUSED(objformat); + + return n_index; +} + +/** + * @brief USBD_MTP_Itf_WriteData + * Write file data to SD card + * @param len: size of data to write + * @param buff: data to write in SD card + * @retval None + */ +static void USBD_MTP_Itf_WriteData(uint16_t len, uint8_t *buff) +{ + UNUSED(len); + UNUSED(buff); + + return; +} + +/** + * @brief USBD_MTP_Itf_GetContainerLength + * Get length of generic container + * @param Param1: object handle + * @retval length of generic container + */ +static uint32_t USBD_MTP_Itf_GetContainerLength(uint32_t Param1) +{ + uint32_t length = 0U; + UNUSED(Param1); + + return length; +} + +/** + * @brief USBD_MTP_Itf_DeleteObject + * delete object from SD card + * @param Param1: object handle (file/folder index) + * @retval response code + */ +static uint16_t USBD_MTP_Itf_DeleteObject(uint32_t Param1) +{ + uint16_t rep_code = 0U; + UNUSED(Param1); + + return rep_code; +} + +/** + * @brief USBD_MTP_Get_idx_to_delete + * Get all files/foldres index to delete with descending order ( max depth) + * @param Param: object handle (file/folder index) + * @param tab: pointer to list of files/folders to delete + * @retval Number of files/folders to delete + */ +/* static uint32_t USBD_MTP_Get_idx_to_delete(uint32_t Param, uint8_t *tab) +{ + uint32_t cnt = 0U; + + return cnt; +} +*/ + +/** + * @brief USBD_MTP_Itf_ReadData + * Read data from SD card + * @param Param1: object handle + * @param buff: pointer to data to be read + * @param temp_length: current data size read + * @retval necessary information for next read/finish reading + */ +static uint32_t USBD_MTP_Itf_ReadData(uint32_t Param1, uint8_t *buff, MTP_DataLengthTypeDef *data_length) +{ + UNUSED(Param1); + UNUSED(buff); + UNUSED(data_length); + + return 0U; +} + +/** + * @brief USBD_MTP_Itf_Cancel + * Close opened folder/file while cancelling transaction + * @param MTP_ResponsePhase: MTP current state + * @retval None + */ +static void USBD_MTP_Itf_Cancel(uint32_t Phase) +{ + UNUSED(Phase); + + /* Make sure to close open file while canceling transaction */ + + return; +} diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp_opt.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp_opt.c new file mode 100644 index 00000000..9a49717b --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp_opt.c @@ -0,0 +1,1267 @@ +/** + ****************************************************************************** + * @file usbd_mtp_opt.c + * @author MCD Application Team + * @brief This file includes the PTP operations layer + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_mtp_opt.h" + +/* Private typedef -----------------------------------------------------------*/ +/* Private define ------------------------------------------------------------*/ +/* Private macro -------------------------------------------------------------*/ +/* Private variables ---------------------------------------------------------*/ +static uint8_t ObjInfo_buff[255]; +static uint32_t objhandle; +static uint16_t obj_format; +static uint32_t storage_id; + +static MTP_DeviceInfoTypedef MTP_DeviceInfo; +static MTP_StorageIDSTypeDef MTP_StorageIDS; +static MTP_StorageInfoTypedef MTP_StorageInfo; +static MTP_ObjectHandleTypeDef MTP_ObjectHandle; +static MTP_ObjectInfoTypeDef MTP_ObjectInfo; +static MTP_ObjectPropSuppTypeDef MTP_ObjectPropSupp; +static MTP_ObjectPropDescTypeDef MTP_ObjectPropDesc; +static MTP_PropertiesListTypedef MTP_PropertiesList; +static MTP_RefTypeDef MTP_Ref; +static MTP_PropertyValueTypedef MTP_PropertyValue; +static MTP_FileNameTypeDef MTP_FileName; +static MTP_DevicePropDescTypeDef MTP_DevicePropDesc; + +/* Private function prototypes -----------------------------------------------*/ +static void MTP_Get_DeviceInfo(void); +static void MTP_Get_StorageIDS(void); +static void MTP_Get_PayloadContent(USBD_HandleTypeDef *pdev); +static void MTP_Get_ObjectInfo(USBD_HandleTypeDef *pdev); +static void MTP_Get_StorageInfo(USBD_HandleTypeDef *pdev); +static void MTP_Get_ObjectHandle(USBD_HandleTypeDef *pdev); +static void MTP_Get_ObjectPropSupp(void); +static void MTP_Get_ObjectPropDesc(USBD_HandleTypeDef *pdev); +static void MTP_Get_ObjectPropList(USBD_HandleTypeDef *pdev); +static void MTP_Get_DevicePropDesc(void); +static uint8_t *MTP_Get_ObjectPropValue(USBD_HandleTypeDef *pdev); +static uint32_t MTP_build_data_propdesc(USBD_HandleTypeDef *pdev, MTP_ObjectPropDescTypeDef def); +static uint32_t MTP_build_data_ObjInfo(USBD_HandleTypeDef *pdev, MTP_ObjectInfoTypeDef objinfo); +static uint32_t MTP_build_data_proplist(USBD_HandleTypeDef *pdev, + MTP_PropertiesListTypedef proplist, uint32_t idx); + +/* Private functions ---------------------------------------------------------*/ + +/** + * @} + */ + + +/** + * @brief USBD_MTP_OPT_CreateObjectHandle + * Open a new session + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_CreateObjectHandle(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + + if (hmtp->OperationsContainer.Param1 == 0U) /* Param1 == Session ID*/ + { + hmtp->ResponseCode = MTP_RESPONSE_INVALID_PARAMETER; + } + /* driver supports single session */ + else if (hmtp->MTP_SessionState == MTP_SESSION_OPENED) + { + hmtp->ResponseCode = MTP_RESPONSE_SESSION_ALREADY_OPEN; + } + else + { + hmtp->ResponseCode = MTP_RESPONSE_OK; + hmtp->MTP_SessionState = MTP_SESSION_OPENED; + } + + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_RESPONSE; + hmtp->ResponseLength = MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; +} + +/** + * @brief USBD_MTP_OPT_GetDeviceInfo + * Get all device information + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_GetDeviceInfo(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + + if (hmtp->MTP_SessionState == MTP_SESSION_NOT_OPENED) /* no session opened */ + { + /* if GetDevice Info called outside a session then SessionID and Transaction_ID shall be 0x00000000*/ + /* Param1 == session ID*/ + if ((hmtp->OperationsContainer.Param1 == 0U) && (hmtp->OperationsContainer.trans_id == 0U)) + { + hmtp->ResponseCode = MTP_RESPONSE_OK; + } + else + { + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_RESPONSE; + hmtp->ResponseCode = MTP_RESPONSE_INVALID_PARAMETER; + hmtp->GenericContainer.length = MTP_CONT_HEADER_SIZE; + } + } + else + { + hmtp->ResponseCode = MTP_RESPONSE_OK; + } + + if (hmtp->ResponseCode == MTP_RESPONSE_OK) + { + hmtp->GenericContainer.code = MTP_OP_GET_DEVICE_INFO; + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_DATA; + + (void)MTP_Get_PayloadContent(pdev); + + hmtp->ResponseLength = sizeof(MTP_DeviceInfo) + MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; + } +} + +/** + * @brief USBD_MTP_OPT_GetStorageIDS + * Get Storage IDs + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_GetStorageIDS(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + hmtp->GenericContainer.code = MTP_OP_GET_STORAGE_IDS; + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_DATA; + + (void)MTP_Get_PayloadContent(pdev); + + hmtp->ResponseLength = sizeof(MTP_StorageIDS) + MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; + + hmtp->ResponseCode = MTP_RESPONSE_OK; +} + +/** + * @brief USBD_MTP_OPT_GetStorageInfo + * Get Storage information + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_GetStorageInfo(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + hmtp->GenericContainer.code = MTP_OP_GET_STORAGE_INFO; + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_DATA; + + (void)MTP_Get_PayloadContent(pdev); + + hmtp->ResponseLength = sizeof(MTP_StorageInfo) + MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; + + hmtp->ResponseCode = MTP_RESPONSE_OK; +} + +/** + * @brief USBD_MTP_OPT_GetObjectHandle + * Get all object handles + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_GetObjectHandle(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + hmtp->GenericContainer.code = MTP_OP_GET_OBJECT_HANDLES; + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_DATA; + + (void)MTP_Get_PayloadContent(pdev); + + hmtp->ResponseLength = hmtp->ResponseLength + MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; + + hmtp->ResponseCode = MTP_RESPONSE_OK; +} + +/** + * @brief USBD_MTP_OPT_GetObjectInfo + * Get all information about the object + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_GetObjectInfo(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + + hmtp->GenericContainer.code = MTP_OP_GET_OBJECT_INFO; + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_DATA; + + (void)MTP_Get_PayloadContent(pdev); + + hmtp->ResponseLength = hmtp->ResponseLength + MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; + + hmtp->ResponseCode = MTP_RESPONSE_OK; +} + +/** + * @brief USBD_MTP_OPT_GetObjectReferences + * Get object references + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_GetObjectReferences(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + + hmtp->GenericContainer.code = MTP_OP_GET_OBJECT_PROP_REFERENCES; + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_DATA; + + (void)MTP_Get_PayloadContent(pdev); + + hmtp->ResponseLength = sizeof(MTP_Ref) + MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; + + hmtp->ResponseCode = MTP_RESPONSE_OK; +} + +/** + * @brief USBD_MTP_OPT_GetObjectPropSupp + * Get all object properties supported + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_GetObjectPropSupp(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + + hmtp->GenericContainer.code = MTP_OP_GET_OBJECT_PROPS_SUPPORTED; + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_DATA; + + (void)MTP_Get_PayloadContent(pdev); + + hmtp->ResponseLength = sizeof(MTP_ObjectPropSupp) + MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; + + hmtp->ResponseCode = MTP_RESPONSE_OK; +} + +/** + * @brief USBD_MTP_OPT_GetObjectPropDesc + * Get all descriptions about object properties + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_GetObjectPropDesc(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + + hmtp->GenericContainer.code = MTP_OP_GET_OBJECT_PROP_DESC; + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_DATA; + + (void)MTP_Get_PayloadContent(pdev); + + hmtp->ResponseLength = hmtp->ResponseLength + MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; + + hmtp->ResponseCode = MTP_RESPONSE_OK; +} + +/** + * @brief USBD_MTP_OPT_GetObjectPropList + * Get the list of object properties + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_GetObjectPropList(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + + hmtp->GenericContainer.code = MTP_OP_GET_OBJECT_PROPLIST; + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_DATA; + + (void)MTP_Get_PayloadContent(pdev); + + hmtp->ResponseLength = hmtp->ResponseLength + MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; + + hmtp->ResponseCode = MTP_RESPONSE_OK; +} + +/** + * @brief USBD_MTP_OPT_GetObjectPropValue + * Get current value of the object property + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_GetObjectPropValue(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + + hmtp->GenericContainer.code = MTP_OP_GET_OBJECT_PROP_VALUE; + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_DATA; + + (void)MTP_Get_PayloadContent(pdev); + + hmtp->ResponseLength = hmtp->ResponseLength + MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; + + hmtp->ResponseCode = MTP_RESPONSE_OK; +} + +/** + * @brief USBD_MTP_OPT_GetObject + * Get binary data from an object + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_GetObject(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_MTP_ItfTypeDef *hmtpif = (USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId]; + + hmtp->GenericContainer.length = hmtpif->GetContainerLength(hmtp->OperationsContainer.Param1); + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_DATA; + + hmtp->ResponseCode = MTP_RESPONSE_OK; +} + +/** + * @brief USBD_MTP_OPT_GetDevicePropDesc + * Get The DevicePropDesc dataset + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_GetDevicePropDesc(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + + hmtp->GenericContainer.code = MTP_OP_GET_DEVICE_PROP_DESC; + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_DATA; + + (void)MTP_Get_PayloadContent(pdev); + + hmtp->ResponseLength = sizeof(MTP_DevicePropDesc) + MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; + + hmtp->ResponseCode = MTP_RESPONSE_OK; +} + +/** + * @brief USBD_MTP_OPT_SendObject + * Send object from host to MTP device + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_SendObject(USBD_HandleTypeDef *pdev, uint8_t *buff, uint32_t len) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_MTP_ItfTypeDef *hmtpif = (USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId]; + static uint32_t tmp = 0U; + + switch (hmtp->RECEIVE_DATA_STATUS) + { + case RECEIVE_IDLE_STATE: + hmtp->RECEIVE_DATA_STATUS = RECEIVE_COMMAND_DATA; + break; + case RECEIVE_COMMAND_DATA: + hmtp->RECEIVE_DATA_STATUS = RECEIVE_FIRST_DATA; + break; + case RECEIVE_FIRST_DATA: + if ((uint16_t)len < (hmtp->MaxPcktLen - MTP_CONT_HEADER_SIZE)) + { + hmtp->GenericContainer.code = MTP_RESPONSE_OK; + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_RESPONSE; + hmtp->ResponseLength = MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; + + hmtp->RECEIVE_DATA_STATUS = RECEIVE_IDLE_STATE; + } + else + { + hmtp->RECEIVE_DATA_STATUS = RECEIVE_REST_OF_DATA; + } + tmp = (uint32_t)buff; + hmtpif->WriteData(len, (uint8_t *)(tmp + 12U)); + break; + + case RECEIVE_REST_OF_DATA: + hmtpif->WriteData(len, buff); + break; + + case SEND_RESPONSE: + hmtpif->WriteData(0, buff); /* send 0 length to stop write process */ + hmtp->GenericContainer.code = MTP_RESPONSE_OK; + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_RESPONSE; + hmtp->ResponseLength = MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; + + hmtp->RECEIVE_DATA_STATUS = RECEIVE_IDLE_STATE; + break; + + default: + break; + } + + hmtp->ResponseCode = MTP_RESPONSE_OK; +} + +/** + * @brief USBD_MTP_OPT_SendObjectInfo + * Send the object information from host to MTP device + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_SendObjectInfo(USBD_HandleTypeDef *pdev, uint8_t *buff, uint32_t len) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_MTP_ItfTypeDef *hmtpif = (USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId]; + MTP_ObjectInfoTypeDef ObjectInfo; + uint8_t dataLength = offsetof(MTP_ObjectInfoTypeDef, Filename); + uint8_t *tmp; + + switch (hmtp->RECEIVE_DATA_STATUS) + { + case RECEIVE_IDLE_STATE: + hmtp->RECEIVE_DATA_STATUS = RECEIVE_COMMAND_DATA; + break; + + case RECEIVE_COMMAND_DATA: + /* store object handle and storage id for future use */ + if (hmtp->OperationsContainer.Param2 == 0xFFFFFFFFU) + { + objhandle = 0U; + } + else + { + objhandle = hmtp->OperationsContainer.Param2; + } + storage_id = hmtp->OperationsContainer.Param1; + hmtp->RECEIVE_DATA_STATUS = RECEIVE_FIRST_DATA; + break; + + case RECEIVE_FIRST_DATA: + tmp = buff; + + (void)USBD_memcpy(ObjInfo_buff, tmp + 12U, + (uint16_t)(hmtp->MaxPcktLen - MTP_CONT_HEADER_SIZE)); + + hmtp->RECEIVE_DATA_STATUS = RECEIVE_REST_OF_DATA; + break; + + case RECEIVE_REST_OF_DATA: + + (void)USBD_memcpy(ObjInfo_buff + len, buff, hmtp->MaxPcktLen); + + break; + + case SEND_RESPONSE: + (void)USBD_memcpy((uint8_t *)&ObjectInfo, ObjInfo_buff, dataLength); + (void)USBD_memcpy((uint8_t *)&ObjectInfo.Filename, (ObjInfo_buff + dataLength), + ((uint32_t)(ObjectInfo.Filename_len) * 2U)); + + obj_format = ObjectInfo.ObjectFormat; + + hmtp->ResponseCode = hmtpif->Create_NewObject(ObjectInfo, objhandle); + hmtp->GenericContainer.code = (uint16_t)hmtp->ResponseCode; + hmtp->ResponseLength = MTP_CONT_HEADER_SIZE + (sizeof(uint32_t) * 3U); /* Header + 3 Param */ + hmtp->GenericContainer.length = hmtp->ResponseLength; + hmtp->GenericContainer.type = MTP_CONT_TYPE_RESPONSE; + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + + (void)MTP_Get_PayloadContent(pdev); + + hmtp->RECEIVE_DATA_STATUS = RECEIVE_IDLE_STATE; + break; + + default: + break; + } +} + +/** + * @brief USBD_MTP_OPT_DeleteObject + * Delete the object from the device + * @param pdev: device instance + * @retval None + */ +void USBD_MTP_OPT_DeleteObject(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_MTP_ItfTypeDef *hmtpif = (USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId]; + + hmtp->GenericContainer.trans_id = hmtp->OperationsContainer.trans_id; + hmtp->GenericContainer.type = MTP_CONT_TYPE_RESPONSE; + hmtp->ResponseLength = MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; + hmtp->ResponseCode = hmtpif->DeleteObject(hmtp->OperationsContainer.Param1); +} + +/** + * @brief MTP_Get_PayloadContent + * Get the payload data of generic container + * @param pdev: device instance + * @retval None + */ +static void MTP_Get_PayloadContent(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_MTP_ItfTypeDef *hmtpif = (USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId]; + uint8_t *buffer = hmtp->GenericContainer.data; + uint32_t i; + uint32_t n_idx; + + switch (hmtp->OperationsContainer.code) + { + case MTP_OP_GET_DEVICE_INFO: + (void)MTP_Get_DeviceInfo(); + (void)USBD_memcpy(buffer, (const uint8_t *)&MTP_DeviceInfo, sizeof(MTP_DeviceInfo)); + + for (i = 0U; i < sizeof(MTP_StorageIDS); i++) + { + hmtp->GenericContainer.data[i] = buffer[i]; + } + break; + + case MTP_OP_GET_STORAGE_IDS: + (void)MTP_Get_StorageIDS(); + (void)USBD_memcpy(buffer, (const uint8_t *)&MTP_StorageIDS, sizeof(MTP_StorageIDS)); + + for (i = 0U; i < sizeof(MTP_StorageIDS); i++) + { + hmtp->GenericContainer.data[i] = buffer[i]; + } + break; + + case MTP_OP_GET_STORAGE_INFO: + (void)MTP_Get_StorageInfo(pdev); + (void)USBD_memcpy(buffer, (const uint8_t *)&MTP_StorageInfo, sizeof(MTP_StorageInfo)); + + for (i = 0U; i < sizeof(MTP_StorageInfo); i++) + { + hmtp->GenericContainer.data[i] = buffer[i]; + } + break; + + case MTP_OP_GET_OBJECT_HANDLES: + (void)MTP_Get_ObjectHandle(pdev); + (void)USBD_memcpy(buffer, (const uint8_t *)&MTP_ObjectHandle, hmtp->ResponseLength); + + for (i = 0U; i < hmtp->ResponseLength; i++) + { + hmtp->GenericContainer.data[i] = buffer[i]; + } + break; + + case MTP_OP_GET_OBJECT_INFO: + (void)MTP_Get_ObjectInfo(pdev); + break; + + case MTP_OP_GET_OBJECT_PROPS_SUPPORTED: + (void)MTP_Get_ObjectPropSupp(); + (void)USBD_memcpy(buffer, (const uint8_t *)&MTP_ObjectPropSupp, sizeof(MTP_ObjectPropSupp)); + + for (i = 0U; i < sizeof(MTP_ObjectPropSupp); i++) + { + hmtp->GenericContainer.data[i] = buffer[i]; + } + break; + + case MTP_OP_GET_OBJECT_PROP_DESC: + (void)MTP_Get_ObjectPropDesc(pdev); + hmtp->ResponseLength = MTP_build_data_propdesc(pdev, MTP_ObjectPropDesc); + break; + + case MTP_OP_GET_OBJECT_PROP_REFERENCES: + MTP_Ref.ref_len = 0U; + (void)USBD_memcpy(buffer, (const uint8_t *)&MTP_Ref.ref_len, sizeof(MTP_Ref.ref_len)); + + for (i = 0U; i < sizeof(MTP_Ref.ref_len); i++) + { + hmtp->GenericContainer.data[i] = buffer[i]; + } + break; + + case MTP_OP_GET_OBJECT_PROPLIST: + (void)MTP_Get_ObjectPropList(pdev); + break; + + case MTP_OP_GET_OBJECT_PROP_VALUE: + buffer = MTP_Get_ObjectPropValue(pdev); + for (i = 0U; i < hmtp->ResponseLength; i++) + { + hmtp->GenericContainer.data[i] = buffer[i]; + } + + break; + + case MTP_OP_GET_DEVICE_PROP_DESC: + (void)MTP_Get_DevicePropDesc(); + (void)USBD_memcpy(buffer, (const uint8_t *)&MTP_DevicePropDesc, sizeof(MTP_DevicePropDesc)); + for (i = 0U; i < sizeof(MTP_DevicePropDesc); i++) + { + hmtp->GenericContainer.data[i] = buffer[i]; + } + break; + + case MTP_OP_SEND_OBJECT_INFO: + n_idx = hmtpif->GetNewIndex(obj_format); + (void)USBD_memcpy(hmtp->GenericContainer.data, (const uint8_t *)&storage_id, sizeof(uint32_t)); + (void)USBD_memcpy(hmtp->GenericContainer.data + 4U, (const uint8_t *)&objhandle, sizeof(uint32_t)); + (void)USBD_memcpy(hmtp->GenericContainer.data + 8U, (const uint8_t *)&n_idx, sizeof(uint32_t)); + break; + + case MTP_OP_GET_OBJECT: + break; + + default: + break; + } +} + +/** + * @brief MTP_Get_DeviceInfo + * Fill the MTP_DeviceInfo struct + * @param pdev: device instance + * @retval None + */ +static void MTP_Get_DeviceInfo(void) +{ + MTP_DeviceInfo.StandardVersion = STANDARD_VERSION; + MTP_DeviceInfo.VendorExtensionID = VEND_EXT_ID; + MTP_DeviceInfo.VendorExtensionVersion = VEND_EXT_VERSION; + MTP_DeviceInfo.VendorExtensionDesc_len = (uint8_t)VEND_EXT_DESC_LEN; + uint32_t i; + +#if USBD_MTP_VEND_EXT_DESC_SUPPORTED == 1 + for (i = 0U; i < VEND_EXT_DESC_LEN; i++) + { + MTP_DeviceInfo.VendorExtensionDesc[i] = VendExtDesc[i]; + } +#endif /* USBD_MTP_VEND_EXT_DESC_SUPPORTED */ + + MTP_DeviceInfo.FunctionalMode = FUNCTIONAL_MODE; /* device supports one mode , standard mode */ + + /* All supported operation */ + MTP_DeviceInfo.OperationsSupported_len = SUPP_OP_LEN; + for (i = 0U; i < SUPP_OP_LEN; i++) + { + MTP_DeviceInfo.OperationsSupported[i] = SuppOP[i]; + } + + MTP_DeviceInfo.EventsSupported_len = SUPP_EVENTS_LEN; /* event that are currently generated by the device*/ + +#if USBD_MTP_EVENTS_SUPPORTED == 1 + for (i = 0U; i < SUPP_EVENTS_LEN; i++) + { + MTP_DeviceInfo.EventsSupported[i] = SuppEvents[i]; + } +#endif /* USBD_MTP_EVENTS_SUPPORTED */ + + MTP_DeviceInfo.DevicePropertiesSupported_len = SUPP_DEVICE_PROP_LEN; + +#if USBD_MTP_DEVICE_PROP_SUPPORTED == 1 + for (i = 0U; i < SUPP_DEVICE_PROP_LEN; i++) + { + MTP_DeviceInfo.DevicePropertiesSupported[i] = DevicePropSupp[i]; + } +#endif /* USBD_MTP_DEVICE_PROP_SUPPORTED */ + + MTP_DeviceInfo.CaptureFormats_len = SUPP_CAPT_FORMAT_LEN; + +#if USBD_MTP_CAPTURE_FORMAT_SUPPORTED == 1 + for (i = 0U; i < SUPP_CAPT_FORMAT_LEN; i++) + { + MTP_DeviceInfo.CaptureFormats[i] = SuppCaptFormat[i]; + } +#endif /* USBD_MTP_CAPTURE_FORMAT_SUPPORTED */ + + MTP_DeviceInfo.ImageFormats_len = SUPP_IMG_FORMAT_LEN; /* number of image formats that are supported by the device*/ + for (i = 0U; i < SUPP_IMG_FORMAT_LEN; i++) + { + MTP_DeviceInfo.ImageFormats[i] = SuppImgFormat[i]; + } + + MTP_DeviceInfo.Manufacturer_len = (uint8_t)MANUF_LEN; + for (i = 0U; i < MANUF_LEN; i++) + { + MTP_DeviceInfo.Manufacturer[i] = Manuf[i]; + } + + MTP_DeviceInfo.Model_len = (uint8_t)MODEL_LEN; + for (i = 0U; i < MODEL_LEN; i++) + { + MTP_DeviceInfo.Model[i] = Model[i]; + } + + MTP_DeviceInfo.DeviceVersion_len = (uint8_t)DEVICE_VERSION_LEN; + for (i = 0U; i < DEVICE_VERSION_LEN; i++) + { + MTP_DeviceInfo.DeviceVersion[i] = DeviceVers[i]; + } + + MTP_DeviceInfo.SerialNumber_len = (uint8_t)SERIAL_NBR_LEN; + for (i = 0U; i < SERIAL_NBR_LEN; i++) + { + MTP_DeviceInfo.SerialNumber[i] = SerialNbr[i]; + } +} + +/** + * @brief MTP_Get_StorageInfo + * Fill the MTP_StorageInfo struct + * @param pdev: device instance + * @retval None + */ +static void MTP_Get_StorageInfo(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_ItfTypeDef *hmtpif = (USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId]; + + MTP_StorageInfo.StorageType = MTP_STORAGE_REMOVABLE_RAM; + MTP_StorageInfo.FilesystemType = MTP_FILESYSTEM_GENERIC_FLAT; + MTP_StorageInfo.AccessCapability = MTP_ACCESS_CAP_RW; + MTP_StorageInfo.MaxCapability = hmtpif->GetMaxCapability(); + MTP_StorageInfo.FreeSpaceInBytes = hmtpif->GetFreeSpaceInBytes(); + MTP_StorageInfo.FreeSpaceInObjects = FREE_SPACE_IN_OBJ_NOT_USED; /* not used */ + MTP_StorageInfo.StorageDescription = 0U; + MTP_StorageInfo.VolumeLabel = 0U; +} + +/** + * @brief MTP_Get_ObjectHandle + * Fill the MTP_ObjectHandle struct + * @param pdev: device instance + * @retval None + */ +static void MTP_Get_ObjectHandle(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_MTP_ItfTypeDef *hmtpif = (USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId]; + + MTP_ObjectHandle.ObjectHandle_len = (uint32_t)(hmtpif->GetIdx(hmtp->OperationsContainer.Param3, + MTP_ObjectHandle.ObjectHandle)); + + hmtp->ResponseLength = (MTP_ObjectHandle.ObjectHandle_len * sizeof(uint32_t)) + sizeof(uint32_t); +} + +/** + * @brief MTP_Get_ObjectPropSupp + * Fill the MTP_ObjectPropSupp struct + * @param pdev: device instance + * @retval None + */ +static void MTP_Get_ObjectPropSupp(void) +{ + uint32_t i; + + MTP_ObjectPropSupp.ObjectPropSupp_len = SUPP_OBJ_PROP_LEN; + + for (i = 0U; i < SUPP_OBJ_PROP_LEN; i++) + { + MTP_ObjectPropSupp.ObjectPropSupp[i] = ObjectPropSupp[i]; + } +} + +/** + * @brief MTP_Get_ObjectPropDesc + * Fill the MTP_ObjectPropDesc struct + * @param pdev: device instance + * @retval None + */ +static void MTP_Get_ObjectPropDesc(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint16_t undef_format = MTP_OBJ_FORMAT_UNDEFINED; + uint32_t storageid = MTP_STORAGE_ID; + + switch (hmtp->OperationsContainer.Param1) /* switch obj prop code */ + { + case MTP_OB_PROP_OBJECT_FORMAT : + MTP_ObjectPropDesc.ObjectPropertyCode = (uint16_t)(hmtp->OperationsContainer.Param1); + MTP_ObjectPropDesc.DataType = MTP_DATATYPE_UINT16; + MTP_ObjectPropDesc.GetSet = MTP_PROP_GET; + MTP_ObjectPropDesc.DefValue = (uint8_t *)&undef_format; + MTP_ObjectPropDesc.GroupCode = 0U; + MTP_ObjectPropDesc.FormFlag = 0U; + break; + + case MTP_OB_PROP_STORAGE_ID : + MTP_ObjectPropDesc.ObjectPropertyCode = (uint16_t)(hmtp->OperationsContainer.Param1); + MTP_ObjectPropDesc.DataType = MTP_DATATYPE_UINT32; + MTP_ObjectPropDesc.GetSet = MTP_PROP_GET; + MTP_ObjectPropDesc.DefValue = (uint8_t *)&storageid; + MTP_ObjectPropDesc.GroupCode = 0U; + MTP_ObjectPropDesc.FormFlag = 0U; + break; + + case MTP_OB_PROP_OBJ_FILE_NAME : + MTP_ObjectPropDesc.ObjectPropertyCode = (uint16_t)(hmtp->OperationsContainer.Param1); + MTP_ObjectPropDesc.DataType = MTP_DATATYPE_STR; + MTP_ObjectPropDesc.GetSet = MTP_PROP_GET; + MTP_FileName.FileName_len = DEFAULT_FILE_NAME_LEN; + (void)USBD_memcpy((void *) & (MTP_FileName.FileName), (const void *)DefaultFileName, sizeof(DefaultFileName)); + MTP_ObjectPropDesc.DefValue = (uint8_t *)&MTP_FileName; + MTP_ObjectPropDesc.GroupCode = 0U; + MTP_ObjectPropDesc.FormFlag = 0U; + break; + + case MTP_OB_PROP_PARENT_OBJECT : + MTP_ObjectPropDesc.ObjectPropertyCode = (uint16_t)(hmtp->OperationsContainer.Param1); + MTP_ObjectPropDesc.DataType = MTP_DATATYPE_STR; + MTP_ObjectPropDesc.GetSet = MTP_PROP_GET; + MTP_ObjectPropDesc.DefValue = 0U; + MTP_ObjectPropDesc.GroupCode = 0U; + MTP_ObjectPropDesc.FormFlag = 0U; + break; + + case MTP_OB_PROP_OBJECT_SIZE : + MTP_ObjectPropDesc.ObjectPropertyCode = (uint16_t)(hmtp->OperationsContainer.Param1); + MTP_ObjectPropDesc.DataType = MTP_DATATYPE_UINT64; + MTP_ObjectPropDesc.GetSet = MTP_PROP_GET; + MTP_ObjectPropDesc.DefValue = 0U; + MTP_ObjectPropDesc.GroupCode = 0U; + MTP_ObjectPropDesc.FormFlag = 0U; + break; + + case MTP_OB_PROP_NAME : + MTP_ObjectPropDesc.ObjectPropertyCode = (uint16_t)(hmtp->OperationsContainer.Param1); + MTP_ObjectPropDesc.DataType = MTP_DATATYPE_STR; + MTP_ObjectPropDesc.GetSet = MTP_PROP_GET; + MTP_FileName.FileName_len = DEFAULT_FILE_NAME_LEN; + (void)USBD_memcpy((void *) & (MTP_FileName.FileName), + (const void *)DefaultFileName, sizeof(DefaultFileName)); + + MTP_ObjectPropDesc.DefValue = (uint8_t *)&MTP_FileName; + MTP_ObjectPropDesc.GroupCode = 0U; + MTP_ObjectPropDesc.FormFlag = 0U; + break; + + case MTP_OB_PROP_PERS_UNIQ_OBJ_IDEN : + MTP_ObjectPropDesc.ObjectPropertyCode = (uint16_t)(hmtp->OperationsContainer.Param1); + MTP_ObjectPropDesc.DataType = MTP_DATATYPE_UINT128; + MTP_ObjectPropDesc.GetSet = MTP_PROP_GET; + MTP_ObjectPropDesc.DefValue = 0U; + MTP_ObjectPropDesc.GroupCode = 0U; + MTP_ObjectPropDesc.FormFlag = 0U; + break; + + case MTP_OB_PROP_PROTECTION_STATUS : + MTP_ObjectPropDesc.ObjectPropertyCode = (uint16_t)(hmtp->OperationsContainer.Param1); + MTP_ObjectPropDesc.DataType = MTP_DATATYPE_UINT16; + MTP_ObjectPropDesc.GetSet = MTP_PROP_GET_SET; + MTP_ObjectPropDesc.DefValue = 0U; + MTP_ObjectPropDesc.GroupCode = 0U; + MTP_ObjectPropDesc.FormFlag = 0U; + break; + + default: + break; + } +} + +/** + * @brief MTP_Get_ObjectPropValue + * Get the property value + * @param pdev: device instance + * @retval None + */ +static uint8_t *MTP_Get_ObjectPropValue(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_MTP_ItfTypeDef *hmtpif = (USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId]; + static uint8_t buf[512]; + + /* Add all other supported object properties */ + switch (hmtp->OperationsContainer.Param2) + { + case MTP_OB_PROP_STORAGE_ID: + MTP_PropertyValue.u32 = MTP_STORAGE_ID; + (void)USBD_memcpy(buf, (const uint8_t *)&MTP_PropertyValue, sizeof(uint32_t)); + hmtp->ResponseLength = sizeof(uint32_t); + break; + + case MTP_OB_PROP_OBJECT_FORMAT: + MTP_PropertyValue.u16 = hmtpif->GetObjectFormat(hmtp->OperationsContainer.Param1); + (void)USBD_memcpy(buf, (const uint8_t *)&MTP_PropertyValue, sizeof(uint16_t)); + hmtp->ResponseLength = sizeof(uint16_t); + break; + + case MTP_OB_PROP_OBJ_FILE_NAME: + MTP_FileName.FileName_len = hmtpif->GetObjectName_len(hmtp->OperationsContainer.Param1); + hmtpif->GetObjectName(hmtp->OperationsContainer.Param1, MTP_FileName.FileName_len, (uint16_t *)buf); + (void)USBD_memcpy(MTP_FileName.FileName, (uint16_t *)buf, ((uint32_t)MTP_FileName.FileName_len * 2U) + 1U); + + hmtp->ResponseLength = ((uint32_t)MTP_FileName.FileName_len * 2U) + 1U; + break; + + case MTP_OB_PROP_PARENT_OBJECT : + MTP_PropertyValue.u32 = hmtpif->GetParentObject(hmtp->OperationsContainer.Param1); + (void)USBD_memcpy(buf, (const uint8_t *)&MTP_PropertyValue, sizeof(uint32_t)); + hmtp->ResponseLength = sizeof(uint32_t); + break; + + case MTP_OB_PROP_OBJECT_SIZE : + MTP_PropertyValue.u64 = hmtpif->GetObjectSize(hmtp->OperationsContainer.Param1); + (void)USBD_memcpy(buf, (const uint8_t *)&MTP_PropertyValue, sizeof(uint64_t)); + hmtp->ResponseLength = sizeof(uint64_t); + break; + + default: + break; + } + + return buf; +} + +/** + * @brief MTP_Get_ObjectPropList + * Get the object property list data to be transmitted + * @param pdev: device instance + * @retval None + */ +static void MTP_Get_ObjectPropList(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_MTP_ItfTypeDef *hmtpif = (USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId]; + uint16_t filename[255]; + uint32_t storageid = MTP_STORAGE_ID; + uint32_t default_val = 0U; + uint32_t i; + uint16_t format; + uint64_t objsize; + uint32_t parent_proval; + + MTP_PropertiesList.MTP_Properties_len = SUPP_OBJ_PROP_LEN; + hmtp->ResponseLength = 4U; /* size of MTP_PropertiesList.MTP_Properties_len */ + (void)USBD_memcpy(hmtp->GenericContainer.data, + (const uint8_t *)&MTP_PropertiesList.MTP_Properties_len, hmtp->ResponseLength); + + for (i = 0U; i < SUPP_OBJ_PROP_LEN; i++) + { + MTP_PropertiesList.MTP_Properties[i].ObjectHandle = hmtp->OperationsContainer.Param1; + + switch (ObjectPropSupp[i]) + { + case MTP_OB_PROP_STORAGE_ID : + MTP_PropertiesList.MTP_Properties[i].PropertyCode = MTP_OB_PROP_STORAGE_ID; + MTP_PropertiesList.MTP_Properties[i].Datatype = MTP_DATATYPE_UINT32; + MTP_PropertiesList.MTP_Properties[i].propval = (uint8_t *)&storageid; + break; + + case MTP_OB_PROP_OBJECT_FORMAT : + MTP_PropertiesList.MTP_Properties[i].PropertyCode = MTP_OB_PROP_OBJECT_FORMAT; + MTP_PropertiesList.MTP_Properties[i].Datatype = MTP_DATATYPE_UINT16; + format = hmtpif->GetObjectFormat(hmtp->OperationsContainer.Param1); + MTP_PropertiesList.MTP_Properties[i].propval = (uint8_t *)&format; + break; + + case MTP_OB_PROP_OBJ_FILE_NAME: + MTP_PropertiesList.MTP_Properties[i].PropertyCode = MTP_OB_PROP_OBJ_FILE_NAME; + MTP_PropertiesList.MTP_Properties[i].Datatype = MTP_DATATYPE_STR; + /* MTP_FileName.FileName_len value shall be set before USBD_MTP_FS_GetObjectName */ + MTP_FileName.FileName_len = hmtpif->GetObjectName_len(hmtp->OperationsContainer.Param1); + hmtpif->GetObjectName(hmtp->OperationsContainer.Param1, MTP_FileName.FileName_len, filename); + (void)USBD_memcpy(MTP_FileName.FileName, filename, ((uint32_t)MTP_FileName.FileName_len * 2U) + 1U); + MTP_PropertiesList.MTP_Properties[i].propval = (uint8_t *)&MTP_FileName; + break; + + case MTP_OB_PROP_PARENT_OBJECT : + MTP_PropertiesList.MTP_Properties[i].PropertyCode = MTP_OB_PROP_PARENT_OBJECT; + MTP_PropertiesList.MTP_Properties[i].Datatype = MTP_DATATYPE_UINT32; + parent_proval = hmtpif->GetParentObject(hmtp->OperationsContainer.Param1); + MTP_PropertiesList.MTP_Properties[i].propval = (uint8_t *)&parent_proval; + break; + + case MTP_OB_PROP_OBJECT_SIZE : + MTP_PropertiesList.MTP_Properties[i].PropertyCode = MTP_OB_PROP_OBJECT_SIZE; + MTP_PropertiesList.MTP_Properties[i].Datatype = MTP_DATATYPE_UINT64; + objsize = hmtpif->GetObjectSize(hmtp->OperationsContainer.Param1); + MTP_PropertiesList.MTP_Properties[i].propval = (uint8_t *)&objsize; + break; + + case MTP_OB_PROP_NAME : + MTP_PropertiesList.MTP_Properties[i].PropertyCode = MTP_OB_PROP_NAME; + MTP_PropertiesList.MTP_Properties[i].Datatype = MTP_DATATYPE_STR; + /* MTP_FileName.FileName_len value shall be set before USBD_MTP_FS_GetObjectName */ + MTP_FileName.FileName_len = hmtpif->GetObjectName_len(hmtp->OperationsContainer.Param1); + hmtpif->GetObjectName(hmtp->OperationsContainer.Param1, MTP_FileName.FileName_len, filename); + (void)USBD_memcpy(MTP_FileName.FileName, filename, ((uint32_t)MTP_FileName.FileName_len * 2U) + 1U); + MTP_PropertiesList.MTP_Properties[i].propval = (uint8_t *)&MTP_FileName; + break; + + case MTP_OB_PROP_PERS_UNIQ_OBJ_IDEN : + MTP_PropertiesList.MTP_Properties[i].PropertyCode = MTP_OB_PROP_PERS_UNIQ_OBJ_IDEN; + MTP_PropertiesList.MTP_Properties[i].Datatype = MTP_DATATYPE_UINT128; + MTP_PropertiesList.MTP_Properties[i].propval = (uint8_t *)&hmtp->OperationsContainer.Param1; + break; + + case MTP_OB_PROP_PROTECTION_STATUS : + MTP_PropertiesList.MTP_Properties[i].PropertyCode = MTP_OB_PROP_PROTECTION_STATUS; + MTP_PropertiesList.MTP_Properties[i].Datatype = MTP_DATATYPE_UINT16; + MTP_PropertiesList.MTP_Properties[i].propval = (uint8_t *)&default_val; + break; + + default: + break; + } + + hmtp->ResponseLength = MTP_build_data_proplist(pdev, MTP_PropertiesList, i); + } +} + +/** + * @brief MTP_Get_DevicePropDesc + * Fill the MTP_DevicePropDesc struct + * @param pdev: device instance + * @retval None + */ +static void MTP_Get_DevicePropDesc(void) +{ + MTP_DevicePropDesc.DevicePropertyCode = MTP_DEV_PROP_DEVICE_FRIENDLY_NAME; + MTP_DevicePropDesc.DataType = MTP_DATATYPE_STR; + MTP_DevicePropDesc.GetSet = MTP_PROP_GET_SET; + MTP_DevicePropDesc.DefValue_len = DEVICE_PROP_DESC_DEF_LEN; + uint32_t i; + + for (i = 0U; i < (sizeof(DevicePropDefVal) / 2U); i++) + { + MTP_DevicePropDesc.DefValue[i] = DevicePropDefVal[i]; + } + + MTP_DevicePropDesc.curDefValue_len = DEVICE_PROP_DESC_CUR_LEN; + + for (i = 0U; i < (sizeof(DevicePropCurDefVal) / 2U); i++) + { + MTP_DevicePropDesc.curDefValue[i] = DevicePropCurDefVal[i]; + } + + MTP_DevicePropDesc.FormFlag = 0U; +} + +/** + * @brief MTP_Get_ObjectInfo + * Fill the MTP_ObjectInfo struct + * @param pdev: device instance + * @retval None + */ +static void MTP_Get_ObjectInfo(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_ItfTypeDef *hmtpif = (USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId]; + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint16_t filename[255]; + + MTP_ObjectInfo.Storage_id = MTP_STORAGE_ID; + MTP_ObjectInfo.ObjectFormat = hmtpif->GetObjectFormat(hmtp->OperationsContainer.Param1); + MTP_ObjectInfo.ObjectCompressedSize = hmtpif->GetObjectSize(hmtp->OperationsContainer.Param1); + MTP_ObjectInfo.ProtectionStatus = 0U; + MTP_ObjectInfo.ThumbFormat = MTP_OBJ_FORMAT_UNDEFINED; + MTP_ObjectInfo.ThumbCompressedSize = 0U; + MTP_ObjectInfo.ThumbPixWidth = 0U; /* not supported or not an image */ + MTP_ObjectInfo.ThumbPixHeight = 0U; + MTP_ObjectInfo.ImagePixWidth = 0U; + MTP_ObjectInfo.ImagePixHeight = 0U; + MTP_ObjectInfo.ImageBitDepth = 0U; + MTP_ObjectInfo.ParentObject = hmtpif->GetParentObject(hmtp->OperationsContainer.Param1); + MTP_ObjectInfo.AssociationType = 0U; + MTP_ObjectInfo.AssociationDesc = 0U; + MTP_ObjectInfo.SequenceNumber = 0U; + + /* we have to get this value before MTP_ObjectInfo.Filename */ + MTP_ObjectInfo.Filename_len = hmtpif->GetObjectName_len(hmtp->OperationsContainer.Param1); + hmtpif->GetObjectName(hmtp->OperationsContainer.Param1, MTP_ObjectInfo.Filename_len, filename); + (void)USBD_memcpy(MTP_ObjectInfo.Filename, filename, ((uint32_t)MTP_FileName.FileName_len * 2U) + 1U); + + MTP_ObjectInfo.CaptureDate = 0U; + MTP_ObjectInfo.ModificationDate = 0U; + MTP_ObjectInfo.Keywords = 0U; + hmtp->ResponseLength = MTP_build_data_ObjInfo(pdev, MTP_ObjectInfo); +} + +/** + * @brief MTP_Get_StorageIDS + * Fill the MTP_StorageIDS struct + * @param pdev: device instance + * @retval None + */ +static void MTP_Get_StorageIDS(void) +{ + MTP_StorageIDS.StorageIDS_len = MTP_NBR_STORAGE_ID; + MTP_StorageIDS.StorageIDS[0] = MTP_STORAGE_ID; +} + +/** + * @brief MTP_build_data_propdesc + * Copy the MTP_ObjectPropDesc dataset to the payload data + * @param pdev: device instance + * @retval None + */ +static uint32_t MTP_build_data_propdesc(USBD_HandleTypeDef *pdev, MTP_ObjectPropDescTypeDef def) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint8_t DefValue_size = (MTP_FileName.FileName_len * 2U) + 1U; + uint32_t dataLength = offsetof(MTP_ObjectPropDescTypeDef, DefValue); + + (void)USBD_memcpy(hmtp->GenericContainer.data, (const uint8_t *)&def, dataLength); + + switch (def.DataType) + { + case MTP_DATATYPE_UINT16: + (void)USBD_memcpy(hmtp->GenericContainer.data + dataLength, def.DefValue, sizeof(uint16_t)); + dataLength += sizeof(uint16_t); + break; + + case MTP_DATATYPE_UINT32: + (void)USBD_memcpy(hmtp->GenericContainer.data + dataLength, def.DefValue, sizeof(uint32_t)); + dataLength += sizeof(uint32_t); + break; + + case MTP_DATATYPE_UINT64: + (void)USBD_memcpy(hmtp->GenericContainer.data + dataLength, def.DefValue, sizeof(uint64_t)); + dataLength += sizeof(uint64_t); + break; + + case MTP_DATATYPE_STR: + (void)USBD_memcpy(hmtp->GenericContainer.data + dataLength, def.DefValue, DefValue_size); + dataLength += DefValue_size; + break; + + case MTP_DATATYPE_UINT128: + (void)USBD_memcpy(hmtp->GenericContainer.data + dataLength, def.DefValue, (sizeof(uint64_t) * 2U)); + dataLength += (sizeof(uint64_t) * 2U); + break; + + default: + break; + } + + (void)USBD_memcpy(hmtp->GenericContainer.data + dataLength, + (const uint8_t *)&MTP_ObjectPropDesc.GroupCode, sizeof(MTP_ObjectPropDesc.GroupCode)); + + dataLength += sizeof(MTP_ObjectPropDesc.GroupCode); + + (void)USBD_memcpy(hmtp->GenericContainer.data + dataLength, + (const uint8_t *)&MTP_ObjectPropDesc.FormFlag, sizeof(MTP_ObjectPropDesc.FormFlag)); + + dataLength += sizeof(MTP_ObjectPropDesc.FormFlag); + + return dataLength; +} + +/** + * @brief MTP_build_data_proplist + * Copy the MTP_PropertiesList dataset to the payload data + * @param pdev: device instance + * @retval None + */ +static uint32_t MTP_build_data_proplist(USBD_HandleTypeDef *pdev, + MTP_PropertiesListTypedef proplist, uint32_t idx) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint8_t propval_size = (MTP_FileName.FileName_len * 2U) + 1U; + uint32_t dataLength; + + dataLength = offsetof(MTP_PropertiesTypedef, propval); + + (void)USBD_memcpy(hmtp->GenericContainer.data + hmtp->ResponseLength, + (const uint8_t *)&proplist.MTP_Properties[idx], dataLength); + + dataLength += hmtp->ResponseLength; + + switch (proplist.MTP_Properties[idx].Datatype) + { + case MTP_DATATYPE_UINT16: + (void)USBD_memcpy(hmtp->GenericContainer.data + dataLength, + proplist.MTP_Properties[idx].propval, sizeof(uint16_t)); + + dataLength += sizeof(uint16_t); + break; + + case MTP_DATATYPE_UINT32: + (void)USBD_memcpy(hmtp->GenericContainer.data + dataLength, + proplist.MTP_Properties[idx].propval, sizeof(uint32_t)); + + dataLength += sizeof(uint32_t); + break; + + case MTP_DATATYPE_STR: + (void)USBD_memcpy(hmtp->GenericContainer.data + dataLength, + proplist.MTP_Properties[idx].propval, propval_size); + + dataLength += propval_size; + break; + + case MTP_DATATYPE_UINT64: + (void)USBD_memcpy(hmtp->GenericContainer.data + dataLength, + proplist.MTP_Properties[idx].propval, sizeof(uint64_t)); + + dataLength += sizeof(uint64_t); + break; + + case MTP_DATATYPE_UINT128: + (void)USBD_memcpy(hmtp->GenericContainer.data + dataLength, + proplist.MTP_Properties[idx].propval, (sizeof(uint64_t) * 2U)); + + dataLength += (sizeof(uint64_t) * 2U); + break; + + default: + break; + } + + return dataLength; +} + +/** + * @brief MTP_build_data_ObjInfo + * Copy the MTP_ObjectInfo dataset to the payload data + * @param pdev: device instance + * @retval None + */ +static uint32_t MTP_build_data_ObjInfo(USBD_HandleTypeDef *pdev, MTP_ObjectInfoTypeDef objinfo) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint32_t ObjInfo_len = offsetof(MTP_ObjectInfoTypeDef, Filename); + + (void)USBD_memcpy(hmtp->GenericContainer.data, (const uint8_t *)&objinfo, ObjInfo_len); + (void)USBD_memcpy(hmtp->GenericContainer.data + ObjInfo_len, + (const uint8_t *)&objinfo.Filename, objinfo.Filename_len * sizeof(uint16_t)); + + ObjInfo_len = ObjInfo_len + (objinfo.Filename_len * sizeof(uint16_t)); + + (void)USBD_memcpy(hmtp->GenericContainer.data + ObjInfo_len, + (const uint8_t *)&objinfo.CaptureDate, sizeof(objinfo.CaptureDate)); + + ObjInfo_len = ObjInfo_len + sizeof(objinfo.CaptureDate); + + return ObjInfo_len; +} diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp_storage.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp_storage.c new file mode 100644 index 00000000..e09cd392 --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/MTP/Src/usbd_mtp_storage.c @@ -0,0 +1,472 @@ +/** + ****************************************************************************** + * @file usbd_mtp_storage.c + * @author MCD Application Team + * @brief This file provides all the transfer command functions for MTP + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** + */ + +/* Includes ------------------------------------------------------------------*/ +#include "usbd_mtp_storage.h" + +/* Private typedef -----------------------------------------------------------*/ +/* Private define ------------------------------------------------------------*/ +/* Private macro -------------------------------------------------------------*/ +extern uint8_t MTPInEpAdd; +extern uint8_t MTPOutEpAdd; + +/* Private variables ---------------------------------------------------------*/ +static MTP_DataLengthTypeDef MTP_DataLength; +static MTP_READ_DATA_STATUS ReadDataStatus; + +/* Private function prototypes -----------------------------------------------*/ +static uint8_t USBD_MTP_STORAGE_DecodeOperations(USBD_HandleTypeDef *pdev); +static uint8_t USBD_MTP_STORAGE_ReceiveContainer(USBD_HandleTypeDef *pdev, uint32_t *pDst, uint32_t len); +static uint8_t USBD_MTP_STORAGE_SendData(USBD_HandleTypeDef *pdev, uint8_t *buf, uint32_t len); + +/* Private functions ---------------------------------------------------------*/ +/** + * @} + */ + +/** + * @brief USBD_MTP_STORAGE_Init + * Initialize the MTP USB Layer + * @param pdev: device instance + * @retval status value + */ +uint8_t USBD_MTP_STORAGE_Init(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MTPOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + /* Initialize the HW layyer of the file system */ + (void)((USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId])->Init(); + + /* Prepare EP to Receive First Operation */ + (void)USBD_LL_PrepareReceive(pdev, MTPOutEpAdd, (uint8_t *)&hmtp->rx_buff, + hmtp->MaxPcktLen); + + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_MTP_STORAGE_DeInit + * Uninitialize the MTP Machine + * @param pdev: device instance + * @retval status value + */ +uint8_t USBD_MTP_STORAGE_DeInit(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + + /* DeInit physical Interface components */ + hmtp->MTP_SessionState = MTP_SESSION_NOT_OPENED; + + /* Stop low layer file system operations if any */ + USBD_MTP_STORAGE_Cancel(pdev, MTP_PHASE_IDLE); + + /* Free low layer file system resources */ + (void)((USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId])->DeInit(); + + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_MTP_STORAGE_ReadData + * Read data from device objects and send it to the host + * @param pdev: device instance + * @retval status value + */ +uint8_t USBD_MTP_STORAGE_ReadData(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint32_t *data_buff; + + /* Get the data buffer pointer from the low layer interface */ + data_buff = ((USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId])->ScratchBuff; + + switch (ReadDataStatus) + { + case READ_FIRST_DATA: + /* Reset the data length */ + MTP_DataLength.temp_length = 0U; + + /* Perform the low layer read operation on the scratch buffer */ + (void)((USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId])->ReadData(hmtp->OperationsContainer.Param1, + (uint8_t *)data_buff, &MTP_DataLength); + + /* Add the container header to the data buffer */ + (void)USBD_memcpy((uint8_t *)data_buff, (uint8_t *)&hmtp->GenericContainer, MTP_CONT_HEADER_SIZE); + + /* Start USB data transmission to the host */ + (void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)data_buff, + MTP_DataLength.readbytes + MTP_CONT_HEADER_SIZE); + + /* Check if this will be the last packet to send ? */ + if (MTP_DataLength.readbytes < ((uint32_t)hmtp->MaxPcktLen - MTP_CONT_HEADER_SIZE)) + { + /* Move to response phase */ + hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE; + } + else + { + /* Continue to the next packets sending */ + ReadDataStatus = READ_REST_OF_DATA; + } + break; + + case READ_REST_OF_DATA: + /* Perform the low layer read operation on the scratch buffer */ + (void)((USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId])->ReadData(hmtp->OperationsContainer.Param1, + (uint8_t *)data_buff, &MTP_DataLength); + + /* Check if more data need to be sent */ + if (MTP_DataLength.temp_length == MTP_DataLength.totallen) + { + /* Start USB data transmission to the host */ + (void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)data_buff, MTP_DataLength.readbytes); + + /* Move to response phase */ + hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE; + + /* Reset the stat machine */ + ReadDataStatus = READ_FIRST_DATA; + } + else + { + /* Start USB data transmission to the host */ + (void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)data_buff, MTP_DataLength.readbytes); + + /* Keep the state machine into sending next packet of data */ + ReadDataStatus = READ_REST_OF_DATA; + } + break; + + default: + break; + } + + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_MTP_STORAGE_SendContainer + * Send generic container to the host + * @param pdev: device instance + * @retval status value + */ +uint8_t USBD_MTP_STORAGE_SendContainer(USBD_HandleTypeDef *pdev, MTP_CONTAINER_TYPE CONT_TYPE) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + switch (CONT_TYPE) + { + case DATA_TYPE: + /* send header + data : hmtp->ResponseLength = header size + data size */ + (void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)&hmtp->GenericContainer, hmtp->ResponseLength); + break; + case REP_TYPE: + /* send header without data */ + hmtp->GenericContainer.code = (uint16_t)hmtp->ResponseCode; + hmtp->ResponseLength = MTP_CONT_HEADER_SIZE; + hmtp->GenericContainer.length = hmtp->ResponseLength; + hmtp->GenericContainer.type = MTP_CONT_TYPE_RESPONSE; + + (void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)&hmtp->GenericContainer, hmtp->ResponseLength); + break; + default: + break; + } + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_MTP_STORAGE_ReceiveOpt + * Data length Packet Received from host + * @param pdev: device instance + * @retval status value + */ +uint8_t USBD_MTP_STORAGE_ReceiveOpt(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint32_t *pMsgBuffer; +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MTPOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + MTP_DataLength.rx_length = USBD_GetRxCount(pdev, MTPOutEpAdd); + + switch (hmtp->RECEIVE_DATA_STATUS) + { + case RECEIVE_REST_OF_DATA: + /* we don't need to do anything here because we receive only data without operation header*/ + break; + + case RECEIVE_FIRST_DATA: + /* Expected Data Length Packet Received */ + pMsgBuffer = (uint32_t *) &hmtp->OperationsContainer; + + /* Fill hmtp->OperationsContainer Data Buffer from USB Buffer */ + (void)USBD_MTP_STORAGE_ReceiveContainer(pdev, pMsgBuffer, MTP_DataLength.rx_length); + break; + + default: + /* Expected Data Length Packet Received */ + pMsgBuffer = (uint32_t *) &hmtp->OperationsContainer; + + /* Fill hmtp->OperationsContainer Data Buffer from USB Buffer */ + (void)USBD_MTP_STORAGE_ReceiveContainer(pdev, pMsgBuffer, MTP_DataLength.rx_length); + (void)USBD_MTP_STORAGE_DecodeOperations(pdev); + break; + + } + return (uint8_t)USBD_OK; +} + + +/** + * @brief USBD_MTP_STORAGE_ReceiveData + * Receive objects or object info from host + * @param pdev: device instance + * @retval status value + */ +uint8_t USBD_MTP_STORAGE_ReceiveData(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + switch (hmtp->RECEIVE_DATA_STATUS) + { + case RECEIVE_COMMAND_DATA : + if (hmtp->OperationsContainer.type == MTP_CONT_TYPE_COMMAND) + { + MTP_DataLength.temp_length = 0; + MTP_DataLength.prv_len = 0; + (void)USBD_MTP_STORAGE_DecodeOperations(pdev); + + } + break; + + case RECEIVE_FIRST_DATA : + if (hmtp->OperationsContainer.type == MTP_CONT_TYPE_DATA) + { + MTP_DataLength.totallen = hmtp->OperationsContainer.length; + MTP_DataLength.temp_length = MTP_DataLength.rx_length; + MTP_DataLength.rx_length = MTP_DataLength.temp_length - MTP_CONT_HEADER_SIZE; + (void)USBD_MTP_STORAGE_DecodeOperations(pdev); + + if (MTP_DataLength.temp_length < hmtp->MaxPcktLen) /* we received all data, we don't need to go to next state */ + { + hmtp->RECEIVE_DATA_STATUS = SEND_RESPONSE; + (void)USBD_MTP_STORAGE_DecodeOperations(pdev); + + /* send response header after receiving all data successfully */ + (void)USBD_MTP_STORAGE_SendContainer(pdev, DATA_TYPE); + } + } + + break; + + case RECEIVE_REST_OF_DATA : + MTP_DataLength.prv_len = MTP_DataLength.temp_length - MTP_CONT_HEADER_SIZE; + (void)USBD_MTP_STORAGE_DecodeOperations(pdev); + MTP_DataLength.temp_length = MTP_DataLength.temp_length + MTP_DataLength.rx_length; + + if (MTP_DataLength.temp_length == MTP_DataLength.totallen) /* we received all data*/ + { + hmtp->RECEIVE_DATA_STATUS = SEND_RESPONSE; + (void)USBD_MTP_STORAGE_DecodeOperations(pdev); + + /* send response header after receiving all data successfully */ + (void)USBD_MTP_STORAGE_SendContainer(pdev, DATA_TYPE); + } + break; + + default : + break; + } + + return (uint8_t)USBD_OK; +} + + +/** + * @brief USBD_MTP_STORAGE_DecodeOperations + * Parse the operations and Process operations + * @param pdev: device instance + * @retval status value + */ +static uint8_t USBD_MTP_STORAGE_DecodeOperations(USBD_HandleTypeDef *pdev) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + switch (hmtp->OperationsContainer.code) + { + case MTP_OP_GET_DEVICE_INFO: + USBD_MTP_OPT_GetDeviceInfo(pdev); + hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE; + break; + + case MTP_OP_OPEN_SESSION: + USBD_MTP_OPT_CreateObjectHandle(pdev); + hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE; + break; + + case MTP_OP_GET_STORAGE_IDS: + USBD_MTP_OPT_GetStorageIDS(pdev); + hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE; + break; + + case MTP_OP_GET_STORAGE_INFO: + USBD_MTP_OPT_GetStorageInfo(pdev); + hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE; + break; + + case MTP_OP_GET_OBJECT_HANDLES: + USBD_MTP_OPT_GetObjectHandle(pdev); + hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE; + break; + + case MTP_OP_GET_OBJECT_INFO: + USBD_MTP_OPT_GetObjectInfo(pdev); + hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE; + break; + + case MTP_OP_GET_OBJECT_PROP_REFERENCES: + USBD_MTP_OPT_GetObjectReferences(pdev); + hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE; + break; + + case MTP_OP_GET_OBJECT_PROPS_SUPPORTED: + USBD_MTP_OPT_GetObjectPropSupp(pdev); + hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE; + break; + + case MTP_OP_GET_OBJECT_PROP_DESC: + USBD_MTP_OPT_GetObjectPropDesc(pdev); + hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE; + break; + + case MTP_OP_GET_OBJECT_PROPLIST: + USBD_MTP_OPT_GetObjectPropList(pdev); + hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE; + break; + + case MTP_OP_GET_OBJECT_PROP_VALUE: + USBD_MTP_OPT_GetObjectPropValue(pdev); + hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE; + break; + + case MTP_OP_GET_DEVICE_PROP_DESC: + USBD_MTP_OPT_GetDevicePropDesc(pdev); + hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE; + break; + + case MTP_OP_GET_OBJECT: + USBD_MTP_OPT_GetObject(pdev); + hmtp->MTP_ResponsePhase = MTP_READ_DATA; + break; + + case MTP_OP_SEND_OBJECT_INFO: + USBD_MTP_OPT_SendObjectInfo(pdev, (uint8_t *)(hmtp->rx_buff), MTP_DataLength.prv_len); + hmtp->MTP_ResponsePhase = MTP_RECEIVE_DATA; + break; + + case MTP_OP_SEND_OBJECT: + USBD_MTP_OPT_SendObject(pdev, (uint8_t *)(hmtp->rx_buff), MTP_DataLength.rx_length); + hmtp->MTP_ResponsePhase = MTP_RECEIVE_DATA; + break; + + case MTP_OP_DELETE_OBJECT: + USBD_MTP_OPT_DeleteObject(pdev); + hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE; + break; + + default: + break; + } + + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_MTP_STORAGE_ReceiveContainer + * Receive the Data from USB BulkOut Buffer to Pointer + * @param pdev: device instance + * @param pDst: destination address to copy the buffer + * @param len: length of data to copy + * @retval status value + */ +static uint8_t USBD_MTP_STORAGE_ReceiveContainer(USBD_HandleTypeDef *pdev, + uint32_t *pDst, uint32_t len) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint32_t Counter; + uint32_t *pdst = pDst; + + for (Counter = 0; Counter < len; Counter++) + { + *pdst = (hmtp->rx_buff[Counter]); + pdst++; + } + return (uint8_t)USBD_OK; +} + +/** + * @brief USBD_MTP_STORAGE_Cancel + * Reinitialize all states and cancel transfer through Bulk transfer + * @param pdev: device instance + * @param MTP_ResponsePhase: MTP current state + * @retval None + */ +void USBD_MTP_STORAGE_Cancel(USBD_HandleTypeDef *pdev, + MTP_ResponsePhaseTypeDef MTP_ResponsePhase) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + + hmtp->MTP_ResponsePhase = MTP_PHASE_IDLE; + ReadDataStatus = READ_FIRST_DATA; + hmtp->RECEIVE_DATA_STATUS = RECEIVE_IDLE_STATE; + + if (MTP_ResponsePhase == MTP_RECEIVE_DATA) + { + ((USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId])->Cancel(1U); + } + else + { + ((USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId])->Cancel(0U); + } +} + +/** + * @brief USBD_MTP_STORAGE_SendData + * Send the data on bulk-in EP + * @param pdev: device instance + * @param buf: pointer to data buffer + * @param len: Data Length + * @retval status value + */ +static uint8_t USBD_MTP_STORAGE_SendData(USBD_HandleTypeDef *pdev, uint8_t *buf, + uint32_t len) +{ + USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + uint32_t length = MIN(hmtp->GenericContainer.length, len); +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + MTPInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + + (void)USBD_LL_Transmit(pdev, MTPInEpAdd, buf, length); + + return (uint8_t)USBD_OK; +} diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Inc/usbd_printer.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Inc/usbd_printer.h index ec73edca..bf00b9c1 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Inc/usbd_printer.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Inc/usbd_printer.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2021 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -114,8 +113,7 @@ typedef struct __IO uint32_t TxState; __IO uint32_t RxState; -} -USBD_PRNT_HandleTypeDef; +} USBD_PRNT_HandleTypeDef; @@ -161,4 +159,3 @@ uint8_t USBD_PRNT_ReceivePacket(USBD_HandleTypeDef *pdev); * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Inc/usbd_printer_if_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Inc/usbd_printer_if_template.h index 7037afad..7765eed5 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Inc/usbd_printer_if_template.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Inc/usbd_printer_if_template.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2021 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * http://www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -42,4 +41,3 @@ extern USBD_PRNT_ItfTypeDef USBD_PRNT_Template_fops; #endif /* __USBD_PRNT_IF_TEMPLATE_H */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Src/usbd_printer.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Src/usbd_printer.c index 246674f7..7c1a4ba8 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Src/usbd_printer.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Src/usbd_printer.c @@ -10,6 +10,17 @@ * - Command OUT transfer (class requests management) * - Error management * + ****************************************************************************** + * @attention + * + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** * @verbatim * * =================================================================== @@ -37,17 +48,6 @@ * @endverbatim * ****************************************************************************** - * @attention - * - * <h2><center>© Copyright (c) 2021 STMicroelectronics. - * All rights reserved.</center></h2> - * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** */ /* BSPDependencies @@ -102,27 +102,13 @@ static uint8_t USBD_PRNT_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *r static uint8_t USBD_PRNT_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum); static uint8_t USBD_PRNT_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum); +#ifndef USE_USBD_COMPOSITE static uint8_t *USBD_PRNT_GetFSCfgDesc(uint16_t *length); static uint8_t *USBD_PRNT_GetHSCfgDesc(uint16_t *length); static uint8_t *USBD_PRNT_GetOtherSpeedCfgDesc(uint16_t *length); static uint8_t *USBD_PRNT_GetOtherSpeedCfgDesc(uint16_t *length); uint8_t *USBD_PRNT_GetDeviceQualifierDescriptor(uint16_t *length); - -/* USB Standard Device Descriptor */ -__ALIGN_BEGIN static uint8_t USBD_PRNT_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_DESC] __ALIGN_END = -{ - USB_LEN_DEV_QUALIFIER_DESC, - USB_DESC_TYPE_DEVICE_QUALIFIER, - 0x00, - 0x02, - 0x00, - 0x00, - 0x00, - 0x40, - 0x01, - 0x00, -}; - +#endif /* USE_USBD_COMPOSITE */ /** * @} */ @@ -144,63 +130,22 @@ USBD_ClassTypeDef USBD_PRNT = NULL, NULL, NULL, +#ifdef USE_USBD_COMPOSITE + NULL, + NULL, + NULL, + NULL, +#else USBD_PRNT_GetHSCfgDesc, USBD_PRNT_GetFSCfgDesc, USBD_PRNT_GetOtherSpeedCfgDesc, USBD_PRNT_GetDeviceQualifierDescriptor, +#endif /* USE_USBD_COMPOSITE */ }; +#ifndef USE_USBD_COMPOSITE /* USB PRNT device Configuration Descriptor */ -__ALIGN_BEGIN static uint8_t USBD_PRNT_CfgHSDesc[] __ALIGN_END = -{ - /*Configuration Descriptor*/ - 0x09, /* bLength: Configuration Descriptor size */ - USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ - USB_PRNT_CONFIG_DESC_SIZ, /* wTotalLength:no of returned bytes */ - 0x00, - 0x01, /* bNumInterfaces: 1 interface */ - 0x01, /* bConfigurationValue: Configuration value */ - 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ -#if (USBD_SELF_POWERED == 1U) - 0xC0, /* bmAttributes: Self Powered according to user configuration */ -#else - 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif - USBD_MAX_POWER, /* MaxPower in mA */ - - /* Interface Descriptor */ - 0x09, /* bLength: Interface Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: Interface */ - 0x00, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x02, /* bNumEndpoints: 2 endpoints used */ - 0x07, /* bInterfaceClass: Communication Interface Class */ - 0x01, /* bInterfaceSubClass: Abstract Control Model */ - USB_PRNT_BIDIRECTIONAL, /* bDeviceProtocol */ - 0x00, /* iInterface */ - - /* Endpoint IN Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - PRNT_IN_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - LOBYTE(PRNT_DATA_HS_IN_PACKET_SIZE), /* wMaxPacketSize */ - HIBYTE(PRNT_DATA_HS_IN_PACKET_SIZE), - 0x00, /* bInterval */ - - /* Endpoint OUT Descriptor */ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - PRNT_OUT_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - LOBYTE(PRNT_DATA_HS_OUT_PACKET_SIZE),/* wMaxPacketSize */ - HIBYTE(PRNT_DATA_HS_OUT_PACKET_SIZE), - 0x00 /* bInterval */ -}; - - -/* USB PRNT device Configuration Descriptor */ -__ALIGN_BEGIN static uint8_t USBD_PRNT_CfgFSDesc[] __ALIGN_END = +__ALIGN_BEGIN static uint8_t USBD_PRNT_CfgDesc[] __ALIGN_END = { /*Configuration Descriptor*/ 0x09, /* bLength: Configuration Descriptor size */ @@ -214,7 +159,7 @@ __ALIGN_BEGIN static uint8_t USBD_PRNT_CfgFSDesc[] __ALIGN_END = 0xC0, /* bmAttributes: Self Powered according to user configuration */ #else 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif +#endif /* USBD_SELF_POWERED */ USBD_MAX_POWER, /* MaxPower in mA */ /*Interface Descriptor */ @@ -247,52 +192,24 @@ __ALIGN_BEGIN static uint8_t USBD_PRNT_CfgFSDesc[] __ALIGN_END = 0x00 /* bInterval */ }; -__ALIGN_BEGIN static uint8_t USBD_PRNT_OtherSpeedCfgDesc[] __ALIGN_END = +/* USB Standard Device Descriptor */ +__ALIGN_BEGIN static uint8_t USBD_PRNT_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_DESC] __ALIGN_END = { - /*Configuration Descriptor*/ - 0x09, /* bLength: Configuration Descriptor size */ - USB_DESC_TYPE_CONFIGURATION, /* bDescriptorType: Configuration */ - USB_PRNT_CONFIG_DESC_SIZ, /* wTotalLength:no of returned bytes */ + USB_LEN_DEV_QUALIFIER_DESC, + USB_DESC_TYPE_DEVICE_QUALIFIER, + 0x00, + 0x02, + 0x00, + 0x00, + 0x00, + 0x40, + 0x01, 0x00, - 0x01, /* bNumInterfaces: 1 interface */ - 0x01, /* bConfigurationValue: Configuration value */ - 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ -#if (USBD_SELF_POWERED == 1U) - 0xC0, /* bmAttributes: Self Powered according to user configuration */ -#else - 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif - USBD_MAX_POWER, /* MaxPower in mA */ - - /*Interface Descriptor */ - 0x09, /* bLength: Interface Descriptor size */ - USB_DESC_TYPE_INTERFACE, /* bDescriptorType: Interface */ - 0x00, /* bInterfaceNumber: Number of Interface */ - 0x00, /* bAlternateSetting: Alternate setting */ - 0x02, /* bNumEndpoints: 2 endpoints used */ - 0x07, /* bInterfaceClass: Communication Interface Class */ - 0x01, /* bInterfaceSubClass: Abstract Control Model */ - USB_PRNT_BIDIRECTIONAL, /* bDeviceProtocol */ - 0x00, /* iInterface */ - - /*Endpoint IN Descriptor*/ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - PRNT_IN_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - LOBYTE(PRNT_DATA_FS_IN_PACKET_SIZE), /* wMaxPacketSize */ - HIBYTE(PRNT_DATA_FS_IN_PACKET_SIZE), - 0x00, /* bInterval */ - - /*Endpoint OUT Descriptor*/ - 0x07, /* bLength: Endpoint Descriptor size */ - USB_DESC_TYPE_ENDPOINT, /* bDescriptorType: Endpoint */ - PRNT_OUT_EP, /* bEndpointAddress */ - 0x02, /* bmAttributes: Bulk */ - LOBYTE(PRNT_DATA_FS_OUT_PACKET_SIZE),/* wMaxPacketSize */ - HIBYTE(PRNT_DATA_FS_OUT_PACKET_SIZE), - 0x00 /* bInterval */ }; +#endif /* USE_USBD_COMPOSITE */ + +static uint8_t PRNTInEpAdd = PRNT_IN_EP; +static uint8_t PRNTOutEpAdd = PRNT_OUT_EP; /** * @} @@ -315,16 +232,26 @@ static uint8_t USBD_PRNT_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) USBD_PRNT_HandleTypeDef *hPRNT; uint16_t mps; + hPRNT = (USBD_PRNT_HandleTypeDef *)USBD_malloc(sizeof(USBD_PRNT_HandleTypeDef)); if (hPRNT == NULL) { - pdev->pClassData = NULL; + pdev->pClassDataCmsit[pdev->classId] = NULL; return (uint8_t)USBD_EMEM; } + (void)USBD_memset(hPRNT, 0, sizeof(USBD_PRNT_HandleTypeDef)); + /* Setup the pClassData pointer */ - pdev->pClassData = (void *)hPRNT; + pdev->pClassDataCmsit[pdev->classId] = (void *)hPRNT; + pdev->pClassData = pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + PRNTInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + PRNTOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ /* Setup the max packet size according to selected speed */ if (pdev->dev_speed == USBD_SPEED_HIGH) @@ -337,22 +264,29 @@ static uint8_t USBD_PRNT_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) } /* Open EP IN */ - (void)USBD_LL_OpenEP(pdev, PRNT_IN_EP, USBD_EP_TYPE_BULK, mps); + (void)USBD_LL_OpenEP(pdev, PRNTInEpAdd, USBD_EP_TYPE_BULK, mps); /* Set endpoint as used */ - pdev->ep_in[PRNT_IN_EP & 0xFU].is_used = 1U; + pdev->ep_in[PRNTInEpAdd & 0xFU].is_used = 1U; /* Open EP OUT */ - (void)USBD_LL_OpenEP(pdev, PRNT_OUT_EP, USBD_EP_TYPE_BULK, mps); + (void)USBD_LL_OpenEP(pdev, PRNTOutEpAdd, USBD_EP_TYPE_BULK, mps); /* Set endpoint as used */ - pdev->ep_out[PRNT_OUT_EP & 0xFU].is_used = 1U; + pdev->ep_out[PRNTOutEpAdd & 0xFU].is_used = 1U; + + hPRNT->RxBuffer = NULL; /* Init physical Interface components */ - ((USBD_PRNT_ItfTypeDef *)pdev->pUserData)->Init(); + ((USBD_PRNT_ItfTypeDef *)pdev->pUserData[pdev->classId])->Init(); + + if (hPRNT->RxBuffer == NULL) + { + return (uint8_t)USBD_EMEM; + } /* Prepare Out endpoint to receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, PRNT_OUT_EP, hPRNT->RxBuffer, mps); + (void)USBD_LL_PrepareReceive(pdev, PRNTOutEpAdd, hPRNT->RxBuffer, mps); /* End of initialization phase */ return (uint8_t)USBD_OK; @@ -369,19 +303,26 @@ static uint8_t USBD_PRNT_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) { UNUSED(cfgidx); +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + PRNTInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + PRNTOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + /* Close EP IN */ - (void)USBD_LL_CloseEP(pdev, PRNT_IN_EP); - pdev->ep_in[PRNT_IN_EP & 0xFU].is_used = 0U; + (void)USBD_LL_CloseEP(pdev, PRNTInEpAdd); + pdev->ep_in[PRNTInEpAdd & 0xFU].is_used = 0U; /* Close EP OUT */ - (void)USBD_LL_CloseEP(pdev, PRNT_OUT_EP); - pdev->ep_out[PRNT_OUT_EP & 0xFU].is_used = 0U; + (void)USBD_LL_CloseEP(pdev, PRNTOutEpAdd); + pdev->ep_out[PRNTOutEpAdd & 0xFU].is_used = 0U; /* DeInit physical Interface components */ - if (pdev->pClassData != NULL) + if (pdev->pClassDataCmsit[pdev->classId] != NULL) { - ((USBD_PRNT_ItfTypeDef *)pdev->pUserData)->DeInit(); - (void)USBD_free(pdev->pClassData); + ((USBD_PRNT_ItfTypeDef *)pdev->pUserData[pdev->classId])->DeInit(); + (void)USBD_free(pdev->pClassDataCmsit[pdev->classId]); + pdev->pClassDataCmsit[pdev->classId] = NULL; pdev->pClassData = NULL; } @@ -397,8 +338,8 @@ static uint8_t USBD_PRNT_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) */ static uint8_t USBD_PRNT_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_PRNT_HandleTypeDef *hPRNT = (USBD_PRNT_HandleTypeDef *)pdev->pClassData; - USBD_PRNT_ItfTypeDef *hPRNTitf = (USBD_PRNT_ItfTypeDef *)pdev->pUserData; + USBD_PRNT_HandleTypeDef *hPRNT = (USBD_PRNT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + USBD_PRNT_ItfTypeDef *hPRNTitf = (USBD_PRNT_ItfTypeDef *)pdev->pUserData[pdev->classId]; USBD_StatusTypeDef ret = USBD_OK; uint16_t status_info = 0U; @@ -495,7 +436,7 @@ static uint8_t USBD_PRNT_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *r */ static uint8_t USBD_PRNT_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) { - USBD_PRNT_HandleTypeDef *hPRNT = (USBD_PRNT_HandleTypeDef *)pdev->pClassData; + USBD_PRNT_HandleTypeDef *hPRNT = (USBD_PRNT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; PCD_HandleTypeDef *hpcd = pdev->pData; if (hPRNT == NULL) @@ -503,11 +444,11 @@ static uint8_t USBD_PRNT_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) return (uint8_t)USBD_FAIL; } - if ((pdev->ep_in[epnum].total_length > 0U) && - ((pdev->ep_in[epnum].total_length % hpcd->IN_ep[epnum].maxpacket) == 0U)) + if ((pdev->ep_in[epnum & 0xFU].total_length > 0U) && + ((pdev->ep_in[epnum & 0xFU].total_length % hpcd->IN_ep[epnum & 0xFU].maxpacket) == 0U)) { /* Update the packet total length */ - pdev->ep_in[epnum].total_length = 0U; + pdev->ep_in[epnum & 0xFU].total_length = 0U; /* Send ZLP */ (void) USBD_LL_Transmit(pdev, epnum, NULL, 0U); @@ -528,7 +469,7 @@ static uint8_t USBD_PRNT_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) */ static uint8_t USBD_PRNT_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) { - USBD_PRNT_HandleTypeDef *hPRNT = (USBD_PRNT_HandleTypeDef *)pdev->pClassData; + USBD_PRNT_HandleTypeDef *hPRNT = (USBD_PRNT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; if (hPRNT == NULL) { @@ -540,11 +481,12 @@ static uint8_t USBD_PRNT_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) /* USB data will be immediately processed, this allow next USB traffic being NAKed till the end of the application Xfer */ - ((USBD_PRNT_ItfTypeDef *)pdev->pUserData)->Receive(hPRNT->RxBuffer, &hPRNT->RxLength); + ((USBD_PRNT_ItfTypeDef *)pdev->pUserData[pdev->classId])->Receive(hPRNT->RxBuffer, &hPRNT->RxLength); return (uint8_t)USBD_OK; } +#ifndef USE_USBD_COMPOSITE /** * @brief USBD_PRNT_GetFSCfgDesc * Return configuration descriptor @@ -553,8 +495,21 @@ static uint8_t USBD_PRNT_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) */ static uint8_t *USBD_PRNT_GetFSCfgDesc(uint16_t *length) { - *length = (uint16_t) sizeof(USBD_PRNT_CfgFSDesc); - return USBD_PRNT_CfgFSDesc; + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_PRNT_CfgDesc, PRNT_IN_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_PRNT_CfgDesc, PRNT_OUT_EP); + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = PRNT_DATA_FS_IN_PACKET_SIZE; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = PRNT_DATA_FS_OUT_PACKET_SIZE; + } + + *length = (uint16_t) sizeof(USBD_PRNT_CfgDesc); + return USBD_PRNT_CfgDesc; } /** @@ -565,8 +520,21 @@ static uint8_t *USBD_PRNT_GetFSCfgDesc(uint16_t *length) */ static uint8_t *USBD_PRNT_GetHSCfgDesc(uint16_t *length) { - *length = (uint16_t) sizeof(USBD_PRNT_CfgHSDesc); - return USBD_PRNT_CfgHSDesc; + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_PRNT_CfgDesc, PRNT_IN_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_PRNT_CfgDesc, PRNT_OUT_EP); + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = PRNT_DATA_HS_IN_PACKET_SIZE; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = PRNT_DATA_HS_OUT_PACKET_SIZE; + } + + *length = (uint16_t) sizeof(USBD_PRNT_CfgDesc); + return USBD_PRNT_CfgDesc; } /** @@ -577,8 +545,21 @@ static uint8_t *USBD_PRNT_GetHSCfgDesc(uint16_t *length) */ static uint8_t *USBD_PRNT_GetOtherSpeedCfgDesc(uint16_t *length) { - *length = (uint16_t) sizeof(USBD_PRNT_OtherSpeedCfgDesc); - return USBD_PRNT_OtherSpeedCfgDesc; + USBD_EpDescTypeDef *pEpInDesc = USBD_GetEpDesc(USBD_PRNT_CfgDesc, PRNT_IN_EP); + USBD_EpDescTypeDef *pEpOutDesc = USBD_GetEpDesc(USBD_PRNT_CfgDesc, PRNT_OUT_EP); + + if (pEpInDesc != NULL) + { + pEpInDesc->wMaxPacketSize = PRNT_DATA_FS_IN_PACKET_SIZE; + } + + if (pEpOutDesc != NULL) + { + pEpOutDesc->wMaxPacketSize = PRNT_DATA_FS_OUT_PACKET_SIZE; + } + + *length = (uint16_t) sizeof(USBD_PRNT_CfgDesc); + return USBD_PRNT_CfgDesc; } /** @@ -592,6 +573,7 @@ uint8_t *USBD_PRNT_GetDeviceQualifierDescriptor(uint16_t *length) *length = (uint16_t)sizeof(USBD_PRNT_DeviceQualifierDesc); return USBD_PRNT_DeviceQualifierDesc; } +#endif /* USE_USBD_COMPOSITE */ /** * @brief USBD_PRNT_RegisterInterface @@ -608,7 +590,7 @@ uint8_t USBD_PRNT_RegisterInterface(USBD_HandleTypeDef *pdev, USBD_PRNT_ItfTypeD } /* Setup the fops pointer */ - pdev->pUserData = fops; + pdev->pUserData[pdev->classId] = fops; return (uint8_t)USBD_OK; } @@ -621,7 +603,7 @@ uint8_t USBD_PRNT_RegisterInterface(USBD_HandleTypeDef *pdev, USBD_PRNT_ItfTypeD */ uint8_t USBD_PRNT_SetRxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff) { - USBD_PRNT_HandleTypeDef *hPRNT = (USBD_PRNT_HandleTypeDef *) pdev->pClassData; + USBD_PRNT_HandleTypeDef *hPRNT = (USBD_PRNT_HandleTypeDef *) pdev->pClassDataCmsit[pdev->classId]; hPRNT->RxBuffer = pbuff; @@ -636,7 +618,13 @@ uint8_t USBD_PRNT_SetRxBuffer(USBD_HandleTypeDef *pdev, uint8_t *pbuff) */ uint8_t USBD_PRNT_ReceivePacket(USBD_HandleTypeDef *pdev) { - USBD_PRNT_HandleTypeDef *hPRNT = (USBD_PRNT_HandleTypeDef *)pdev->pClassData; + USBD_PRNT_HandleTypeDef *hPRNT = (USBD_PRNT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + PRNTInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); + PRNTOutEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_OUT, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ if (hPRNT == NULL) { @@ -646,18 +634,19 @@ uint8_t USBD_PRNT_ReceivePacket(USBD_HandleTypeDef *pdev) if (pdev->dev_speed == USBD_SPEED_HIGH) { /* Prepare Out endpoint to receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, PRNT_OUT_EP, hPRNT->RxBuffer, + (void)USBD_LL_PrepareReceive(pdev, PRNTOutEpAdd, hPRNT->RxBuffer, PRNT_DATA_HS_OUT_PACKET_SIZE); } else { /* Prepare Out endpoint to receive next packet */ - (void)USBD_LL_PrepareReceive(pdev, PRNT_OUT_EP, hPRNT->RxBuffer, + (void)USBD_LL_PrepareReceive(pdev, PRNTOutEpAdd, hPRNT->RxBuffer, PRNT_DATA_FS_OUT_PACKET_SIZE); } return (uint8_t)USBD_OK; } + /** * @} */ @@ -669,5 +658,3 @@ uint8_t USBD_PRNT_ReceivePacket(USBD_HandleTypeDef *pdev) /** * @} */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Src/usbd_printer_if_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Src/usbd_printer_if_template.c index 064c41e8..8b63db06 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Src/usbd_printer_if_template.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Printer/Src/usbd_printer_if_template.c @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2021 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2021 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * http://www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -71,11 +70,11 @@ USBD_PRNT_ItfTypeDef USBD_PRNT_Template_fops = { TEMPLATE_Init, TEMPLATE_DeInit, - TEMPLATE_Control_req + TEMPLATE_Control_req, TEMPLATE_Receive }; -static uint8_t PRNT_DEVICE_ID[DEVICE_ID_LEN] = +static uint8_t PRNT_DEVICE_ID[] = { 0x00, 0x6D, 'M', 'A', 'N', 'U', 'F', 'A', 'C', 'T', 'U', 'R', 'E', 'R', ':', @@ -128,7 +127,7 @@ static int8_t TEMPLATE_DeInit(void) * * @note * This function will issue a NAK packet on any OUT packet received on - * USB endpoint untill exiting this function. If you exit this function + * USB endpoint until exiting this function. If you exit this function * before transfer is complete on PRNT interface (ie. using DMA controller) * it will result in receiving more data while previous ones are still * not sent. @@ -147,16 +146,16 @@ static int8_t TEMPLATE_Receive(uint8_t *Buf, uint32_t *Len) /** - * @brief TEMPLATE_PRNT_Itf_Control_req + * @brief TEMPLATE_Control_req * Manage the PRNT class requests * @param req: Command code * @param pbuf: Buffer containing command data (request parameters) * @param length: Number of data to be sent (in bytes) * @retval Result of the operation: USBD_OK if all operations are OK else USBD_FAIL */ -static int8_t TEMPLATE_PRNT_Itf_Control_req(uint8_t req, uint8_t *pbuf, uint16_t *length) +static int8_t TEMPLATE_Control_req(uint8_t req, uint8_t *pbuf, uint16_t *length) { - uint32_t i = 0; + uint32_t i = 0U; /* Check on the setup request value */ switch (req) @@ -169,7 +168,7 @@ static int8_t TEMPLATE_PRNT_Itf_Control_req(uint8_t req, uint8_t *pbuf, uint16_t pbuf[i] = PRNT_DEVICE_ID[i]; i++; } - *length = i; + *length = (uint16_t)i; break; /* Get Printer current status */ @@ -183,25 +182,16 @@ static int8_t TEMPLATE_PRNT_Itf_Control_req(uint8_t req, uint8_t *pbuf, uint16_t /* Printer SOFT RESET request: cleanup pending tasks */ case PRNT_SOFT_RESET: - (void)f_close(&hSD.MyFile); break; default: - /* Unkown commands are not managed */ + /* Unknown commands are not managed */ break; -} + } -/** -* @brief TEMPLATE_PRNT_PageEndManager, defined by user -* Call this function frequently to check if data is received. -* @param Buf: Buffer of data to be received -* @param Len: Number of data received (in bytes) -*/ -void TEMPLATE_PRNT_PageEndManager(uint8_t *Buf, uint32_t Len) -{ - UNUSED(Buf); - UNUSED(Len); + return (0); } + /** * @} */ @@ -214,5 +204,3 @@ void TEMPLATE_PRNT_PageEndManager(uint8_t *Buf, uint32_t Len) * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ - diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Template/Inc/usbd_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Template/Inc/usbd_template.h index 8e9163bb..be314376 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Template/Inc/usbd_template.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Template/Inc/usbd_template.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -97,5 +96,3 @@ extern USBD_ClassTypeDef USBD_TEMPLATE_ClassDriver; /** * @} */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Template/Src/usbd_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Template/Src/usbd_template.c index 10ced3fc..947ae911 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Template/Src/usbd_template.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/Template/Src/usbd_template.c @@ -4,6 +4,17 @@ * @author MCD Application Team * @brief This file provides the HID core functions. * + ****************************************************************************** + * @attention + * + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** * @verbatim * * =================================================================== @@ -22,17 +33,6 @@ * @endverbatim * ****************************************************************************** - * @attention - * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> - * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** */ /* Includes ------------------------------------------------------------------*/ @@ -124,7 +124,7 @@ USBD_ClassTypeDef USBD_TEMPLATE_ClassDriver = #if defined ( __ICCARM__ ) /*!< IAR Compiler */ #pragma data_alignment=4 -#endif +#endif /* __ICCARM__ */ /* USB TEMPLATE device Configuration Descriptor */ __ALIGN_BEGIN static uint8_t USBD_TEMPLATE_CfgDesc[USB_TEMPLATE_CONFIG_DESC_SIZ] __ALIGN_END = { @@ -146,7 +146,7 @@ __ALIGN_BEGIN static uint8_t USBD_TEMPLATE_CfgDesc[USB_TEMPLATE_CONFIG_DESC_SIZ] #if defined ( __ICCARM__ ) /*!< IAR Compiler */ #pragma data_alignment=4 -#endif +#endif /* __ICCARM__ */ /* USB Standard Device Descriptor */ __ALIGN_BEGIN static uint8_t USBD_TEMPLATE_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_DESC] __ALIGN_END = { @@ -252,12 +252,12 @@ static uint8_t *USBD_TEMPLATE_GetCfgDesc(uint16_t *length) } /** - * @brief DeviceQualifierDescriptor + * @brief USBD_TEMPLATE_GetDeviceQualifierDesc * return Device Qualifier descriptor * @param length : pointer data length * @retval pointer to descriptor buffer */ -uint8_t *USBD_TEMPLATE_DeviceQualifierDescriptor(uint16_t *length) +uint8_t *USBD_TEMPLATE_GetDeviceQualifierDesc(uint16_t *length) { *length = (uint16_t)sizeof(USBD_TEMPLATE_DeviceQualifierDesc); return USBD_TEMPLATE_DeviceQualifierDesc; @@ -350,19 +350,6 @@ static uint8_t USBD_TEMPLATE_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum) return (uint8_t)USBD_OK; } -/** - * @brief DeviceQualifierDescriptor - * return Device Qualifier descriptor - * @param length : pointer data length - * @retval pointer to descriptor buffer - */ -uint8_t *USBD_TEMPLATE_GetDeviceQualifierDesc(uint16_t *length) -{ - *length = (uint16_t)sizeof(USBD_TEMPLATE_DeviceQualifierDesc); - - return USBD_TEMPLATE_DeviceQualifierDesc; -} - /** * @} */ @@ -376,5 +363,3 @@ uint8_t *USBD_TEMPLATE_GetDeviceQualifierDesc(uint16_t *length) /** * @} */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Inc/usbd_video.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Inc/usbd_video.h index c92ff761..8ece002b 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Inc/usbd_video.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Inc/usbd_video.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2020 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2020 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -47,12 +46,12 @@ extern "C" { #define UVC_VERSION 0x0100U /* UVC 1.0 */ #else #define UVC_VERSION 0x0110U /* UVC 1.1 */ -#endif +#endif /* UVC_1_0 */ /* bEndpointAddress in Endpoint Descriptor */ #ifndef UVC_IN_EP #define UVC_IN_EP 0x81U -#endif /* VIDEO_IN_EP */ +#endif /* UVC_IN_EP */ /* These defines shall be updated in the usbd_conf.h file */ #ifndef UVC_WIDTH @@ -111,15 +110,15 @@ extern "C" { #ifndef UVC_ISO_FS_MPS #define UVC_ISO_FS_MPS 256U -#endif +#endif /* UVC_ISO_FS_MPS */ #ifndef UVC_ISO_HS_MPS #define UVC_ISO_HS_MPS 512U -#endif +#endif /* UVC_ISO_HS_MPS */ #ifndef UVC_HEADER_PACKET_CNT #define UVC_HEADER_PACKET_CNT 0x01U -#endif +#endif /* UVC_HEADER_PACKET_CNT */ #define UVC_REQ_READ_MASK 0x80U @@ -131,7 +130,9 @@ extern "C" { #define UVC_CONFIG_DESC_SIZ (0x88U + 0x16U) #else #define UVC_CONFIG_DESC_SIZ 0x88U -#endif +#endif /* USBD_UVC_FORMAT_UNCOMPRESSED */ + +#define USBD_VC_GIUD_FORMAT_SIZE 16U #define UVC_TOTAL_BUF_SIZE 0x04U @@ -148,10 +149,10 @@ extern "C" { #ifndef WBVAL #define WBVAL(x) ((x) & 0xFFU),(((x) >> 8) & 0xFFU) -#endif +#endif /* WBVAL */ #ifndef DBVAL #define DBVAL(x) ((x)& 0xFFU),(((x) >> 8) & 0xFFU),(((x)>> 16) & 0xFFU),(((x) >> 24) & 0xFFU) -#endif +#endif /* DBVAL */ /* Video Interface Protocol Codes */ #define PC_PROTOCOL_UNDEFINED 0x00U @@ -183,7 +184,7 @@ extern "C" { #define VC_HEADER_SIZE (VIDEO_VS_IF_IN_HEADER_DESC_SIZE + \ VS_FORMAT_DESC_SIZE + \ VS_FRAME_DESC_SIZE) -#endif +#endif /* USBD_UVC_FORMAT_UNCOMPRESSED */ /* * Video Class specification release 1.1 @@ -364,32 +365,6 @@ typedef enum VIDEO_OFFSET_UNKNOWN, } VIDEO_OffsetTypeDef; -typedef struct _VIDEO_DescHeader -{ - uint8_t bLength; - uint8_t bDescriptorType; - uint8_t bDescriptorSubType; -} USBD_VIDEO_DescHeader_t; - -typedef struct -{ - uint8_t bLength; - uint8_t bDescriptorType; - uint8_t bDescriptorSubType; - uint8_t bFrameIndex; - uint8_t bmCapabilities; - uint16_t wWidth; - uint16_t wHeight; - uint32_t dwMinBitRate; - uint32_t dwMaxBitRate; - uint32_t dwMaxVideoFrameBufSize; - uint32_t dwDefaultFrameInterval; - uint8_t bFrameIntervalType; - uint32_t dwMinFrameInterval; - uint32_t dwMaxFrameInterval; - uint32_t dwFrameIntervalStep; -} __PACKED USBD_VIDEO_VSFrameDescTypeDef; - typedef struct { uint8_t cmd; @@ -413,7 +388,6 @@ typedef struct int8_t (* DeInit)(void); int8_t (* Control)(uint8_t, uint8_t *, uint16_t); int8_t (* Data)(uint8_t **, uint16_t *, uint16_t *); - uint8_t *pStrDesc; } USBD_VIDEO_ItfTypeDef; /* UVC uses only 26 first bytes */ @@ -437,6 +411,130 @@ typedef struct uint8_t bMaxVersion; } __PACKED USBD_VideoControlTypeDef; +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bInterfaceNumber; + uint8_t bAlternateSetting; + uint8_t bNumEndpoints; + uint8_t bInterfaceClass; + uint8_t bInterfaceSubClass; + uint8_t bInterfaceProtocol; + uint8_t iFunction; +} __PACKED USBD_StandardVCIfDescTypeDef; + +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubtype; + uint16_t bcdUVC; + uint16_t wTotalLength; + uint32_t dwClockFrequency; + uint8_t baInterfaceNr; + uint8_t iTerminal; +} __PACKED USBD_specificVCInDescTypeDef; + +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubtype; + uint8_t bTerminalID; + uint16_t wTerminalType; + uint8_t bAssocTerminal; + uint8_t iTerminal; +} __PACKED USBD_InputTerminalDescTypeDef; + +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubtype; + uint8_t bTerminalID; + uint16_t wTerminalType; + uint8_t bAssocTerminal; + uint8_t bSourceID; + uint8_t iTerminal; +} __PACKED USBD_OutputTerminalDescTypeDef; + +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubtype; + uint16_t bNumFormats; + uint8_t bVideoControlSize; + uint8_t bEndPointAddress; + uint8_t bmInfo; + uint8_t bTerminalLink; + uint8_t bStillCaptureMethod; + uint8_t bTriggerSupport; + uint8_t bTriggerUsage; + uint8_t bControlSize; + uint8_t bmaControls; +} __PACKED USBD_ClassSpecificVsHeaderDescTypeDef; + +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubType; + uint8_t bFormatIndex; + uint8_t bNumFrameDescriptor; +#ifdef USBD_UVC_FORMAT_UNCOMPRESSED + uint8_t pGiudFormat[USBD_VC_GIUD_FORMAT_SIZE]; + uint8_t bBitsPerPixel; +#else + uint8_t bmFlags; +#endif /* USBD_UVC_FORMAT_UNCOMPRESSED */ + uint8_t bDefaultFrameIndex; + uint8_t bAspectRatioX; + uint8_t bAspectRatioY; + uint8_t bInterlaceFlags; + uint8_t bCopyProtect; +} __PACKED USBD_PayloadFormatDescTypeDef; + +#ifdef USBD_UVC_FORMAT_UNCOMPRESSED +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubType; + uint8_t bColorPrimarie; + uint8_t bTransferCharacteristics; + uint8_t bMatrixCoefficients; +} __PACKED USBD_ColorMatchingDescTypeDef; +#endif /* USBD_UVC_FORMAT_UNCOMPRESSED */ + +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bEndpointAddress; + uint8_t bmAttributes; + uint16_t wMaxPacketSize; + uint8_t bInterval; +} __PACKED USBD_StandardVCDataEPDescTypeDef; + +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubType; + uint8_t bFrameIndex; + uint8_t bmCapabilities; + uint16_t wWidth; + uint16_t wHeight; + uint32_t dwMinBitRate; + uint32_t dwMaxBitRate; + uint32_t dwMaxVideoFrameBufSize; + uint32_t dwDefaultFrameInterval; + uint8_t bFrameIntervalType; + uint32_t dwMinFrameInterval; +} __PACKED USBD_VIDEO_VSFrameDescTypeDef; + extern USBD_ClassTypeDef USBD_VIDEO; #define USBD_VIDEO_CLASS &USBD_VIDEO /** diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Inc/usbd_video_if_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Inc/usbd_video_if_template.h index 036c7aad..bc5b8faf 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Inc/usbd_video_if_template.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Inc/usbd_video_if_template.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2020 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2020 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -114,6 +113,25 @@ void TransferComplete_CallBack_FS(void); */ void HalfTransfer_CallBack_FS(void); + + +#define IMG_NBR 1U +#define IMAGE_SIZE 0x1U + +const uint8_t image[] = {0x00}; +const uint8_t *tImagesList[] = {image}; +uint16_t tImagesSizes[] = {IMAGE_SIZE}; + +/* Time laps between video frames in ms. + Please adjust this value depending on required speed. + Please note that this define uses the system HAL_Delay() which uses the systick. + In case of changes on HAL_Delay, please ensure the values in ms correspond. */ +#ifdef USE_USB_HS +#define USBD_VIDEO_IMAGE_LAPS 160U +#else +#define USBD_VIDEO_IMAGE_LAPS 80U +#endif /* USE_USB_HS */ + /* USER CODE BEGIN EXPORTED_FUNCTIONS */ /* USER CODE END EXPORTED_FUNCTIONS */ @@ -136,7 +154,3 @@ void HalfTransfer_CallBack_FS(void); #endif /* USBD_VIDEO_IF_H_ */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ - - - diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Src/usbd_video.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Src/usbd_video.c index a414ed6f..a2cb571d 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Src/usbd_video.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Src/usbd_video.c @@ -4,6 +4,17 @@ * @author MCD Application Team * @brief This file provides the Video core functions. * + ****************************************************************************** + * @attention + * + * Copyright (c) 2020 STMicroelectronics. + * All rights reserved. + * + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. + * + ****************************************************************************** * @verbatim * * =================================================================== @@ -35,17 +46,6 @@ * @endverbatim * ****************************************************************************** - * @attention - * - * <h2><center>© Copyright (c) 2020 STMicroelectronics. - * All rights reserved.</center></h2> - * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** */ /* Includes ------------------------------------------------------------------*/ @@ -87,10 +87,14 @@ static uint8_t USBD_VIDEO_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx); static uint8_t USBD_VIDEO_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx); static uint8_t USBD_VIDEO_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); + +#ifndef USE_USBD_COMPOSITE static uint8_t *USBD_VIDEO_GetFSCfgDesc(uint16_t *length); static uint8_t *USBD_VIDEO_GetHSCfgDesc(uint16_t *length); static uint8_t *USBD_VIDEO_GetOtherSpeedCfgDesc(uint16_t *length); static uint8_t *USBD_VIDEO_GetDeviceQualifierDesc(uint16_t *length); +#endif /* USE_USBD_COMPOSITE */ + static uint8_t USBD_VIDEO_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum); static uint8_t USBD_VIDEO_SOF(USBD_HandleTypeDef *pdev); static uint8_t USBD_VIDEO_IsoINIncomplete(USBD_HandleTypeDef *pdev, uint8_t epnum); @@ -99,10 +103,11 @@ static uint8_t USBD_VIDEO_IsoINIncomplete(USBD_HandleTypeDef *pdev, uint8_t epnu static void VIDEO_REQ_GetCurrent(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); static void VIDEO_REQ_SetCurrent(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req); -static USBD_VIDEO_DescHeader_t *USBD_VIDEO_GetNextDesc(uint8_t *pbuf, uint16_t *ptr); -static void *USBD_VIDEO_GetEpDesc(uint8_t *pConfDesc, uint8_t EpAddr); +#ifndef USE_USBD_COMPOSITE static void *USBD_VIDEO_GetVSFrameDesc(uint8_t *pConfDesc); +#endif /* USE_USBD_COMPOSITE */ +static void *USBD_VIDEO_GetVideoHeaderDesc(uint8_t *pConfDesc); /** * @} @@ -124,10 +129,17 @@ USBD_ClassTypeDef USBD_VIDEO = USBD_VIDEO_SOF, USBD_VIDEO_IsoINIncomplete, NULL, +#ifdef USE_USBD_COMPOSITE + NULL, + NULL, + NULL, + NULL, +#else USBD_VIDEO_GetHSCfgDesc, USBD_VIDEO_GetFSCfgDesc, USBD_VIDEO_GetOtherSpeedCfgDesc, USBD_VIDEO_GetDeviceQualifierDesc, +#endif /* USE_USBD_COMPOSITE */ }; /* USB VIDEO device Configuration Descriptor (same for all speeds thanks to user defines) */ @@ -140,12 +152,13 @@ __ALIGN_BEGIN static uint8_t USBD_VIDEO_CfgDesc[] __ALIGN_END = HIBYTE(UVC_CONFIG_DESC_SIZ), 0x02, /* bNumInterfaces: 2 interfaces */ 0x01, /* bConfigurationValue: Configuration value */ - 0x00, /* iConfiguration: Index of string descriptor describing the configuration */ + 0x00, /* iConfiguration: Index of string descriptor + describing the configuration */ #if (USBD_SELF_POWERED == 1U) 0xC0, /* bmAttributes: Bus Powered according to user configuration */ #else 0x80, /* bmAttributes: Bus Powered according to user configuration */ -#endif +#endif /* USBD_SELF_POWERED */ USBD_MAX_POWER, /* bMaxPower in mA according to user configuration */ /* Interface Association Descriptor */ @@ -182,8 +195,8 @@ __ALIGN_BEGIN static uint8_t USBD_VIDEO_CfgDesc[] __ALIGN_END = 0xDC, 0x02, 0x01, /* bInCollection: number of streaming interfaces */ - 0x01, /* baInterfaceNr(1): VideoStreaming interface 1 is part of VC interface */ - + 0x01, /* baInterfaceNr(1): VideoStreaming interface 1 is part + of VC interface */ /* Input Terminal Descriptor */ VIDEO_IN_TERMINAL_DESC_SIZE, /* bLength: Input terminal descriptor size */ CS_INTERFACE, /* bDescriptorType: INTERFACE */ @@ -248,8 +261,9 @@ __ALIGN_BEGIN static uint8_t USBD_VIDEO_CfgDesc[] __ALIGN_END = UVC_BITS_PER_PIXEL, /* bBitsPerPixel : Number of bits per pixel */ #else 0x01, /* bmFlags: FixedSizeSamples */ -#endif - 0x01, /* bDefaultFrameIndex: default frame used is frame 1 (only one frame used) */ +#endif /* USBD_UVC_FORMAT_UNCOMPRESSED */ + 0x01, /* bDefaultFrameIndex: default frame used is frame 1 + (only one frame used) */ 0x00, /* bAspectRatioX: not required by specification */ 0x00, /* bAspectRatioY: not required by specification */ 0x00, /* bInterlaceFlags: non interlaced stream */ @@ -264,7 +278,7 @@ __ALIGN_BEGIN static uint8_t USBD_VIDEO_CfgDesc[] __ALIGN_END = 0x00, /* bmCapabilities: no till image capture */ #else 0x02, /* bmCapabilities: fixed frame rate supported */ -#endif +#endif /* USBD_UVC_FORMAT_UNCOMPRESSED */ WBVAL(UVC_WIDTH), /* wWidth: Image Frame Width */ WBVAL(UVC_HEIGHT), /* wHeight: Image Frame Height */ DBVAL(UVC_MIN_BIT_RATE(UVC_CAM_FPS_FS)), /* dwMinBitRate: Minimum supported bit rate in bits/s */ @@ -282,7 +296,7 @@ __ALIGN_BEGIN static uint8_t USBD_VIDEO_CfgDesc[] __ALIGN_END = UVC_COLOR_PRIMARIE, /* bColorPrimarie: 1: BT.709, sRGB (default) */ UVC_TFR_CHARACTERISTICS, /* bTransferCharacteristics: 1: BT.709 (default) */ UVC_MATRIX_COEFFICIENTS, /* bMatrixCoefficients: 4: BT.601, (default) */ -#endif +#endif /* USBD_UVC_FORMAT_UNCOMPRESSED */ /* Standard VS Interface Descriptor = interface 1, alternate setting 1 = data transfer mode */ USB_IF_DESC_SIZE, /* bLength */ @@ -305,6 +319,7 @@ __ALIGN_BEGIN static uint8_t USBD_VIDEO_CfgDesc[] __ALIGN_END = 0x01, /* bInterval: 1 frame interval */ }; +#ifndef USE_USBD_COMPOSITE /* USB Standard Device Descriptor */ __ALIGN_BEGIN static uint8_t USBD_VIDEO_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIER_DESC] __ALIGN_END = { @@ -319,6 +334,9 @@ __ALIGN_BEGIN static uint8_t USBD_VIDEO_DeviceQualifierDesc[USB_LEN_DEV_QUALIFIE 0x01, 0x00, }; +#endif /* USE_USBD_COMPOSITE */ + +static uint8_t VIDEOinEpAdd = UVC_IN_EP; /* Video Commit data structure */ static USBD_VideoControlTypeDef video_Commit_Control = @@ -377,7 +395,7 @@ static USBD_VideoControlTypeDef video_Probe_Control = * @param cfgidx: Configuration index * @retval status */ -static uint8_t USBD_VIDEO_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) +static uint8_t USBD_VIDEO_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) { USBD_VIDEO_HandleTypeDef *hVIDEO; @@ -391,26 +409,32 @@ static uint8_t USBD_VIDEO_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) } /* Assign the pClassData pointer to the allocated structure */ - pdev->pClassData = (void *)hVIDEO; + pdev->pClassDataCmsit[pdev->classId] = (void *)hVIDEO; + pdev->pClassData = pdev->pClassDataCmsit[pdev->classId]; + +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + VIDEOinEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_ISOC, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ /* Open EP IN */ if (pdev->dev_speed == USBD_SPEED_HIGH) { - (void)USBD_LL_OpenEP(pdev, UVC_IN_EP, USBD_EP_TYPE_ISOC, UVC_ISO_HS_MPS); + (void)USBD_LL_OpenEP(pdev, VIDEOinEpAdd, USBD_EP_TYPE_ISOC, UVC_ISO_HS_MPS); - pdev->ep_in[UVC_IN_EP & 0xFU].is_used = 1U; - pdev->ep_in[UVC_IN_EP & 0xFU].maxpacket = UVC_ISO_HS_MPS; + pdev->ep_in[VIDEOinEpAdd & 0xFU].is_used = 1U; + pdev->ep_in[VIDEOinEpAdd & 0xFU].maxpacket = UVC_ISO_HS_MPS; } else { - (void)USBD_LL_OpenEP(pdev, UVC_IN_EP, USBD_EP_TYPE_ISOC, UVC_ISO_FS_MPS); + (void)USBD_LL_OpenEP(pdev, VIDEOinEpAdd, USBD_EP_TYPE_ISOC, UVC_ISO_FS_MPS); - pdev->ep_in[UVC_IN_EP & 0xFU].is_used = 1U; - pdev->ep_in[UVC_IN_EP & 0xFU].maxpacket = UVC_ISO_FS_MPS; + pdev->ep_in[VIDEOinEpAdd & 0xFU].is_used = 1U; + pdev->ep_in[VIDEOinEpAdd & 0xFU].maxpacket = UVC_ISO_FS_MPS; } /* Init physical Interface components */ - ((USBD_VIDEO_ItfTypeDef *)pdev->pUserData)->Init(); + ((USBD_VIDEO_ItfTypeDef *)pdev->pUserData[pdev->classId])->Init(); /* Init Xfer states */ hVIDEO->interface = 0U; @@ -430,23 +454,29 @@ static uint8_t USBD_VIDEO_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx) * @param cfgidx: Configuration index * @retval status */ -static uint8_t USBD_VIDEO_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) +static uint8_t USBD_VIDEO_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) { UNUSED(cfgidx); /* Check if the video structure pointer is valid */ - if (pdev->pClassData == NULL) + if (pdev->pClassDataCmsit[pdev->classId] == NULL) { return (uint8_t)USBD_FAIL; } +#ifdef USE_USBD_COMPOSITE + /* Get the Endpoints addresses allocated for this class instance */ + VIDEOinEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_ISOC, (uint8_t)pdev->classId); +#endif /* USE_USBD_COMPOSITE */ + /* Close EP IN */ - (void)USBD_LL_CloseEP(pdev, UVC_IN_EP); - pdev->ep_in[UVC_IN_EP & 0xFU].is_used = 0U; + (void)USBD_LL_CloseEP(pdev, VIDEOinEpAdd); + pdev->ep_in[VIDEOinEpAdd & 0xFU].is_used = 0U; /* DeInit physical Interface components */ - ((USBD_VIDEO_ItfTypeDef *)pdev->pUserData)->DeInit(); - USBD_free(pdev->pClassData); + ((USBD_VIDEO_ItfTypeDef *)pdev->pUserData[pdev->classId])->DeInit(); + USBD_free(pdev->pClassDataCmsit[pdev->classId]); + pdev->pClassDataCmsit[pdev->classId] = NULL; pdev->pClassData = NULL; /* Exit with no error code */ @@ -460,13 +490,13 @@ static uint8_t USBD_VIDEO_DeInit(USBD_HandleTypeDef *pdev, uint8_t cfgidx) * @param req: usb requests * @retval status */ -static uint8_t USBD_VIDEO_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) +static uint8_t USBD_VIDEO_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_VIDEO_HandleTypeDef *hVIDEO = (USBD_VIDEO_HandleTypeDef *) pdev->pClassData; + USBD_VIDEO_HandleTypeDef *hVIDEO = (USBD_VIDEO_HandleTypeDef *) pdev->pClassDataCmsit[pdev->classId]; uint8_t ret = (uint8_t)USBD_OK; - uint16_t len = 0U; - uint8_t *pbuf = NULL; uint16_t status_info = 0U; + uint16_t len; + uint8_t *pbuf; switch (req->bmRequest & USB_REQ_TYPE_MASK) { @@ -480,13 +510,16 @@ static uint8_t USBD_VIDEO_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef case UVC_GET_MAX: VIDEO_REQ_GetCurrent(pdev, req); break; + case UVC_GET_RES: case UVC_GET_LEN: case UVC_GET_INFO: break; + case UVC_SET_CUR: VIDEO_REQ_SetCurrent(pdev, req); break; + default: (void) USBD_CtlError(pdev, req); ret = (uint8_t)USBD_FAIL; @@ -513,10 +546,18 @@ static uint8_t USBD_VIDEO_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef case USB_REQ_GET_DESCRIPTOR: if ((req->wValue >> 8) == CS_DEVICE) { - pbuf = USBD_VIDEO_CfgDesc + 18; - len = MIN((uint16_t)USB_CONF_DESC_SIZE, (uint16_t)req->wLength); + pbuf = (uint8_t *)USBD_VIDEO_GetVideoHeaderDesc(pdev->pConfDesc); + if (pbuf != NULL) + { + len = MIN((uint16_t)USB_CONF_DESC_SIZE, (uint16_t)req->wLength); + (void)USBD_CtlSendData(pdev, pbuf, len); + } + else + { + USBD_CtlError(pdev, req); + ret = (uint8_t)USBD_FAIL; + } } - (void)USBD_CtlSendData(pdev, pbuf, len); break; case USB_REQ_GET_INTERFACE : @@ -540,14 +581,14 @@ static uint8_t USBD_VIDEO_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef if (hVIDEO->interface == 1U) { /* Start Streaming (First endpoint writing will be done on next SOF) */ - (void)USBD_LL_FlushEP(pdev, UVC_IN_EP); + (void)USBD_LL_FlushEP(pdev, VIDEOinEpAdd); hVIDEO->uvc_state = UVC_PLAY_STATUS_READY; } else { /* Stop Streaming */ hVIDEO->uvc_state = UVC_PLAY_STATUS_STOP; - (void)USBD_LL_FlushEP(pdev, UVC_IN_EP); + (void)USBD_LL_FlushEP(pdev, VIDEOinEpAdd); } } else @@ -592,20 +633,21 @@ static uint8_t USBD_VIDEO_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef */ static uint8_t USBD_VIDEO_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) { - USBD_VIDEO_HandleTypeDef *hVIDEO = (USBD_VIDEO_HandleTypeDef *) pdev->pClassData; + USBD_VIDEO_HandleTypeDef *hVIDEO = (USBD_VIDEO_HandleTypeDef *) pdev->pClassDataCmsit[pdev->classId]; static uint8_t packet[UVC_PACKET_SIZE + (UVC_HEADER_PACKET_CNT * 2U)] = {0x00U}; static uint8_t *Pcktdata = packet; static uint16_t PcktIdx = 0U; static uint16_t PcktSze = UVC_PACKET_SIZE; static uint8_t payload_header[2] = {0x02U, 0x00U}; uint8_t i = 0U; - uint32_t RemainData, DataOffset = 0U; + uint32_t RemainData = 0U; + uint32_t DataOffset = 0U; /* Check if the Streaming has already been started */ if (hVIDEO->uvc_state == UVC_PLAY_STATUS_STREAMING) { /* Get the current packet buffer, index and size from the application layer */ - ((USBD_VIDEO_ItfTypeDef *)pdev->pUserData)->Data(&Pcktdata, &PcktSze, &PcktIdx); + ((USBD_VIDEO_ItfTypeDef *)pdev->pUserData[pdev->classId])->Data(&Pcktdata, &PcktSze, &PcktIdx); /* Check if end of current image has been reached */ if (PcktSze > 2U) @@ -624,9 +666,10 @@ static uint8_t USBD_VIDEO_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) { packet[((DataOffset + 0U) * i)] = payload_header[0]; packet[((DataOffset + 0U) * i) + 1U] = payload_header[1]; - if (RemainData > pdev->ep_in[UVC_IN_EP & 0xFU].maxpacket) + + if (RemainData > pdev->ep_in[VIDEOinEpAdd & 0xFU].maxpacket) { - DataOffset = pdev->ep_in[UVC_IN_EP & 0xFU].maxpacket; + DataOffset = pdev->ep_in[VIDEOinEpAdd & 0xFU].maxpacket; (void)USBD_memcpy((packet + ((DataOffset + 0U) * i) + 2U), Pcktdata + ((DataOffset - 2U) * i), (DataOffset - 2U)); @@ -666,14 +709,14 @@ static uint8_t USBD_VIDEO_DataIn(USBD_HandleTypeDef *pdev, uint8_t epnum) */ static uint8_t USBD_VIDEO_SOF(USBD_HandleTypeDef *pdev) { - USBD_VIDEO_HandleTypeDef *hVIDEO = (USBD_VIDEO_HandleTypeDef *) pdev->pClassData; + USBD_VIDEO_HandleTypeDef *hVIDEO = (USBD_VIDEO_HandleTypeDef *) pdev->pClassDataCmsit[pdev->classId]; uint8_t payload[2] = {0x02U, 0x00U}; /* Check if the Streaming has already been started by SetInterface AltSetting 1 */ if (hVIDEO->uvc_state == UVC_PLAY_STATUS_READY) { /* Transmit the first packet indicating that Streaming is starting */ - (void)USBD_LL_Transmit(pdev, UVC_IN_EP, (uint8_t *)payload, 2U); + (void)USBD_LL_Transmit(pdev, VIDEOinEpAdd, (uint8_t *)payload, 2U); /* Enable Streaming state */ hVIDEO->uvc_state = UVC_PLAY_STATUS_STREAMING; @@ -708,7 +751,7 @@ static uint8_t USBD_VIDEO_IsoINIncomplete(USBD_HandleTypeDef *pdev, uint8_t epnu static void VIDEO_REQ_GetCurrent(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { USBD_VIDEO_HandleTypeDef *hVIDEO; - hVIDEO = (USBD_VIDEO_HandleTypeDef *)(pdev->pClassData); + hVIDEO = (USBD_VIDEO_HandleTypeDef *)(pdev->pClassDataCmsit[pdev->classId]); static __IO uint8_t EntityStatus[8] = {0}; /* Reset buffer to zeros */ @@ -757,7 +800,7 @@ static void VIDEO_REQ_GetCurrent(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef /* Probe Request */ (void)USBD_CtlSendData(pdev, (uint8_t *)&video_Probe_Control, - MIN(req->wLength, sizeof(USBD_VideoControlTypeDef))); + MIN(req->wLength, sizeof(USBD_VideoControlTypeDef))); } else if (LOBYTE(req->wValue) == (uint8_t)VS_COMMIT_CONTROL) { @@ -774,7 +817,7 @@ static void VIDEO_REQ_GetCurrent(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef /* Commit Request */ (void)USBD_CtlSendData(pdev, (uint8_t *)&video_Commit_Control, - MIN(req->wLength, sizeof(USBD_VideoControlTypeDef))); + MIN(req->wLength, sizeof(USBD_VideoControlTypeDef))); } else { @@ -794,19 +837,18 @@ static void VIDEO_REQ_GetCurrent(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef */ static void VIDEO_REQ_SetCurrent(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { - USBD_VIDEO_HandleTypeDef *hVIDEO = (USBD_VIDEO_HandleTypeDef *)(pdev->pClassData); /* Check that the request has control data */ if (req->wLength > 0U) { /* Prepare the reception of the buffer over EP0 */ - if (LOBYTE(req->wValue) == (uint8_t)VS_PROBE_CONTROL) + if (req->wValue == (uint16_t)VS_PROBE_CONTROL) { /* Probe Request */ (void) USBD_CtlPrepareRx(pdev, (uint8_t *)&video_Probe_Control, MIN(req->wLength, sizeof(USBD_VideoControlTypeDef))); } - else if (LOBYTE(req->wValue) == (uint8_t)VS_COMMIT_CONTROL) + else if (req->wValue == (uint16_t)VS_COMMIT_CONTROL) { /* Commit Request */ (void) USBD_CtlPrepareRx(pdev, (uint8_t *)&video_Commit_Control, @@ -814,13 +856,12 @@ static void VIDEO_REQ_SetCurrent(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef } else { - /* Prepare the reception of the buffer over EP0 */ - (void) USBD_CtlPrepareRx(pdev, hVIDEO->control.data, - MIN(req->wLength, USB_MAX_EP0_SIZE)); + (void)USBD_LL_StallEP(pdev, 0x80U); } } } +#ifndef USE_USBD_COMPOSITE /** * @brief USBD_VIDEO_GetFSCfgDesc * return configuration descriptor @@ -829,7 +870,7 @@ static void VIDEO_REQ_SetCurrent(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef */ static uint8_t *USBD_VIDEO_GetFSCfgDesc(uint16_t *length) { - USBD_EpDescTypedef *pEpDesc = USBD_VIDEO_GetEpDesc(USBD_VIDEO_CfgDesc, UVC_IN_EP); + USBD_EpDescTypeDef *pEpDesc = USBD_GetEpDesc(USBD_VIDEO_CfgDesc, VIDEOinEpAdd); USBD_VIDEO_VSFrameDescTypeDef *pVSFrameDesc = USBD_VIDEO_GetVSFrameDesc(USBD_VIDEO_CfgDesc); if (pEpDesc != NULL) @@ -857,7 +898,7 @@ static uint8_t *USBD_VIDEO_GetFSCfgDesc(uint16_t *length) */ static uint8_t *USBD_VIDEO_GetHSCfgDesc(uint16_t *length) { - USBD_EpDescTypedef *pEpDesc = USBD_VIDEO_GetEpDesc(USBD_VIDEO_CfgDesc, UVC_IN_EP); + USBD_EpDescTypeDef *pEpDesc = USBD_GetEpDesc(USBD_VIDEO_CfgDesc, VIDEOinEpAdd); USBD_VIDEO_VSFrameDescTypeDef *pVSFrameDesc = USBD_VIDEO_GetVSFrameDesc(USBD_VIDEO_CfgDesc); if (pEpDesc != NULL) @@ -885,7 +926,7 @@ static uint8_t *USBD_VIDEO_GetHSCfgDesc(uint16_t *length) */ static uint8_t *USBD_VIDEO_GetOtherSpeedCfgDesc(uint16_t *length) { - USBD_EpDescTypedef *pEpDesc = USBD_VIDEO_GetEpDesc(USBD_VIDEO_CfgDesc, UVC_IN_EP); + USBD_EpDescTypeDef *pEpDesc = USBD_GetEpDesc(USBD_VIDEO_CfgDesc, VIDEOinEpAdd); USBD_VIDEO_VSFrameDescTypeDef *pVSFrameDesc = USBD_VIDEO_GetVSFrameDesc(USBD_VIDEO_CfgDesc); if (pEpDesc != NULL) @@ -917,23 +958,6 @@ static uint8_t *USBD_VIDEO_GetDeviceQualifierDesc(uint16_t *length) return USBD_VIDEO_DeviceQualifierDesc; } -/** - * @brief USBD_VIDEO_GetNextDesc - * This function return the next descriptor header - * @param buf: Buffer where the descriptor is available - * @param ptr: data pointer inside the descriptor - * @retval next header - */ -static USBD_VIDEO_DescHeader_t *USBD_VIDEO_GetNextDesc(uint8_t *pbuf, uint16_t *ptr) -{ - USBD_VIDEO_DescHeader_t *pnext = (USBD_VIDEO_DescHeader_t *)(void *)pbuf; - - *ptr += pnext->bLength; - pnext = (USBD_VIDEO_DescHeader_t *)(void *)(pbuf + pnext->bLength); - - return (pnext); -} - /** * @brief USBD_VIDEO_GetVSFrameDesc * This function return the Video Endpoint descriptor @@ -943,8 +967,8 @@ static USBD_VIDEO_DescHeader_t *USBD_VIDEO_GetNextDesc(uint8_t *pbuf, uint16_t * */ static void *USBD_VIDEO_GetVSFrameDesc(uint8_t *pConfDesc) { - USBD_VIDEO_DescHeader_t *pdesc = (USBD_VIDEO_DescHeader_t *)(void *)pConfDesc; - USBD_ConfigDescTypedef *desc = (USBD_ConfigDescTypedef *)(void *)pConfDesc; + USBD_DescHeaderTypeDef *pdesc = (USBD_DescHeaderTypeDef *)(void *)pConfDesc; + USBD_ConfigDescTypeDef *desc = (USBD_ConfigDescTypeDef *)(void *)pConfDesc; USBD_VIDEO_VSFrameDescTypeDef *pVSFrameDesc = NULL; uint16_t ptr; @@ -954,10 +978,10 @@ static void *USBD_VIDEO_GetVSFrameDesc(uint8_t *pConfDesc) while (ptr < desc->wTotalLength) { - pdesc = USBD_VIDEO_GetNextDesc((uint8_t *)pdesc, &ptr); + pdesc = USBD_GetNextDesc((uint8_t *)pdesc, &ptr); if (((pdesc->bDescriptorSubType == VS_FRAME_MJPEG) || - (pdesc->bDescriptorSubType == VS_FRAME_UNCOMPRESSED)) && + (pdesc->bDescriptorSubType == VS_FRAME_UNCOMPRESSED)) && (pdesc->bLength == VS_FRAME_DESC_SIZE)) { pVSFrameDesc = (USBD_VIDEO_VSFrameDescTypeDef *)(void *)pdesc; @@ -968,20 +992,20 @@ static void *USBD_VIDEO_GetVSFrameDesc(uint8_t *pConfDesc) return (void *)pVSFrameDesc; } +#endif /* USE_USBD_COMPOSITE */ /** - * @brief USBD_VIDEO_GetEpDesc - * This function return the Video Endpoint descriptor + * @brief USBD_VIDEO_GetVideoHeaderDesc + * This function return the Video Header descriptor * @param pdev: device instance * @param pConfDesc: pointer to Bos descriptor - * @param EpAddr: endpoint address - * @retval pointer to video endpoint descriptor + * @retval pointer to the Video Header descriptor */ -static void *USBD_VIDEO_GetEpDesc(uint8_t *pConfDesc, uint8_t EpAddr) +static void *USBD_VIDEO_GetVideoHeaderDesc(uint8_t *pConfDesc) { - USBD_VIDEO_DescHeader_t *pdesc = (USBD_VIDEO_DescHeader_t *)(void *)pConfDesc; - USBD_ConfigDescTypedef *desc = (USBD_ConfigDescTypedef *)(void *)pConfDesc; - USBD_EpDescTypedef *pEpDesc = NULL; + USBD_ConfigDescTypeDef *desc = (USBD_ConfigDescTypeDef *)(void *)pConfDesc; + USBD_DescHeaderTypeDef *pdesc = (USBD_DescHeaderTypeDef *)(void *)pConfDesc; + uint8_t *pVideoDesc = NULL; uint16_t ptr; if (desc->wTotalLength > desc->bLength) @@ -990,25 +1014,16 @@ static void *USBD_VIDEO_GetEpDesc(uint8_t *pConfDesc, uint8_t EpAddr) while (ptr < desc->wTotalLength) { - pdesc = USBD_VIDEO_GetNextDesc((uint8_t *)pdesc, &ptr); - - if (pdesc->bDescriptorType == USB_DESC_TYPE_ENDPOINT) + pdesc = USBD_GetNextDesc((uint8_t *)pdesc, &ptr); + if ((pdesc->bDescriptorType == CS_INTERFACE) && + (pdesc->bDescriptorSubType == VC_HEADER)) { - pEpDesc = (USBD_EpDescTypedef *)(void *)pdesc; - - if (pEpDesc->bEndpointAddress == EpAddr) - { - break; - } - else - { - pEpDesc = NULL; - } + pVideoDesc = (uint8_t *)pdesc; + break; } } } - - return (void *)pEpDesc; + return pVideoDesc; } /** @@ -1026,7 +1041,7 @@ uint8_t USBD_VIDEO_RegisterInterface(USBD_HandleTypeDef *pdev, USBD_VIDEO_ItfT } /* Assign the FOPS pointer */ - pdev->pUserData = fops; + pdev->pUserData[pdev->classId] = fops; /* Exit with no error code */ return (uint8_t)USBD_OK; @@ -1045,7 +1060,3 @@ uint8_t USBD_VIDEO_RegisterInterface(USBD_HandleTypeDef *pdev, USBD_VIDEO_ItfT /** * @} */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ - - diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Src/usbd_video_if_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Src/usbd_video_if_template.c index 70b8d3aa..5b6f2aeb 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Src/usbd_video_if_template.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Class/VIDEO/Src/usbd_video_if_template.c @@ -6,19 +6,18 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2020 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2020 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ /* Includes ------------------------------------------------------------------*/ -#include "usbd_video_if.h" +#include "usbd_video_if_template.h" /* Include you image binary file here Binary image template shall provide: @@ -41,7 +40,7 @@ /* USER CODE BEGIN PV */ /* Private variables ---------------------------------------------------------*/ - +uint8_t img_count; /* USER CODE END PV */ /** @addtogroup STM32_USB_OTG_DEVICE_LIBRARY @@ -189,6 +188,9 @@ static int8_t VIDEO_Itf_DeInit(void) */ static int8_t VIDEO_Itf_Control(uint8_t cmd, uint8_t *pbuf, uint16_t length) { + UNUSED(cmd); + UNUSED(pbuf); + UNUSED(length); return (0); } @@ -221,9 +223,9 @@ static int8_t VIDEO_Itf_Data(uint8_t **pbuf, uint16_t *psize, uint16_t *pcktidx) */ const uint8_t *(*ImagePtr) = tImagesList; - uint32_t packet_count = (tImagesSizes[img_count]) / ((uint16_t)(UVC_PACKET_SIZE - (UVC_HEADER_PACKET_CNT * 2U))); - uint32_t packet_remainder = (tImagesSizes[img_count]) % ((uint16_t)(UVC_PACKET_SIZE - (UVC_HEADER_PACKET_CNT * 2U))); - static uint8_t packet_index = 0; + uint32_t packet_count = (tImagesSizes[img_count]) / ((UVC_PACKET_SIZE - (UVC_HEADER_PACKET_CNT * 2U))); + uint32_t packet_remainder = (tImagesSizes[img_count]) % ((UVC_PACKET_SIZE - (UVC_HEADER_PACKET_CNT * 2U))); + static uint8_t packet_index = 0U; /* Check if end of current image has been reached */ if (packet_index < packet_count) @@ -232,17 +234,19 @@ static int8_t VIDEO_Itf_Data(uint8_t **pbuf, uint16_t *psize, uint16_t *pcktidx) *psize = (uint16_t)UVC_PACKET_SIZE; /* Get the pointer to the next packet to be transmitted */ - *pbuf = (uint8_t *)(*(ImagePtr + img_count) + packet_index * ((uint16_t)(UVC_PACKET_SIZE - (UVC_HEADER_PACKET_CNT * 2U)))); + *pbuf = (uint8_t *)(*(ImagePtr + img_count) + \ + (packet_index * ((uint16_t)(UVC_PACKET_SIZE - (UVC_HEADER_PACKET_CNT * 2U))))); } else if ((packet_index == packet_count)) { if (packet_remainder != 0U) { /* Get the pointer to the next packet to be transmitted */ - *pbuf = (uint8_t *)(*(ImagePtr + img_count) + packet_index * ((uint16_t)(UVC_PACKET_SIZE - (UVC_HEADER_PACKET_CNT * 2U)))); + *pbuf = (uint8_t *)(*(ImagePtr + img_count) + \ + (packet_index * ((uint16_t)(UVC_PACKET_SIZE - (UVC_HEADER_PACKET_CNT * 2U))))); /* Set the current packet size */ - *psize = packet_remainder + (UVC_HEADER_PACKET_CNT * 2U); + *psize = (uint16_t)(packet_remainder + (UVC_HEADER_PACKET_CNT * 2U)); } else { @@ -262,7 +266,7 @@ static int8_t VIDEO_Itf_Data(uint8_t **pbuf, uint16_t *psize, uint16_t *pcktidx) *pcktidx = packet_index; /* Increment the packet count and check if it reached the end of current image buffer */ - if (packet_index++ >= (packet_count + 1)) + if (packet_index++ >= (packet_count + 1U)) { /* Reset the packet count to zero */ packet_index = 0U; @@ -270,7 +274,7 @@ static int8_t VIDEO_Itf_Data(uint8_t **pbuf, uint16_t *psize, uint16_t *pcktidx) /* Move to the next image in the images table */ img_count++; - HAL_Delay(USBD_VIDEO_IMAGE_LAPS); + USBD_Delay(USBD_VIDEO_IMAGE_LAPS); /* Check if images count has been reached, then reset to zero (go back to first image in circular loop) */ if (img_count == IMG_NBR) { @@ -292,5 +296,3 @@ static int8_t VIDEO_Itf_Data(uint8_t **pbuf, uint16_t *psize, uint16_t *pcktidx) /** * @} */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_conf_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_conf_template.h index 0527cf27..b288b66c 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_conf_template.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_conf_template.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -77,6 +76,11 @@ extern "C" { #define USBD_CUSTOMHID_OUTREPORT_BUF_SIZE 0x02U #define USBD_CUSTOM_HID_REPORT_DESC_SIZE 163U +/* #define USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED */ +/* #define USBD_CUSTOMHID_OUT_PREPARE_RECEIVE_DISABLED */ +/* #define USBD_CUSTOMHID_EP0_OUT_PREPARE_RECEIVE_DISABLED */ +/* #define USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED */ + /* VIDEO Class Config */ #define UVC_1_1 /* #define UVC_1_0 */ @@ -91,7 +95,7 @@ extern "C" { #define UVC_COLOR_PRIMARIE 0x01U #define UVC_TFR_CHARACTERISTICS 0x01U #define UVC_MATRIX_COEFFICIENTS 0x04U -#endif +#endif /* USBD_UVC_FORMAT_UNCOMPRESSED */ /* Video Stream frame width and height */ #define UVC_WIDTH 176U @@ -141,7 +145,7 @@ extern "C" { } while (0) #else #define USBD_UsrLog(...) do {} while (0) -#endif +#endif /* (USBD_DEBUG_LEVEL > 0U) */ #if (USBD_DEBUG_LEVEL > 1U) @@ -152,7 +156,7 @@ extern "C" { } while (0) #else #define USBD_ErrLog(...) do {} while (0) -#endif +#endif /* (USBD_DEBUG_LEVEL > 1U) */ #if (USBD_DEBUG_LEVEL > 2U) #define USBD_DbgLog(...) do { \ @@ -162,7 +166,7 @@ extern "C" { } while (0) #else #define USBD_DbgLog(...) do {} while (0) -#endif +#endif /* (USBD_DEBUG_LEVEL > 2U) */ /** * @} @@ -221,4 +225,3 @@ void USBD_static_free(void *p); /** * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_core.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_core.h index 92f72705..d6016724 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_core.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_core.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -88,6 +87,17 @@ USBD_StatusTypeDef USBD_Start(USBD_HandleTypeDef *pdev); USBD_StatusTypeDef USBD_Stop(USBD_HandleTypeDef *pdev); USBD_StatusTypeDef USBD_RegisterClass(USBD_HandleTypeDef *pdev, USBD_ClassTypeDef *pclass); +#ifdef USE_USBD_COMPOSITE +USBD_StatusTypeDef USBD_RegisterClassComposite(USBD_HandleTypeDef *pdev, USBD_ClassTypeDef *pclass, + USBD_CompositeClassTypeDef classtype, uint8_t *EpAddr); + +USBD_StatusTypeDef USBD_UnRegisterClassComposite(USBD_HandleTypeDef *pdev); +uint8_t USBD_CoreGetEPAdd(USBD_HandleTypeDef *pdev, uint8_t ep_dir, uint8_t ep_type, uint8_t ClassId); +#endif /* USE_USBD_COMPOSITE */ + +uint8_t USBD_CoreFindIF(USBD_HandleTypeDef *pdev, uint8_t index); +uint8_t USBD_CoreFindEP(USBD_HandleTypeDef *pdev, uint8_t index); + USBD_StatusTypeDef USBD_RunTestMode(USBD_HandleTypeDef *pdev); USBD_StatusTypeDef USBD_SetClassConfig(USBD_HandleTypeDef *pdev, uint8_t cfgidx); USBD_StatusTypeDef USBD_ClrClassConfig(USBD_HandleTypeDef *pdev, uint8_t cfgidx); @@ -129,11 +139,18 @@ USBD_StatusTypeDef USBD_LL_Transmit(USBD_HandleTypeDef *pdev, uint8_t ep_addr, USBD_StatusTypeDef USBD_LL_PrepareReceive(USBD_HandleTypeDef *pdev, uint8_t ep_addr, uint8_t *pbuf, uint32_t size); +#ifdef USBD_HS_TESTMODE_ENABLE +USBD_StatusTypeDef USBD_LL_SetTestMode(USBD_HandleTypeDef *pdev, uint8_t testmode); +#endif /* USBD_HS_TESTMODE_ENABLE */ + uint8_t USBD_LL_IsStallEP(USBD_HandleTypeDef *pdev, uint8_t ep_addr); uint32_t USBD_LL_GetRxDataSize(USBD_HandleTypeDef *pdev, uint8_t ep_addr); void USBD_LL_Delay(uint32_t Delay); +void *USBD_GetEpDesc(uint8_t *pConfDesc, uint8_t EpAddr); +USBD_DescHeaderTypeDef *USBD_GetNextDesc(uint8_t *pbuf, uint16_t *ptr); + /** * @} */ @@ -152,7 +169,4 @@ void USBD_LL_Delay(uint32_t Delay); * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ - - diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_ctlreq.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_ctlreq.h index 3495a978..6c45d6ce 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_ctlreq.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_ctlreq.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -100,4 +99,3 @@ void USBD_GetString(uint8_t *desc, uint8_t *unicode, uint16_t *len); */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_def.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_def.h index 4c4de689..89468195 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_def.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_def.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -53,6 +52,24 @@ extern "C" { #define USBD_MAX_NUM_CONFIGURATION 1U #endif /* USBD_MAX_NUM_CONFIGURATION */ +#ifdef USE_USBD_COMPOSITE +#ifndef USBD_MAX_SUPPORTED_CLASS +#define USBD_MAX_SUPPORTED_CLASS 4U +#endif /* USBD_MAX_SUPPORTED_CLASS */ +#else +#ifndef USBD_MAX_SUPPORTED_CLASS +#define USBD_MAX_SUPPORTED_CLASS 1U +#endif /* USBD_MAX_SUPPORTED_CLASS */ +#endif /* USE_USBD_COMPOSITE */ + +#ifndef USBD_MAX_CLASS_ENDPOINTS +#define USBD_MAX_CLASS_ENDPOINTS 5U +#endif /* USBD_MAX_CLASS_ENDPOINTS */ + +#ifndef USBD_MAX_CLASS_INTERFACES +#define USBD_MAX_CLASS_INTERFACES 5U +#endif /* USBD_MAX_CLASS_INTERFACES */ + #ifndef USBD_LPM_ENABLED #define USBD_LPM_ENABLED 0U #endif /* USBD_LPM_ENABLED */ @@ -160,6 +177,14 @@ extern "C" { #define USBD_EP_TYPE_BULK 0x02U #define USBD_EP_TYPE_INTR 0x03U +#ifdef USE_USBD_COMPOSITE +#define USBD_EP_IN 0x80U +#define USBD_EP_OUT 0x00U +#define USBD_FUNC_DESCRIPTOR_TYPE 0x24U +#define USBD_DESC_SUBTYPE_ACM 0x0FU +#define USBD_DESC_ECM_BCD_LOW 0x00U +#define USBD_DESC_ECM_BCD_HIGH 0x10U +#endif /* USE_USBD_COMPOSITE */ /** * @} */ @@ -188,7 +213,7 @@ typedef struct uint8_t iConfiguration; uint8_t bmAttributes; uint8_t bMaxPower; -} USBD_ConfigDescTypedef; +} __PACKED USBD_ConfigDescTypeDef; typedef struct { @@ -196,7 +221,7 @@ typedef struct uint8_t bDescriptorType; uint16_t wTotalLength; uint8_t bNumDeviceCaps; -} USBD_BosDescTypedef; +} USBD_BosDescTypeDef; typedef struct { @@ -206,7 +231,14 @@ typedef struct uint8_t bmAttributes; uint16_t wMaxPacketSize; uint8_t bInterval; -} USBD_EpDescTypedef; +} __PACKED USBD_EpDescTypeDef; + +typedef struct +{ + uint8_t bLength; + uint8_t bDescriptorType; + uint8_t bDescriptorSubType; +} USBD_DescHeaderTypeDef; struct _USBD_HandleTypeDef; @@ -231,7 +263,7 @@ typedef struct _Device_cb uint8_t *(*GetDeviceQualifierDescriptor)(uint16_t *length); #if (USBD_SUPPORT_USER_STRING_DESC == 1U) uint8_t *(*GetUsrStrDescriptor)(struct _USBD_HandleTypeDef *pdev, uint8_t index, uint16_t *length); -#endif +#endif /* USBD_SUPPORT_USER_STRING_DESC */ } USBD_ClassTypeDef; @@ -264,10 +296,10 @@ typedef struct uint8_t *(*GetInterfaceStrDescriptor)(USBD_SpeedTypeDef speed, uint16_t *length); #if (USBD_CLASS_USER_STRING_DESC == 1) uint8_t *(*GetUserStrDescriptor)(USBD_SpeedTypeDef speed, uint8_t idx, uint16_t *length); -#endif +#endif /* USBD_CLASS_USER_STRING_DESC */ #if ((USBD_LPM_ENABLED == 1U) || (USBD_CLASS_BOS_ENABLED == 1)) uint8_t *(*GetBOSDescriptor)(USBD_SpeedTypeDef speed, uint16_t *length); -#endif +#endif /* (USBD_LPM_ENABLED == 1U) || (USBD_CLASS_BOS_ENABLED == 1) */ } USBD_DescriptorsTypeDef; /* USB Device handle structure */ @@ -281,6 +313,49 @@ typedef struct uint16_t bInterval; } USBD_EndpointTypeDef; +#ifdef USE_USBD_COMPOSITE +typedef enum +{ + CLASS_TYPE_NONE = 0, + CLASS_TYPE_HID = 1, + CLASS_TYPE_CDC = 2, + CLASS_TYPE_MSC = 3, + CLASS_TYPE_DFU = 4, + CLASS_TYPE_CHID = 5, + CLASS_TYPE_AUDIO = 6, + CLASS_TYPE_ECM = 7, + CLASS_TYPE_RNDIS = 8, + CLASS_TYPE_MTP = 9, + CLASS_TYPE_VIDEO = 10, + CLASS_TYPE_PRINTER = 11, + CLASS_TYPE_CCID = 12, +} USBD_CompositeClassTypeDef; + + +/* USB Device handle structure */ +typedef struct +{ + uint8_t add; + uint8_t type; + uint8_t size; + uint8_t is_used; +} USBD_EPTypeDef; + +/* USB Device handle structure */ +typedef struct +{ + USBD_CompositeClassTypeDef ClassType; + uint32_t ClassId; + uint32_t Active; + uint32_t NumEps; + USBD_EPTypeDef Eps[USBD_MAX_CLASS_ENDPOINTS]; + uint8_t *EpAdd; + uint32_t NumIf; + uint8_t Ifs[USBD_MAX_CLASS_INTERFACES]; + uint32_t CurrPcktSze; +} USBD_CompositeElementTypeDef; +#endif /* USE_USBD_COMPOSITE */ + /* USB Device handle structure */ typedef struct _USBD_HandleTypeDef { @@ -303,14 +378,33 @@ typedef struct _USBD_HandleTypeDef USBD_SetupReqTypedef request; USBD_DescriptorsTypeDef *pDesc; - USBD_ClassTypeDef *pClass; + USBD_ClassTypeDef *pClass[USBD_MAX_SUPPORTED_CLASS]; void *pClassData; - void *pUserData; + void *pClassDataCmsit[USBD_MAX_SUPPORTED_CLASS]; + void *pUserData[USBD_MAX_SUPPORTED_CLASS]; void *pData; void *pBosDesc; void *pConfDesc; + uint32_t classId; + uint32_t NumClasses; +#ifdef USE_USBD_COMPOSITE + USBD_CompositeElementTypeDef tclasslist[USBD_MAX_SUPPORTED_CLASS]; +#endif /* USE_USBD_COMPOSITE */ } USBD_HandleTypeDef; +/* USB Device endpoint direction */ +typedef enum +{ + OUT = 0x00, + IN = 0x80, +} USBD_EPDirectionTypeDef; + +typedef enum +{ + NETWORK_CONNECTION = 0x00, + RESPONSE_AVAILABLE = 0x01, + CONNECTION_SPEED_CHANGE = 0x2A +} USBD_CDC_NotifCodeTypeDef; /** * @} */ @@ -322,7 +416,9 @@ typedef struct _USBD_HandleTypeDef */ __STATIC_INLINE uint16_t SWAPBYTE(uint8_t *addr) { - uint16_t _SwapVal, _Byte1, _Byte2; + uint16_t _SwapVal; + uint16_t _Byte1; + uint16_t _Byte2; uint8_t *_pbuff = addr; _Byte1 = *(uint8_t *)_pbuff; @@ -336,19 +432,19 @@ __STATIC_INLINE uint16_t SWAPBYTE(uint8_t *addr) #ifndef LOBYTE #define LOBYTE(x) ((uint8_t)((x) & 0x00FFU)) -#endif +#endif /* LOBYTE */ #ifndef HIBYTE #define HIBYTE(x) ((uint8_t)(((x) & 0xFF00U) >> 8U)) -#endif +#endif /* HIBYTE */ #ifndef MIN #define MIN(a, b) (((a) < (b)) ? (a) : (b)) -#endif +#endif /* MIN */ #ifndef MAX #define MAX(a, b) (((a) > (b)) ? (a) : (b)) -#endif +#endif /* MAX */ #if defined ( __GNUC__ ) #ifndef __weak @@ -417,4 +513,4 @@ __STATIC_INLINE uint16_t SWAPBYTE(uint8_t *addr) /** * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ + diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_desc_template.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_desc_template.h index 8fbbfaa6..e3923055 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_desc_template.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_desc_template.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -44,7 +43,7 @@ #define USBD_BB_URL_STR_DESC (uint8_t *)"www.st.com" #define USBD_BB_ALTMODE0_STR_DESC (uint8_t *)"STM32 Alternate0 Mode" #define USBD_BB_ALTMODE1_STR_DESC (uint8_t *)"STM32 Alternate1 Mode" -#endif +#endif /* USBD_CLASS_USER_STRING_DESC */ #define USB_SIZ_STRING_SERIAL 0x1AU @@ -52,7 +51,7 @@ #define USB_SIZ_BOS_DESC 0x0CU #elif (USBD_CLASS_BOS_ENABLED == 1) #define USB_SIZ_BOS_DESC 0x5DU -#endif +#endif /* USBD_LPM_ENABLED */ /* Exported macro ------------------------------------------------------------*/ /* Exported functions ------------------------------------------------------- */ @@ -60,4 +59,3 @@ extern USBD_DescriptorsTypeDef XXX_Desc; /* Replace 'XXX_Desc' with your active #endif /* __USBD_DESC_TEMPLATE_H*/ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_ioreq.h b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_ioreq.h index c896b5ae..15197b92 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_ioreq.h +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Inc/usbd_ioreq.h @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -111,4 +110,4 @@ uint32_t USBD_GetRxCount(USBD_HandleTypeDef *pdev, uint8_t ep_addr); /** * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ + diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_conf_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_conf_template.c index 381988b4..74ff4302 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_conf_template.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_conf_template.c @@ -8,13 +8,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -234,6 +233,22 @@ uint32_t USBD_LL_GetRxDataSize(USBD_HandleTypeDef *pdev, uint8_t ep_addr) return 0U; } +#ifdef USBD_HS_TESTMODE_ENABLE +/** + * @brief Set High speed Test mode. + * @param pdev: Device handle + * @param testmode: test mode + * @retval USBD Status + */ +USBD_StatusTypeDef USBD_LL_SetTestMode(USBD_HandleTypeDef *pdev, uint8_t testmode) +{ + UNUSED(pdev); + UNUSED(testmode); + + return USBD_OK; +} +#endif /* USBD_HS_TESTMODE_ENABLE */ + /** * @brief Static single allocation. * @param size: Size of allocated memory @@ -241,6 +256,7 @@ uint32_t USBD_LL_GetRxDataSize(USBD_HandleTypeDef *pdev, uint8_t ep_addr) */ void *USBD_static_malloc(uint32_t size) { + UNUSED(size); static uint32_t mem[(sizeof(USBD_HID_HandleTypeDef) / 4) + 1]; /* On 32-bit boundary */ return mem; } @@ -252,7 +268,7 @@ void *USBD_static_malloc(uint32_t size) */ void USBD_static_free(void *p) { - + UNUSED(p); } /** @@ -264,5 +280,4 @@ void USBD_LL_Delay(uint32_t Delay) { UNUSED(Delay); } -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_core.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_core.c index a6154ea8..3c0610a1 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_core.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_core.c @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -20,6 +19,10 @@ /* Includes ------------------------------------------------------------------*/ #include "usbd_core.h" +#ifdef USE_USBD_COMPOSITE +#include "usbd_composite_builder.h" +#endif /* USE_USBD_COMPOSITE */ + /** @addtogroup STM32_USBD_DEVICE_LIBRARY * @{ */ @@ -96,13 +99,29 @@ USBD_StatusTypeDef USBD_Init(USBD_HandleTypeDef *pdev, { #if (USBD_DEBUG_LEVEL > 1U) USBD_ErrLog("Invalid Device handle"); -#endif +#endif /* (USBD_DEBUG_LEVEL > 1U) */ return USBD_FAIL; } - /* Unlink previous class resources */ - pdev->pClass = NULL; - pdev->pUserData = NULL; +#ifdef USE_USBD_COMPOSITE + /* Parse the table of classes in use */ + for (uint32_t i = 0; i < USBD_MAX_SUPPORTED_CLASS; i++) + { + /* Unlink previous class*/ + pdev->pClass[i] = NULL; + pdev->pUserData[i] = NULL; + + /* Set class as inactive */ + pdev->tclasslist[i].Active = 0; + pdev->NumClasses = 0; + pdev->classId = 0; + } +#else + /* Unlink previous class*/ + pdev->pClass[0] = NULL; + pdev->pUserData[0] = NULL; +#endif /* USE_USBD_COMPOSITE */ + pdev->pConfDesc = NULL; /* Assign USBD Descriptors */ @@ -137,14 +156,32 @@ USBD_StatusTypeDef USBD_DeInit(USBD_HandleTypeDef *pdev) /* Set Default State */ pdev->dev_state = USBD_STATE_DEFAULT; +#ifdef USE_USBD_COMPOSITE + /* Parse the table of classes in use */ + for (uint32_t i = 0; i < USBD_MAX_SUPPORTED_CLASS; i++) + { + /* Check if current class is in use */ + if ((pdev->tclasslist[i].Active) == 1U) + { + if (pdev->pClass[i] != NULL) + { + pdev->classId = i; + /* Free Class Resources */ + pdev->pClass[i]->DeInit(pdev, (uint8_t)pdev->dev_config); + } + } + } +#else /* Free Class Resources */ - if (pdev->pClass != NULL) + if (pdev->pClass[0] != NULL) { - pdev->pClass->DeInit(pdev, (uint8_t)pdev->dev_config); - pdev->pClass = NULL; - pdev->pUserData = NULL; + pdev->pClass[0]->DeInit(pdev, (uint8_t)pdev->dev_config); } + pdev->pUserData[0] = NULL; + +#endif /* USE_USBD_COMPOSITE */ + /* Free Device descriptors resources */ pdev->pDesc = NULL; pdev->pConfDesc = NULL; @@ -170,29 +207,161 @@ USBD_StatusTypeDef USBD_RegisterClass(USBD_HandleTypeDef *pdev, USBD_ClassTypeDe { #if (USBD_DEBUG_LEVEL > 1U) USBD_ErrLog("Invalid Class handle"); -#endif +#endif /* (USBD_DEBUG_LEVEL > 1U) */ return USBD_FAIL; } /* link the class to the USB Device handle */ - pdev->pClass = pclass; + pdev->pClass[0] = pclass; /* Get Device Configuration Descriptor */ #ifdef USE_USB_HS - if (pdev->pClass->GetHSConfigDescriptor != NULL) + if (pdev->pClass[pdev->classId]->GetHSConfigDescriptor != NULL) { - pdev->pConfDesc = (void *)pdev->pClass->GetHSConfigDescriptor(&len); + pdev->pConfDesc = (void *)pdev->pClass[pdev->classId]->GetHSConfigDescriptor(&len); } #else /* Default USE_USB_FS */ - if (pdev->pClass->GetFSConfigDescriptor != NULL) + if (pdev->pClass[pdev->classId]->GetFSConfigDescriptor != NULL) { - pdev->pConfDesc = (void *)pdev->pClass->GetFSConfigDescriptor(&len); + pdev->pConfDesc = (void *)pdev->pClass[pdev->classId]->GetFSConfigDescriptor(&len); } #endif /* USE_USB_FS */ + /* Increment the NumClasses */ + pdev->NumClasses ++; + return USBD_OK; } +#ifdef USE_USBD_COMPOSITE +/** + * @brief USBD_RegisterClassComposite + * Link class driver to Device Core. + * @param pdev : Device Handle + * @param pclass: Class handle + * @param classtype: Class type + * @param EpAddr: Endpoint Address handle + * @retval USBD Status + */ +USBD_StatusTypeDef USBD_RegisterClassComposite(USBD_HandleTypeDef *pdev, USBD_ClassTypeDef *pclass, + USBD_CompositeClassTypeDef classtype, uint8_t *EpAddr) +{ + USBD_StatusTypeDef ret = USBD_OK; + uint16_t len = 0U; + + if ((pdev->classId < USBD_MAX_SUPPORTED_CLASS) && (pdev->NumClasses < USBD_MAX_SUPPORTED_CLASS)) + { + if ((uint32_t)pclass != 0U) + { + /* Link the class to the USB Device handle */ + pdev->pClass[pdev->classId] = pclass; + ret = USBD_OK; + + pdev->tclasslist[pdev->classId].EpAdd = EpAddr; + + /* Call the composite class builder */ + (void)USBD_CMPSIT_AddClass(pdev, pclass, classtype, 0); + + /* Increment the ClassId for the next occurrence */ + pdev->classId ++; + pdev->NumClasses ++; + } + else + { +#if (USBD_DEBUG_LEVEL > 1U) + USBD_ErrLog("Invalid Class handle"); +#endif /* (USBD_DEBUG_LEVEL > 1U) */ + ret = USBD_FAIL; + } + } + + if (ret == USBD_OK) + { + /* Get Device Configuration Descriptor */ +#ifdef USE_USB_HS + pdev->pConfDesc = USBD_CMPSIT.GetHSConfigDescriptor(&len); +#else /* Default USE_USB_FS */ + pdev->pConfDesc = USBD_CMPSIT.GetFSConfigDescriptor(&len); +#endif /* USE_USB_FS */ + } + + return ret; +} + +/** + * @brief USBD_UnRegisterClassComposite + * UnLink all composite class drivers from Device Core. + * @param pDevice : Device Handle + * @retval USBD Status + */ +USBD_StatusTypeDef USBD_UnRegisterClassComposite(USBD_HandleTypeDef *pdev) +{ + USBD_StatusTypeDef ret = USBD_FAIL; + uint8_t idx1; + uint8_t idx2; + + /* Unroll all activated classes */ + for (idx1 = 0; idx1 < pdev->NumClasses; idx1++) + { + /* Check if the class correspond to the requested type and if it is active */ + if (pdev->tclasslist[idx1].Active == 1U) + { + /* Set the new class ID */ + pdev->classId = idx1; + + /* Free resources used by the selected class */ + if (pdev->pClass[pdev->classId] != NULL) + { + /* Free Class Resources */ + if (pdev->pClass[pdev->classId]->DeInit(pdev, (uint8_t)pdev->dev_config) != 0U) + { +#if (USBD_DEBUG_LEVEL > 1U) + USBD_ErrLog("Class DeInit didn't succeed!, can't unregister selected class"); +#endif /* (USBD_DEBUG_LEVEL > 1U) */ + + ret = USBD_FAIL; + } + } + + /* Free the class pointer */ + pdev->pClass[pdev->classId] = NULL; + + /* Free the class location in classes table and reset its parameters to zero */ + pdev->tclasslist[pdev->classId].ClassType = CLASS_TYPE_NONE; + pdev->tclasslist[pdev->classId].ClassId = 0U; + pdev->tclasslist[pdev->classId].Active = 0U; + pdev->tclasslist[pdev->classId].NumEps = 0U; + pdev->tclasslist[pdev->classId].NumIf = 0U; + pdev->tclasslist[pdev->classId].CurrPcktSze = 0U; + + for (idx2 = 0U; idx2 < USBD_MAX_CLASS_ENDPOINTS; idx2++) + { + pdev->tclasslist[pdev->classId].Eps[idx2].add = 0U; + pdev->tclasslist[pdev->classId].Eps[idx2].type = 0U; + pdev->tclasslist[pdev->classId].Eps[idx2].size = 0U; + pdev->tclasslist[pdev->classId].Eps[idx2].is_used = 0U; + } + + for (idx2 = 0U; idx2 < USBD_MAX_CLASS_INTERFACES; idx2++) + { + pdev->tclasslist[pdev->classId].Ifs[idx2] = 0U; + } + } + } + + /* Reset the configuration descriptor */ + (void)USBD_CMPST_ClearConfDesc(pdev); + + /* Reset the class ID and number of classes */ + pdev->classId = 0U; + pdev->NumClasses = 0U; + + return ret; +} + + +#endif /* USE_USBD_COMPOSITE */ + /** * @brief USBD_Start * Start the USB Device Core. @@ -201,6 +370,10 @@ USBD_StatusTypeDef USBD_RegisterClass(USBD_HandleTypeDef *pdev, USBD_ClassTypeDe */ USBD_StatusTypeDef USBD_Start(USBD_HandleTypeDef *pdev) { +#ifdef USE_USBD_COMPOSITE + pdev->classId = 0U; +#endif /* USE_USBD_COMPOSITE */ + /* Start the low level driver */ return USBD_LL_Start(pdev); } @@ -217,10 +390,30 @@ USBD_StatusTypeDef USBD_Stop(USBD_HandleTypeDef *pdev) (void)USBD_LL_Stop(pdev); /* Free Class Resources */ - if (pdev->pClass != NULL) +#ifdef USE_USBD_COMPOSITE + /* Parse the table of classes in use */ + for (uint32_t i = 0U; i < USBD_MAX_SUPPORTED_CLASS; i++) + { + /* Check if current class is in use */ + if ((pdev->tclasslist[i].Active) == 1U) + { + if (pdev->pClass[i] != NULL) + { + pdev->classId = i; + /* Free Class Resources */ + (void)pdev->pClass[i]->DeInit(pdev, (uint8_t)pdev->dev_config); + } + } + } + + /* Reset the class ID */ + pdev->classId = 0U; +#else + if (pdev->pClass[0] != NULL) { - (void)pdev->pClass->DeInit(pdev, (uint8_t)pdev->dev_config); + (void)pdev->pClass[0]->DeInit(pdev, (uint8_t)pdev->dev_config); } +#endif /* USE_USBD_COMPOSITE */ return USBD_OK; } @@ -231,12 +424,21 @@ USBD_StatusTypeDef USBD_Stop(USBD_HandleTypeDef *pdev) * @param pdev: device instance * @retval status */ -USBD_StatusTypeDef USBD_RunTestMode(USBD_HandleTypeDef *pdev) +USBD_StatusTypeDef USBD_RunTestMode(USBD_HandleTypeDef *pdev) { +#ifdef USBD_HS_TESTMODE_ENABLE + USBD_StatusTypeDef ret; + + /* Run USB HS test mode */ + ret = USBD_LL_SetTestMode(pdev, pdev->dev_test_mode); + + return ret; +#else /* Prevent unused argument compilation warning */ UNUSED(pdev); return USBD_OK; +#endif /* USBD_HS_TESTMODE_ENABLE */ } /** @@ -249,13 +451,33 @@ USBD_StatusTypeDef USBD_RunTestMode(USBD_HandleTypeDef *pdev) USBD_StatusTypeDef USBD_SetClassConfig(USBD_HandleTypeDef *pdev, uint8_t cfgidx) { - USBD_StatusTypeDef ret = USBD_FAIL; + USBD_StatusTypeDef ret = USBD_OK; - if (pdev->pClass != NULL) +#ifdef USE_USBD_COMPOSITE + /* Parse the table of classes in use */ + for (uint32_t i = 0U; i < USBD_MAX_SUPPORTED_CLASS; i++) + { + /* Check if current class is in use */ + if ((pdev->tclasslist[i].Active) == 1U) + { + if (pdev->pClass[i] != NULL) + { + pdev->classId = i; + /* Set configuration and Start the Class*/ + if (pdev->pClass[i]->Init(pdev, cfgidx) != 0U) + { + ret = USBD_FAIL; + } + } + } + } +#else + if (pdev->pClass[0] != NULL) { /* Set configuration and Start the Class */ - ret = (USBD_StatusTypeDef)pdev->pClass->Init(pdev, cfgidx); + ret = (USBD_StatusTypeDef)pdev->pClass[0]->Init(pdev, cfgidx); } +#endif /* USE_USBD_COMPOSITE */ return ret; } @@ -269,13 +491,35 @@ USBD_StatusTypeDef USBD_SetClassConfig(USBD_HandleTypeDef *pdev, uint8_t cfgidx) */ USBD_StatusTypeDef USBD_ClrClassConfig(USBD_HandleTypeDef *pdev, uint8_t cfgidx) { - /* Clear configuration and De-initialize the Class process */ - if (pdev->pClass != NULL) + USBD_StatusTypeDef ret = USBD_OK; + +#ifdef USE_USBD_COMPOSITE + /* Parse the table of classes in use */ + for (uint32_t i = 0U; i < USBD_MAX_SUPPORTED_CLASS; i++) + { + /* Check if current class is in use */ + if ((pdev->tclasslist[i].Active) == 1U) + { + if (pdev->pClass[i] != NULL) + { + pdev->classId = i; + /* Clear configuration and De-initialize the Class process */ + if (pdev->pClass[i]->DeInit(pdev, cfgidx) != 0U) + { + ret = USBD_FAIL; + } + } + } + } +#else + /* Clear configuration and De-initialize the Class process */ + if (pdev->pClass[0]->DeInit(pdev, cfgidx) != 0U) { - pdev->pClass->DeInit(pdev, cfgidx); + ret = USBD_FAIL; } +#endif /* USE_USBD_COMPOSITE */ - return USBD_OK; + return ret; } @@ -329,7 +573,8 @@ USBD_StatusTypeDef USBD_LL_DataOutStage(USBD_HandleTypeDef *pdev, uint8_t epnum, uint8_t *pdata) { USBD_EndpointTypeDef *pep; - USBD_StatusTypeDef ret; + USBD_StatusTypeDef ret = USBD_OK; + uint8_t idx; if (epnum == 0U) { @@ -345,44 +590,66 @@ USBD_StatusTypeDef USBD_LL_DataOutStage(USBD_HandleTypeDef *pdev, } else { - if (pdev->dev_state == USBD_STATE_CONFIGURED) + /* Find the class ID relative to the current request */ + switch (pdev->request.bmRequest & 0x1FU) + { + case USB_REQ_RECIPIENT_DEVICE: + /* Device requests must be managed by the first instantiated class + (or duplicated by all classes for simplicity) */ + idx = 0U; + break; + + case USB_REQ_RECIPIENT_INTERFACE: + idx = USBD_CoreFindIF(pdev, LOBYTE(pdev->request.wIndex)); + break; + + case USB_REQ_RECIPIENT_ENDPOINT: + idx = USBD_CoreFindEP(pdev, LOBYTE(pdev->request.wIndex)); + break; + + default: + /* Back to the first class in case of doubt */ + idx = 0U; + break; + } + + if (idx < USBD_MAX_SUPPORTED_CLASS) { - if (pdev->pClass->EP0_RxReady != NULL) + /* Setup the class ID and route the request to the relative class function */ + if (pdev->dev_state == USBD_STATE_CONFIGURED) { - pdev->pClass->EP0_RxReady(pdev); + if (pdev->pClass[idx]->EP0_RxReady != NULL) + { + pdev->classId = idx; + pdev->pClass[idx]->EP0_RxReady(pdev); + } } } (void)USBD_CtlSendStatus(pdev); } } - else - { -#if 0 - if (pdev->ep0_state == USBD_EP0_STATUS_OUT) - { - /* - * STATUS PHASE completed, update ep0_state to idle - */ - pdev->ep0_state = USBD_EP0_IDLE; - (void)USBD_LL_StallEP(pdev, 0U); - } -#endif - } } else { - if (pdev->dev_state == USBD_STATE_CONFIGURED) + /* Get the class index relative to this interface */ + idx = USBD_CoreFindEP(pdev, (epnum & 0x7FU)); + + if (((uint16_t)idx != 0xFFU) && (idx < USBD_MAX_SUPPORTED_CLASS)) { - if (pdev->pClass->DataOut != NULL) + /* Call the class data out function to manage the request */ + if (pdev->dev_state == USBD_STATE_CONFIGURED) { - ret = (USBD_StatusTypeDef)pdev->pClass->DataOut(pdev, epnum); - - if (ret != USBD_OK) + if (pdev->pClass[idx]->DataOut != NULL) { - return ret; + pdev->classId = idx; + ret = (USBD_StatusTypeDef)pdev->pClass[idx]->DataOut(pdev, epnum); } } + if (ret != USBD_OK) + { + return ret; + } } } @@ -401,6 +668,7 @@ USBD_StatusTypeDef USBD_LL_DataInStage(USBD_HandleTypeDef *pdev, { USBD_EndpointTypeDef *pep; USBD_StatusTypeDef ret; + uint8_t idx; if (epnum == 0U) { @@ -434,33 +702,19 @@ USBD_StatusTypeDef USBD_LL_DataInStage(USBD_HandleTypeDef *pdev, { if (pdev->dev_state == USBD_STATE_CONFIGURED) { - if (pdev->pClass->EP0_TxSent != NULL) + if (pdev->pClass[0]->EP0_TxSent != NULL) { - pdev->pClass->EP0_TxSent(pdev); + pdev->classId = 0U; + pdev->pClass[0]->EP0_TxSent(pdev); } } - /* - * Fix EPO STALL issue in STM32 USB Device library 2.5.1 - * Issue raised by @makarenya, see #388 - * USB Specification EP0 should never STALL. - */ - /* (void)USBD_LL_StallEP(pdev, 0x80U); */ + (void)USBD_LL_StallEP(pdev, 0x80U); (void)USBD_CtlReceiveStatus(pdev); } } } - else - { -#if 0 - if ((pdev->ep0_state == USBD_EP0_STATUS_IN) || - (pdev->ep0_state == USBD_EP0_IDLE)) - { - (void)USBD_LL_StallEP(pdev, 0x80U); - } -#endif - } - if (pdev->dev_test_mode == 1U) + if (pdev->dev_test_mode != 0U) { (void)USBD_RunTestMode(pdev); pdev->dev_test_mode = 0U; @@ -468,15 +722,23 @@ USBD_StatusTypeDef USBD_LL_DataInStage(USBD_HandleTypeDef *pdev, } else { - if (pdev->dev_state == USBD_STATE_CONFIGURED) + /* Get the class index relative to this interface */ + idx = USBD_CoreFindEP(pdev, ((uint8_t)epnum | 0x80U)); + + if (((uint16_t)idx != 0xFFU) && (idx < USBD_MAX_SUPPORTED_CLASS)) { - if (pdev->pClass->DataIn != NULL) + /* Call the class data out function to manage the request */ + if (pdev->dev_state == USBD_STATE_CONFIGURED) { - ret = (USBD_StatusTypeDef)pdev->pClass->DataIn(pdev, epnum); - - if (ret != USBD_OK) + if (pdev->pClass[idx]->DataIn != NULL) { - return ret; + pdev->classId = idx; + ret = (USBD_StatusTypeDef)pdev->pClass[idx]->DataIn(pdev, epnum); + + if (ret != USBD_OK) + { + return ret; + } } } } @@ -494,24 +756,50 @@ USBD_StatusTypeDef USBD_LL_DataInStage(USBD_HandleTypeDef *pdev, USBD_StatusTypeDef USBD_LL_Reset(USBD_HandleTypeDef *pdev) { + USBD_StatusTypeDef ret = USBD_OK; + /* Upon Reset call user call back */ pdev->dev_state = USBD_STATE_DEFAULT; pdev->ep0_state = USBD_EP0_IDLE; pdev->dev_config = 0U; pdev->dev_remote_wakeup = 0U; + pdev->dev_test_mode = 0U; - if (pdev->pClass == NULL) +#ifdef USE_USBD_COMPOSITE + /* Parse the table of classes in use */ + for (uint32_t i = 0U; i < USBD_MAX_SUPPORTED_CLASS; i++) { - return USBD_FAIL; + /* Check if current class is in use */ + if ((pdev->tclasslist[i].Active) == 1U) + { + if (pdev->pClass[i] != NULL) + { + pdev->classId = i; + /* Clear configuration and De-initialize the Class process*/ + + if (pdev->pClass[i]->DeInit != NULL) + { + if (pdev->pClass[i]->DeInit(pdev, (uint8_t)pdev->dev_config) != USBD_OK) + { + ret = USBD_FAIL; + } + } + } + } } +#else - if (pdev->pClassData != NULL) + if (pdev->pClass[0] != NULL) { - if (pdev->pClass->DeInit != NULL) + if (pdev->pClass[0]->DeInit != NULL) { - (void)pdev->pClass->DeInit(pdev, (uint8_t)pdev->dev_config); + if (pdev->pClass[0]->DeInit(pdev, (uint8_t)pdev->dev_config) != USBD_OK) + { + ret = USBD_FAIL; + } } } +#endif /* USE_USBD_COMPOSITE */ /* Open EP0 OUT */ (void)USBD_LL_OpenEP(pdev, 0x00U, USBD_EP_TYPE_CTRL, USB_MAX_EP0_SIZE); @@ -525,7 +813,7 @@ USBD_StatusTypeDef USBD_LL_Reset(USBD_HandleTypeDef *pdev) pdev->ep_in[0].maxpacket = USB_MAX_EP0_SIZE; - return USBD_OK; + return ret; } /** @@ -551,7 +839,11 @@ USBD_StatusTypeDef USBD_LL_SetSpeed(USBD_HandleTypeDef *pdev, USBD_StatusTypeDef USBD_LL_Suspend(USBD_HandleTypeDef *pdev) { - pdev->dev_old_state = pdev->dev_state; + if (pdev->dev_state != USBD_STATE_SUSPENDED) + { + pdev->dev_old_state = pdev->dev_state; + } + pdev->dev_state = USBD_STATE_SUSPENDED; return USBD_OK; @@ -583,17 +875,35 @@ USBD_StatusTypeDef USBD_LL_Resume(USBD_HandleTypeDef *pdev) USBD_StatusTypeDef USBD_LL_SOF(USBD_HandleTypeDef *pdev) { - if (pdev->pClass == NULL) - { - return USBD_FAIL; - } - + /* The SOF event can be distributed for all classes that support it */ if (pdev->dev_state == USBD_STATE_CONFIGURED) { - if (pdev->pClass->SOF != NULL) +#ifdef USE_USBD_COMPOSITE + /* Parse the table of classes in use */ + for (uint32_t i = 0; i < USBD_MAX_SUPPORTED_CLASS; i++) { - (void)pdev->pClass->SOF(pdev); + /* Check if current class is in use */ + if ((pdev->tclasslist[i].Active) == 1U) + { + if (pdev->pClass[i] != NULL) + { + if (pdev->pClass[i]->SOF != NULL) + { + pdev->classId = i; + (void)pdev->pClass[i]->SOF(pdev); + } + } + } } +#else + if (pdev->pClass[0] != NULL) + { + if (pdev->pClass[0]->SOF != NULL) + { + (void)pdev->pClass[0]->SOF(pdev); + } + } +#endif /* USE_USBD_COMPOSITE */ } return USBD_OK; @@ -608,16 +918,16 @@ USBD_StatusTypeDef USBD_LL_SOF(USBD_HandleTypeDef *pdev) USBD_StatusTypeDef USBD_LL_IsoINIncomplete(USBD_HandleTypeDef *pdev, uint8_t epnum) { - if (pdev->pClass == NULL) + if (pdev->pClass[pdev->classId] == NULL) { return USBD_FAIL; } if (pdev->dev_state == USBD_STATE_CONFIGURED) { - if (pdev->pClass->IsoINIncomplete != NULL) + if (pdev->pClass[pdev->classId]->IsoINIncomplete != NULL) { - (void)pdev->pClass->IsoINIncomplete(pdev, epnum); + (void)pdev->pClass[pdev->classId]->IsoINIncomplete(pdev, epnum); } } @@ -633,16 +943,16 @@ USBD_StatusTypeDef USBD_LL_IsoINIncomplete(USBD_HandleTypeDef *pdev, USBD_StatusTypeDef USBD_LL_IsoOUTIncomplete(USBD_HandleTypeDef *pdev, uint8_t epnum) { - if (pdev->pClass == NULL) + if (pdev->pClass[pdev->classId] == NULL) { return USBD_FAIL; } if (pdev->dev_state == USBD_STATE_CONFIGURED) { - if (pdev->pClass->IsoOUTIncomplete != NULL) + if (pdev->pClass[pdev->classId]->IsoOUTIncomplete != NULL) { - (void)pdev->pClass->IsoOUTIncomplete(pdev, epnum); + (void)pdev->pClass[pdev->classId]->IsoOUTIncomplete(pdev, epnum); } } @@ -671,20 +981,210 @@ USBD_StatusTypeDef USBD_LL_DevConnected(USBD_HandleTypeDef *pdev) */ USBD_StatusTypeDef USBD_LL_DevDisconnected(USBD_HandleTypeDef *pdev) { + USBD_StatusTypeDef ret = USBD_OK; + /* Free Class Resources */ pdev->dev_state = USBD_STATE_DEFAULT; - if (pdev->pClass != NULL) +#ifdef USE_USBD_COMPOSITE + /* Parse the table of classes in use */ + for (uint32_t i = 0; i < USBD_MAX_SUPPORTED_CLASS; i++) + { + /* Check if current class is in use */ + if ((pdev->tclasslist[i].Active) == 1U) + { + if (pdev->pClass[i] != NULL) + { + pdev->classId = i; + /* Clear configuration and De-initialize the Class process*/ + if (pdev->pClass[i]->DeInit(pdev, (uint8_t)pdev->dev_config) != 0U) + { + ret = USBD_FAIL; + } + } + } + } +#else + if (pdev->pClass[0] != NULL) { - (void)pdev->pClass->DeInit(pdev, (uint8_t)pdev->dev_config); + if (pdev->pClass[0]->DeInit(pdev, (uint8_t)pdev->dev_config) != 0U) + { + ret = USBD_FAIL; + } } +#endif /* USE_USBD_COMPOSITE */ - return USBD_OK; + return ret; } + /** - * @} + * @brief USBD_CoreFindIF + * return the class index relative to the selected interface + * @param pdev: device instance + * @param index : selected interface number + * @retval index of the class using the selected interface number. OxFF if no class found. */ +uint8_t USBD_CoreFindIF(USBD_HandleTypeDef *pdev, uint8_t index) +{ +#ifdef USE_USBD_COMPOSITE + /* Parse the table of classes in use */ + for (uint32_t i = 0U; i < USBD_MAX_SUPPORTED_CLASS; i++) + { + /* Check if current class is in use */ + if ((pdev->tclasslist[i].Active) == 1U) + { + /* Parse all interfaces listed in the current class */ + for (uint32_t j = 0U; j < pdev->tclasslist[i].NumIf; j++) + { + /* Check if requested Interface matches the current class interface */ + if (pdev->tclasslist[i].Ifs[j] == index) + { + if (pdev->pClass[i]->Setup != NULL) + { + return (uint8_t)i; + } + } + } + } + } + + return 0xFFU; +#else + UNUSED(pdev); + UNUSED(index); + return 0x00U; +#endif /* USE_USBD_COMPOSITE */ +} + +/** + * @brief USBD_CoreFindEP + * return the class index relative to the selected endpoint + * @param pdev: device instance + * @param index : selected endpoint number + * @retval index of the class using the selected endpoint number. 0xFF if no class found. + */ +uint8_t USBD_CoreFindEP(USBD_HandleTypeDef *pdev, uint8_t index) +{ +#ifdef USE_USBD_COMPOSITE + /* Parse the table of classes in use */ + for (uint32_t i = 0U; i < USBD_MAX_SUPPORTED_CLASS; i++) + { + /* Check if current class is in use */ + if ((pdev->tclasslist[i].Active) == 1U) + { + /* Parse all endpoints listed in the current class */ + for (uint32_t j = 0U; j < pdev->tclasslist[i].NumEps; j++) + { + /* Check if requested endpoint matches the current class endpoint */ + if (pdev->tclasslist[i].Eps[j].add == index) + { + if (pdev->pClass[i]->Setup != NULL) + { + return (uint8_t)i; + } + } + } + } + } + + return 0xFFU; +#else + UNUSED(pdev); + UNUSED(index); + + return 0x00U; +#endif /* USE_USBD_COMPOSITE */ +} + +#ifdef USE_USBD_COMPOSITE +/** + * @brief USBD_CoreGetEPAdd + * Get the endpoint address relative to a selected class + * @param pdev: device instance + * @param ep_dir: USBD_EP_IN or USBD_EP_OUT + * @param ep_type: USBD_EP_TYPE_CTRL, USBD_EP_TYPE_ISOC, USBD_EP_TYPE_BULK or USBD_EP_TYPE_INTR + * @param ClassId: The Class ID + * @retval Address of the selected endpoint or 0xFFU if no endpoint found. + */ +uint8_t USBD_CoreGetEPAdd(USBD_HandleTypeDef *pdev, uint8_t ep_dir, uint8_t ep_type, uint8_t ClassId) +{ + uint8_t idx; + + /* Find the EP address in the selected class table */ + for (idx = 0; idx < pdev->tclasslist[ClassId].NumEps; idx++) + { + if (((pdev->tclasslist[ClassId].Eps[idx].add & USBD_EP_IN) == ep_dir) && \ + (pdev->tclasslist[ClassId].Eps[idx].type == ep_type) && \ + (pdev->tclasslist[ClassId].Eps[idx].is_used != 0U)) + { + return (pdev->tclasslist[ClassId].Eps[idx].add); + } + } + + /* If reaching this point, then no endpoint was found */ + return 0xFFU; +} +#endif /* USE_USBD_COMPOSITE */ + +/** + * @brief USBD_GetEpDesc + * This function return the Endpoint descriptor + * @param pdev: device instance + * @param pConfDesc: pointer to Bos descriptor + * @param EpAddr: endpoint address + * @retval pointer to video endpoint descriptor + */ +void *USBD_GetEpDesc(uint8_t *pConfDesc, uint8_t EpAddr) +{ + USBD_DescHeaderTypeDef *pdesc = (USBD_DescHeaderTypeDef *)(void *)pConfDesc; + USBD_ConfigDescTypeDef *desc = (USBD_ConfigDescTypeDef *)(void *)pConfDesc; + USBD_EpDescTypeDef *pEpDesc = NULL; + uint16_t ptr; + + if (desc->wTotalLength > desc->bLength) + { + ptr = desc->bLength; + + while (ptr < desc->wTotalLength) + { + pdesc = USBD_GetNextDesc((uint8_t *)pdesc, &ptr); + + if (pdesc->bDescriptorType == USB_DESC_TYPE_ENDPOINT) + { + pEpDesc = (USBD_EpDescTypeDef *)(void *)pdesc; + + if (pEpDesc->bEndpointAddress == EpAddr) + { + break; + } + else + { + pEpDesc = NULL; + } + } + } + } + + return (void *)pEpDesc; +} + +/** + * @brief USBD_GetNextDesc + * This function return the next descriptor header + * @param buf: Buffer where the descriptor is available + * @param ptr: data pointer inside the descriptor + * @retval next header + */ +USBD_DescHeaderTypeDef *USBD_GetNextDesc(uint8_t *pbuf, uint16_t *ptr) +{ + USBD_DescHeaderTypeDef *pnext = (USBD_DescHeaderTypeDef *)(void *)pbuf; + + *ptr += pnext->bLength; + pnext = (USBD_DescHeaderTypeDef *)(void *)(pbuf + pnext->bLength); + + return (pnext); +} /** * @} @@ -695,5 +1195,8 @@ USBD_StatusTypeDef USBD_LL_DevDisconnected(USBD_HandleTypeDef *pdev) * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ + +/** + * @} + */ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_ctlreq.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_ctlreq.c index b7098546..899bc706 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_ctlreq.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_ctlreq.c @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -21,6 +20,9 @@ #include "usbd_ctlreq.h" #include "usbd_ioreq.h" +#ifdef USE_USBD_COMPOSITE +#include "usbd_composite_builder.h" +#endif /* USE_USBD_COMPOSITE */ /** @addtogroup STM32_USBD_STATE_DEVICE_LIBRARY * @{ @@ -105,7 +107,7 @@ USBD_StatusTypeDef USBD_StdDevReq(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef { case USB_REQ_TYPE_CLASS: case USB_REQ_TYPE_VENDOR: - ret = (USBD_StatusTypeDef)pdev->pClass->Setup(pdev, req); + ret = (USBD_StatusTypeDef)pdev->pClass[pdev->classId]->Setup(pdev, req); break; case USB_REQ_TYPE_STANDARD: @@ -163,6 +165,7 @@ USBD_StatusTypeDef USBD_StdDevReq(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef USBD_StatusTypeDef USBD_StdItfReq(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) { USBD_StatusTypeDef ret = USBD_OK; + uint8_t idx; switch (req->bmRequest & USB_REQ_TYPE_MASK) { @@ -177,7 +180,27 @@ USBD_StatusTypeDef USBD_StdItfReq(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef if (LOBYTE(req->wIndex) <= USBD_MAX_NUM_INTERFACES) { - ret = (USBD_StatusTypeDef)pdev->pClass->Setup(pdev, req); + /* Get the class index relative to this interface */ + idx = USBD_CoreFindIF(pdev, LOBYTE(req->wIndex)); + if (((uint8_t)idx != 0xFFU) && (idx < USBD_MAX_SUPPORTED_CLASS)) + { + /* Call the class data out function to manage the request */ + if (pdev->pClass[idx]->Setup != NULL) + { + pdev->classId = idx; + ret = (USBD_StatusTypeDef)(pdev->pClass[idx]->Setup(pdev, req)); + } + else + { + /* should never reach this condition */ + ret = USBD_FAIL; + } + } + else + { + /* No relative interface found */ + ret = USBD_FAIL; + } if ((req->wLength == 0U) && (ret == USBD_OK)) { @@ -215,14 +238,26 @@ USBD_StatusTypeDef USBD_StdEPReq(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef { USBD_EndpointTypeDef *pep; uint8_t ep_addr; + uint8_t idx; USBD_StatusTypeDef ret = USBD_OK; + ep_addr = LOBYTE(req->wIndex); switch (req->bmRequest & USB_REQ_TYPE_MASK) { case USB_REQ_TYPE_CLASS: case USB_REQ_TYPE_VENDOR: - ret = (USBD_StatusTypeDef)pdev->pClass->Setup(pdev, req); + /* Get the class index relative to this endpoint */ + idx = USBD_CoreFindEP(pdev, ep_addr); + if (((uint8_t)idx != 0xFFU) && (idx < USBD_MAX_SUPPORTED_CLASS)) + { + pdev->classId = idx; + /* Call the class data out function to manage the request */ + if (pdev->pClass[idx]->Setup != NULL) + { + ret = (USBD_StatusTypeDef)pdev->pClass[idx]->Setup(pdev, req); + } + } break; case USB_REQ_TYPE_STANDARD: @@ -285,7 +320,18 @@ USBD_StatusTypeDef USBD_StdEPReq(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef (void)USBD_LL_ClearStallEP(pdev, ep_addr); } (void)USBD_CtlSendStatus(pdev); - ret = (USBD_StatusTypeDef)pdev->pClass->Setup(pdev, req); + + /* Get the class index relative to this interface */ + idx = USBD_CoreFindEP(pdev, ep_addr); + if (((uint8_t)idx != 0xFFU) && (idx < USBD_MAX_SUPPORTED_CLASS)) + { + pdev->classId = idx; + /* Call the class data out function to manage the request */ + if (pdev->pClass[idx]->Setup != NULL) + { + ret = (USBD_StatusTypeDef)(pdev->pClass[idx]->Setup(pdev, req)); + } + } } break; @@ -397,7 +443,7 @@ static void USBD_GetDescriptor(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *r err++; } break; -#endif +#endif /* (USBD_LPM_ENABLED == 1U) || (USBD_CLASS_BOS_ENABLED == 1U) */ case USB_DESC_TYPE_DEVICE: pbuf = pdev->pDesc->GetDeviceDescriptor(pdev->dev_speed, &len); break; @@ -405,12 +451,30 @@ static void USBD_GetDescriptor(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *r case USB_DESC_TYPE_CONFIGURATION: if (pdev->dev_speed == USBD_SPEED_HIGH) { - pbuf = pdev->pClass->GetHSConfigDescriptor(&len); +#ifdef USE_USBD_COMPOSITE + if ((uint8_t)(pdev->NumClasses) > 0U) + { + pbuf = (uint8_t *)USBD_CMPSIT.GetHSConfigDescriptor(&len); + } + else +#endif /* USE_USBD_COMPOSITE */ + { + pbuf = (uint8_t *)pdev->pClass[0]->GetHSConfigDescriptor(&len); + } pbuf[1] = USB_DESC_TYPE_CONFIGURATION; } else { - pbuf = pdev->pClass->GetFSConfigDescriptor(&len); +#ifdef USE_USBD_COMPOSITE + if ((uint8_t)(pdev->NumClasses) > 0U) + { + pbuf = (uint8_t *)USBD_CMPSIT.GetFSConfigDescriptor(&len); + } + else +#endif /* USE_USBD_COMPOSITE */ + { + pbuf = (uint8_t *)pdev->pClass[0]->GetFSConfigDescriptor(&len); + } pbuf[1] = USB_DESC_TYPE_CONFIGURATION; } break; @@ -492,16 +556,28 @@ static void USBD_GetDescriptor(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *r default: #if (USBD_SUPPORT_USER_STRING_DESC == 1U) - if (pdev->pClass->GetUsrStrDescriptor != NULL) - { - pbuf = pdev->pClass->GetUsrStrDescriptor(pdev, (req->wValue), &len); - } - else + pbuf = NULL; + + + for (uint32_t idx = 0U; (idx < pdev->NumClasses); idx++) { - USBD_CtlError(pdev, req); - err++; + if (pdev->pClass[idx]->GetUsrStrDescriptor != NULL) + { + pdev->classId = idx; + pbuf = pdev->pClass[idx]->GetUsrStrDescriptor(pdev, LOBYTE(req->wValue), &len); + + if (pbuf == NULL) /* This means that no class recognized the string index */ + { + continue; + } + else + { + break; + } + } } -#endif + +#endif /* USBD_SUPPORT_USER_STRING_DESC */ #if (USBD_CLASS_USER_STRING_DESC == 1U) if (pdev->pDesc->GetUserStrDescriptor != NULL) @@ -513,12 +589,12 @@ static void USBD_GetDescriptor(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *r USBD_CtlError(pdev, req); err++; } -#endif +#endif /* USBD_SUPPORT_USER_STRING_DESC */ #if ((USBD_CLASS_USER_STRING_DESC == 0U) && (USBD_SUPPORT_USER_STRING_DESC == 0U)) USBD_CtlError(pdev, req); err++; -#endif +#endif /* (USBD_CLASS_USER_STRING_DESC == 0U) && (USBD_SUPPORT_USER_STRING_DESC == 0U) */ break; } break; @@ -526,7 +602,16 @@ static void USBD_GetDescriptor(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *r case USB_DESC_TYPE_DEVICE_QUALIFIER: if (pdev->dev_speed == USBD_SPEED_HIGH) { - pbuf = pdev->pClass->GetDeviceQualifierDescriptor(&len); +#ifdef USE_USBD_COMPOSITE + if ((uint8_t)(pdev->NumClasses) > 0U) + { + pbuf = (uint8_t *)USBD_CMPSIT.GetDeviceQualifierDescriptor(&len); + } + else +#endif /* USE_USBD_COMPOSITE */ + { + pbuf = (uint8_t *)pdev->pClass[0]->GetDeviceQualifierDescriptor(&len); + } } else { @@ -538,7 +623,16 @@ static void USBD_GetDescriptor(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *r case USB_DESC_TYPE_OTHER_SPEED_CONFIGURATION: if (pdev->dev_speed == USBD_SPEED_HIGH) { - pbuf = pdev->pClass->GetOtherSpeedConfigDescriptor(&len); +#ifdef USE_USBD_COMPOSITE + if ((uint8_t)(pdev->NumClasses) > 0U) + { + pbuf = (uint8_t *)USBD_CMPSIT.GetOtherSpeedConfigDescriptor(&len); + } + else +#endif /* USE_USBD_COMPOSITE */ + { + pbuf = (uint8_t *)pdev->pClass[0]->GetOtherSpeedConfigDescriptor(&len); + } pbuf[1] = USB_DESC_TYPE_OTHER_SPEED_CONFIGURATION; } else @@ -651,6 +745,7 @@ static USBD_StatusTypeDef USBD_SetConfig(USBD_HandleTypeDef *pdev, USBD_SetupReq if (ret != USBD_OK) { USBD_CtlError(pdev, req); + pdev->dev_state = USBD_STATE_ADDRESSED; } else { @@ -767,7 +862,7 @@ static void USBD_GetStatus(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) pdev->dev_config_status = USB_CONFIG_SELF_POWERED; #else pdev->dev_config_status = 0U; -#endif +#endif /* USBD_SELF_POWERED */ if (pdev->dev_remote_wakeup != 0U) { @@ -798,6 +893,15 @@ static void USBD_SetFeature(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req) pdev->dev_remote_wakeup = 1U; (void)USBD_CtlSendStatus(pdev); } + else if (req->wValue == USB_FEATURE_TEST_MODE) + { + pdev->dev_test_mode = (uint8_t)(req->wIndex >> 8); + (void)USBD_CtlSendStatus(pdev); + } + else + { + USBD_CtlError(pdev, req); + } } @@ -945,4 +1049,3 @@ static uint8_t USBD_GetLen(uint8_t *buf) * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_desc_template.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_desc_template.c index 8558ebd7..a9f995fa 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_desc_template.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_desc_template.c @@ -8,13 +8,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -28,7 +27,7 @@ /* Private define ------------------------------------------------------------*/ #define USBD_VID 0x0483 #define USBD_PID 0xaaaa /* Replace '0xaaaa' with your device product ID */ -#define USBD_LANGID_STRING 0xbbb /* Replace '0xbbb' with your device language ID */ +#define USBD_LANGID_STRING 0xbbb /* Replace '0xbbb' with your device language ID */ #define USBD_MANUFACTURER_STRING "xxxxx" /* Add your manufacturer string */ #define USBD_PRODUCT_HS_STRING "xxxxx" /* Add your product High Speed string */ #define USBD_PRODUCT_FS_STRING "xxxxx" /* Add your product Full Speed string */ @@ -53,7 +52,7 @@ uint8_t *USBD_Class_UserStrDescriptor(USBD_SpeedTypeDef speed, uint8_t idx, uint #if ((USBD_LPM_ENABLED == 1) || (USBD_CLASS_BOS_ENABLED == 1)) uint8_t *USBD_USR_BOSDescriptor(USBD_SpeedTypeDef speed, uint16_t *length); -#endif +#endif /* (USBD_LPM_ENABLED == 1) || (USBD_CLASS_BOS_ENABLED == 1) */ /* Private variables ---------------------------------------------------------*/ USBD_DescriptorsTypeDef Class_Desc = @@ -67,17 +66,17 @@ USBD_DescriptorsTypeDef Class_Desc = USBD_Class_InterfaceStrDescriptor, #if (USBD_CLASS_USER_STRING_DESC == 1) USBD_CLASS_UserStrDescriptor, -#endif +#endif /* USB_CLASS_USER_STRING_DESC */ #if ((USBD_LPM_ENABLED == 1) || (USBD_CLASS_BOS_ENABLED == 1)) USBD_USR_BOSDescriptor, -#endif +#endif /* (USBD_LPM_ENABLED == 1) || (USBD_CLASS_BOS_ENABLED == 1) */ }; /* USB Standard Device Descriptor */ #if defined ( __ICCARM__ ) /*!< IAR Compiler */ #pragma data_alignment=4 -#endif +#endif /* __ICCARM__ */ __ALIGN_BEGIN uint8_t USBD_DeviceDesc[USB_LEN_DEV_DESC] __ALIGN_END = { 0x12, /* bLength */ @@ -87,7 +86,7 @@ __ALIGN_BEGIN uint8_t USBD_DeviceDesc[USB_LEN_DEV_DESC] __ALIGN_END = in order to support BOS Desc */ #else 0x00, /* bcdUSB */ -#endif +#endif /* (USBD_LPM_ENABLED == 1) || (USBD_CLASS_BOS_ENABLED == 1) */ 0x02, 0x00, /* bDeviceClass */ 0x00, /* bDeviceSubClass */ @@ -110,7 +109,7 @@ __ALIGN_BEGIN uint8_t USBD_DeviceDesc[USB_LEN_DEV_DESC] __ALIGN_END = #if (USBD_LPM_ENABLED == 1) #if defined ( __ICCARM__ ) /*!< IAR Compiler */ #pragma data_alignment=4 -#endif +#endif /* __ICCARM__ */ __ALIGN_BEGIN uint8_t USBD_BOSDesc[USB_SIZ_BOS_DESC] __ALIGN_END = { 0x5, @@ -127,13 +126,13 @@ __ALIGN_BEGIN uint8_t USBD_BOSDesc[USB_SIZ_BOS_DESC] __ALIGN_END = 0x0, 0x0 }; -#endif +#endif /* USBD_LPM_ENABLED */ /* USB Device Billboard BOS descriptor Template */ #if (USBD_CLASS_BOS_ENABLED == 1) #if defined ( __ICCARM__ ) /*!< IAR Compiler */ #pragma data_alignment=4 -#endif +#endif /* __ICCARM__ */ __ALIGN_BEGIN uint8_t USBD_BOSDesc[USB_SIZ_BOS_DESC] __ALIGN_END = { 0x05, /* bLength */ @@ -156,14 +155,16 @@ __ALIGN_BEGIN uint8_t USBD_BOSDesc[USB_SIZ_BOS_DESC] __ALIGN_END = 0x34, /* bLength */ 0x10, /* bDescriptorType: DEVICE CAPABILITY Type */ 0x0D, /* bDevCapabilityType: BILLBOARD_CAPABILITY */ - USBD_BB_URL_STRING_INDEX, /* iAddtionalInfoURL: Index of string descriptor providing a URL where the user can go to get more - detailed information about the product and the various Alternate Modes it supports */ + USBD_BB_URL_STRING_INDEX, /* iAddtionalInfoURL: Index of string descriptor providing a URL where the user + can go to get more detailed information about the product and the various + Alternate Modes it supports */ 0x02, /* bNumberOfAlternateModes: Number of Alternate modes supported. The maximum value that this field can be set to is MAX_NUM_ALT_MODE. */ 0x00, /* bPreferredAlternateMode: Index of the preferred Alternate Mode. System - software may use this information to provide the user with a better user experience. */ + software may use this information to provide the user with a better + user experience. */ 0x00, 0x00, /* VCONN Power needed by the adapter for full functionality 000b = 1W */ @@ -184,8 +185,8 @@ __ALIGN_BEGIN uint8_t USBD_BOSDesc[USB_SIZ_BOS_DESC] __ALIGN_END = 0x00, /* bAlternateMode[0] Index of the Alternate Mode within the SVID as returned in response to a Discover Modes command. Example: - 0 – first Mode entry - 1 – second mode entry */ + 0 first Mode entry + 1 second mode entry */ USBD_BB_ALTMODE0_STRING_INDEX, /* iAlternateModeString[0]: Index of string descriptor describing protocol. It is optional to support this string. */ @@ -195,10 +196,10 @@ __ALIGN_BEGIN uint8_t USBD_BOSDesc[USB_SIZ_BOS_DESC] __ALIGN_END = 0x01, /* bAlternateMode[1] Index of the Alternate Mode within the SVID as returned in response to a Discover Modes command. Example: - 0 – first Mode entry - 1 – second Mode entry */ + 0 first Mode entry + 1 second Mode entry */ - USBD_BB_ALTMODE1_STRING_INDEX, /* iAlternateModeString[1]: Index of string descriptor describing protocol. + USBD_BB_ALTMODE1_STRING_INDEX, /* iAlternateModeString[1]: Index of string descriptor describing protocol. It is optional to support this string. */ /* Alternate Mode Desc */ /* ----------- Device Capability Descriptor: BillBoard Alternate Mode Desc ---------- */ @@ -206,21 +207,23 @@ __ALIGN_BEGIN uint8_t USBD_BOSDesc[USB_SIZ_BOS_DESC] __ALIGN_END = 0x10, /* bDescriptorType: Device Descriptor Type */ 0x0F, /* bDevCapabilityType: BILLBOARD ALTERNATE MODE CAPABILITY */ 0x00, /* bIndex: Index of Alternate Mode described in the Billboard Capability Desc */ - 0x10, 0x00, 0x00, 0x00, /* dwAlternateModeVdo: contents of the Mode VDO for the alternate mode identified by bIndex */ + 0x10, 0x00, 0x00, 0x00, /* dwAlternateModeVdo: contents of the Mode VDO for the alternate mode + identified by bIndex */ 0x08, /* bLength */ 0x10, /* bDescriptorType: Device Descriptor Type */ 0x0F, /* bDevCapabilityType: BILLBOARD ALTERNATE MODE CAPABILITY */ 0x01, /* bIndex: Index of Alternate Mode described in the Billboard Capability Desc */ - 0x20, 0x00, 0x00, 0x00, /* dwAlternateModeVdo: contents of the Mode VDO for the alternate mode identified by bIndex */ + 0x20, 0x00, 0x00, 0x00, /* dwAlternateModeVdo: contents of the Mode VDO for the alternate mode + identified by bIndex */ }; -#endif +#endif /* USBD_CLASS_BOS_ENABLED */ /* USB Standard Device Descriptor */ #if defined ( __ICCARM__ ) /*!< IAR Compiler */ #pragma data_alignment=4 -#endif +#endif /* __ICCARM__ */ __ALIGN_BEGIN uint8_t USBD_LangIDDesc[USB_LEN_LANGID_STR_DESC] __ALIGN_END = { USB_LEN_LANGID_STR_DESC, @@ -231,7 +234,7 @@ __ALIGN_BEGIN uint8_t USBD_LangIDDesc[USB_LEN_LANGID_STR_DESC] __ALIGN_END = #if defined ( __ICCARM__ ) /*!< IAR Compiler */ #pragma data_alignment=4 -#endif +#endif /* __ICCARM__ */ __ALIGN_BEGIN uint8_t USBD_StringSerial[USB_SIZ_STRING_SERIAL] = { USB_SIZ_STRING_SERIAL, @@ -240,7 +243,7 @@ __ALIGN_BEGIN uint8_t USBD_StringSerial[USB_SIZ_STRING_SERIAL] = #if defined ( __ICCARM__ ) /*!< IAR Compiler */ #pragma data_alignment=4 -#endif +#endif /* __ICCARM__ */ __ALIGN_BEGIN uint8_t USBD_StrDesc[USBD_MAX_STR_DESC_SIZ] __ALIGN_END; /* Private functions ---------------------------------------------------------*/ @@ -371,7 +374,9 @@ uint8_t *USBD_Class_InterfaceStrDescriptor(USBD_SpeedTypeDef speed, uint16_t *le */ static void Get_SerialNum(void) { - uint32_t deviceserial0, deviceserial1, deviceserial2; + uint32_t deviceserial0; + uint32_t deviceserial1; + uint32_t deviceserial2; deviceserial0 = *(uint32_t *)DEVICE_ID1; deviceserial1 = *(uint32_t *)DEVICE_ID2; @@ -400,7 +405,7 @@ uint8_t *USBD_USR_BOSDescriptor(USBD_SpeedTypeDef speed, uint16_t *length) *length = sizeof(USBD_BOSDesc); return (uint8_t *)USBD_BOSDesc; } -#endif +#endif /* (USBD_LPM_ENABLED == 1) || (USBD_CLASS_BOS_ENABLED == 1) */ #if (USBD_CLASS_USER_STRING_DESC == 1) @@ -417,7 +422,7 @@ uint8_t *USBD_Class_UserStrDescriptor(USBD_SpeedTypeDef speed, uint8_t idx, uint return USBD_StrDesc; } -#endif +#endif /* USBD_CLASS_USER_STRING_DESC */ /** @@ -447,5 +452,3 @@ static void IntToUnicode(uint32_t value, uint8_t *pbuf, uint8_t len) pbuf[2U * idx + 1] = 0U; } } - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_ioreq.c b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_ioreq.c index 1e7e62d4..7c8004ad 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_ioreq.c +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Core/Src/usbd_ioreq.c @@ -6,13 +6,12 @@ ****************************************************************************** * @attention * - * <h2><center>© Copyright (c) 2015 STMicroelectronics. - * All rights reserved.</center></h2> + * Copyright (c) 2015 STMicroelectronics. + * All rights reserved. * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 + * This software is licensed under terms that can be found in the LICENSE file + * in the root directory of this software component. + * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ @@ -95,7 +94,7 @@ USBD_StatusTypeDef USBD_CtlSendData(USBD_HandleTypeDef *pdev, pdev->ep_in[0].rem_length = 0U; #else pdev->ep_in[0].rem_length = len; -#endif +#endif /* USBD_AVOID_PACKET_SPLIT_MPS */ /* Start the transfer */ (void)USBD_LL_Transmit(pdev, 0x00U, pbuf, len); @@ -139,7 +138,7 @@ USBD_StatusTypeDef USBD_CtlPrepareRx(USBD_HandleTypeDef *pdev, pdev->ep_out[0].rem_length = 0U; #else pdev->ep_out[0].rem_length = len; -#endif +#endif /* USBD_AVOID_PACKET_SPLIT_MPS */ /* Start the transfer */ (void)USBD_LL_PrepareReceive(pdev, 0U, pbuf, len); @@ -223,4 +222,3 @@ uint32_t USBD_GetRxCount(USBD_HandleTypeDef *pdev, uint8_t ep_addr) * @} */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/LICENSE.md b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/LICENSE.md new file mode 100644 index 00000000..1af52330 --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/LICENSE.md @@ -0,0 +1,80 @@ +SLA0044 Rev5/February 2018 + +## Software license agreement + +### __ULTIMATE LIBERTY SOFTWARE LICENSE AGREEMENT__ + +BY INSTALLING, COPYING, DOWNLOADING, ACCESSING OR OTHERWISE USING THIS SOFTWARE +OR ANY PART THEREOF (AND THE RELATED DOCUMENTATION) FROM STMICROELECTRONICS +INTERNATIONAL N.V, SWISS BRANCH AND/OR ITS AFFILIATED COMPANIES +(STMICROELECTRONICS), THE RECIPIENT, ON BEHALF OF HIMSELF OR HERSELF, OR ON +BEHALF OF ANY ENTITY BY WHICH SUCH RECIPIENT IS EMPLOYED AND/OR ENGAGED AGREES +TO BE BOUND BY THIS SOFTWARE LICENSE AGREEMENT. + +Under STMicroelectronics’ intellectual property rights, the redistribution, +reproduction and use in source and binary forms of the software or any part +thereof, with or without modification, are permitted provided that the following +conditions are met: + +1. Redistribution of source code (modified or not) must retain any copyright +notice, this list of conditions and the disclaimer set forth below as items 10 +and 11. + +2. Redistributions in binary form, except as embedded into microcontroller or +microprocessor device manufactured by or for STMicroelectronics or a software +update for such device, must reproduce any copyright notice provided with the +binary code, this list of conditions, and the disclaimer set forth below as +items 10 and 11, in documentation and/or other materials provided with the +distribution. + +3. Neither the name of STMicroelectronics nor the names of other contributors to +this software may be used to endorse or promote products derived from this +software or part thereof without specific written permission. + +4. This software or any part thereof, including modifications and/or derivative +works of this software, must be used and execute solely and exclusively on or in +combination with a microcontroller or microprocessor device manufactured by or +for STMicroelectronics. + +5. No use, reproduction or redistribution of this software partially or totally +may be done in any manner that would subject this software to any Open Source +Terms. “Open Source Terms†shall mean any open source license which requires as +part of distribution of software that the source code of such software is +distributed therewith or otherwise made available, or open source license that +substantially complies with the Open Source definition specified at +www.opensource.org and any other comparable open source license such as for +example GNU General Public License (GPL), Eclipse Public License (EPL), Apache +Software License, BSD license or MIT license. + +6. STMicroelectronics has no obligation to provide any maintenance, support or +updates for the software. + +7. The software is and will remain the exclusive property of STMicroelectronics +and its licensors. The recipient will not take any action that jeopardizes +STMicroelectronics and its licensors' proprietary rights or acquire any rights +in the software, except the limited rights specified hereunder. + +8. The recipient shall comply with all applicable laws and regulations affecting +the use of the software or any part thereof including any applicable export +control law or regulation. + +9. Redistribution and use of this software or any part thereof other than as +permitted under this license is void and will automatically terminate your +rights under this license. + +10. THIS SOFTWARE IS PROVIDED BY STMICROELECTRONICS AND CONTRIBUTORS "AS IS" AND +ANY EXPRESS, IMPLIED OR STATUTORY WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +IMPLIED WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NON-INFRINGEMENT OF THIRD PARTY INTELLECTUAL PROPERTY RIGHTS, WHICH ARE +DISCLAIMED TO THE FULLEST EXTENT PERMITTED BY LAW. IN NO EVENT SHALL +STMICROELECTRONICS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, +INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR +PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE +OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF +ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +11. EXCEPT AS EXPRESSLY PERMITTED HEREUNDER, NO LICENSE OR OTHER RIGHTS, WHETHER +EXPRESS OR IMPLIED, ARE GRANTED UNDER ANY PATENT OR OTHER INTELLECTUAL PROPERTY +RIGHTS OF STMICROELECTRONICS OR ANY THIRD PARTY. diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/README.md b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/README.md new file mode 100644 index 00000000..c25a9191 --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/README.md @@ -0,0 +1,36 @@ +# Middleware USB Device MCU Component + +![latest tag](https://img.shields.io/github/v/tag/STMicroelectronics/stm32_mw_usb_device.svg?color=brightgreen) + +## Overview + +**STM32Cube** is an STMicroelectronics original initiative to ease developers' life by reducing efforts, time and cost. + +**STM32Cube** covers the overall STM32 products portfolio. It includes a comprehensive embedded software platform delivered for each STM32 series. + * The CMSIS modules (core and device) corresponding to the ARM(tm) core implemented in this STM32 product. + * The STM32 HAL-LL drivers, an abstraction layer offering a set of APIs ensuring maximized portability across the STM32 portfolio. + * The BSP drivers of each evaluation, demonstration or nucleo board provided for this STM32 series. + * A consistent set of middleware libraries such as RTOS, USB, FatFS, graphics, touch sensing library... + * A full set of software projects (basic examples, applications, and demonstrations) for each board provided for this STM32 series. + +Two models of publication are proposed for the STM32Cube embedded software: + * The monolithic **MCU Package**: all STM32Cube software modules of one STM32 series are present (Drivers, Middleware, Projects, Utilities) in the repository (usual name **STM32Cubexx**, xx corresponding to the STM32 series). + * The **MCU component**: each STM32Cube software module being part of the STM32Cube MCU Package, is delivered as an individual repository, allowing the user to select and get only the required software functions. + +## Description + +This **stm32_mw_usb_device** MCU component repository is one element **common to all** STM32Cube MCU embedded software packages, providing the **USB Device MCU Middleware** part. + +It represents ST offer to ensure the support of USB Devices on STM32 MCUs. +It includes two main modules: + * **Core** module for the USB device standard peripheral control APIs. It includes the files ensuring USB 2.0 standard code implementation for an USB device. +These files’ APIs will be called within every USB device application regardless of the desired functionality. + * **Class** module for the commonly supported classes APIs. it includes the files including different USB device classes. All STM32 USB classes are implemented according to the USB 2.0 and every class’s specifications. These files’ APIs will be called within USB device applications according to the desired functionality. + +## Release note + +Details about the content of this release are available in the release note [here](https://htmlpreview.github.io/?https://github.com/STMicroelectronics/stm32_mw_usb_device/blob/master/Release_Notes.html). + +## Troubleshooting + +Please refer to the [CONTRIBUTING.md](CONTRIBUTING.md) guide. \ No newline at end of file diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Release_Notes.html b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Release_Notes.html index cf3544b2..9744d73f 100644 --- a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Release_Notes.html +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/Release_Notes.html @@ -1,1517 +1,1020 @@ -<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> -<html xmlns:v="urn:schemas-microsoft-com:vml" xmlns:o="urn:schemas-microsoft-com:office:office" xmlns:w="urn:schemas-microsoft-com:office:word" xmlns:m="http://schemas.microsoft.com/office/2004/12/omml" xmlns="http://www.w3.org/TR/REC-html40"><head> - - - - - - - - - - - - - - - - -<meta http-equiv="Content-Type" content="text/html; charset=windows-1252"> -<link rel="File-List" href="Release_Notes_for_STM32F2xx_StdPeriph_Driver_files/filelist.xml"> -<link rel="Edit-Time-Data" href="Release_Notes_for_STM32F2xx_StdPeriph_Driver_files/editdata.mso"><!--[if !mso]> -<style> -v\:* {behavior:url(#default#VML);} -o\:* {behavior:url(#default#VML);} -w\:* {behavior:url(#default#VML);} -.shape {behavior:url(#default#VML);} -</style> -<![endif]--><title>Release Notes for STM32 USB Device Library - - - - - - - - - -
- -

 

- -
- - - - + + +
- - - - - - - -
-

Back to Release page

-
-

Release Notes for STM32 USB Device Library

-

Copyright - 2015 STMicroelectronics

-

-
-

 

- - - - -
-

Update History

- -

V2.8.0 / 10-Mars-2021

Main -Changes

  • Integration of  new USB device Class driver:
    • USB Printer Class driver based on Universal Serial Bus -Device Class Definition -for -Printing Devices Version 1.1
  • USB All Classes:
    • Fix USB buffer overflow vulnerability for CDC, CDC-ECM, CDC-RNDIS, DFU, AUDIO, CustomHID, and Video Classes
    • fix compilation warning with C++ due to missing casting during class handler allocation
    • Enhance comments of USB configuration descriptors fields
  • USB Video Class:
    • fix missing closing bracket for extern "C" in usbd_video.h
    • fix USBCV test with Uncompressed video format support

V2.7.1 / 18-August-2020

-

Main -Changes

-
    -
  • USB All Class:
  • -
      -
    • Add NULL pointer access check to Class handler
      -
    • -
    -
- -

V2.7.0 / 12-August-2020

-

Main -Changes

  • Integration of  new USB device Class driver:
    • USB video Class driver based on USB-IF video class definition version 1.1
- - - - - - - - - - - - - - - - - - - - - - - -
  • USB Core:
    • Enhance NULL pointer check in Core APIs
    • Allow supporting both USER and USER Class string desc
    • Add support of USB controller which handles packet-size splitting by hardware
    • Avoid compilation warning due macro redefinition
    • change -added to USBD_HandleTypeDef structure: dev_state, old_dev_state and -ep0_state declaration become volatile to disable compiler optimization
    • Word spelling correction and file indentation improved
    • usbd_conf.h/c Template file updated to suggest using by default a static memory allocation for Class handler
  • USB All Classes
    • Word spelling correction and file indentation improved
    • Allow updating device config descriptor Max power from user code usbd_conf.h using USBD_MAX_POWER define
    • Fix device config descriptor bmAttributes value which depends on user code define USBD_SELF_POWERED
  • USB CDC Class:
    • Class specific request, add protection to limit the maximum data length to be sent by the CDC device
  • USB CustomHID Class:
    • Allow changing CustomHID data EP size from user code
-

V2.6.1 / 05-June-2020

- - - -

Main -Changes

- - - - - - - - - - - - - - - - - - - - - - - -
    -
  • USB Core:
  • -
      -
    • minor rework on USBD_Init() USBD_DeInit()
    • -
    • Fix warning issue with Keil due to missing return value of setup API
      -
    • -
    -
  • USB CDC Class:
  • -
      -
    • Fix file indentation
    • -
    • Avoid accessing to NULL pointer in case + + + + + + + Release Notes for STM32Cube USB Device Library + + + + + + +
      +
      +
      +

      Release Notes +for STM32Cube USB Device Library

      +

      Copyright © 2015 STMicroelectronics
      +

      + +
      +

      Purpose

      +

      The USB device library comes on top of the STM32Cubeâ„¢ USB device HAL +driver and offers all the APIs required to develop an USB device +application.

      +

      The main USB device library features are:

      +
        +
      • Support of multi packet transfer features allowing sending big +amount of data without splitting it into max packet size transfers.
      • +
      • Support of most common USB Class drivers (HID, MSC, DFU, CDC-ACM, +CDC-ECM, RNDIS, MTP, AUDIO1.0, Printer, Video, Composite)
      • +
      • Configuration files to interface with Cube HAL and change the +library configuration without changing the library code (Read +Only).
      • +
      • 32-bits aligned data structures to handle DMA based transfer in High +speed modes.
      • +
      +

      Here is the list of references to user documents:

      +
        +
      • UM1734 +: STM32Cube USB device library User Manual
      • +
      • Wiki +Page : STM32Cube USB Wiki Page
      • +
      +
      +
      +

      Update History

      +
      + + +

      Main Changes

      + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
      Headline
      Improvement of the memory management +
      USB Core:
      Fix some compilation warnings related +to unused parameters
      Improve some code parts style
      Add check on the USB Device status in +USBD_LL_Suspend before suspending it
      USB CDC-ACM Class:
      Remove redundant prototype declaration of +USBD_CDC_GetOtherSpeedCfgDesc()
      USB CompositeBuilder, CCID, CDC_ECM, CDC_RNDIS, +CustomHID, MSC & Video Classes:
      Improve some code parts style
      +
      +
      + + +

      Main Changes

      + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
      Headline
      USB VIDEO Class:
      Correction of the support of +VS_PROBE_CONTROL & VS_COMMIT_CONTROL requets
      USB AUDIO Class:
      Correction of the check on +AUDIO_TOTAL_BUF_SIZE to avoid vulnerabilities
      USB HID Class:
      Modification of some constants names to +avoid duplication versus USB host library
      USB CustomHID Class:
      Add support of Get Report control +request
      Allow disabling EP OUT prepare receive +using a dedicated macros that can be defined in usbd_conf.h application +file
      Add support of Report Descriptor with +length greater than 255 bytes
      USB CCID Class:
      Fix minor Code Spelling warning
      USB All Classes:
      Update all classes to support composite +multi-instance using the class id parameter
      Fix code spelling and improve code +style
      fix misraC 2012 rule 10.3
      +
      +
      + + +

      Main Changes

      + + + + + + + + + + + + + + + + + + + + + + + + + + +
      Headline
      Integration of new USB device +Class driver:
      Adding support of Composite devices with +an auto generation of composite device configuration descriptors
      USB All Classes:
      Fix Code Spelling and improve Code +Style
      Update device class drivers to support +Composite devices
      Improve declaration of USB configuration +descriptor table which is allocated if the composite builder is not +selected
      +
      +
      + + +

      Main Changes

      + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
      Headline
      Integration of new USB device +Class driver:
      USB CCID Class driver based on Universal +Serial Bus Device Class Definition for Integrated Circuit(s) Cards +Interface Devices Revision 1.1
      USB MTP Class driver based on Universal +Serial Bus Device Class Media Transfer Protocol Revision 1.1
      USB All Classes:
      Fix Code Spelling and improve Code +Style
      Update the way to declare licenses
      USB CDC/RNDIS/ECM +Classes:
      Fix compilation warning with C++ due to +missing casting during class handler allocation
      +
      +
      + + +

      Main Changes

      + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
      Headline
      Integration of new USB device +Class driver:
      USB Printer Class driver based on +Universal Serial Bus Device Class Definition for Printing Devices +Version 1.1
      USB All Classes:
      Fix USB buffer overflow vulnerability for +CDC, CDC-ECM, CDC-RNDIS, DFU, AUDIO, CustomHID, and Video Classes
      Fix compilation warning with C++ due to +missing casting during class handler allocation
      Enhance comments of USB configuration +descriptors fields
      USB Video Class:
      Fix missing closing bracket for extern “C†+in usbd_video.h
      Fix USBCV test with Uncompressed video +format support
      +
      +
      + + +

      Main Changes

      + + + + + + + + + + + +
      Headline
      USB All Class: Add NULL pointer access +check to Class handler
      +
      +
      + + +

      Main Changes

      + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
      Headline
      Integration of new USB device +Class driver:
      USB video Class driver based on USB-IF +video class definition version 1.1
      USB Core:
      Enhance NULL pointer check in Core +APIs
      Allow supporting both USER and USER Class +string desc
      Add support of USB controller which +handles packet-size splitting by hardware
      Avoid compilation warning due macro +redefinition
      change added to USBD_HandleTypeDef +structure: dev_state, old_dev_state and ep0_state declaration become +volatile to disable compiler optimization
      Word spelling correction and file +indentation improved
      usbd_conf.h/c Template file updated to +suggest using by default a static memory allocation for Class +handler
      USB All Classes:
      Word spelling correction and file +indentation improved
      Allow updating device config descriptor +Max power from user code usbd_conf.h using USBD_MAX_POWER define
      Fix device config descriptor bmAttributes +value which depends on user code define USBD_SELF_POWERED
      USB CDC Class:
      Class specific request, add protection to +limit the maximum data length to be sent by the CDC device
      USB CustomHID Class:
      Allow changing CustomHID data EP size from +user code
      +
      +
      + + +

      Main Changes

      + + + + + + + + + + + + + + + + + + + + + + + + + + + - -
      Headline
      Fix minor misra-c 2012 violations
      USB Core:
      minor rework on USBD_Init() +USBD_DeInit()
      Fix warning issue with Keil due to missing +return value of setup API
      USB CDC Class:
      Fix file indentation
      Avoid accessing to NULL pointer in case TransmitCplt() user fops is not defined to allow application -compatibility with device library version below v2.6.0
      - - - -
      -
        -
      • Fix minor misra-c 2012 violations
      • -
      -

      V2.6.0 / 27-December-2019

      -

      Main -Changes

      - - - - - - - - - - - - - - - - - - - - - -
      • Integration of three new USB device Class drivers:
        • USB CDC ECM Class driver
        • USB CDC RNDIS Microsoft Class driver
        • USB Billboard Class driver
      • Fix mandatory misra-c 2012 violations
      • update user core and class template files
      • USB Core:
        • Fix unexpected EP0 stall during enumeration phase 
        • Improve APIs error management and prevent accessing NULL pointers
      • USB MSC Class:
        • Fix USBCV specific class tests
        • Fix multiple error with SCSI commands handling
        • Protect medium access when host ask for medium ejection
      • USB CDC Class:
        • Add new function to inform user that current IN transfer is completed
        • update transmit and receive APIs to transfer up to 64KB
      • USB AUDIO Class:
        • Fix audio sync start buffer size
        • update user callback periodicTC args by adding pointer to user buffer and size
      • USB CustomHID Class:
        • Rework the OUT transfer complete and prevent automatically re-enabling the OUT EP 
        • Add new user API to restart the OUT transfer: USBD_CUSTOM_HID_ReceivePacket()

      V2.5.3 / 30-April-2019

      Main -Changes

      - - - - - - - - - - - - - - - - - - - - - -
      • Fix misra-c 2012 high severity violations
      • Core driver:
        • protect shared macros __ALIGN_BEGIN, __ALIGN_END with C directive #ifndef
        • update Core driver and DFU Class driver to use USBD_SUPPORT_USER_STRING_DESC instead of  USBD_SUPPORT_USER_STRING
        •  prevent accessing to NULL pointer if the get descriptor functions are not defined
        • Update on USBD_LL_Resume(),  restore the device state only if the current state is USBD_STATE_SUSPENDED

      V2.5.2 / 27-Mars-2019

      Main -Changes

      - - - - - - - - - - - - - - - - - - - - - -
      • DFU Class:
        • fix compilation warning due to unreachable - instruction code introduced with CMSIS V5.4.0 NVIC_SystemReset() prototype change

      V2.5.1 / 03-August-2018
      -

      - - - - - - - - - - - - - - - - - -

      Main -Changes

      - - - - - - - - - - - - - - - - - - - - - -
      • Update license section by adding path to get copy of ST Ultimate Liberty license
      • Core: Fix unexpected stall during status OUT phase
      • DFU Class:
        • rework hdfu struct to prevent unaligned addresses
      • MSC Class:
        • fix lba address overflow during large file transfers > 4Go
      • Template Class:
        • add missing Switch case Break on USBD_Template_Setup API

      V2.5.0 / 15-December-2017
      -

      - - - - - - - - - - - - - - - - - -

      Main -Changes

      - - - - - - - - - - - - - - - - - - - - - -
      • Update license section
      • Update some functions to be MISRAC 2004 compliant
      • Add HS and OtherSpeed configuration descriptor for HID and CustomHID classes
      • Correct error handling in all class setup function
      • Add usbd_desc_template.c/ usbd_desc_template.h templates files
      • Add support of class and vendor request
      • CDC Class: fix zero-length packet issue in bulk IN transfer
      • Fix compilation warning with unused arguments for some functions
      • Improve USB Core enumeration state machine

      V2.4.2 / 11-December-2015
      -

      - - - - - - - - - - - - - - - - - -

      Main -Changes

      - - - - - - - - - - - - - - - - - - - - - -
        -
      • CDC Class
        • usbd_cdc.c: change #include "USBD_CDC.h" by #include "usbd_cdc.h"
        -
      -
      - -

      V2.4.1 / 19-June-2015
      -

      - - - - - - - - - - - - - - - -

      Main -Changes

      - - - - - - - - - - - - - - - - - - - -
        -
      • CDC Class
      • -
          -
        • usbd_cdc.c: comments update
        • -
        -
      • MSC Class
      • -
          -
        • usbd_msc_bot.h: update to be C++ compliant
        • -
        -
      • AUDIO Class
      • -
          -
        • usbd_audio.c: fix issue when Host sends GetInterface command it gets a wrong value
        • -
        -
          -
        • usbd_audio.c: remove useless management of DMA half transfer
          -
        • -
        -
      - - - -

      V2.4.0 / 28-February-2015
      -

      - - - - - - - - - - - - - -

      Main -Changes

      - - - - - - - - - - - - - - - - - -
        -
      • Core Driver
      • -
          -
        • Add support of Link Power Management (LPM): add new API GetBOSDescriptor(), that is used only if USBD_LPM_ENABLED switch is enabled in usbd_conf.h file
        • usbd_core.c: -Fix bug of unsupported premature Host Out stage during data In stage -(ie. when endpoint 0 maximum data size is 8 and Host requests -GetDeviceDescriptor for the first time)
        • usbd_ctlreq.c: Fix bug of unsupported Endpoint Class requests (ie. Audio SetCurrent request for endpoint sampling rate setting)
        • -
        -
      • HID Class
      • -
          -
        • Updating Polling time API USBD_HID_GetPollingInterval() to query this period for HS and FS
        • usbd_hid.c: Fix USBD_LL_CloseEP() function call in USBD_HID_DeInit() replacing endpoint size by endpoint address.
        • -
      • CDC Class
        • usbd_cdc.c: 
          • Add missing GetInterface request management in USBD_CDC_Setup() function
          • Update -USBD_CDC_Setup() function to allow correct user implementation of -CDC_SET_CONTROL_LINE_STATE and similar no-data setup requests.
        -
      - -

      V2.3.0 / 04-November-2014
      -

      - - - - - - - - - - - -

      Main -Changes

      - - - - - - - - - - - - - - - -
        -
      • Update all drivers to be C++ compliant
        -
      • -
      • CDC Class
      • -
          -
        • usbd_cdc.c: fix clear flag issue in USBD_CDC_TransmitPacket() function
        • -
        -
          -
        • usbd_cdc_if_template.c: update TEMPLATE_Receive() function header comment
          -
        • -
        -
      • Miscellaneous source code comments update
      • -
      -

      V2.2.0 / 13-June-2014

      - - - - - - - - - -

      Main -Changes

      - - - - - - - - - - - - - -
        -
      • Source code comments review and update
      • -
      • HID class
      • -
          -
        • Remove unused API USBD_HID_DeviceQualifierDescriptor()
        • -
        • Add a new API in the HID class to query the poll time USBD_HID_GetPollingInterval()
          -
        • -
        - -
      • CDC class
      • -
          -
        • Bug fix: missing handling ZeroLength Setup request
        • -
        -
      • All classes
        -
      • - -
          -
        • Add alias for the class definition, it's defined as macro with capital letter
        • -
        -
      -
      ex. for the HID, the USBD_HID_CLASS macro is defined this way #define USBD_HID_CLASS  &USBD_HID
        and the application code can use the previous definition: &USBD_HID ex. USBD_RegisterClass(&USBD_Device, &USBD_HID) or the new USBD_HID_CLASS ex. USBD_RegisterClass(&USBD_Device, USBD_HID_CLASS)
      -

      V2.1.0 / 22-April-2014

      - - - - - - - - -

      Main -Changes

      - - - - - - - - - - - -
        -
      • usbd_conf_template.c: update file with the right content (it was using MSC memory management layer)
        -
      • -
      • usbd_conf_template.h: change include of stm32f4xx.h by stm32xxx.h and add comment to inform user to adapt it to the device used
      • -
      • Several enhancements in CustomHID class
      • -
          -
        • Update the Custom HID class driver to simplify the link with user processes
        • -
        • Optimize the Custom HID class driver and reduce footprint
        • -
        • Add USBD_CUSTOM_HID_RegisterInterface() API to link user process to custom HID class
        • -
        • Add Custom HID interface template file usbd_customhid_if_template.c/h
        • -
        -
      • Miscellaneous comments update
        -
      • - -
      - -

      V2.0.0 / 18-February-2014

      - - - - - -

      Main -Changes

      - - - - - - - - - -
        -
      • Major update -based on STM32Cube specification: Library Core, Classes architecture and APIs -modified vs. V1.1.0, and thus the 2 versions are not compatible.
        -
      • This version has to be used only with STM32Cube based development
      • -
      - - -

      V1.1.0 / 19-March-2012

      -

      Main -Changes

      - -
      • Official support of STM32F4xx devices
      • All source files: license disclaimer text update and add link to the License file on ST Internet.
      • Handle test mode in the set feature request
      • Handle dynamically the USB SELF POWERED feature
      • Handle correctly the USBD_CtlError process to take into account error during Control OUT stage
      • Miscellaneous bug fix

      V1.0.0 / 22-July-2011

      Main -Changes

      -
      • First official version for STM32F105/7xx and STM32F2xx devices

      -

      License

      This -software component is licensed by ST under Ultimate Liberty license -SLA0044, the "License"; You may not use this component except in -compliance with the License. You may obtain a copy of the License at:

      http://www.st.com/SLA0044

      - -
      -
      -
      -

      For - complete documentation on STM32 - Microcontrollers visit www.st.com/STM32

      -
      -

      -
- +compatibility with device library version below v2.6.0
- -

 

- +
+ + +

Main Changes

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Headline
Integration of three new USB device Class +drivers:CDC ECM , CDC RNDIS Microsoft, USB Billboard
Fix mandatory misra-c 2012 violations
update user core and class template +files
USB Core:
Fix unexpected EP0 stall during +enumeration phase
Improve APIs error management and prevent +accessing NULL pointers
USB MSC Class:
Fix USBCV specific class tests
Fix multiple error with SCSI commands +handling
Protect medium access when host ask for +medium ejection
USB CDC Class:
Add new function to inform user that +current IN transfer is completed
update transmit and receive APIs to +transfer up to 64KB
USB AUDIO Class:
Fix audio sync start buffer size
update user callback periodicTC args by +adding pointer to user buffer and size
USB CustomHID Class:
Rework the OUT transfer complete and +prevent automatically re-enabling the OUT EP
Add new user API to restart the OUT +transfer: USBD_CUSTOM_HID_ReceivePacket()
+
+ + +

Main Changes

+ + + + + + + + + + + + + + + + + + + + + + + + + + +
Headline
Fix misra-c 2012 high severity +violations
Core driver:
protect shared macros __ALIGN_BEGIN, +__ALIGN_END with C directive #ifndef
update Core driver and DFU Class driver to +use USBD_SUPPORT_USER_STRING_DESC instead of +USBD_SUPPORT_USER_STRING
prevent accessing to NULL pointer if the +get descriptor functions are not defined
Update on USBD_LL_Resume(), restore the +device state only if the current state is USBD_STATE_SUSPENDED
+
+
+ + +

Main Changes

+ + + + + + + + + + + +
Headline
DFU Class: fix compilation warning due to +unreachable instruction code introduced with CMSIS V5.4.0 +NVIC_SystemReset() prototype change
+
+
+ + +

Main Changes

+ + + + + + + + + + + + + + + + + + + + + + + +
Headline
Update license section by adding path to +get copy of ST Ultimate Liberty license
Core: Fix unexpected stall during status +OUT phase
DFU Class: rework hdfu struct to prevent +unaligned addresses
MSC Class: fix lba address overflow during +large file transfers greater than 4Go
Template Class: add missing Switch case +Break on USBD_Template_Setup API
+
+
+ + +

Main Changes

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Headline
Update license section
Update some functions to be MISRAC 2004 +compliant
Add HS and OtherSpeed configuration +descriptor for HID and CustomHID classes
Correct error handling in all class setup +function
Add usbd_desc_template.c/ +usbd_desc_template.h templates files
Add support of class and vendor +request
CDC Class: fix zero-length packet issue in +bulk IN transfer
Fix compilation warning with unused +arguments for some functions
Improve USB Core enumeration state +machine
+
+
+ + +

Main Changes

+ + + + + + + + + + + + + + +
Headline
CDC Class
usbd_cdc.c: change #include “USBD_CDC.h†+by #include “usbd_cdc.hâ€
+
+
+ + +

Main Changes

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Headline
CDC Class
usbd_cdc.c: comments update
MSC Class
usbd_msc_bot.h: update to be C++ +compliant
AUDIO Class
usbd_audio.c: fix issue when Host sends +GetInterface command it gets a wrong value
usbd_audio.c: remove useless management of +DMA half transfer
+
+
+ + +

Main Changes

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Headline
Core Driver
Add support of Link Power Management +(LPM): add new API GetBOSDescriptor(), that is used only if +USBD_LPM_ENABLED switch is enabled in usbd_conf.h file
usbd_core.c: Fix bug of unsupported +premature Host Out stage during data In stage (ie. when endpoint 0 +maximum data size is 8 and Host requests GetDeviceDescriptor for the +first time)
usbd_ctlreq.c: Fix bug of unsupported +Endpoint Class requests (ie. Audio SetCurrent request for endpoint +sampling rate setting)
HID Class
Updating Polling time API +USBD_HID_GetPollingInterval() to query this period for HS and FS
usbd_hid.c: Fix USBD_LL_CloseEP() function +call in USBD_HID_DeInit() replacing endpoint size by endpoint +address.
CDC Class
usbd_cdc.c: Add missing GetInterface +request management in USBD_CDC_Setup() function
usbd_cdc.c: Update USBD_CDC_Setup() +function to allow correct user implementation of +CDC_SET_CONTROL_LINE_STATE and similar no-data setup requests.
+
+
+ + +

Main Changes

+ + + + + + + + + + + + + + + + + + + + + + + +
Headline
Update all drivers to be C++ +compliant
CDC Class
usbd_cdc.c: fix clear flag issue in +USBD_CDC_TransmitPacket() function
usbd_cdc_if_template.c: update +TEMPLATE_Receive() function header comment
Miscellaneous source code comments +update
+
+
+ + +

Main Changes

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Headline
Source code comments review and +update
HID class
Remove unused API +USBD_HID_DeviceQualifierDescriptor()
Add a new API in the HID class to query +the poll time USBD_HID_GetPollingInterval()
CDC class
Bug fix: missing handling ZeroLength Setup +request
All classes
Add alias for the class definition, it’s +defined as macro with capital letter
ex. for the HID, the USBD_HID_CLASS macro +is defined this way #define USBD_HID_CLASS &USBD_HID
and the application code can use the +previous definition: &USBD_HID ex. +USBD_RegisterClass(&USBD_Device, &USBD_HID) or the new +USBD_HID_CLASS ex. USBD_RegisterClass(&USBD_Device, +USBD_HID_CLASS)
+
+
+ + +

Main Changes

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Headline
usbd_conf_template.c: update file with the +right content (it was using MSC memory management layer)
usbd_conf_template.h: change include of +stm32f4xx.h by stm32xxx.h and add comment to inform user to adapt it to +the device used
Several enhancements in CustomHID +class
Update the Custom HID class driver to +simplify the link with user processes
Optimize the Custom HID class driver and +reduce footprint
Add USBD_CUSTOM_HID_RegisterInterface() +API to link user process to custom HID class
Add Custom HID interface template file +usbd_customhid_if_template.c/h
Miscellaneous comments update
+
+
+ + +

Main Changes

+

Major update based on STM32Cube specification.

+ + + + + + + + + + + +
Headline
Library Core, Classes architecture and +APIs modified vs. V1.1.0, and thus the 2 versions are not +compatible.
+

This version has to be used only with STM32Cube based +development

+
+
+ + +

Main Changes

+ + + + + + + + + + + + + + + + + + + + + + + + + + +
Headline
Official support of STM32F4xx devices
All source files: license disclaimer text +update and add link to the License file on ST Internet.
Handle test mode in the set feature +request
Handle dynamically the USB SELF POWERED +feature
Handle correctly the USBD_CtlError process +to take into account error during Control OUT stage
Miscellaneous bug fix
+
+
+ + +

Main Changes

+

First official version for STM32F105/7xx and STM32F2xx devices

+
+ +
+
+
+
- \ No newline at end of file +
+

Info

+
+
+
+ + diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/_htmresc/favicon.png b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/_htmresc/favicon.png new file mode 100644 index 0000000000000000000000000000000000000000..06713eec4974e141c6e9b4156d34e61e89f282ca GIT binary patch literal 4126 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 literal 0 HcmV?d00001 diff --git a/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/_htmresc/mini-st.css b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/_htmresc/mini-st.css similarity index 77% rename from stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/_htmresc/mini-st.css rename to stm32/system/Middlewares/ST/STM32_USB_Device_Library/_htmresc/mini-st.css index eb41d56c..986f4d42 100644 --- a/stm32/system/Drivers/CMSIS/Device/ST/STM32F4xx/_htmresc/mini-st.css +++ b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/_htmresc/mini-st.css @@ -1,39 +1,39 @@ @charset "UTF-8"; /* - Flavor name: Default (mini-default) - Author: Angelos Chalaris (chalarangelo@gmail.com) - Maintainers: Angelos Chalaris - mini.css version: v3.0.0-alpha.3 + Flavor name: Custom (mini-custom) + Generated online - https://minicss.org/flavors + mini.css version: v3.0.1 */ /* Browsers resets and base typography. */ /* Core module CSS variable definitions */ :root { - --fore-color: #111; - --secondary-fore-color: #444; - --back-color: #f8f8f8; - --secondary-back-color: #f0f0f0; - --blockquote-color: #f57c00; - --pre-color: #1565c0; - --border-color: #aaa; - --secondary-border-color: #ddd; - --heading-ratio: 1.19; + --fore-color: #03234b; + --secondary-fore-color: #03234b; + --back-color: #ffffff; + --secondary-back-color: #ffffff; + --blockquote-color: #e6007e; + --pre-color: #e6007e; + --border-color: #3cb4e6; + --secondary-border-color: #3cb4e6; + --heading-ratio: 1.2; --universal-margin: 0.5rem; - --universal-padding: 0.125rem; - --universal-border-radius: 0.125rem; - --a-link-color: #0277bd; - --a-visited-color: #01579b; } + --universal-padding: 0.25rem; + --universal-border-radius: 0.075rem; + --background-margin: 1.5%; + --a-link-color: #3cb4e6; + --a-visited-color: #8c0078; } html { - font-size: 14px; } + font-size: 13.5px; } a, b, del, em, i, ins, q, span, strong, u { font-size: 1em; } html, * { - font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Ubuntu, "Helvetica Neue", Helvetica, sans-serif; - line-height: 1.4; + font-family: -apple-system, BlinkMacSystemFont, Helvetica, arial, sans-serif; + line-height: 1.25; -webkit-text-size-adjust: 100%; } * { @@ -42,7 +42,10 @@ html, * { body { margin: 0; color: var(--fore-color); - background: var(--back-color); } + @background: var(--back-color); + background: var(--back-color) linear-gradient(#ffd200, #ffd200) repeat-y left top; + background-size: var(--background-margin); + } details { display: block; } @@ -62,9 +65,9 @@ img { height: auto; } h1, h2, h3, h4, h5, h6 { - line-height: 1.2; + line-height: 1.25; margin: calc(1.5 * var(--universal-margin)) var(--universal-margin); - font-weight: 500; } + font-weight: 400; } h1 small, h2 small, h3 small, h4 small, h5 small, h6 small { color: var(--secondary-fore-color); display: block; @@ -74,21 +77,15 @@ h1 { font-size: calc(1rem * var(--heading-ratio) * var(--heading-ratio) * var(--heading-ratio)); } h2 { - font-size: calc(1rem * var(--heading-ratio) * var(--heading-ratio); ); - background: var(--mark-back-color); - font-weight: 600; - padding: 0.1em 0.5em 0.2em 0.5em; - color: var(--mark-fore-color); } - + font-size: calc(1rem * var(--heading-ratio) * var(--heading-ratio) ); + border-style: none none solid none ; + border-width: thin; + border-color: var(--border-color); } h3 { - font-size: calc(1rem * var(--heading-ratio)); - padding-left: calc(2 * var(--universal-margin)); - /* background: var(--border-color); */ - } + font-size: calc(1rem * var(--heading-ratio) ); } h4 { - font-size: 1rem;); - padding-left: calc(4 * var(--universal-margin)); } + font-size: calc(1rem * var(--heading-ratio)); } h5 { font-size: 1rem; } @@ -101,7 +98,7 @@ p { ol, ul { margin: var(--universal-margin); - padding-left: calc(6 * var(--universal-margin)); } + padding-left: calc(3 * var(--universal-margin)); } b, strong { font-weight: 700; } @@ -111,7 +108,7 @@ hr { border: 0; line-height: 1.25em; margin: var(--universal-margin); - height: 0.0625rem; + height: 0.0714285714rem; background: linear-gradient(to right, transparent, var(--border-color) 20%, var(--border-color) 80%, transparent); } blockquote { @@ -121,16 +118,16 @@ blockquote { color: var(--secondary-fore-color); margin: var(--universal-margin); padding: calc(3 * var(--universal-padding)); - border: 0.0625rem solid var(--secondary-border-color); - border-left: 0.375rem solid var(--blockquote-color); + border: 0.0714285714rem solid var(--secondary-border-color); + border-left: 0.3rem solid var(--blockquote-color); border-radius: 0 var(--universal-border-radius) var(--universal-border-radius) 0; } blockquote:before { position: absolute; top: calc(0rem - var(--universal-padding)); left: 0; font-family: sans-serif; - font-size: 3rem; - font-weight: 700; + font-size: 2rem; + font-weight: 800; content: "\201c"; color: var(--blockquote-color); } blockquote[cite]:after { @@ -160,8 +157,8 @@ pre { background: var(--secondary-back-color); padding: calc(1.5 * var(--universal-padding)); margin: var(--universal-margin); - border: 0.0625rem solid var(--secondary-border-color); - border-left: 0.25rem solid var(--pre-color); + border: 0.0714285714rem solid var(--secondary-border-color); + border-left: 0.2857142857rem solid var(--pre-color); border-radius: 0 var(--universal-border-radius) var(--universal-border-radius) 0; } sup, sub, code, kbd { @@ -204,7 +201,8 @@ a { box-sizing: border-box; display: flex; flex: 0 1 auto; - flex-flow: row wrap; } + flex-flow: row wrap; + margin: 0 0 0 var(--background-margin); } .col-sm, [class^='col-sm-'], @@ -565,9 +563,9 @@ a { order: 999; } } /* Card component CSS variable definitions */ :root { - --card-back-color: #f8f8f8; - --card-fore-color: #111; - --card-border-color: #ddd; } + --card-back-color: #3cb4e6; + --card-fore-color: #03234b; + --card-border-color: #03234b; } .card { display: flex; @@ -578,7 +576,7 @@ a { width: 100%; background: var(--card-back-color); color: var(--card-fore-color); - border: 0.0625rem solid var(--card-border-color); + border: 0.0714285714rem solid var(--card-border-color); border-radius: var(--universal-border-radius); margin: var(--universal-margin); overflow: hidden; } @@ -592,7 +590,7 @@ a { margin: 0; border: 0; border-radius: 0; - border-bottom: 0.0625rem solid var(--card-border-color); + border-bottom: 0.0714285714rem solid var(--card-border-color); padding: var(--universal-padding); width: 100%; } .card > .sectione.media { @@ -617,17 +615,18 @@ a { width: auto; } .card.warning { -/* --card-back-color: #ffca28; */ --card-back-color: #e5b8b7; - --card-border-color: #e8b825; } + --card-fore-color: #3b234b; + --card-border-color: #8c0078; } .card.error { - --card-back-color: #b71c1c; - --card-fore-color: #f8f8f8; - --card-border-color: #a71a1a; } + --card-back-color: #464650; + --card-fore-color: #ffffff; + --card-border-color: #8c0078; } .card > .sectione.dark { - --card-back-color: #e0e0e0; } + --card-back-color: #3b234b; + --card-fore-color: #ffffff; } .card > .sectione.double-padded { padding: calc(1.5 * var(--universal-padding)); } @@ -637,12 +636,12 @@ a { */ /* Input_control module CSS variable definitions */ :root { - --form-back-color: #f0f0f0; - --form-fore-color: #111; - --form-border-color: #ddd; - --input-back-color: #f8f8f8; - --input-fore-color: #111; - --input-border-color: #ddd; + --form-back-color: #ffe97f; + --form-fore-color: #03234b; + --form-border-color: #3cb4e6; + --input-back-color: #ffffff; + --input-fore-color: #03234b; + --input-border-color: #3cb4e6; --input-focus-color: #0288d1; --input-invalid-color: #d32f2f; --button-back-color: #e2e2e2; @@ -655,13 +654,13 @@ a { form { background: var(--form-back-color); color: var(--form-fore-color); - border: 0.0625rem solid var(--form-border-color); + border: 0.0714285714rem solid var(--form-border-color); border-radius: var(--universal-border-radius); margin: var(--universal-margin); padding: calc(2 * var(--universal-padding)) var(--universal-padding); } fieldset { - border: 0.0625rem solid var(--form-border-color); + border: 0.0714285714rem solid var(--form-border-color); border-radius: var(--universal-border-radius); margin: calc(var(--universal-margin) / 4); padding: var(--universal-padding); } @@ -671,7 +670,7 @@ legend { display: table; max-width: 100%; white-space: normal; - font-weight: 700; + font-weight: 500; padding: calc(var(--universal-padding) / 2); } label { @@ -716,7 +715,7 @@ input:not([type]), [type="text"], [type="email"], [type="number"], [type="search box-sizing: border-box; background: var(--input-back-color); color: var(--input-fore-color); - border: 0.0625rem solid var(--input-border-color); + border: 0.0714285714rem solid var(--input-border-color); border-radius: var(--universal-border-radius); margin: calc(var(--universal-margin) / 2); padding: var(--universal-padding) calc(1.5 * var(--universal-padding)); } @@ -763,8 +762,8 @@ option { [type="radio"]:checked:before { border-radius: 100%; content: ''; - top: calc(0.0625rem + var(--universal-padding) / 2); - left: calc(0.0625rem + var(--universal-padding) / 2); + top: calc(0.0714285714rem + var(--universal-padding) / 2); + left: calc(0.0714285714rem + var(--universal-padding) / 2); background: var(--input-fore-color); width: 0.5rem; height: 0.5rem; } @@ -793,7 +792,7 @@ a[role="button"], label[role="button"], [role="button"] { display: inline-block; background: var(--button-back-color); color: var(--button-fore-color); - border: 0.0625rem solid var(--button-border-color); + border: 0.0714285714rem solid var(--button-border-color); border-radius: var(--universal-border-radius); padding: var(--universal-padding) calc(1.5 * var(--universal-padding)); margin: var(--universal-margin); @@ -814,7 +813,7 @@ input:disabled, input[disabled], textarea:disabled, textarea[disabled], select:d .button-group { display: flex; - border: 0.0625rem solid var(--button-group-border-color); + border: 0.0714285714rem solid var(--button-group-border-color); border-radius: var(--universal-border-radius); margin: var(--universal-margin); } .button-group > button, .button-group [type="button"], .button-group > [type="submit"], .button-group > [type="reset"], .button-group > .button, .button-group > [role="button"] { @@ -826,13 +825,13 @@ input:disabled, input[disabled], textarea:disabled, textarea[disabled], select:d border-radius: 0; box-shadow: none; } .button-group > :not(:first-child) { - border-left: 0.0625rem solid var(--button-group-border-color); } + border-left: 0.0714285714rem solid var(--button-group-border-color); } @media screen and (max-width: 499px) { .button-group { flex-direction: column; } .button-group > :not(:first-child) { border: 0; - border-top: 0.0625rem solid var(--button-group-border-color); } } + border-top: 0.0714285714rem solid var(--button-group-border-color); } } /* Custom elements for forms and input elements. @@ -874,29 +873,29 @@ button.large, [type="button"].large, [type="submit"].large, [type="reset"].large */ /* Navigation module CSS variable definitions */ :root { - --header-back-color: #f8f8f8; - --header-hover-back-color: #f0f0f0; - --header-fore-color: #444; - --header-border-color: #ddd; - --nav-back-color: #f8f8f8; - --nav-hover-back-color: #f0f0f0; - --nav-fore-color: #444; - --nav-border-color: #ddd; - --nav-link-color: #0277bd; - --footer-fore-color: #444; - --footer-back-color: #f8f8f8; - --footer-border-color: #ddd; - --footer-link-color: #0277bd; - --drawer-back-color: #f8f8f8; - --drawer-hover-back-color: #f0f0f0; - --drawer-border-color: #ddd; - --drawer-close-color: #444; } + --header-back-color: #03234b; + --header-hover-back-color: #ffd200; + --header-fore-color: #ffffff; + --header-border-color: #3cb4e6; + --nav-back-color: #ffffff; + --nav-hover-back-color: #ffe97f; + --nav-fore-color: #e6007e; + --nav-border-color: #3cb4e6; + --nav-link-color: #3cb4e6; + --footer-fore-color: #ffffff; + --footer-back-color: #03234b; + --footer-border-color: #3cb4e6; + --footer-link-color: #3cb4e6; + --drawer-back-color: #ffffff; + --drawer-hover-back-color: #ffe97f; + --drawer-border-color: #3cb4e6; + --drawer-close-color: #e6007e; } header { - height: 3.1875rem; + height: 2.75rem; background: var(--header-back-color); color: var(--header-fore-color); - border-bottom: 0.0625rem solid var(--header-border-color); + border-bottom: 0.0714285714rem solid var(--header-border-color); padding: calc(var(--universal-padding) / 4) 0; white-space: nowrap; overflow-x: auto; @@ -927,7 +926,7 @@ header { nav { background: var(--nav-back-color); color: var(--nav-fore-color); - border: 0.0625rem solid var(--nav-border-color); + border: 0.0714285714rem solid var(--nav-border-color); border-radius: var(--universal-border-radius); margin: var(--universal-margin); } nav * { @@ -946,10 +945,10 @@ nav { nav .sublink-1:before { position: absolute; left: calc(var(--universal-padding) - 1 * var(--universal-padding)); - top: -0.0625rem; + top: -0.0714285714rem; content: ''; height: 100%; - border: 0.0625rem solid var(--nav-border-color); + border: 0.0714285714rem solid var(--nav-border-color); border-left: 0; } nav .sublink-2 { position: relative; @@ -957,16 +956,16 @@ nav { nav .sublink-2:before { position: absolute; left: calc(var(--universal-padding) - 3 * var(--universal-padding)); - top: -0.0625rem; + top: -0.0714285714rem; content: ''; height: 100%; - border: 0.0625rem solid var(--nav-border-color); + border: 0.0714285714rem solid var(--nav-border-color); border-left: 0; } footer { background: var(--footer-back-color); color: var(--footer-fore-color); - border-top: 0.0625rem solid var(--footer-border-color); + border-top: 0.0714285714rem solid var(--footer-border-color); padding: calc(2 * var(--universal-padding)) var(--universal-padding); font-size: 0.875rem; } footer a, footer a:visited { @@ -1013,7 +1012,7 @@ footer.sticky { height: 100vh; overflow-y: auto; background: var(--drawer-back-color); - border: 0.0625rem solid var(--drawer-border-color); + border: 0.0714285714rem solid var(--drawer-border-color); border-radius: 0; margin: 0; z-index: 1110; @@ -1060,38 +1059,36 @@ footer.sticky { */ /* Table module CSS variable definitions. */ :root { - --table-border-color: #aaa; - --table-border-separator-color: #666; - --table-head-back-color: #e6e6e6; - --table-head-fore-color: #111; - --table-body-back-color: #f8f8f8; - --table-body-fore-color: #111; - --table-body-alt-back-color: #eee; } + --table-border-color: #03234b; + --table-border-separator-color: #03234b; + --table-head-back-color: #03234b; + --table-head-fore-color: #ffffff; + --table-body-back-color: #ffffff; + --table-body-fore-color: #03234b; + --table-body-alt-back-color: #f4f4f4; } table { border-collapse: separate; border-spacing: 0; - : margin: calc(1.5 * var(--universal-margin)) var(--universal-margin); + margin: 0; display: flex; flex: 0 1 auto; flex-flow: row wrap; padding: var(--universal-padding); - padding-top: 0; - margin: calc(1.5 * var(--universal-margin)) var(--universal-margin); } + padding-top: 0; } table caption { - font-size: 1.25 * rem; + font-size: 1rem; margin: calc(2 * var(--universal-margin)) 0; max-width: 100%; - flex: 0 0 100%; - text-align: left;} + flex: 0 0 100%; } table thead, table tbody { display: flex; flex-flow: row wrap; - border: 0.0625rem solid var(--table-border-color); } + border: 0.0714285714rem solid var(--table-border-color); } table thead { z-index: 999; border-radius: var(--universal-border-radius) var(--universal-border-radius) 0 0; - border-bottom: 0.0625rem solid var(--table-border-separator-color); } + border-bottom: 0.0714285714rem solid var(--table-border-separator-color); } table tbody { border-top: 0; margin-top: calc(0 - var(--universal-margin)); @@ -1109,11 +1106,11 @@ table { table td { background: var(--table-body-back-color); color: var(--table-body-fore-color); - border-top: 0.0625rem solid var(--table-border-color); } + border-top: 0.0714285714rem solid var(--table-border-color); } table:not(.horizontal) { overflow: auto; - max-height: 850px; } + max-height: 100%; } table:not(.horizontal) thead, table:not(.horizontal) tbody { max-width: 100%; flex: 0 0 100%; } @@ -1134,32 +1131,33 @@ table.horizontal { border: 0; } table.horizontal thead, table.horizontal tbody { border: 0; + flex: .2 0 0; flex-flow: row nowrap; } table.horizontal tbody { overflow: auto; justify-content: space-between; - flex: 1 0 0; - margin-left: calc( 4 * var(--universal-margin)); + flex: .8 0 0; + margin-left: 0; padding-bottom: calc(var(--universal-padding) / 4); } table.horizontal tr { flex-direction: column; flex: 1 0 auto; } table.horizontal th, table.horizontal td { - width: 100%; + width: auto; border: 0; - border-bottom: 0.0625rem solid var(--table-border-color); } + border-bottom: 0.0714285714rem solid var(--table-border-color); } table.horizontal th:not(:first-child), table.horizontal td:not(:first-child) { border-top: 0; } table.horizontal th { text-align: right; - border-left: 0.0625rem solid var(--table-border-color); - border-right: 0.0625rem solid var(--table-border-separator-color); } + border-left: 0.0714285714rem solid var(--table-border-color); + border-right: 0.0714285714rem solid var(--table-border-separator-color); } table.horizontal thead tr:first-child { padding-left: 0; } table.horizontal th:first-child, table.horizontal td:first-child { - border-top: 0.0625rem solid var(--table-border-color); } + border-top: 0.0714285714rem solid var(--table-border-color); } table.horizontal tbody tr:last-child td { - border-right: 0.0625rem solid var(--table-border-color); } + border-right: 0.0714285714rem solid var(--table-border-color); } table.horizontal tbody tr:last-child td:first-child { border-top-right-radius: 0.25rem; } table.horizontal tbody tr:last-child td:last-child { @@ -1191,12 +1189,12 @@ table.horizontal { display: table-row-group; } table tr, table.horizontal tr { display: block; - border: 0.0625rem solid var(--table-border-color); + border: 0.0714285714rem solid var(--table-border-color); border-radius: var(--universal-border-radius); - background: #fafafa; + background: #ffffff; padding: var(--universal-padding); margin: var(--universal-margin); - margin-bottom: calc(2 * var(--universal-margin)); } + margin-bottom: calc(1 * var(--universal-margin)); } table th, table td, table.horizontal th, table.horizontal td { width: auto; } table td, table.horizontal td { @@ -1211,9 +1209,6 @@ table.horizontal { border-top: 0; } table tbody tr:last-child td, table.horizontal tbody tr:last-child td { border-right: 0; } } -:root { - --table-body-alt-back-color: #eee; } - table tr:nth-of-type(2n) > td { background: var(--table-body-alt-back-color); } @@ -1234,8 +1229,8 @@ table.hoverable tr:hover, table.hoverable tr:hover > td, table.hoverable tr:focu */ /* Contextual module CSS variable definitions */ :root { - --mark-back-color: #0277bd; - --mark-fore-color: #fafafa; } + --mark-back-color: #3cb4e6; + --mark-fore-color: #ffffff; } mark { background: var(--mark-back-color); @@ -1243,11 +1238,11 @@ mark { font-size: 0.95em; line-height: 1em; border-radius: var(--universal-border-radius); - padding: calc(var(--universal-padding) / 4) calc(var(--universal-padding) / 2); } + padding: calc(var(--universal-padding) / 4) var(--universal-padding); } mark.inline-block { display: inline-block; font-size: 1em; - line-height: 1.5; + line-height: 1.4; padding: calc(var(--universal-padding) / 2) var(--universal-padding); } :root { @@ -1314,8 +1309,8 @@ mark { :root { --modal-overlay-color: rgba(0, 0, 0, 0.45); - --modal-close-color: #444; - --modal-close-hover-color: #f0f0f0; } + --modal-close-color: #e6007e; + --modal-close-hover-color: #ffe97f; } [type="checkbox"].modal { height: 1px; @@ -1368,13 +1363,14 @@ mark { z-index: 1211; } :root { - --collapse-label-back-color: #e8e8e8; - --collapse-label-fore-color: #212121; - --collapse-label-hover-back-color: #f0f0f0; - --collapse-selected-label-back-color: #ececec; - --collapse-border-color: #ddd; - --collapse-content-back-color: #fafafa; - --collapse-selected-label-border-color: #0277bd; } + --collapse-label-back-color: #03234b; + --collapse-label-fore-color: #ffffff; + --collapse-label-hover-back-color: #3cb4e6; + --collapse-selected-label-back-color: #3cb4e6; + --collapse-border-color: var(--collapse-label-back-color); + --collapse-selected-border-color: #ceecf8; + --collapse-content-back-color: #ffffff; + --collapse-selected-label-border-color: #3cb4e6; } .collapse { width: calc(100% - 2 * var(--universal-margin)); @@ -1395,13 +1391,13 @@ mark { .collapse > label { flex-grow: 1; display: inline-block; - height: 1.5rem; + height: 1.25rem; cursor: pointer; - transition: background 0.3s; + transition: background 0.2s; color: var(--collapse-label-fore-color); background: var(--collapse-label-back-color); - border: 0.0625rem solid var(--collapse-border-color); - padding: calc(1.5 * var(--universal-padding)); } + border: 0.0714285714rem solid var(--collapse-selected-border-color); + padding: calc(1.25 * var(--universal-padding)); } .collapse > label:hover, .collapse > label:focus { background: var(--collapse-label-hover-back-color); } .collapse > label + div { @@ -1418,7 +1414,7 @@ mark { max-height: 1px; } .collapse > :checked + label { background: var(--collapse-selected-label-back-color); - border-bottom-color: var(--collapse-selected-label-border-color); } + border-color: var(--collapse-selected-label-border-color); } .collapse > :checked + label + div { box-sizing: border-box; position: relative; @@ -1427,13 +1423,13 @@ mark { overflow: auto; margin: 0; background: var(--collapse-content-back-color); - border: 0.0625rem solid var(--collapse-border-color); + border: 0.0714285714rem solid var(--collapse-selected-border-color); border-top: 0; padding: var(--universal-padding); clip: auto; -webkit-clip-path: inset(0%); clip-path: inset(0%); - max-height: 850px; } + max-height: 100%; } .collapse > label:not(:first-of-type) { border-top: 0; } .collapse > label:first-of-type { @@ -1450,11 +1446,8 @@ mark { /* Custom elements for contextual background elements, toasts and tooltips. */ -mark.secondary { - --mark-back-color: #d32f2f; } - mark.tertiary { - --mark-back-color: #308732; } + --mark-back-color: #3cb4e6; } mark.tag { padding: calc(var(--universal-padding)/2) var(--universal-padding); @@ -1465,7 +1458,7 @@ mark.tag { */ /* Progress module CSS variable definitions */ :root { - --progress-back-color: #ddd; + --progress-back-color: #3cb4e6; --progress-fore-color: #555; } progress { @@ -1558,45 +1551,53 @@ span[class^='icon-'] { filter: invert(100%); } span.icon-alert { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='8' x2='12' y2='12'%3E%3C/line%3E%3Cline x1='12' y1='16' x2='12' y2='16'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='8' x2='12' y2='12'%3E%3C/line%3E%3Cline x1='12' y1='16' x2='12' y2='16'%3E%3C/line%3E%3C/svg%3E"); } span.icon-bookmark { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M19 21l-7-5-7 5V5a2 2 0 0 1 2-2h10a2 2 0 0 1 2 2z'%3E%3C/path%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M19 21l-7-5-7 5V5a2 2 0 0 1 2-2h10a2 2 0 0 1 2 2z'%3E%3C/path%3E%3C/svg%3E"); } span.icon-calendar { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='3' y='4' width='18' height='18' rx='2' ry='2'%3E%3C/rect%3E%3Cline x1='16' y1='2' x2='16' y2='6'%3E%3C/line%3E%3Cline x1='8' y1='2' x2='8' y2='6'%3E%3C/line%3E%3Cline x1='3' y1='10' x2='21' y2='10'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='3' y='4' width='18' height='18' rx='2' ry='2'%3E%3C/rect%3E%3Cline x1='16' y1='2' x2='16' y2='6'%3E%3C/line%3E%3Cline x1='8' y1='2' x2='8' y2='6'%3E%3C/line%3E%3Cline x1='3' y1='10' x2='21' y2='10'%3E%3C/line%3E%3C/svg%3E"); } span.icon-credit { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='1' y='4' width='22' height='16' rx='2' ry='2'%3E%3C/rect%3E%3Cline x1='1' y1='10' x2='23' y2='10'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='1' y='4' width='22' height='16' rx='2' ry='2'%3E%3C/rect%3E%3Cline x1='1' y1='10' x2='23' y2='10'%3E%3C/line%3E%3C/svg%3E"); } span.icon-edit { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M20 14.66V20a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V6a2 2 0 0 1 2-2h5.34'%3E%3C/path%3E%3Cpolygon points='18 2 22 6 12 16 8 16 8 12 18 2'%3E%3C/polygon%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M20 14.66V20a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V6a2 2 0 0 1 2-2h5.34'%3E%3C/path%3E%3Cpolygon points='18 2 22 6 12 16 8 16 8 12 18 2'%3E%3C/polygon%3E%3C/svg%3E"); } span.icon-link { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6'%3E%3C/path%3E%3Cpolyline points='15 3 21 3 21 9'%3E%3C/polyline%3E%3Cline x1='10' y1='14' x2='21' y2='3'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6'%3E%3C/path%3E%3Cpolyline points='15 3 21 3 21 9'%3E%3C/polyline%3E%3Cline x1='10' y1='14' x2='21' y2='3'%3E%3C/line%3E%3C/svg%3E"); } span.icon-help { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M9.09 9a3 3 0 0 1 5.83 1c0 2-3 3-3 3'%3E%3C/path%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='17' x2='12' y2='17'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M9.09 9a3 3 0 0 1 5.83 1c0 2-3 3-3 3'%3E%3C/path%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='17' x2='12' y2='17'%3E%3C/line%3E%3C/svg%3E"); } span.icon-home { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M3 9l9-7 9 7v11a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2z'%3E%3C/path%3E%3Cpolyline points='9 22 9 12 15 12 15 22'%3E%3C/polyline%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M3 9l9-7 9 7v11a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2z'%3E%3C/path%3E%3Cpolyline points='9 22 9 12 15 12 15 22'%3E%3C/polyline%3E%3C/svg%3E"); } span.icon-info { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='16' x2='12' y2='12'%3E%3C/line%3E%3Cline x1='12' y1='8' x2='12' y2='8'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='16' x2='12' y2='12'%3E%3C/line%3E%3Cline x1='12' y1='8' x2='12' y2='8'%3E%3C/line%3E%3C/svg%3E"); } span.icon-lock { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='3' y='11' width='18' height='11' rx='2' ry='2'%3E%3C/rect%3E%3Cpath d='M7 11V7a5 5 0 0 1 10 0v4'%3E%3C/path%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='3' y='11' width='18' height='11' rx='2' ry='2'%3E%3C/rect%3E%3Cpath d='M7 11V7a5 5 0 0 1 10 0v4'%3E%3C/path%3E%3C/svg%3E"); } span.icon-mail { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M4 4h16c1.1 0 2 .9 2 2v12c0 1.1-.9 2-2 2H4c-1.1 0-2-.9-2-2V6c0-1.1.9-2 2-2z'%3E%3C/path%3E%3Cpolyline points='22,6 12,13 2,6'%3E%3C/polyline%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M4 4h16c1.1 0 2 .9 2 2v12c0 1.1-.9 2-2 2H4c-1.1 0-2-.9-2-2V6c0-1.1.9-2 2-2z'%3E%3C/path%3E%3Cpolyline points='22,6 12,13 2,6'%3E%3C/polyline%3E%3C/svg%3E"); } span.icon-location { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M21 10c0 7-9 13-9 13s-9-6-9-13a9 9 0 0 1 18 0z'%3E%3C/path%3E%3Ccircle cx='12' cy='10' r='3'%3E%3C/circle%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M21 10c0 7-9 13-9 13s-9-6-9-13a9 9 0 0 1 18 0z'%3E%3C/path%3E%3Ccircle cx='12' cy='10' r='3'%3E%3C/circle%3E%3C/svg%3E"); } span.icon-phone { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M22 16.92v3a2 2 0 0 1-2.18 2 19.79 19.79 0 0 1-8.63-3.07 19.5 19.5 0 0 1-6-6 19.79 19.79 0 0 1-3.07-8.67A2 2 0 0 1 4.11 2h3a2 2 0 0 1 2 1.72 12.84 12.84 0 0 0 .7 2.81 2 2 0 0 1-.45 2.11L8.09 9.91a16 16 0 0 0 6 6l1.27-1.27a2 2 0 0 1 2.11-.45 12.84 12.84 0 0 0 2.81.7A2 2 0 0 1 22 16.92z'%3E%3C/path%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M22 16.92v3a2 2 0 0 1-2.18 2 19.79 19.79 0 0 1-8.63-3.07 19.5 19.5 0 0 1-6-6 19.79 19.79 0 0 1-3.07-8.67A2 2 0 0 1 4.11 2h3a2 2 0 0 1 2 1.72 12.84 12.84 0 0 0 .7 2.81 2 2 0 0 1-.45 2.11L8.09 9.91a16 16 0 0 0 6 6l1.27-1.27a2 2 0 0 1 2.11-.45 12.84 12.84 0 0 0 2.81.7A2 2 0 0 1 22 16.92z'%3E%3C/path%3E%3C/svg%3E"); } span.icon-rss { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M4 11a9 9 0 0 1 9 9'%3E%3C/path%3E%3Cpath d='M4 4a16 16 0 0 1 16 16'%3E%3C/path%3E%3Ccircle cx='5' cy='19' r='1'%3E%3C/circle%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M4 11a9 9 0 0 1 9 9'%3E%3C/path%3E%3Cpath d='M4 4a16 16 0 0 1 16 16'%3E%3C/path%3E%3Ccircle cx='5' cy='19' r='1'%3E%3C/circle%3E%3C/svg%3E"); } span.icon-search { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='11' cy='11' r='8'%3E%3C/circle%3E%3Cline x1='21' y1='21' x2='16.65' y2='16.65'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='11' cy='11' r='8'%3E%3C/circle%3E%3Cline x1='21' y1='21' x2='16.65' y2='16.65'%3E%3C/line%3E%3C/svg%3E"); } span.icon-settings { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='3'%3E%3C/circle%3E%3Cpath d='M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z'%3E%3C/path%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='3'%3E%3C/circle%3E%3Cpath d='M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z'%3E%3C/path%3E%3C/svg%3E"); } span.icon-share { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='18' cy='5' r='3'%3E%3C/circle%3E%3Ccircle cx='6' cy='12' r='3'%3E%3C/circle%3E%3Ccircle cx='18' cy='19' r='3'%3E%3C/circle%3E%3Cline x1='8.59' y1='13.51' x2='15.42' y2='17.49'%3E%3C/line%3E%3Cline x1='15.41' y1='6.51' x2='8.59' y2='10.49'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='18' cy='5' r='3'%3E%3C/circle%3E%3Ccircle cx='6' cy='12' r='3'%3E%3C/circle%3E%3Ccircle cx='18' cy='19' r='3'%3E%3C/circle%3E%3Cline x1='8.59' y1='13.51' x2='15.42' y2='17.49'%3E%3C/line%3E%3Cline x1='15.41' y1='6.51' x2='8.59' y2='10.49'%3E%3C/line%3E%3C/svg%3E"); } span.icon-cart { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='9' cy='21' r='1'%3E%3C/circle%3E%3Ccircle cx='20' cy='21' r='1'%3E%3C/circle%3E%3Cpath d='M1 1h4l2.68 13.39a2 2 0 0 0 2 1.61h9.72a2 2 0 0 0 2-1.61L23 6H6'%3E%3C/path%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='9' cy='21' r='1'%3E%3C/circle%3E%3Ccircle cx='20' cy='21' r='1'%3E%3C/circle%3E%3Cpath d='M1 1h4l2.68 13.39a2 2 0 0 0 2 1.61h9.72a2 2 0 0 0 2-1.61L23 6H6'%3E%3C/path%3E%3C/svg%3E"); } span.icon-upload { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4'%3E%3C/path%3E%3Cpolyline points='17 8 12 3 7 8'%3E%3C/polyline%3E%3Cline x1='12' y1='3' x2='12' y2='15'%3E%3C/line%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4'%3E%3C/path%3E%3Cpolyline points='17 8 12 3 7 8'%3E%3C/polyline%3E%3Cline x1='12' y1='3' x2='12' y2='15'%3E%3C/line%3E%3C/svg%3E"); } span.icon-user { - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%23111' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M20 21v-2a4 4 0 0 0-4-4H8a4 4 0 0 0-4 4v2'%3E%3C/path%3E%3Ccircle cx='12' cy='7' r='4'%3E%3C/circle%3E%3C/svg%3E"); } + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M20 21v-2a4 4 0 0 0-4-4H8a4 4 0 0 0-4 4v2'%3E%3C/path%3E%3Ccircle cx='12' cy='7' r='4'%3E%3C/circle%3E%3C/svg%3E"); } + +/* + Definitions for STMicroelectronics icons (https://brandportal.st.com/document/26). +*/ +span.icon-st-update { + background-image: url("Update.svg"); } +span.icon-st-add { + background-image: url("Add button.svg"); } /* Definitions for utilities and helper classes. @@ -1604,7 +1605,7 @@ span.icon-user { /* Utility module CSS variable definitions */ :root { --generic-border-color: rgba(0, 0, 0, 0.3); - --generic-box-shadow: 0 0.25rem 0.25rem 0 rgba(0, 0, 0, 0.125), 0 0.125rem 0.125rem -0.125rem rgba(0, 0, 0, 0.25); } + --generic-box-shadow: 0 0.2857142857rem 0.2857142857rem 0 rgba(0, 0, 0, 0.125), 0 0.1428571429rem 0.1428571429rem -0.1428571429rem rgba(0, 0, 0, 0.125); } .hidden { display: none !important; } @@ -1622,7 +1623,7 @@ span.icon-user { overflow: hidden !important; } .bordered { - border: 0.0625rem solid var(--generic-border-color) !important; } + border: 0.0714285714rem solid var(--generic-border-color) !important; } .rounded { border-radius: var(--universal-border-radius) !important; } @@ -1697,4 +1698,14 @@ span.icon-user { clip-path: inset(100%) !important; overflow: hidden !important; } } -/*# sourceMappingURL=mini-default.css.map */ +/*# sourceMappingURL=mini-custom.css.map */ + +img[alt="ST logo"] { display: block; margin: auto; width: 75%; max-width: 250px; min-width: 71px; } +img[alt="Cube logo"] { float: right; width: 30%; max-width: 10rem; min-width: 8rem; padding-right: 1rem;} + +.figure { + display: block; + margin-left: auto; + margin-right: auto; + text-align: center; +} \ No newline at end of file diff --git a/stm32/system/Middlewares/ST/STM32_USB_Device_Library/_htmresc/st_logo_2020.png b/stm32/system/Middlewares/ST/STM32_USB_Device_Library/_htmresc/st_logo_2020.png new file mode 100644 index 0000000000000000000000000000000000000000..d6cebb5ac70e0594cbe37a6b60036a80ef3bed37 GIT binary patch literal 7520 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"NJQ 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 b^}%0^qYmp2WD(pA|JtZ4>nPPKybJpu3@X?! literal 0 HcmV?d00001 diff --git a/stm32/system/Middlewares/ST/STM32_USB_Host_Library/_htmresc/favicon.png b/stm32/system/Middlewares/ST/STM32_USB_Host_Library/_htmresc/favicon.png new file mode 100644 index 0000000000000000000000000000000000000000..06713eec4974e141c6e9b4156d34e61e89f282ca GIT binary patch literal 4126 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 literal 0 HcmV?d00001 diff --git a/stm32/system/Middlewares/ST/STM32_USB_Host_Library/_htmresc/mini-st.css b/stm32/system/Middlewares/ST/STM32_USB_Host_Library/_htmresc/mini-st.css new file mode 100644 index 00000000..986f4d42 --- /dev/null +++ b/stm32/system/Middlewares/ST/STM32_USB_Host_Library/_htmresc/mini-st.css @@ -0,0 +1,1711 @@ +@charset "UTF-8"; +/* + Flavor name: Custom (mini-custom) + Generated online - https://minicss.org/flavors + mini.css version: v3.0.1 +*/ +/* + Browsers resets and base typography. +*/ +/* Core module CSS variable definitions */ +:root { + --fore-color: #03234b; + --secondary-fore-color: #03234b; + --back-color: #ffffff; + --secondary-back-color: #ffffff; + --blockquote-color: #e6007e; + --pre-color: #e6007e; + --border-color: #3cb4e6; + --secondary-border-color: #3cb4e6; + --heading-ratio: 1.2; + --universal-margin: 0.5rem; + --universal-padding: 0.25rem; + --universal-border-radius: 0.075rem; + --background-margin: 1.5%; + --a-link-color: #3cb4e6; + --a-visited-color: #8c0078; } + +html { + font-size: 13.5px; } + +a, b, del, em, i, ins, q, span, strong, u { + font-size: 1em; } + +html, * { + font-family: -apple-system, BlinkMacSystemFont, Helvetica, arial, sans-serif; + line-height: 1.25; + -webkit-text-size-adjust: 100%; } + +* { + font-size: 1rem; } + +body { + margin: 0; + color: var(--fore-color); + @background: var(--back-color); + background: var(--back-color) linear-gradient(#ffd200, #ffd200) repeat-y left top; + background-size: var(--background-margin); + } + +details { + display: block; } + +summary { + display: list-item; } + +abbr[title] { + border-bottom: none; + text-decoration: underline dotted; } + +input { + overflow: visible; } + +img { + max-width: 100%; + height: auto; } + +h1, h2, h3, h4, h5, h6 { + line-height: 1.25; + margin: calc(1.5 * var(--universal-margin)) var(--universal-margin); + font-weight: 400; } + h1 small, h2 small, h3 small, h4 small, h5 small, h6 small { + color: var(--secondary-fore-color); + display: block; + margin-top: -0.25rem; } + +h1 { + font-size: calc(1rem * var(--heading-ratio) * var(--heading-ratio) * var(--heading-ratio)); } + +h2 { + font-size: calc(1rem * var(--heading-ratio) * var(--heading-ratio) ); + border-style: none none solid none ; + border-width: thin; + border-color: var(--border-color); } +h3 { + font-size: calc(1rem * var(--heading-ratio) ); } + +h4 { + font-size: calc(1rem * var(--heading-ratio)); } + +h5 { + font-size: 1rem; } + +h6 { + font-size: calc(1rem / var(--heading-ratio)); } + +p { + margin: var(--universal-margin); } + +ol, ul { + margin: var(--universal-margin); + padding-left: calc(3 * var(--universal-margin)); } + +b, strong { + font-weight: 700; } + +hr { + box-sizing: content-box; + border: 0; + line-height: 1.25em; + margin: var(--universal-margin); + height: 0.0714285714rem; + background: linear-gradient(to right, transparent, var(--border-color) 20%, var(--border-color) 80%, transparent); } + +blockquote { + display: block; + position: relative; + font-style: italic; + color: var(--secondary-fore-color); + margin: var(--universal-margin); + padding: calc(3 * var(--universal-padding)); + border: 0.0714285714rem solid var(--secondary-border-color); + border-left: 0.3rem solid var(--blockquote-color); + border-radius: 0 var(--universal-border-radius) var(--universal-border-radius) 0; } + blockquote:before { + position: absolute; + top: calc(0rem - var(--universal-padding)); + left: 0; + font-family: sans-serif; + font-size: 2rem; + font-weight: 800; + content: "\201c"; + color: var(--blockquote-color); } + blockquote[cite]:after { + font-style: normal; + font-size: 0.75em; + font-weight: 700; + content: "\a— " attr(cite); + white-space: pre; } + +code, kbd, pre, samp { + font-family: Menlo, Consolas, monospace; + font-size: 0.85em; } + +code { + background: var(--secondary-back-color); + border-radius: var(--universal-border-radius); + padding: calc(var(--universal-padding) / 4) calc(var(--universal-padding) / 2); } + +kbd { + background: var(--fore-color); + color: var(--back-color); + border-radius: var(--universal-border-radius); + padding: calc(var(--universal-padding) / 4) calc(var(--universal-padding) / 2); } + +pre { + overflow: auto; + background: var(--secondary-back-color); + padding: calc(1.5 * var(--universal-padding)); + margin: var(--universal-margin); + border: 0.0714285714rem solid var(--secondary-border-color); + border-left: 0.2857142857rem solid var(--pre-color); + border-radius: 0 var(--universal-border-radius) var(--universal-border-radius) 0; } + +sup, sub, code, kbd { + line-height: 0; + position: relative; + vertical-align: baseline; } + +small, sup, sub, figcaption { + font-size: 0.75em; } + +sup { + top: -0.5em; } + +sub { + bottom: -0.25em; } + +figure { + margin: var(--universal-margin); } + +figcaption { + color: var(--secondary-fore-color); } + +a { + text-decoration: none; } + a:link { + color: var(--a-link-color); } + a:visited { + color: var(--a-visited-color); } + a:hover, a:focus { + text-decoration: underline; } + +/* + Definitions for the grid system, cards and containers. +*/ +.container { + margin: 0 auto; + padding: 0 calc(1.5 * var(--universal-padding)); } + +.row { + box-sizing: border-box; + display: flex; + flex: 0 1 auto; + flex-flow: row wrap; + margin: 0 0 0 var(--background-margin); } + +.col-sm, +[class^='col-sm-'], +[class^='col-sm-offset-'], +.row[class*='cols-sm-'] > * { + box-sizing: border-box; + flex: 0 0 auto; + padding: 0 calc(var(--universal-padding) / 2); } + +.col-sm, +.row.cols-sm > * { + max-width: 100%; + flex-grow: 1; + flex-basis: 0; } + +.col-sm-1, +.row.cols-sm-1 > * { + max-width: 8.3333333333%; + flex-basis: 8.3333333333%; } + +.col-sm-offset-0 { + margin-left: 0; } + +.col-sm-2, +.row.cols-sm-2 > * { + max-width: 16.6666666667%; + flex-basis: 16.6666666667%; } + +.col-sm-offset-1 { + margin-left: 8.3333333333%; } + +.col-sm-3, +.row.cols-sm-3 > * { + max-width: 25%; + flex-basis: 25%; } + +.col-sm-offset-2 { + margin-left: 16.6666666667%; } + +.col-sm-4, +.row.cols-sm-4 > * { + max-width: 33.3333333333%; + flex-basis: 33.3333333333%; } + +.col-sm-offset-3 { + margin-left: 25%; } + +.col-sm-5, +.row.cols-sm-5 > * { + max-width: 41.6666666667%; + flex-basis: 41.6666666667%; } + +.col-sm-offset-4 { + margin-left: 33.3333333333%; } + +.col-sm-6, +.row.cols-sm-6 > * { + max-width: 50%; + flex-basis: 50%; } + +.col-sm-offset-5 { + margin-left: 41.6666666667%; } + +.col-sm-7, +.row.cols-sm-7 > * { + max-width: 58.3333333333%; + flex-basis: 58.3333333333%; } + +.col-sm-offset-6 { + margin-left: 50%; } + +.col-sm-8, +.row.cols-sm-8 > * { + max-width: 66.6666666667%; + flex-basis: 66.6666666667%; } + +.col-sm-offset-7 { + margin-left: 58.3333333333%; } + +.col-sm-9, +.row.cols-sm-9 > * { + max-width: 75%; + flex-basis: 75%; } + +.col-sm-offset-8 { + margin-left: 66.6666666667%; } + +.col-sm-10, +.row.cols-sm-10 > * { + max-width: 83.3333333333%; + flex-basis: 83.3333333333%; } + +.col-sm-offset-9 { + margin-left: 75%; } + +.col-sm-11, +.row.cols-sm-11 > * { + max-width: 91.6666666667%; + flex-basis: 91.6666666667%; } + +.col-sm-offset-10 { + margin-left: 83.3333333333%; } + +.col-sm-12, +.row.cols-sm-12 > * { + max-width: 100%; + flex-basis: 100%; } + +.col-sm-offset-11 { + margin-left: 91.6666666667%; } + +.col-sm-normal { + order: initial; } + +.col-sm-first { + order: -999; } + +.col-sm-last { + order: 999; } + +@media screen and (min-width: 500px) { + .col-md, + [class^='col-md-'], + [class^='col-md-offset-'], + .row[class*='cols-md-'] > * { + box-sizing: border-box; + flex: 0 0 auto; + padding: 0 calc(var(--universal-padding) / 2); } + + .col-md, + .row.cols-md > * { + max-width: 100%; + flex-grow: 1; + flex-basis: 0; } + + .col-md-1, + .row.cols-md-1 > * { + max-width: 8.3333333333%; + flex-basis: 8.3333333333%; } + + .col-md-offset-0 { + margin-left: 0; } + + .col-md-2, + .row.cols-md-2 > * { + max-width: 16.6666666667%; + flex-basis: 16.6666666667%; } + + .col-md-offset-1 { + margin-left: 8.3333333333%; } + + .col-md-3, + .row.cols-md-3 > * { + max-width: 25%; + flex-basis: 25%; } + + .col-md-offset-2 { + margin-left: 16.6666666667%; } + + .col-md-4, + .row.cols-md-4 > * { + max-width: 33.3333333333%; + flex-basis: 33.3333333333%; } + + .col-md-offset-3 { + margin-left: 25%; } + + .col-md-5, + .row.cols-md-5 > * { + max-width: 41.6666666667%; + flex-basis: 41.6666666667%; } + + .col-md-offset-4 { + margin-left: 33.3333333333%; } + + .col-md-6, + .row.cols-md-6 > * { + max-width: 50%; + flex-basis: 50%; } + + .col-md-offset-5 { + margin-left: 41.6666666667%; } + + .col-md-7, + .row.cols-md-7 > * { + max-width: 58.3333333333%; + flex-basis: 58.3333333333%; } + + .col-md-offset-6 { + margin-left: 50%; } + + .col-md-8, + .row.cols-md-8 > * { + max-width: 66.6666666667%; + flex-basis: 66.6666666667%; } + + .col-md-offset-7 { + margin-left: 58.3333333333%; } + + .col-md-9, + .row.cols-md-9 > * { + max-width: 75%; + flex-basis: 75%; } + + .col-md-offset-8 { + margin-left: 66.6666666667%; } + + .col-md-10, + .row.cols-md-10 > * { + max-width: 83.3333333333%; + flex-basis: 83.3333333333%; } + + .col-md-offset-9 { + margin-left: 75%; } + + .col-md-11, + .row.cols-md-11 > * { + max-width: 91.6666666667%; + flex-basis: 91.6666666667%; } + + .col-md-offset-10 { + margin-left: 83.3333333333%; } + + .col-md-12, + .row.cols-md-12 > * { + max-width: 100%; + flex-basis: 100%; } + + .col-md-offset-11 { + margin-left: 91.6666666667%; } + + .col-md-normal { + order: initial; } + + .col-md-first { + order: -999; } + + .col-md-last { + order: 999; } } +@media screen and (min-width: 1280px) { + .col-lg, + [class^='col-lg-'], + [class^='col-lg-offset-'], + .row[class*='cols-lg-'] > * { + box-sizing: border-box; + flex: 0 0 auto; + padding: 0 calc(var(--universal-padding) / 2); } + + .col-lg, + .row.cols-lg > * { + max-width: 100%; + flex-grow: 1; + flex-basis: 0; } + + .col-lg-1, + .row.cols-lg-1 > * { + max-width: 8.3333333333%; + flex-basis: 8.3333333333%; } + + .col-lg-offset-0 { + margin-left: 0; } + + .col-lg-2, + .row.cols-lg-2 > * { + max-width: 16.6666666667%; + flex-basis: 16.6666666667%; } + + .col-lg-offset-1 { + margin-left: 8.3333333333%; } + + .col-lg-3, + .row.cols-lg-3 > * { + max-width: 25%; + flex-basis: 25%; } + + .col-lg-offset-2 { + margin-left: 16.6666666667%; } + + .col-lg-4, + .row.cols-lg-4 > * { + max-width: 33.3333333333%; + flex-basis: 33.3333333333%; } + + .col-lg-offset-3 { + margin-left: 25%; } + + .col-lg-5, + .row.cols-lg-5 > * { + max-width: 41.6666666667%; + flex-basis: 41.6666666667%; } + + .col-lg-offset-4 { + margin-left: 33.3333333333%; } + + .col-lg-6, + .row.cols-lg-6 > * { + max-width: 50%; + flex-basis: 50%; } + + .col-lg-offset-5 { + margin-left: 41.6666666667%; } + + .col-lg-7, + .row.cols-lg-7 > * { + max-width: 58.3333333333%; + flex-basis: 58.3333333333%; } + + .col-lg-offset-6 { + margin-left: 50%; } + + .col-lg-8, + .row.cols-lg-8 > * { + max-width: 66.6666666667%; + flex-basis: 66.6666666667%; } + + .col-lg-offset-7 { + margin-left: 58.3333333333%; } + + .col-lg-9, + .row.cols-lg-9 > * { + max-width: 75%; + flex-basis: 75%; } + + .col-lg-offset-8 { + margin-left: 66.6666666667%; } + + .col-lg-10, + .row.cols-lg-10 > * { + max-width: 83.3333333333%; + flex-basis: 83.3333333333%; } + + .col-lg-offset-9 { + margin-left: 75%; } + + .col-lg-11, + .row.cols-lg-11 > * { + max-width: 91.6666666667%; + flex-basis: 91.6666666667%; } + + .col-lg-offset-10 { + margin-left: 83.3333333333%; } + + .col-lg-12, + .row.cols-lg-12 > * { + max-width: 100%; + flex-basis: 100%; } + + .col-lg-offset-11 { + margin-left: 91.6666666667%; } + + .col-lg-normal { + order: initial; } + + .col-lg-first { + order: -999; } + + .col-lg-last { + order: 999; } } +/* Card component CSS variable definitions */ +:root { + --card-back-color: #3cb4e6; + --card-fore-color: #03234b; + --card-border-color: #03234b; } + +.card { + display: flex; + flex-direction: column; + justify-content: space-between; + align-self: center; + position: relative; + width: 100%; + background: var(--card-back-color); + color: var(--card-fore-color); + border: 0.0714285714rem solid var(--card-border-color); + border-radius: var(--universal-border-radius); + margin: var(--universal-margin); + overflow: hidden; } + @media screen and (min-width: 320px) { + .card { + max-width: 320px; } } + .card > .sectione { + background: var(--card-back-color); + color: var(--card-fore-color); + box-sizing: border-box; + margin: 0; + border: 0; + border-radius: 0; + border-bottom: 0.0714285714rem solid var(--card-border-color); + padding: var(--universal-padding); + width: 100%; } + .card > .sectione.media { + height: 200px; + padding: 0; + -o-object-fit: cover; + object-fit: cover; } + .card > .sectione:last-child { + border-bottom: 0; } + +/* + Custom elements for card elements. +*/ +@media screen and (min-width: 240px) { + .card.small { + max-width: 240px; } } +@media screen and (min-width: 480px) { + .card.large { + max-width: 480px; } } +.card.fluid { + max-width: 100%; + width: auto; } + +.card.warning { + --card-back-color: #e5b8b7; + --card-fore-color: #3b234b; + --card-border-color: #8c0078; } + +.card.error { + --card-back-color: #464650; + --card-fore-color: #ffffff; + --card-border-color: #8c0078; } + +.card > .sectione.dark { + --card-back-color: #3b234b; + --card-fore-color: #ffffff; } + +.card > .sectione.double-padded { + padding: calc(1.5 * var(--universal-padding)); } + +/* + Definitions for forms and input elements. +*/ +/* Input_control module CSS variable definitions */ +:root { + --form-back-color: #ffe97f; + --form-fore-color: #03234b; + --form-border-color: #3cb4e6; + --input-back-color: #ffffff; + --input-fore-color: #03234b; + --input-border-color: #3cb4e6; + --input-focus-color: #0288d1; + --input-invalid-color: #d32f2f; + --button-back-color: #e2e2e2; + --button-hover-back-color: #dcdcdc; + --button-fore-color: #212121; + --button-border-color: transparent; + --button-hover-border-color: transparent; + --button-group-border-color: rgba(124, 124, 124, 0.54); } + +form { + background: var(--form-back-color); + color: var(--form-fore-color); + border: 0.0714285714rem solid var(--form-border-color); + border-radius: var(--universal-border-radius); + margin: var(--universal-margin); + padding: calc(2 * var(--universal-padding)) var(--universal-padding); } + +fieldset { + border: 0.0714285714rem solid var(--form-border-color); + border-radius: var(--universal-border-radius); + margin: calc(var(--universal-margin) / 4); + padding: var(--universal-padding); } + +legend { + box-sizing: border-box; + display: table; + max-width: 100%; + white-space: normal; + font-weight: 500; + padding: calc(var(--universal-padding) / 2); } + +label { + padding: calc(var(--universal-padding) / 2) var(--universal-padding); } + +.input-group { + display: inline-block; } + .input-group.fluid { + display: flex; + align-items: center; + justify-content: center; } + .input-group.fluid > input { + max-width: 100%; + flex-grow: 1; + flex-basis: 0px; } + @media screen and (max-width: 499px) { + .input-group.fluid { + align-items: stretch; + flex-direction: column; } } + .input-group.vertical { + display: flex; + align-items: stretch; + flex-direction: column; } + .input-group.vertical > input { + max-width: 100%; + flex-grow: 1; + flex-basis: 0px; } + +[type="number"]::-webkit-inner-spin-button, [type="number"]::-webkit-outer-spin-button { + height: auto; } + +[type="search"] { + -webkit-appearance: textfield; + outline-offset: -2px; } + +[type="search"]::-webkit-search-cancel-button, +[type="search"]::-webkit-search-decoration { + -webkit-appearance: none; } + +input:not([type]), [type="text"], [type="email"], [type="number"], [type="search"], +[type="password"], [type="url"], [type="tel"], [type="checkbox"], [type="radio"], textarea, select { + box-sizing: border-box; + background: var(--input-back-color); + color: var(--input-fore-color); + border: 0.0714285714rem solid var(--input-border-color); + border-radius: var(--universal-border-radius); + margin: calc(var(--universal-margin) / 2); + padding: var(--universal-padding) calc(1.5 * var(--universal-padding)); } + +input:not([type="button"]):not([type="submit"]):not([type="reset"]):hover, input:not([type="button"]):not([type="submit"]):not([type="reset"]):focus, textarea:hover, textarea:focus, select:hover, select:focus { + border-color: var(--input-focus-color); + box-shadow: none; } +input:not([type="button"]):not([type="submit"]):not([type="reset"]):invalid, input:not([type="button"]):not([type="submit"]):not([type="reset"]):focus:invalid, textarea:invalid, textarea:focus:invalid, select:invalid, select:focus:invalid { + border-color: var(--input-invalid-color); + box-shadow: none; } +input:not([type="button"]):not([type="submit"]):not([type="reset"])[readonly], textarea[readonly], select[readonly] { + background: var(--secondary-back-color); } + +select { + max-width: 100%; } + +option { + overflow: hidden; + text-overflow: ellipsis; } + +[type="checkbox"], [type="radio"] { + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; + position: relative; + height: calc(1rem + var(--universal-padding) / 2); + width: calc(1rem + var(--universal-padding) / 2); + vertical-align: text-bottom; + padding: 0; + flex-basis: calc(1rem + var(--universal-padding) / 2) !important; + flex-grow: 0 !important; } + [type="checkbox"]:checked:before, [type="radio"]:checked:before { + position: absolute; } + +[type="checkbox"]:checked:before { + content: '\2713'; + font-family: sans-serif; + font-size: calc(1rem + var(--universal-padding) / 2); + top: calc(0rem - var(--universal-padding)); + left: calc(var(--universal-padding) / 4); } + +[type="radio"] { + border-radius: 100%; } + [type="radio"]:checked:before { + border-radius: 100%; + content: ''; + top: calc(0.0714285714rem + var(--universal-padding) / 2); + left: calc(0.0714285714rem + var(--universal-padding) / 2); + background: var(--input-fore-color); + width: 0.5rem; + height: 0.5rem; } + +:placeholder-shown { + color: var(--input-fore-color); } + +::-ms-placeholder { + color: var(--input-fore-color); + opacity: 0.54; } + +button::-moz-focus-inner, [type="button"]::-moz-focus-inner, [type="reset"]::-moz-focus-inner, [type="submit"]::-moz-focus-inner { + border-style: none; + padding: 0; } + +button, html [type="button"], [type="reset"], [type="submit"] { + -webkit-appearance: button; } + +button { + overflow: visible; + text-transform: none; } + +button, [type="button"], [type="submit"], [type="reset"], +a.button, label.button, .button, +a[role="button"], label[role="button"], [role="button"] { + display: inline-block; + background: var(--button-back-color); + color: var(--button-fore-color); + border: 0.0714285714rem solid var(--button-border-color); + border-radius: var(--universal-border-radius); + padding: var(--universal-padding) calc(1.5 * var(--universal-padding)); + margin: var(--universal-margin); + text-decoration: none; + cursor: pointer; + transition: background 0.3s; } + button:hover, button:focus, [type="button"]:hover, [type="button"]:focus, [type="submit"]:hover, [type="submit"]:focus, [type="reset"]:hover, [type="reset"]:focus, + a.button:hover, + a.button:focus, label.button:hover, label.button:focus, .button:hover, .button:focus, + a[role="button"]:hover, + a[role="button"]:focus, label[role="button"]:hover, label[role="button"]:focus, [role="button"]:hover, [role="button"]:focus { + background: var(--button-hover-back-color); + border-color: var(--button-hover-border-color); } + +input:disabled, input[disabled], textarea:disabled, textarea[disabled], select:disabled, select[disabled], button:disabled, button[disabled], .button:disabled, .button[disabled], [role="button"]:disabled, [role="button"][disabled] { + cursor: not-allowed; + opacity: 0.75; } + +.button-group { + display: flex; + border: 0.0714285714rem solid var(--button-group-border-color); + border-radius: var(--universal-border-radius); + margin: var(--universal-margin); } + .button-group > button, .button-group [type="button"], .button-group > [type="submit"], .button-group > [type="reset"], .button-group > .button, .button-group > [role="button"] { + margin: 0; + max-width: 100%; + flex: 1 1 auto; + text-align: center; + border: 0; + border-radius: 0; + box-shadow: none; } + .button-group > :not(:first-child) { + border-left: 0.0714285714rem solid var(--button-group-border-color); } + @media screen and (max-width: 499px) { + .button-group { + flex-direction: column; } + .button-group > :not(:first-child) { + border: 0; + border-top: 0.0714285714rem solid var(--button-group-border-color); } } + +/* + Custom elements for forms and input elements. +*/ +button.primary, [type="button"].primary, [type="submit"].primary, [type="reset"].primary, .button.primary, [role="button"].primary { + --button-back-color: #1976d2; + --button-fore-color: #f8f8f8; } + button.primary:hover, button.primary:focus, [type="button"].primary:hover, [type="button"].primary:focus, [type="submit"].primary:hover, [type="submit"].primary:focus, [type="reset"].primary:hover, [type="reset"].primary:focus, .button.primary:hover, .button.primary:focus, [role="button"].primary:hover, [role="button"].primary:focus { + --button-hover-back-color: #1565c0; } + +button.secondary, [type="button"].secondary, [type="submit"].secondary, [type="reset"].secondary, .button.secondary, [role="button"].secondary { + --button-back-color: #d32f2f; + --button-fore-color: #f8f8f8; } + button.secondary:hover, button.secondary:focus, [type="button"].secondary:hover, [type="button"].secondary:focus, [type="submit"].secondary:hover, [type="submit"].secondary:focus, [type="reset"].secondary:hover, [type="reset"].secondary:focus, .button.secondary:hover, .button.secondary:focus, [role="button"].secondary:hover, [role="button"].secondary:focus { + --button-hover-back-color: #c62828; } + +button.tertiary, [type="button"].tertiary, [type="submit"].tertiary, [type="reset"].tertiary, .button.tertiary, [role="button"].tertiary { + --button-back-color: #308732; + --button-fore-color: #f8f8f8; } + button.tertiary:hover, button.tertiary:focus, [type="button"].tertiary:hover, [type="button"].tertiary:focus, [type="submit"].tertiary:hover, [type="submit"].tertiary:focus, [type="reset"].tertiary:hover, [type="reset"].tertiary:focus, .button.tertiary:hover, .button.tertiary:focus, [role="button"].tertiary:hover, [role="button"].tertiary:focus { + --button-hover-back-color: #277529; } + +button.inverse, [type="button"].inverse, [type="submit"].inverse, [type="reset"].inverse, .button.inverse, [role="button"].inverse { + --button-back-color: #212121; + --button-fore-color: #f8f8f8; } + button.inverse:hover, button.inverse:focus, [type="button"].inverse:hover, [type="button"].inverse:focus, [type="submit"].inverse:hover, [type="submit"].inverse:focus, [type="reset"].inverse:hover, [type="reset"].inverse:focus, .button.inverse:hover, .button.inverse:focus, [role="button"].inverse:hover, [role="button"].inverse:focus { + --button-hover-back-color: #111; } + +button.small, [type="button"].small, [type="submit"].small, [type="reset"].small, .button.small, [role="button"].small { + padding: calc(0.5 * var(--universal-padding)) calc(0.75 * var(--universal-padding)); + margin: var(--universal-margin); } + +button.large, [type="button"].large, [type="submit"].large, [type="reset"].large, .button.large, [role="button"].large { + padding: calc(1.5 * var(--universal-padding)) calc(2 * var(--universal-padding)); + margin: var(--universal-margin); } + +/* + Definitions for navigation elements. +*/ +/* Navigation module CSS variable definitions */ +:root { + --header-back-color: #03234b; + --header-hover-back-color: #ffd200; + --header-fore-color: #ffffff; + --header-border-color: #3cb4e6; + --nav-back-color: #ffffff; + --nav-hover-back-color: #ffe97f; + --nav-fore-color: #e6007e; + --nav-border-color: #3cb4e6; + --nav-link-color: #3cb4e6; + --footer-fore-color: #ffffff; + --footer-back-color: #03234b; + --footer-border-color: #3cb4e6; + --footer-link-color: #3cb4e6; + --drawer-back-color: #ffffff; + --drawer-hover-back-color: #ffe97f; + --drawer-border-color: #3cb4e6; + --drawer-close-color: #e6007e; } + +header { + height: 2.75rem; + background: var(--header-back-color); + color: var(--header-fore-color); + border-bottom: 0.0714285714rem solid var(--header-border-color); + padding: calc(var(--universal-padding) / 4) 0; + white-space: nowrap; + overflow-x: auto; + overflow-y: hidden; } + header.row { + box-sizing: content-box; } + header .logo { + color: var(--header-fore-color); + font-size: 1.75rem; + padding: var(--universal-padding) calc(2 * var(--universal-padding)); + text-decoration: none; } + header button, header [type="button"], header .button, header [role="button"] { + box-sizing: border-box; + position: relative; + top: calc(0rem - var(--universal-padding) / 4); + height: calc(3.1875rem + var(--universal-padding) / 2); + background: var(--header-back-color); + line-height: calc(3.1875rem - var(--universal-padding) * 1.5); + text-align: center; + color: var(--header-fore-color); + border: 0; + border-radius: 0; + margin: 0; + text-transform: uppercase; } + header button:hover, header button:focus, header [type="button"]:hover, header [type="button"]:focus, header .button:hover, header .button:focus, header [role="button"]:hover, header [role="button"]:focus { + background: var(--header-hover-back-color); } + +nav { + background: var(--nav-back-color); + color: var(--nav-fore-color); + border: 0.0714285714rem solid var(--nav-border-color); + border-radius: var(--universal-border-radius); + margin: var(--universal-margin); } + nav * { + padding: var(--universal-padding) calc(1.5 * var(--universal-padding)); } + nav a, nav a:visited { + display: block; + color: var(--nav-link-color); + border-radius: var(--universal-border-radius); + transition: background 0.3s; } + nav a:hover, nav a:focus, nav a:visited:hover, nav a:visited:focus { + text-decoration: none; + background: var(--nav-hover-back-color); } + nav .sublink-1 { + position: relative; + margin-left: calc(2 * var(--universal-padding)); } + nav .sublink-1:before { + position: absolute; + left: calc(var(--universal-padding) - 1 * var(--universal-padding)); + top: -0.0714285714rem; + content: ''; + height: 100%; + border: 0.0714285714rem solid var(--nav-border-color); + border-left: 0; } + nav .sublink-2 { + position: relative; + margin-left: calc(4 * var(--universal-padding)); } + nav .sublink-2:before { + position: absolute; + left: calc(var(--universal-padding) - 3 * var(--universal-padding)); + top: -0.0714285714rem; + content: ''; + height: 100%; + border: 0.0714285714rem solid var(--nav-border-color); + border-left: 0; } + +footer { + background: var(--footer-back-color); + color: var(--footer-fore-color); + border-top: 0.0714285714rem solid var(--footer-border-color); + padding: calc(2 * var(--universal-padding)) var(--universal-padding); + font-size: 0.875rem; } + footer a, footer a:visited { + color: var(--footer-link-color); } + +header.sticky { + position: -webkit-sticky; + position: sticky; + z-index: 1101; + top: 0; } + +footer.sticky { + position: -webkit-sticky; + position: sticky; + z-index: 1101; + bottom: 0; } + +.drawer-toggle:before { + display: inline-block; + position: relative; + vertical-align: bottom; + content: '\00a0\2261\00a0'; + font-family: sans-serif; + font-size: 1.5em; } +@media screen and (min-width: 500px) { + .drawer-toggle:not(.persistent) { + display: none; } } + +[type="checkbox"].drawer { + height: 1px; + width: 1px; + margin: -1px; + overflow: hidden; + position: absolute; + clip: rect(0 0 0 0); + -webkit-clip-path: inset(100%); + clip-path: inset(100%); } + [type="checkbox"].drawer + * { + display: block; + box-sizing: border-box; + position: fixed; + top: 0; + width: 320px; + height: 100vh; + overflow-y: auto; + background: var(--drawer-back-color); + border: 0.0714285714rem solid var(--drawer-border-color); + border-radius: 0; + margin: 0; + z-index: 1110; + right: -320px; + transition: right 0.3s; } + [type="checkbox"].drawer + * .drawer-close { + position: absolute; + top: var(--universal-margin); + right: var(--universal-margin); + z-index: 1111; + width: 2rem; + height: 2rem; + border-radius: var(--universal-border-radius); + padding: var(--universal-padding); + margin: 0; + cursor: pointer; + transition: background 0.3s; } + [type="checkbox"].drawer + * .drawer-close:before { + display: block; + content: '\00D7'; + color: var(--drawer-close-color); + position: relative; + font-family: sans-serif; + font-size: 2rem; + line-height: 1; + text-align: center; } + [type="checkbox"].drawer + * .drawer-close:hover, [type="checkbox"].drawer + * .drawer-close:focus { + background: var(--drawer-hover-back-color); } + @media screen and (max-width: 320px) { + [type="checkbox"].drawer + * { + width: 100%; } } + [type="checkbox"].drawer:checked + * { + right: 0; } + @media screen and (min-width: 500px) { + [type="checkbox"].drawer:not(.persistent) + * { + position: static; + height: 100%; + z-index: 1100; } + [type="checkbox"].drawer:not(.persistent) + * .drawer-close { + display: none; } } + +/* + Definitions for the responsive table component. +*/ +/* Table module CSS variable definitions. */ +:root { + --table-border-color: #03234b; + --table-border-separator-color: #03234b; + --table-head-back-color: #03234b; + --table-head-fore-color: #ffffff; + --table-body-back-color: #ffffff; + --table-body-fore-color: #03234b; + --table-body-alt-back-color: #f4f4f4; } + +table { + border-collapse: separate; + border-spacing: 0; + margin: 0; + display: flex; + flex: 0 1 auto; + flex-flow: row wrap; + padding: var(--universal-padding); + padding-top: 0; } + table caption { + font-size: 1rem; + margin: calc(2 * var(--universal-margin)) 0; + max-width: 100%; + flex: 0 0 100%; } + table thead, table tbody { + display: flex; + flex-flow: row wrap; + border: 0.0714285714rem solid var(--table-border-color); } + table thead { + z-index: 999; + border-radius: var(--universal-border-radius) var(--universal-border-radius) 0 0; + border-bottom: 0.0714285714rem solid var(--table-border-separator-color); } + table tbody { + border-top: 0; + margin-top: calc(0 - var(--universal-margin)); + border-radius: 0 0 var(--universal-border-radius) var(--universal-border-radius); } + table tr { + display: flex; + padding: 0; } + table th, table td { + padding: calc(0.5 * var(--universal-padding)); + font-size: 0.9rem; } + table th { + text-align: left; + background: var(--table-head-back-color); + color: var(--table-head-fore-color); } + table td { + background: var(--table-body-back-color); + color: var(--table-body-fore-color); + border-top: 0.0714285714rem solid var(--table-border-color); } + +table:not(.horizontal) { + overflow: auto; + max-height: 100%; } + table:not(.horizontal) thead, table:not(.horizontal) tbody { + max-width: 100%; + flex: 0 0 100%; } + table:not(.horizontal) tr { + flex-flow: row wrap; + flex: 0 0 100%; } + table:not(.horizontal) th, table:not(.horizontal) td { + flex: 1 0 0%; + overflow: hidden; + text-overflow: ellipsis; } + table:not(.horizontal) thead { + position: sticky; + top: 0; } + table:not(.horizontal) tbody tr:first-child td { + border-top: 0; } + +table.horizontal { + border: 0; } + table.horizontal thead, table.horizontal tbody { + border: 0; + flex: .2 0 0; + flex-flow: row nowrap; } + table.horizontal tbody { + overflow: auto; + justify-content: space-between; + flex: .8 0 0; + margin-left: 0; + padding-bottom: calc(var(--universal-padding) / 4); } + table.horizontal tr { + flex-direction: column; + flex: 1 0 auto; } + table.horizontal th, table.horizontal td { + width: auto; + border: 0; + border-bottom: 0.0714285714rem solid var(--table-border-color); } + table.horizontal th:not(:first-child), table.horizontal td:not(:first-child) { + border-top: 0; } + table.horizontal th { + text-align: right; + border-left: 0.0714285714rem solid var(--table-border-color); + border-right: 0.0714285714rem solid var(--table-border-separator-color); } + table.horizontal thead tr:first-child { + padding-left: 0; } + table.horizontal th:first-child, table.horizontal td:first-child { + border-top: 0.0714285714rem solid var(--table-border-color); } + table.horizontal tbody tr:last-child td { + border-right: 0.0714285714rem solid var(--table-border-color); } + table.horizontal tbody tr:last-child td:first-child { + border-top-right-radius: 0.25rem; } + table.horizontal tbody tr:last-child td:last-child { + border-bottom-right-radius: 0.25rem; } + table.horizontal thead tr:first-child th:first-child { + border-top-left-radius: 0.25rem; } + table.horizontal thead tr:first-child th:last-child { + border-bottom-left-radius: 0.25rem; } + +@media screen and (max-width: 499px) { + table, table.horizontal { + border-collapse: collapse; + border: 0; + width: 100%; + display: table; } + table thead, table th, table.horizontal thead, table.horizontal th { + border: 0; + height: 1px; + width: 1px; + margin: -1px; + overflow: hidden; + padding: 0; + position: absolute; + clip: rect(0 0 0 0); + -webkit-clip-path: inset(100%); + clip-path: inset(100%); } + table tbody, table.horizontal tbody { + border: 0; + display: table-row-group; } + table tr, table.horizontal tr { + display: block; + border: 0.0714285714rem solid var(--table-border-color); + border-radius: var(--universal-border-radius); + background: #ffffff; + padding: var(--universal-padding); + margin: var(--universal-margin); + margin-bottom: calc(1 * var(--universal-margin)); } + table th, table td, table.horizontal th, table.horizontal td { + width: auto; } + table td, table.horizontal td { + display: block; + border: 0; + text-align: right; } + table td:before, table.horizontal td:before { + content: attr(data-label); + float: left; + font-weight: 600; } + table th:first-child, table td:first-child, table.horizontal th:first-child, table.horizontal td:first-child { + border-top: 0; } + table tbody tr:last-child td, table.horizontal tbody tr:last-child td { + border-right: 0; } } +table tr:nth-of-type(2n) > td { + background: var(--table-body-alt-back-color); } + +@media screen and (max-width: 500px) { + table tr:nth-of-type(2n) { + background: var(--table-body-alt-back-color); } } +:root { + --table-body-hover-back-color: #90caf9; } + +table.hoverable tr:hover, table.hoverable tr:hover > td, table.hoverable tr:focus, table.hoverable tr:focus > td { + background: var(--table-body-hover-back-color); } + +@media screen and (max-width: 500px) { + table.hoverable tr:hover, table.hoverable tr:hover > td, table.hoverable tr:focus, table.hoverable tr:focus > td { + background: var(--table-body-hover-back-color); } } +/* + Definitions for contextual background elements, toasts and tooltips. +*/ +/* Contextual module CSS variable definitions */ +:root { + --mark-back-color: #3cb4e6; + --mark-fore-color: #ffffff; } + +mark { + background: var(--mark-back-color); + color: var(--mark-fore-color); + font-size: 0.95em; + line-height: 1em; + border-radius: var(--universal-border-radius); + padding: calc(var(--universal-padding) / 4) var(--universal-padding); } + mark.inline-block { + display: inline-block; + font-size: 1em; + line-height: 1.4; + padding: calc(var(--universal-padding) / 2) var(--universal-padding); } + +:root { + --toast-back-color: #424242; + --toast-fore-color: #fafafa; } + +.toast { + position: fixed; + bottom: calc(var(--universal-margin) * 3); + left: 50%; + transform: translate(-50%, -50%); + z-index: 1111; + color: var(--toast-fore-color); + background: var(--toast-back-color); + border-radius: calc(var(--universal-border-radius) * 16); + padding: var(--universal-padding) calc(var(--universal-padding) * 3); } + +:root { + --tooltip-back-color: #212121; + --tooltip-fore-color: #fafafa; } + +.tooltip { + position: relative; + display: inline-block; } + .tooltip:before, .tooltip:after { + position: absolute; + opacity: 0; + clip: rect(0 0 0 0); + -webkit-clip-path: inset(100%); + clip-path: inset(100%); + transition: all 0.3s; + z-index: 1010; + left: 50%; } + .tooltip:not(.bottom):before, .tooltip:not(.bottom):after { + bottom: 75%; } + .tooltip.bottom:before, .tooltip.bottom:after { + top: 75%; } + .tooltip:hover:before, .tooltip:hover:after, .tooltip:focus:before, .tooltip:focus:after { + opacity: 1; + clip: auto; + -webkit-clip-path: inset(0%); + clip-path: inset(0%); } + .tooltip:before { + content: ''; + background: transparent; + border: var(--universal-margin) solid transparent; + left: calc(50% - var(--universal-margin)); } + .tooltip:not(.bottom):before { + border-top-color: #212121; } + .tooltip.bottom:before { + border-bottom-color: #212121; } + .tooltip:after { + content: attr(aria-label); + color: var(--tooltip-fore-color); + background: var(--tooltip-back-color); + border-radius: var(--universal-border-radius); + padding: var(--universal-padding); + white-space: nowrap; + transform: translateX(-50%); } + .tooltip:not(.bottom):after { + margin-bottom: calc(2 * var(--universal-margin)); } + .tooltip.bottom:after { + margin-top: calc(2 * var(--universal-margin)); } + +:root { + --modal-overlay-color: rgba(0, 0, 0, 0.45); + --modal-close-color: #e6007e; + --modal-close-hover-color: #ffe97f; } + +[type="checkbox"].modal { + height: 1px; + width: 1px; + margin: -1px; + overflow: hidden; + position: absolute; + clip: rect(0 0 0 0); + -webkit-clip-path: inset(100%); + clip-path: inset(100%); } + [type="checkbox"].modal + div { + position: fixed; + top: 0; + left: 0; + display: none; + width: 100vw; + height: 100vh; + background: var(--modal-overlay-color); } + [type="checkbox"].modal + div .card { + margin: 0 auto; + max-height: 50vh; + overflow: auto; } + [type="checkbox"].modal + div .card .modal-close { + position: absolute; + top: 0; + right: 0; + width: 1.75rem; + height: 1.75rem; + border-radius: var(--universal-border-radius); + padding: var(--universal-padding); + margin: 0; + cursor: pointer; + transition: background 0.3s; } + [type="checkbox"].modal + div .card .modal-close:before { + display: block; + content: '\00D7'; + color: var(--modal-close-color); + position: relative; + font-family: sans-serif; + font-size: 1.75rem; + line-height: 1; + text-align: center; } + [type="checkbox"].modal + div .card .modal-close:hover, [type="checkbox"].modal + div .card .modal-close:focus { + background: var(--modal-close-hover-color); } + [type="checkbox"].modal:checked + div { + display: flex; + flex: 0 1 auto; + z-index: 1200; } + [type="checkbox"].modal:checked + div .card .modal-close { + z-index: 1211; } + +:root { + --collapse-label-back-color: #03234b; + --collapse-label-fore-color: #ffffff; + --collapse-label-hover-back-color: #3cb4e6; + --collapse-selected-label-back-color: #3cb4e6; + --collapse-border-color: var(--collapse-label-back-color); + --collapse-selected-border-color: #ceecf8; + --collapse-content-back-color: #ffffff; + --collapse-selected-label-border-color: #3cb4e6; } + +.collapse { + width: calc(100% - 2 * var(--universal-margin)); + opacity: 1; + display: flex; + flex-direction: column; + margin: var(--universal-margin); + border-radius: var(--universal-border-radius); } + .collapse > [type="radio"], .collapse > [type="checkbox"] { + height: 1px; + width: 1px; + margin: -1px; + overflow: hidden; + position: absolute; + clip: rect(0 0 0 0); + -webkit-clip-path: inset(100%); + clip-path: inset(100%); } + .collapse > label { + flex-grow: 1; + display: inline-block; + height: 1.25rem; + cursor: pointer; + transition: background 0.2s; + color: var(--collapse-label-fore-color); + background: var(--collapse-label-back-color); + border: 0.0714285714rem solid var(--collapse-selected-border-color); + padding: calc(1.25 * var(--universal-padding)); } + .collapse > label:hover, .collapse > label:focus { + background: var(--collapse-label-hover-back-color); } + .collapse > label + div { + flex-basis: auto; + height: 1px; + width: 1px; + margin: -1px; + overflow: hidden; + position: absolute; + clip: rect(0 0 0 0); + -webkit-clip-path: inset(100%); + clip-path: inset(100%); + transition: max-height 0.3s; + max-height: 1px; } + .collapse > :checked + label { + background: var(--collapse-selected-label-back-color); + border-color: var(--collapse-selected-label-border-color); } + .collapse > :checked + label + div { + box-sizing: border-box; + position: relative; + width: 100%; + height: auto; + overflow: auto; + margin: 0; + background: var(--collapse-content-back-color); + border: 0.0714285714rem solid var(--collapse-selected-border-color); + border-top: 0; + padding: var(--universal-padding); + clip: auto; + -webkit-clip-path: inset(0%); + clip-path: inset(0%); + max-height: 100%; } + .collapse > label:not(:first-of-type) { + border-top: 0; } + .collapse > label:first-of-type { + border-radius: var(--universal-border-radius) var(--universal-border-radius) 0 0; } + .collapse > label:last-of-type:not(:first-of-type) { + border-radius: 0 0 var(--universal-border-radius) var(--universal-border-radius); } + .collapse > label:last-of-type:first-of-type { + border-radius: var(--universal-border-radius); } + .collapse > :checked:last-of-type:not(:first-of-type) + label { + border-radius: 0; } + .collapse > :checked:last-of-type + label + div { + border-radius: 0 0 var(--universal-border-radius) var(--universal-border-radius); } + +/* + Custom elements for contextual background elements, toasts and tooltips. +*/ +mark.tertiary { + --mark-back-color: #3cb4e6; } + +mark.tag { + padding: calc(var(--universal-padding)/2) var(--universal-padding); + border-radius: 1em; } + +/* + Definitions for progress elements and spinners. +*/ +/* Progress module CSS variable definitions */ +:root { + --progress-back-color: #3cb4e6; + --progress-fore-color: #555; } + +progress { + display: block; + vertical-align: baseline; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; + height: 0.75rem; + width: calc(100% - 2 * var(--universal-margin)); + margin: var(--universal-margin); + border: 0; + border-radius: calc(2 * var(--universal-border-radius)); + background: var(--progress-back-color); + color: var(--progress-fore-color); } + progress::-webkit-progress-value { + background: var(--progress-fore-color); + border-top-left-radius: calc(2 * var(--universal-border-radius)); + border-bottom-left-radius: calc(2 * var(--universal-border-radius)); } + progress::-webkit-progress-bar { + background: var(--progress-back-color); } + progress::-moz-progress-bar { + background: var(--progress-fore-color); + border-top-left-radius: calc(2 * var(--universal-border-radius)); + border-bottom-left-radius: calc(2 * var(--universal-border-radius)); } + progress[value="1000"]::-webkit-progress-value { + border-radius: calc(2 * var(--universal-border-radius)); } + progress[value="1000"]::-moz-progress-bar { + border-radius: calc(2 * var(--universal-border-radius)); } + progress.inline { + display: inline-block; + vertical-align: middle; + width: 60%; } + +:root { + --spinner-back-color: #ddd; + --spinner-fore-color: #555; } + +@keyframes spinner-donut-anim { + 0% { + transform: rotate(0deg); } + 100% { + transform: rotate(360deg); } } +.spinner { + display: inline-block; + margin: var(--universal-margin); + border: 0.25rem solid var(--spinner-back-color); + border-left: 0.25rem solid var(--spinner-fore-color); + border-radius: 50%; + width: 1.25rem; + height: 1.25rem; + animation: spinner-donut-anim 1.2s linear infinite; } + +/* + Custom elements for progress bars and spinners. +*/ +progress.primary { + --progress-fore-color: #1976d2; } + +progress.secondary { + --progress-fore-color: #d32f2f; } + +progress.tertiary { + --progress-fore-color: #308732; } + +.spinner.primary { + --spinner-fore-color: #1976d2; } + +.spinner.secondary { + --spinner-fore-color: #d32f2f; } + +.spinner.tertiary { + --spinner-fore-color: #308732; } + +/* + Definitions for icons - powered by Feather (https://feathericons.com/). +*/ +span[class^='icon-'] { + display: inline-block; + height: 1em; + width: 1em; + vertical-align: -0.125em; + background-size: contain; + margin: 0 calc(var(--universal-margin) / 4); } + span[class^='icon-'].secondary { + -webkit-filter: invert(25%); + filter: invert(25%); } + span[class^='icon-'].inverse { + -webkit-filter: invert(100%); + filter: invert(100%); } + +span.icon-alert { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='8' x2='12' y2='12'%3E%3C/line%3E%3Cline x1='12' y1='16' x2='12' y2='16'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-bookmark { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M19 21l-7-5-7 5V5a2 2 0 0 1 2-2h10a2 2 0 0 1 2 2z'%3E%3C/path%3E%3C/svg%3E"); } +span.icon-calendar { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='3' y='4' width='18' height='18' rx='2' ry='2'%3E%3C/rect%3E%3Cline x1='16' y1='2' x2='16' y2='6'%3E%3C/line%3E%3Cline x1='8' y1='2' x2='8' y2='6'%3E%3C/line%3E%3Cline x1='3' y1='10' x2='21' y2='10'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-credit { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='1' y='4' width='22' height='16' rx='2' ry='2'%3E%3C/rect%3E%3Cline x1='1' y1='10' x2='23' y2='10'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-edit { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M20 14.66V20a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V6a2 2 0 0 1 2-2h5.34'%3E%3C/path%3E%3Cpolygon points='18 2 22 6 12 16 8 16 8 12 18 2'%3E%3C/polygon%3E%3C/svg%3E"); } +span.icon-link { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6'%3E%3C/path%3E%3Cpolyline points='15 3 21 3 21 9'%3E%3C/polyline%3E%3Cline x1='10' y1='14' x2='21' y2='3'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-help { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M9.09 9a3 3 0 0 1 5.83 1c0 2-3 3-3 3'%3E%3C/path%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='17' x2='12' y2='17'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-home { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M3 9l9-7 9 7v11a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2z'%3E%3C/path%3E%3Cpolyline points='9 22 9 12 15 12 15 22'%3E%3C/polyline%3E%3C/svg%3E"); } +span.icon-info { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='16' x2='12' y2='12'%3E%3C/line%3E%3Cline x1='12' y1='8' x2='12' y2='8'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-lock { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='3' y='11' width='18' height='11' rx='2' ry='2'%3E%3C/rect%3E%3Cpath d='M7 11V7a5 5 0 0 1 10 0v4'%3E%3C/path%3E%3C/svg%3E"); } +span.icon-mail { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M4 4h16c1.1 0 2 .9 2 2v12c0 1.1-.9 2-2 2H4c-1.1 0-2-.9-2-2V6c0-1.1.9-2 2-2z'%3E%3C/path%3E%3Cpolyline points='22,6 12,13 2,6'%3E%3C/polyline%3E%3C/svg%3E"); } +span.icon-location { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M21 10c0 7-9 13-9 13s-9-6-9-13a9 9 0 0 1 18 0z'%3E%3C/path%3E%3Ccircle cx='12' cy='10' r='3'%3E%3C/circle%3E%3C/svg%3E"); } +span.icon-phone { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M22 16.92v3a2 2 0 0 1-2.18 2 19.79 19.79 0 0 1-8.63-3.07 19.5 19.5 0 0 1-6-6 19.79 19.79 0 0 1-3.07-8.67A2 2 0 0 1 4.11 2h3a2 2 0 0 1 2 1.72 12.84 12.84 0 0 0 .7 2.81 2 2 0 0 1-.45 2.11L8.09 9.91a16 16 0 0 0 6 6l1.27-1.27a2 2 0 0 1 2.11-.45 12.84 12.84 0 0 0 2.81.7A2 2 0 0 1 22 16.92z'%3E%3C/path%3E%3C/svg%3E"); } +span.icon-rss { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M4 11a9 9 0 0 1 9 9'%3E%3C/path%3E%3Cpath d='M4 4a16 16 0 0 1 16 16'%3E%3C/path%3E%3Ccircle cx='5' cy='19' r='1'%3E%3C/circle%3E%3C/svg%3E"); } +span.icon-search { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='11' cy='11' r='8'%3E%3C/circle%3E%3Cline x1='21' y1='21' x2='16.65' y2='16.65'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-settings { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='3'%3E%3C/circle%3E%3Cpath d='M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z'%3E%3C/path%3E%3C/svg%3E"); } +span.icon-share { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='18' cy='5' r='3'%3E%3C/circle%3E%3Ccircle cx='6' cy='12' r='3'%3E%3C/circle%3E%3Ccircle cx='18' cy='19' r='3'%3E%3C/circle%3E%3Cline x1='8.59' y1='13.51' x2='15.42' y2='17.49'%3E%3C/line%3E%3Cline x1='15.41' y1='6.51' x2='8.59' y2='10.49'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-cart { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='9' cy='21' r='1'%3E%3C/circle%3E%3Ccircle cx='20' cy='21' r='1'%3E%3C/circle%3E%3Cpath d='M1 1h4l2.68 13.39a2 2 0 0 0 2 1.61h9.72a2 2 0 0 0 2-1.61L23 6H6'%3E%3C/path%3E%3C/svg%3E"); } +span.icon-upload { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4'%3E%3C/path%3E%3Cpolyline points='17 8 12 3 7 8'%3E%3C/polyline%3E%3Cline x1='12' y1='3' x2='12' y2='15'%3E%3C/line%3E%3C/svg%3E"); } +span.icon-user { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='%2303234b' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M20 21v-2a4 4 0 0 0-4-4H8a4 4 0 0 0-4 4v2'%3E%3C/path%3E%3Ccircle cx='12' cy='7' r='4'%3E%3C/circle%3E%3C/svg%3E"); } + +/* + Definitions for STMicroelectronics icons (https://brandportal.st.com/document/26). +*/ +span.icon-st-update { + background-image: url("Update.svg"); } +span.icon-st-add { + background-image: url("Add button.svg"); } + +/* + Definitions for utilities and helper classes. +*/ +/* Utility module CSS variable definitions */ +:root { + --generic-border-color: rgba(0, 0, 0, 0.3); + --generic-box-shadow: 0 0.2857142857rem 0.2857142857rem 0 rgba(0, 0, 0, 0.125), 0 0.1428571429rem 0.1428571429rem -0.1428571429rem rgba(0, 0, 0, 0.125); } + +.hidden { + display: none !important; } + +.visually-hidden { + position: absolute !important; + width: 1px !important; + height: 1px !important; + margin: -1px !important; + border: 0 !important; + padding: 0 !important; + clip: rect(0 0 0 0) !important; + -webkit-clip-path: inset(100%) !important; + clip-path: inset(100%) !important; + overflow: hidden !important; } + +.bordered { + border: 0.0714285714rem solid var(--generic-border-color) !important; } + +.rounded { + border-radius: var(--universal-border-radius) !important; } + +.circular { + border-radius: 50% !important; } + +.shadowed { + box-shadow: var(--generic-box-shadow) !important; } + +.responsive-margin { + margin: calc(var(--universal-margin) / 4) !important; } + @media screen and (min-width: 500px) { + .responsive-margin { + margin: calc(var(--universal-margin) / 2) !important; } } + @media screen and (min-width: 1280px) { + .responsive-margin { + margin: var(--universal-margin) !important; } } + +.responsive-padding { + padding: calc(var(--universal-padding) / 4) !important; } + @media screen and (min-width: 500px) { + .responsive-padding { + padding: calc(var(--universal-padding) / 2) !important; } } + @media screen and (min-width: 1280px) { + .responsive-padding { + padding: var(--universal-padding) !important; } } + +@media screen and (max-width: 499px) { + .hidden-sm { + display: none !important; } } +@media screen and (min-width: 500px) and (max-width: 1279px) { + .hidden-md { + display: none !important; } } +@media screen and (min-width: 1280px) { + .hidden-lg { + display: none !important; } } +@media screen and (max-width: 499px) { + .visually-hidden-sm { + position: absolute !important; + width: 1px !important; + height: 1px !important; + margin: -1px !important; + border: 0 !important; + padding: 0 !important; + clip: rect(0 0 0 0) !important; + -webkit-clip-path: inset(100%) !important; + clip-path: inset(100%) !important; + overflow: hidden !important; } } +@media screen and (min-width: 500px) and (max-width: 1279px) { + .visually-hidden-md { + position: absolute !important; + width: 1px !important; + height: 1px !important; + margin: -1px !important; + border: 0 !important; + padding: 0 !important; + clip: rect(0 0 0 0) !important; + -webkit-clip-path: inset(100%) !important; + clip-path: inset(100%) !important; + overflow: hidden !important; } } +@media screen and (min-width: 1280px) { + .visually-hidden-lg { + position: absolute !important; + width: 1px !important; + height: 1px !important; + margin: -1px !important; + border: 0 !important; + padding: 0 !important; + clip: rect(0 0 0 0) !important; + -webkit-clip-path: inset(100%) !important; + clip-path: inset(100%) !important; + overflow: hidden !important; } } + +/*# sourceMappingURL=mini-custom.css.map */ + +img[alt="ST logo"] { display: block; margin: auto; width: 75%; max-width: 250px; min-width: 71px; } +img[alt="Cube logo"] { float: right; width: 30%; max-width: 10rem; min-width: 8rem; padding-right: 1rem;} + +.figure { + display: block; + margin-left: auto; + margin-right: auto; + text-align: center; +} \ No newline at end of file diff --git a/stm32/system/Middlewares/ST/STM32_USB_Host_Library/_htmresc/st_logo_2020.png b/stm32/system/Middlewares/ST/STM32_USB_Host_Library/_htmresc/st_logo_2020.png new file mode 100644 index 0000000000000000000000000000000000000000..d6cebb5ac70e0594cbe37a6b60036a80ef3bed37 GIT binary patch literal 7520 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"NJQ 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 b^}%0^qYmp2WD(pA|JtZ4>nPPKybJpu3@X?! literal 0 HcmV?d00001 diff --git a/stm32/system/STM32F4xx/stm32f4xx_hal_conf_default.h b/stm32/system/STM32F4xx/stm32f4xx_hal_conf_default.h index 36a7e0e5..0f589ea5 100644 --- a/stm32/system/STM32F4xx/stm32f4xx_hal_conf_default.h +++ b/stm32/system/STM32F4xx/stm32f4xx_hal_conf_default.h @@ -167,45 +167,123 @@ in voltage and temperature. */ #define DATA_CACHE_ENABLE 1U #endif +#if !defined(USE_HAL_ADC_REGISTER_CALLBACKS) #define USE_HAL_ADC_REGISTER_CALLBACKS 0U /* ADC register callback disabled */ +#endif +#if !defined(USE_HAL_CAN_REGISTER_CALLBACKS) #define USE_HAL_CAN_REGISTER_CALLBACKS 0U /* CAN register callback disabled */ +#endif +#if !defined(USE_HAL_CEC_REGISTER_CALLBACKS) #define USE_HAL_CEC_REGISTER_CALLBACKS 0U /* CEC register callback disabled */ +#endif +#if !defined(USE_HAL_CRYP_REGISTER_CALLBACKS) #define USE_HAL_CRYP_REGISTER_CALLBACKS 0U /* CRYP register callback disabled */ +#endif +#if !defined(USE_HAL_DAC_REGISTER_CALLBACKS) #define USE_HAL_DAC_REGISTER_CALLBACKS 0U /* DAC register callback disabled */ +#endif +#if !defined(USE_HAL_DCMI_REGISTER_CALLBACKS) #define USE_HAL_DCMI_REGISTER_CALLBACKS 0U /* DCMI register callback disabled */ +#endif +#if !defined(USE_HAL_DFSDM_REGISTER_CALLBACKS) #define USE_HAL_DFSDM_REGISTER_CALLBACKS 0U /* DFSDM register callback disabled */ +#endif +#if !defined(USE_HAL_DMA2D_REGISTER_CALLBACKS) #define USE_HAL_DMA2D_REGISTER_CALLBACKS 0U /* DMA2D register callback disabled */ +#endif +#if !defined(USE_HAL_DSI_REGISTER_CALLBACKS) #define USE_HAL_DSI_REGISTER_CALLBACKS 0U /* DSI register callback disabled */ +#endif +#if !defined(USE_HAL_ETH_REGISTER_CALLBACKS) #define USE_HAL_ETH_REGISTER_CALLBACKS 0U /* ETH register callback disabled */ +#endif +#if !defined(USE_HAL_HASH_REGISTER_CALLBACKS) #define USE_HAL_HASH_REGISTER_CALLBACKS 0U /* HASH register callback disabled */ +#endif +#if !defined(USE_HAL_HCD_REGISTER_CALLBACKS) #define USE_HAL_HCD_REGISTER_CALLBACKS 0U /* HCD register callback disabled */ +#endif +#if !defined(USE_HAL_I2C_REGISTER_CALLBACKS) #define USE_HAL_I2C_REGISTER_CALLBACKS 0U /* I2C register callback disabled */ +#endif +#if !defined(USE_HAL_FMPI2C_REGISTER_CALLBACKS) #define USE_HAL_FMPI2C_REGISTER_CALLBACKS 0U /* FMPI2C register callback disabled */ +#endif +#if !defined(USE_HAL_FMPSMBUS_REGISTER_CALLBACKS) #define USE_HAL_FMPSMBUS_REGISTER_CALLBACKS 0U /* FMPSMBUS register callback disabled */ +#endif +#if !defined(USE_HAL_I2S_REGISTER_CALLBACKS) #define USE_HAL_I2S_REGISTER_CALLBACKS 0U /* I2S register callback disabled */ +#endif +#if !defined(USE_HAL_IRDA_REGISTER_CALLBACKS) #define USE_HAL_IRDA_REGISTER_CALLBACKS 0U /* IRDA register callback disabled */ +#endif +#if !defined(USE_HAL_LPTIM_REGISTER_CALLBACKS) #define USE_HAL_LPTIM_REGISTER_CALLBACKS 0U /* LPTIM register callback disabled */ +#endif +#if !defined(USE_HAL_LTDC_REGISTER_CALLBACKS) #define USE_HAL_LTDC_REGISTER_CALLBACKS 0U /* LTDC register callback disabled */ +#endif +#if !defined(USE_HAL_MMC_REGISTER_CALLBACKS) #define USE_HAL_MMC_REGISTER_CALLBACKS 0U /* MMC register callback disabled */ +#endif +#if !defined(USE_HAL_NAND_REGISTER_CALLBACKS) #define USE_HAL_NAND_REGISTER_CALLBACKS 0U /* NAND register callback disabled */ +#endif +#if !defined(USE_HAL_NOR_REGISTER_CALLBACKS) #define USE_HAL_NOR_REGISTER_CALLBACKS 0U /* NOR register callback disabled */ +#endif +#if !defined(USE_HAL_PCCARD_REGISTER_CALLBACKS) #define USE_HAL_PCCARD_REGISTER_CALLBACKS 0U /* PCCARD register callback disabled */ +#endif +#if !defined(USE_HAL_PCD_REGISTER_CALLBACKS) #define USE_HAL_PCD_REGISTER_CALLBACKS 0U /* PCD register callback disabled */ +#endif +#if !defined(USE_HAL_QSPI_REGISTER_CALLBACKS) #define USE_HAL_QSPI_REGISTER_CALLBACKS 0U /* QSPI register callback disabled */ +#endif +#if !defined(USE_HAL_RNG_REGISTER_CALLBACKS) #define USE_HAL_RNG_REGISTER_CALLBACKS 0U /* RNG register callback disabled */ +#endif +#if !defined(USE_HAL_RTC_REGISTER_CALLBACKS) #define USE_HAL_RTC_REGISTER_CALLBACKS 0U /* RTC register callback disabled */ +#endif +#if !defined(USE_HAL_SAI_REGISTER_CALLBACKS) #define USE_HAL_SAI_REGISTER_CALLBACKS 0U /* SAI register callback disabled */ +#endif +#if !defined(USE_HAL_SD_REGISTER_CALLBACKS) #define USE_HAL_SD_REGISTER_CALLBACKS 0U /* SD register callback disabled */ +#endif +#if !defined(USE_HAL_SMARTCARD_REGISTER_CALLBACKS) #define USE_HAL_SMARTCARD_REGISTER_CALLBACKS 0U /* SMARTCARD register callback disabled */ +#endif +#if !defined(USE_HAL_SDRAM_REGISTER_CALLBACKS) #define USE_HAL_SDRAM_REGISTER_CALLBACKS 0U /* SDRAM register callback disabled */ +#endif +#if !defined(USE_HAL_SRAM_REGISTER_CALLBACKS) #define USE_HAL_SRAM_REGISTER_CALLBACKS 0U /* SRAM register callback disabled */ +#endif +#if !defined(USE_HAL_SPDIFRX_REGISTER_CALLBACKS) #define USE_HAL_SPDIFRX_REGISTER_CALLBACKS 0U /* SPDIFRX register callback disabled */ +#endif +#if !defined(USE_HAL_SMBUS_REGISTER_CALLBACKS) #define USE_HAL_SMBUS_REGISTER_CALLBACKS 0U /* SMBUS register callback disabled */ +#endif +#if !defined(USE_HAL_SPI_REGISTER_CALLBACKS) #define USE_HAL_SPI_REGISTER_CALLBACKS 0U /* SPI register callback disabled */ +#endif +#if !defined(USE_HAL_TIM_REGISTER_CALLBACKS) #define USE_HAL_TIM_REGISTER_CALLBACKS 0U /* TIM register callback disabled */ +#endif +#if !defined(USE_HAL_UART_REGISTER_CALLBACKS) #define USE_HAL_UART_REGISTER_CALLBACKS 0U /* UART register callback disabled */ +#endif +#if !defined(USE_HAL_USART_REGISTER_CALLBACKS) #define USE_HAL_USART_REGISTER_CALLBACKS 0U /* USART register callback disabled */ +#endif +#if !defined(USE_HAL_WWDG_REGISTER_CALLBACKS) #define USE_HAL_WWDG_REGISTER_CALLBACKS 0U /* WWDG register callback disabled */ +#endif /* ########################## Assert Selection ############################## */ /** diff --git a/stm32/system/STM32H7xx/stm32h7xx_hal_conf_default.h b/stm32/system/STM32H7xx/stm32h7xx_hal_conf_default.h index 8a2637dd..812af84a 100644 --- a/stm32/system/STM32H7xx/stm32h7xx_hal_conf_default.h +++ b/stm32/system/STM32H7xx/stm32h7xx_hal_conf_default.h @@ -182,54 +182,150 @@ in voltage and temperature.*/ #define USE_SPI_CRC 0U /*!< use CRC in SPI */ #endif +#if !defined(USE_HAL_ADC_REGISTER_CALLBACKS) #define USE_HAL_ADC_REGISTER_CALLBACKS 0U /* ADC register callback disabled */ +#endif +#if !defined(USE_HAL_CEC_REGISTER_CALLBACKS) #define USE_HAL_CEC_REGISTER_CALLBACKS 0U /* CEC register callback disabled */ +#endif +#if !defined(USE_HAL_COMP_REGISTER_CALLBACKS) #define USE_HAL_COMP_REGISTER_CALLBACKS 0U /* COMP register callback disabled */ +#endif +#if !defined(USE_HAL_CORDIC_REGISTER_CALLBACKS) #define USE_HAL_CORDIC_REGISTER_CALLBACKS 0U /* CORDIC register callback disabled */ +#endif +#if !defined(USE_HAL_CRYP_REGISTER_CALLBACKS) #define USE_HAL_CRYP_REGISTER_CALLBACKS 0U /* CRYP register callback disabled */ +#endif +#if !defined(USE_HAL_DAC_REGISTER_CALLBACKS) #define USE_HAL_DAC_REGISTER_CALLBACKS 0U /* DAC register callback disabled */ +#endif +#if !defined(USE_HAL_DCMI_REGISTER_CALLBACKS) #define USE_HAL_DCMI_REGISTER_CALLBACKS 0U /* DCMI register callback disabled */ +#endif +#if !defined(USE_HAL_DFSDM_REGISTER_CALLBACKS) #define USE_HAL_DFSDM_REGISTER_CALLBACKS 0U /* DFSDM register callback disabled */ +#endif +#if !defined(USE_HAL_DMA2D_REGISTER_CALLBACKS) #define USE_HAL_DMA2D_REGISTER_CALLBACKS 0U /* DMA2D register callback disabled */ +#endif +#if !defined(USE_HAL_DSI_REGISTER_CALLBACKS) #define USE_HAL_DSI_REGISTER_CALLBACKS 0U /* DSI register callback disabled */ +#endif +#if !defined(USE_HAL_DTS_REGISTER_CALLBACKS) #define USE_HAL_DTS_REGISTER_CALLBACKS 0U /* DTS register callback disabled */ +#endif +#if !defined(USE_HAL_ETH_REGISTER_CALLBACKS) #define USE_HAL_ETH_REGISTER_CALLBACKS 0U /* ETH register callback disabled */ +#endif +#if !defined(USE_HAL_FDCAN_REGISTER_CALLBACKS) #define USE_HAL_FDCAN_REGISTER_CALLBACKS 0U /* FDCAN register callback disabled */ +#endif +#if !defined(USE_HAL_FMAC_REGISTER_CALLBACKS) #define USE_HAL_FMAC_REGISTER_CALLBACKS 0U /* FMAC register callback disabled */ +#endif +#if !defined(USE_HAL_NAND_REGISTER_CALLBACKS) #define USE_HAL_NAND_REGISTER_CALLBACKS 0U /* NAND register callback disabled */ +#endif +#if !defined(USE_HAL_NOR_REGISTER_CALLBACKS) #define USE_HAL_NOR_REGISTER_CALLBACKS 0U /* NOR register callback disabled */ +#endif +#if !defined(USE_HAL_SDRAM_REGISTER_CALLBACKS) #define USE_HAL_SDRAM_REGISTER_CALLBACKS 0U /* SDRAM register callback disabled */ +#endif +#if !defined(USE_HAL_SRAM_REGISTER_CALLBACKS) #define USE_HAL_SRAM_REGISTER_CALLBACKS 0U /* SRAM register callback disabled */ +#endif +#if !defined(USE_HAL_HASH_REGISTER_CALLBACKS) #define USE_HAL_HASH_REGISTER_CALLBACKS 0U /* HASH register callback disabled */ +#endif +#if !defined(USE_HAL_HCD_REGISTER_CALLBACKS) #define USE_HAL_HCD_REGISTER_CALLBACKS 0U /* HCD register callback disabled */ +#endif +#if !defined(USE_HAL_GFXMMU_REGISTER_CALLBACKS) #define USE_HAL_GFXMMU_REGISTER_CALLBACKS 0U /* GFXMMU register callback disabled */ +#endif +#if !defined(USE_HAL_HRTIM_REGISTER_CALLBACKS) #define USE_HAL_HRTIM_REGISTER_CALLBACKS 0U /* HRTIM register callback disabled */ +#endif +#if !defined(USE_HAL_I2C_REGISTER_CALLBACKS) #define USE_HAL_I2C_REGISTER_CALLBACKS 0U /* I2C register callback disabled */ +#endif +#if !defined(USE_HAL_I2S_REGISTER_CALLBACKS) #define USE_HAL_I2S_REGISTER_CALLBACKS 0U /* I2S register callback disabled */ +#endif +#if !defined(USE_HAL_IRDA_REGISTER_CALLBACKS) #define USE_HAL_IRDA_REGISTER_CALLBACKS 0U /* IRDA register callback disabled */ +#endif +#if !defined(USE_HAL_JPEG_REGISTER_CALLBACKS) #define USE_HAL_JPEG_REGISTER_CALLBACKS 0U /* JPEG register callback disabled */ +#endif +#if !defined(USE_HAL_LPTIM_REGISTER_CALLBACKS) #define USE_HAL_LPTIM_REGISTER_CALLBACKS 0U /* LPTIM register callback disabled */ +#endif +#if !defined(USE_HAL_LTDC_REGISTER_CALLBACKS) #define USE_HAL_LTDC_REGISTER_CALLBACKS 0U /* LTDC register callback disabled */ +#endif +#if !defined(USE_HAL_MDIOS_REGISTER_CALLBACKS) #define USE_HAL_MDIOS_REGISTER_CALLBACKS 0U /* MDIO register callback disabled */ +#endif +#if !defined(USE_HAL_MMC_REGISTER_CALLBACKS) #define USE_HAL_MMC_REGISTER_CALLBACKS 0U /* MMC register callback disabled */ +#endif +#if !defined(USE_HAL_OPAMP_REGISTER_CALLBACKS) #define USE_HAL_OPAMP_REGISTER_CALLBACKS 0U /* MDIO register callback disabled */ +#endif +#if !defined(USE_HAL_OSPI_REGISTER_CALLBACKS) #define USE_HAL_OSPI_REGISTER_CALLBACKS 0U /* OSPI register callback disabled */ +#endif +#if !defined(USE_HAL_OTFDEC_REGISTER_CALLBACKS) #define USE_HAL_OTFDEC_REGISTER_CALLBACKS 0U /* OTFDEC register callback disabled */ +#endif +#if !defined(USE_HAL_PCD_REGISTER_CALLBACKS) #define USE_HAL_PCD_REGISTER_CALLBACKS 0U /* PCD register callback disabled */ +#endif +#if !defined(USE_HAL_QSPI_REGISTER_CALLBACKS) #define USE_HAL_QSPI_REGISTER_CALLBACKS 0U /* QSPI register callback disabled */ +#endif +#if !defined(USE_HAL_RNG_REGISTER_CALLBACKS) #define USE_HAL_RNG_REGISTER_CALLBACKS 0U /* RNG register callback disabled */ +#endif +#if !defined(USE_HAL_RTC_REGISTER_CALLBACKS) #define USE_HAL_RTC_REGISTER_CALLBACKS 0U /* RTC register callback disabled */ +#endif +#if !defined(USE_HAL_SAI_REGISTER_CALLBACKS) #define USE_HAL_SAI_REGISTER_CALLBACKS 0U /* SAI register callback disabled */ +#endif +#if !defined(USE_HAL_SD_REGISTER_CALLBACKS) #define USE_HAL_SD_REGISTER_CALLBACKS 0U /* SD register callback disabled */ +#endif +#if !defined(USE_HAL_SMARTCARD_REGISTER_CALLBACKS) #define USE_HAL_SMARTCARD_REGISTER_CALLBACKS 0U /* SMARTCARD register callback disabled */ +#endif +#if !defined(USE_HAL_SPDIFRX_REGISTER_CALLBACKS) #define USE_HAL_SPDIFRX_REGISTER_CALLBACKS 0U /* SPDIFRX register callback disabled */ +#endif +#if !defined(USE_HAL_SMBUS_REGISTER_CALLBACKS) #define USE_HAL_SMBUS_REGISTER_CALLBACKS 0U /* SMBUS register callback disabled */ +#endif +#if !defined(USE_HAL_SPI_REGISTER_CALLBACKS) #define USE_HAL_SPI_REGISTER_CALLBACKS 0U /* SPI register callback disabled */ +#endif +#if !defined(USE_HAL_SWPMI_REGISTER_CALLBACKS) #define USE_HAL_SWPMI_REGISTER_CALLBACKS 0U /* SWPMI register callback disabled */ +#endif +#if !defined(USE_HAL_TIM_REGISTER_CALLBACKS) #define USE_HAL_TIM_REGISTER_CALLBACKS 0U /* TIM register callback disabled */ +#endif +#if !defined(USE_HAL_UART_REGISTER_CALLBACKS) #define USE_HAL_UART_REGISTER_CALLBACKS 0U /* UART register callback disabled */ +#endif +#if !defined(USE_HAL_USART_REGISTER_CALLBACKS) #define USE_HAL_USART_REGISTER_CALLBACKS 0U /* USART register callback disabled */ +#endif +#if !defined(USE_HAL_WWDG_REGISTER_CALLBACKS) #define USE_HAL_WWDG_REGISTER_CALLBACKS 0U /* WWDG register callback disabled */ +#endif /* ########################### Ethernet Configuration ######################### */ #define ETH_TX_DESC_CNT 4 /* number of Ethernet Tx DMA descriptors */ diff --git a/stm32/system/STM32H7xx/system_stm32h7xx.c b/stm32/system/STM32H7xx/system_stm32h7xx.c index f64f68d8..950ff938 100644 --- a/stm32/system/STM32H7xx/system_stm32h7xx.c +++ b/stm32/system/STM32H7xx/system_stm32h7xx.c @@ -64,6 +64,8 @@ */ /************************* Miscellaneous Configuration ************************/ +/*!< Uncomment the following line if you need to use initialized data in D2 domain SRAM (AHB SRAM) */ +/* #define DATA_IN_D2_SRAM */ /* Note: Following vector table addresses must be defined in line with linker configuration. */ @@ -81,6 +83,7 @@ This value must be a multiple of 0x300. */ #endif +#ifndef VECT_TAB_BASE_ADDRESS #if defined(DUAL_CORE) && defined(CORE_CM4) #if defined(VECT_TAB_SRAM) #define VECT_TAB_BASE_ADDRESS D2_AXISRAM_BASE /*!< Vector Table base address field. @@ -99,7 +102,7 @@ This value must be a multiple of 0x300. */ #endif /* VECT_TAB_SRAM */ #endif /* DUAL_CORE && CORE_CM4 */ - +#endif /* !VECT_TAB_BASE_ADDRESS */ /******************************************************************************/ diff --git a/stm32/variants/KLST_SHEEP/PeripheralPins_KLST_SHEEP.c b/stm32/variants/KLST_SHEEP/PeripheralPins_KLST_SHEEP.c index 411a1ec1..eede6b41 100644 --- a/stm32/variants/KLST_SHEEP/PeripheralPins_KLST_SHEEP.c +++ b/stm32/variants/KLST_SHEEP/PeripheralPins_KLST_SHEEP.c @@ -15,7 +15,7 @@ * STM32H743V(G-I)Hx.xml, STM32H743VGTx.xml * STM32H743VITx.xml, STM32H750VBTx.xml * STM32H753VIHx.xml, STM32H753VITx.xml - * CubeMX DB release 6.0.80 + * CubeMX DB release 6.0.90 */ #if defined(KLST_BOARD_KLST_SHEEP) #include "Arduino.h" @@ -108,6 +108,8 @@ WEAK const PinMap PinMap_I2C_SCL[] = { }; #endif +//*** No I3C *** + //*** TIM *** #ifdef HAL_TIM_MODULE_ENABLED @@ -505,35 +507,113 @@ WEAK const PinMap PinMap_USB_OTG_HS[] = { //*** SD *** #ifdef HAL_SD_MODULE_ENABLED -WEAK const PinMap PinMap_SD[] = { - {PA_0, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF9_SDIO2)}, // SDMMC2_CMD - {PB_3, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF9_SDIO2)}, // SDMMC2_D2 - {PB_4, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF9_SDIO2)}, // SDMMC2_D3 - {PB_8, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF7_SDIO1)}, // SDMMC1_CKIN - {PB_8_ALT1, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D4 - {PB_8_ALT2, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF10_SDIO2)}, // SDMMC2_D4 - {PB_9, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF7_SDIO1)}, // SDMMC1_CDIR - {PB_9_ALT1, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D5 - {PB_9_ALT2, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF10_SDIO2)}, // SDMMC2_D5 - {PB_14, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF9_SDIO2)}, // SDMMC2_D0 - {PB_15, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF9_SDIO2)}, // SDMMC2_D1 - {PC_1, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF9_SDIO2)}, // SDMMC2_CK - {PC_6, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF8_SDIO1)}, // SDMMC1_D0DIR - {PC_6_ALT1, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D6 - {PC_6_ALT2, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF10_SDIO2)}, // SDMMC2_D6 - {PC_7, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF8_SDIO1)}, // SDMMC1_D123DIR - {PC_7_ALT1, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D7 - {PC_7_ALT2, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF10_SDIO2)}, // SDMMC2_D7 - {PC_8, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D0 - {PC_9, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D1 - {PC_10, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D2 - {PC_11, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D3 - {PC_12, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF12_SDIO1)}, // SDMMC1_CK - {PD_2, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF12_SDIO1)}, // SDMMC1_CMD - {PD_6, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF11_SDIO2)}, // SDMMC2_CK - {PD_7, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF11_SDIO2)}, // SDMMC2_CMD +WEAK const PinMap PinMap_SD_CMD[] = { + {PA_0, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF9_SDIO2)}, // SDMMC2_CMD + {PD_2, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF12_SDIO1)}, // SDMMC1_CMD + {PD_7, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF11_SDIO2)}, // SDMMC2_CMD + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_CK[] = { + {PC_1, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF9_SDIO2)}, // SDMMC2_CK + {PC_12, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF12_SDIO1)}, // SDMMC1_CK + {PD_6, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF11_SDIO2)}, // SDMMC2_CK + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA0[] = { + {PB_14, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF9_SDIO2)}, // SDMMC2_D0 + {PC_8, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D0 + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA1[] = { + {PB_15, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF9_SDIO2)}, // SDMMC2_D1 + {PC_9, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D1 + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA2[] = { + {PB_3, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF9_SDIO2)}, // SDMMC2_D2 + {PC_10, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D2 + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA3[] = { + {PB_4, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF9_SDIO2)}, // SDMMC2_D3 + {PC_11, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D3 + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA4[] = { + {PB_8, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D4 + {PB_8_ALT1, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF10_SDIO2)}, // SDMMC2_D4 {NC, NP, 0} }; #endif +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA5[] = { + {PB_9, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D5 + {PB_9_ALT1, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF10_SDIO2)}, // SDMMC2_D5 + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA6[] = { + {PC_6, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D6 + {PC_6_ALT1, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF10_SDIO2)}, // SDMMC2_D6 + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA7[] = { + {PC_7, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO1)}, // SDMMC1_D7 + {PC_7_ALT1, SDMMC2, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF10_SDIO2)}, // SDMMC2_D7 + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_CKIN[] = { + {PB_8, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF7_SDIO1)}, // SDMMC1_CKIN + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_CDIR[] = { + {PB_9, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF7_SDIO1)}, // SDMMC1_CDIR + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_D0DIR[] = { + {PC_6, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF8_SDIO1)}, // SDMMC1_D0DIR + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_D123DIR[] = { + {PC_7, SDMMC1, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF8_SDIO1)}, // SDMMC1_D123DIR + {NC, NP, 0} +}; +#endif + #endif /* KLST_BOARD_KLST_SHEEP */ diff --git a/stm32/variants/KLST_SHEEP/PinNamesVar.h b/stm32/variants/KLST_SHEEP/PinNamesVar.h index 44e37cee..be78aa1a 100644 --- a/stm32/variants/KLST_SHEEP/PinNamesVar.h +++ b/stm32/variants/KLST_SHEEP/PinNamesVar.h @@ -37,9 +37,7 @@ PB_6_ALT1 = PB_6 | ALT1, PB_6_ALT2 = PB_6 | ALT2, PB_7_ALT1 = PB_7 | ALT1, PB_8_ALT1 = PB_8 | ALT1, -PB_8_ALT2 = PB_8 | ALT2, PB_9_ALT1 = PB_9 | ALT1, -PB_9_ALT2 = PB_9 | ALT2, PB_14_ALT1 = PB_14 | ALT1, PB_14_ALT2 = PB_14 | ALT2, PB_15_ALT1 = PB_15 | ALT1, @@ -51,9 +49,7 @@ PC_1_ALT2 = PC_1 | ALT2, PC_4_ALT1 = PC_4 | ALT1, PC_5_ALT1 = PC_5 | ALT1, PC_6_ALT1 = PC_6 | ALT1, -PC_6_ALT2 = PC_6 | ALT2, PC_7_ALT1 = PC_7 | ALT1, -PC_7_ALT2 = PC_7 | ALT2, PC_8_ALT1 = PC_8 | ALT1, PC_9_ALT1 = PC_9 | ALT1, PC_10_ALT1 = PC_10 | ALT1, diff --git a/stm32/variants/KLST_SHEEP/ldscript.ld b/stm32/variants/KLST_SHEEP/ldscript.ld index 7b81c1cf..78f42966 100644 --- a/stm32/variants/KLST_SHEEP/ldscript.ld +++ b/stm32/variants/KLST_SHEEP/ldscript.ld @@ -6,7 +6,7 @@ ** Author : STM32CubeIDE ** ** Abstract : Linker script for STM32H7 series -** 2048Kbytes FLASH and 1056Kbytes RAM +** 2048Kbytes FLASH and 192Kbytes RAM ** ** Set heap size, stack size and stack location according ** to application requirements. @@ -21,12 +21,13 @@ ***************************************************************************** ** @attention ** -** Copyright (c) 2021 STMicroelectronics. +** Copyright (c) 2019 STMicroelectronics. ** All rights reserved. ** -** This software is licensed under terms that can be found in the LICENSE file -** in the root directory of this software component. -** If no LICENSE file comes with this software, it is provided AS-IS. +** This software component is licensed by ST under BSD 3-Clause license, +** the "License"; You may not use this file except in compliance with the +** License. You may obtain a copy of the License at: +** opensource.org/licenses/BSD-3-Clause ** **************************************************************************** */ @@ -35,8 +36,8 @@ ENTRY(Reset_Handler) /* Highest address of the user mode stack */ -_estack = ORIGIN(RAM_D1) + LENGTH(RAM_D1); /* end of RAM */ -/* Generate a link error if heap and stack don't fit into RAM */ +_estack = ORIGIN(RAM_D1) + LENGTH(RAM_D1); /* end of RAM_D1 */ +/* Generate a link error if heap and stack don't fit into RAM_D1 */ _Min_Heap_Size = 0x200 ; /* required amount of heap */ _Min_Stack_Size = 0x400 ; /* required amount of stack */ @@ -44,7 +45,7 @@ _Min_Stack_Size = 0x400 ; /* required amount of stack */ MEMORY { FLASH (rx) : ORIGIN = 0x08000000, LENGTH = 2048K - DTCMRAM (xrw) : ORIGIN = 0x20000000, LENGTH = 128K + RAM (xrw) : ORIGIN = 0x20000000, LENGTH = 128K RAM_D1 (xrw) : ORIGIN = 0x24000000, LENGTH = 512K RAM_D2 (xrw) : ORIGIN = 0x30000000, LENGTH = 288K RAM_D3 (xrw) : ORIGIN = 0x38000000, LENGTH = 64K @@ -101,7 +102,6 @@ SECTIONS KEEP (*(.preinit_array*)) PROVIDE_HIDDEN (__preinit_array_end = .); } >FLASH - .init_array : { PROVIDE_HIDDEN (__init_array_start = .); @@ -109,7 +109,6 @@ SECTIONS KEEP (*(.init_array*)) PROVIDE_HIDDEN (__init_array_end = .); } >FLASH - .fini_array : { PROVIDE_HIDDEN (__fini_array_start = .); @@ -121,7 +120,7 @@ SECTIONS /* used by the startup to initialize data */ _sidata = LOADADDR(.data); - /* Initialized data sections goes into RAM, load LMA copy after code */ + /* Initialized data sections goes into RAM_D1, load LMA copy after code */ .data : { . = ALIGN(4); @@ -167,9 +166,9 @@ SECTIONS . = ALIGN(4); _ebss = .; /* define a global symbol at bss end */ __bss_end__ = _ebss; - } >RAM_D1 AT> FLASH + } >RAM_D1 - /* User_heap_stack section, used to check that there is enough RAM left */ + /* User_heap_stack section, used to check that there is enough RAM_D1 left */ ._user_heap_stack : { . = ALIGN(8); @@ -178,7 +177,7 @@ SECTIONS . = . + _Min_Heap_Size; . = . + _Min_Stack_Size; . = ALIGN(8); - } >RAM_D1 AT> FLASH + } >RAM_D1 /* Remove information from the standard libraries */ /DISCARD/ : diff --git a/stm32/variants/KLST_TINY/PeripheralPins_KLST_TINY.c b/stm32/variants/KLST_TINY/PeripheralPins_KLST_TINY.c index 033c7b0e..820e64df 100644 --- a/stm32/variants/KLST_TINY/PeripheralPins_KLST_TINY.c +++ b/stm32/variants/KLST_TINY/PeripheralPins_KLST_TINY.c @@ -12,7 +12,7 @@ */ /* * Automatically generated from STM32F446R(C-E)Tx.xml - * CubeMX DB release 6.0.80 + * CubeMX DB release 6.0.90 */ #if defined(KLST_BOARD_KLST_TINY) #include "Arduino.h" @@ -111,6 +111,8 @@ WEAK const PinMap PinMap_I2C_SCL[] = { }; #endif +//*** No I3C *** + //*** TIM *** #ifdef HAL_TIM_MODULE_ENABLED @@ -391,22 +393,76 @@ WEAK const PinMap PinMap_USB_OTG_HS[] = { //*** SD *** #ifdef HAL_SD_MODULE_ENABLED -WEAK const PinMap PinMap_SD[] = { - {PB_0, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D1 - {PB_1, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D2 +WEAK const PinMap PinMap_SD_CMD[] = { + {PD_2, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF12_SDIO)}, // SDIO_CMD + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_CK[] = { {PB_2, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF12_SDIO)}, // SDIO_CK - {PB_8, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D4 - {PB_9, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D5 - {PC_6, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D6 - {PC_7, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D7 - {PC_8, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D0 - {PC_9, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D1 + {PC_12, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF12_SDIO)}, // SDIO_CK + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA0[] = { + {PC_8, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D0 + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA1[] = { + {PB_0, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D1 + {PC_9, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D1 + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA2[] = { + {PB_1, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D2 {PC_10, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D2 + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA3[] = { {PC_11, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D3 - {PC_12, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF12_SDIO)}, // SDIO_CK - {PD_2, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_NOPULL, GPIO_AF12_SDIO)}, // SDIO_CMD {NC, NP, 0} }; #endif +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA4[] = { + {PB_8, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D4 + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA5[] = { + {PB_9, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D5 + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA6[] = { + {PC_6, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D6 + {NC, NP, 0} +}; +#endif + +#ifdef HAL_SD_MODULE_ENABLED +WEAK const PinMap PinMap_SD_DATA7[] = { + {PC_7, SDIO, STM_PIN_DATA(STM_MODE_AF_PP, GPIO_PULLUP, GPIO_AF12_SDIO)}, // SDIO_D7 + {NC, NP, 0} +}; +#endif + #endif /* KLST_BOARD_KLST_TINY */ diff --git a/stm32/variants/KLST_TINY/ldscript.ld b/stm32/variants/KLST_TINY/ldscript.ld index 2cf4fa00..11e22051 100644 --- a/stm32/variants/KLST_TINY/ldscript.ld +++ b/stm32/variants/KLST_TINY/ldscript.ld @@ -2,8 +2,8 @@ ****************************************************************************** * @file LinkerScript.ld * @author Auto-generated by STM32CubeIDE - * @brief Linker script for STM32F446RETx Device from STM32F4 series - * 512Kbytes FLASH + * @brief Linker script for STM32F446RCTx Device from STM32F4 series + * 256Kbytes FLASH * 128Kbytes RAM * * Set heap size, stack size and stack location according @@ -30,14 +30,14 @@ ENTRY(Reset_Handler) /* Highest address of the user mode stack */ _estack = ORIGIN(RAM) + LENGTH(RAM); /* end of "RAM" Ram type memory */ -_Min_Heap_Size = 0x200 ; /* required amount of heap */ -_Min_Stack_Size = 0x400 ; /* required amount of stack */ +_Min_Heap_Size = 0x200; /* required amount of heap */ +_Min_Stack_Size = 0x400; /* required amount of stack */ /* Memories definition */ MEMORY { - RAM (xrw) : ORIGIN = 0x20000000, LENGTH = 128K - FLASH (rx) : ORIGIN = 0x8000000, LENGTH = 512K + RAM (xrw) : ORIGIN = 0x20000000, LENGTH = LD_MAX_DATA_SIZE + FLASH (rx) : ORIGIN = 0x8000000 + LD_FLASH_OFFSET, LENGTH = LD_MAX_SIZE - LD_FLASH_OFFSET } /* Sections */
+
+
+

Release Notes for

+

STM32F4xx CMSIS

+

Copyright © 2017 STMicroelectronics
+

+
+
+

Purpose

+

This driver provides the CMSIS device for the stm32f4xx products.

+
+
+

Update History

+
+ +
+

Main Changes

+
    +
  • Added new atomic register access macros in stm32f4xx.h file.
  • +
  • Update FLASH_SCALE2_LATENCY4_FREQ value to 120MHz instead of 12MHz.
  • +
  • Update the GCC startup file to be aligned to IAR/Keil IDE.
  • +
  • STM32F410/412/413/423: +
      +
    • Fix wrong defined value for wake-up pin 3 (PWR_CSR_EWUP3).
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • All source files: update disclaimer to add reference to the new license agreement.
  • +
  • Correct ETH bits definitions to be in line with naming used in the STM32F4 reference manual documents.
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Add missing definition FLASH_CR_ERRIE to the CMSIS header file.
  • +
  • Remove unsupported “GPIOF_BASE†and “GPIOG_BASE†defines from STM32F412Vx device.
  • +
  • Add new atomic register access macros in stm32f4xx.h file.
  • +
  • Add LSI maximum startup time datasheet value: LSI_STARTUP_TIME.
  • +
  • Fix a typo in CMSIS STM32F4xx version macro (__STM32F4xx_CMSIS_VERSION).
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • system_stm32f4xx.c: +
      +
    • Protect Vector table modification following SRAM or FLASH preprocessor directive by a generic preprocessor directive : USER_VECT_TAB_ADDRESS
    • +
    • Update SystemInit_ExtMemCtl() API to initialize the tmpreg variable before each time out loop condition.
    • +
  • +
  • Add License.md and Readme.md files required for GitHub publication
  • +
  • Improve GCC startup files robustness.
  • +
  • Fix wrong value for GPIO_MODER_MODE8_Msk and GPIO_MODER_MODE2_Pos.
  • +
  • Update max number of host channels in FS for STM32F446: +
      +
    • Update USB_OTG_FS_HOST_MAX_CHANNEL_NBR value from 8 to 12.
    • +
  • +
  • Add SMBDEN and SMBHEN bit definition for STM32F410Tx device.
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • All header files +
      +
    • Update to use new BSD License format
    • +
  • +
  • MDK-ARM startup files +
      +
    • Update to fix invalid config wizard annotations
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • stm32f446xx.h file +
      +
    • Update to support HW flow control on UART4 and UART5 instances
    • +
  • +
  • stm32f412xx.h, stm32f413xx.h and stm32f423xx.h files +
      +
    • Remove unused IS_USB_ALL_INSTANCE() assert macro
    • +
  • +
  • All header files +
      +
    • Remove unused IS_TIM_SYNCHRO_INSTANCE() assert macro
    • +
  • +
  • system_stm32f4xx.c file +
      +
    • Update SystemInit() API to don’t reset RCC registers to its reset values
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • CRYP: +
      +
    • Update CMSIS devices with correct CRYP data input register name: DIN instead of DR
    • +
    • Add Bits definition for CRYP CR ALGOMODE AES GCM/CCM
    • +
  • +
  • HASH: +
      +
    • Update HASH_DIGEST_TypeDef structure: resize the HR register
    • +
    • Remove MDMAT Bits definition
    • +
  • +
  • TIM: +
      +
    • Add requires TIM assert macros:
    • +
    • IS_TIM_SYNCHRO_INSTANCE()
    • +
    • IS_TIM_CLOCKSOURCE_TIX_INSTANCE()
    • +
    • IS_TIM_CLOCKSOURCE_ITRX_INSTANCE()
    • +
  • +
  • RCC +
      +
    • Add RCC_CSR_BORRSTF bits definition
    • +
  • +
  • GPIO +
      +
    • Fix GPIO BRR bits definition
    • +
    • Adjust the GPIO present on STM32F412 devices
    • +
  • +
  • SAI +
      +
    • Fix frame length in SAI_xFRCR_FSALL & SAI_xFRCR_FRL bits description
    • +
  • +
  • USB: +
      +
    • Add missing Bits Definitions in USB_OTG_DOEPMSK register
    • +
    • USB_OTG_DOEPMSK_AHBERRM
    • +
    • USB_OTG_DOEPMSK_OTEPSPRM
    • +
    • USB_OTG_DOEPMSK_BERRM
    • +
    • USB_OTG_DOEPMSK_NAKM
    • +
    • USB_OTG_DOEPMSK_NYETM
    • +
    • Add missing Bits Definitions in USB_OTG_DIEPINT register
    • +
    • USB_OTG_DIEPINT_INEPNM
    • +
    • USB_OTG_DIEPINT_AHBERR
    • +
    • USB_OTG_DOEPINT_OUTPKTERR
    • +
    • USB_OTG_DOEPINT_NAK
    • +
    • USB_OTG_DOEPINT_STPKTRX
    • +
    • Add missing Bits Definitions in USB_OTG_DCFG register
    • +
    • USB_OTG_DCFG_XCVRDLY
    • +
    • USB_OTG_DCFG_ERRATIM
    • +
    • Update USB OTG max number of endpoints (6 FS and 9 HS instead of 5 and 8)
    • +
  • +
  • I2C/FMPI2C +
      +
    • Align Bit naming for FMPI2C_CR1 register: FMPI2C_CR1_DFN–> FMPI2C_CR1_DNF
    • +
    • Add IS_SMBUS_ALL_INSTANCE() define
    • +
  • +
  • DFSDM +
      +
    • Align Bit naming for DFSDM_FLTICR register: DFSDM_FLTICR_CLRSCSDF–> DFSDM_FLTICR_CLRSCDF
    • +
  • +
  • PWR +
      +
    • Remove PWR_CSR_WUPP define: feature not available on STM32F469xx/479xx devices
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Remove Date and Version from all header files
  • +
  • USB_OTG register clean up: remove duplicated bits definitions
  • +
  • stm32f401xc.h, stm32f401xe.h, stm32f411xe.h files +
      +
    • Remove BKPSRAM_BASE define: feature not available
    • +
  • +
  • stm32f405xx.h, stm32f407xx.h files +
      +
    • Rename HASH_RNG_IRQn to RNG_IRQn: HASH instance not available
    • +
  • +
  • stm32f410xx.h, stm32f412xx.h, stm32f413xx.h, stm32f423xx.h files +
      +
    • Add missing wake-up pins defines
    • +
  • +
  • stm32f412cx.h files +
      +
    • Add support of USART3 instance
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • General updates in header files to support LL drivers +
      +
    • Align Bit naming for RCC_CSR register (ex: RCC_CSR_PADRSTF –> RCC_CSR_PINRSTF)
    • +
    • Add new defines for RCC features support:
    • +
    • RCC PLLI2S and RCC PLLSAI support
    • +
    • RCC PLLR I2S clock source and RCC PLLR system clock support
    • +
    • RCC SAI1A PLL source and RCC SAI1B PLL source support
    • +
    • RCC AHB2 support
    • +
    • Add RCC_DCKCFGR_PLLI2SDIVQ_X and RCC_DCKCFGR_PLLSAIDIVQ_X bits definition
    • +
    • Add new defines for RCC_PLLI2SCFGR_RST_VALUE, RCC_PLLSAICFGR_RST_VALUE and RCC_PLLCFGR_RST_VALUE
    • +
    • Add new defines for RTC features support:
    • +
    • RTC Tamper 2 support
    • +
    • RTC AF2 mapping support
    • +
    • Align Bit naming for RTC_CR and RTC_TAFCR registers (ex: RTC_CR_BCK –> RTC_CR_BKP)
    • +
    • Add new define to manage RTC backup register number: RTC_BKP_NUMBER
    • +
    • Rename IS_UART_INSTANCE() macro to IS_UART_HALFDUPLEX_INSTANCE()
    • +
    • Add new defines to check LIN instance: IS_UART_LIN_INSTANCE
    • +
    • Remove USART6 instance from STM32F410Tx header file
    • +
    • Rename IS_I2S_ALL_INSTANCE_EXT() macro to IS_I2S_EXT_ALL_INSTANCEE()
    • +
    • Add IS_I2S_APB1_INSTANCE() macro to check if I2S instance mapping: API1 or APB2
    • +
    • Remove SPI_I2S_SUPPORT define for SPI I2S features support: I2S feature is available on all STM32F4xx devices
    • +
    • Add SPI_I2S_FULLDUPLEX_SUPPORT define for STM32F413xx/423xx devices
    • +
    • Align SPI_I2SCFGR bit naming: SPI_I2SCFGR_ASTRTEN bit is missing for STM32F412xx devices
    • +
    • Add new I2S_APB1_APB2_FEATURE define for STM32F4xx devices where I2S IP’s are splited between RCC APB1 and APB2 interfaces
    • +
    • Add new FLASH_SR_RDERR define in FLASH_SR register
    • +
    • Add FLASH_OTP_BASE and FLASH_OTP_END defnes to manage FLASH OPT area
    • +
    • Add bit definitions for ETH_MACDBGR register
    • +
    • Add new defines ADC1_COMMON_BASE and ADC123_COMMON_BASE to replace ADC_BASE define
    • +
    • Add new defines ADC1_COMMON and ADC123_COMMON to replace ADC define
    • +
    • Add new ADC macros: IS_ADC_COMMON_INSTANCE() and IS_ADC_MULTIMODE_MASTER_INSTANCE()
    • +
    • Add new defines for ADC multi mode features support
    • +
    • Add new ADC aliases ADC_CDR_RDATA_MST and ADC_CDR_RDATA_SLV for compatibilities with all STM32 Families
    • +
    • Update TIM CNT and ARR register mask on 32-bits
    • +
    • Add new TIM_OR_TI1_RMP define in TIM_OR register
    • +
    • Add new TIM macros to check TIM feature instance support:
    • +
    • IS_TIM_COUNTER_MODE_SELECT_INSTANCE()
    • +
    • IS_TIM_CLOCK_DIVISION_INSTANCE()
    • +
    • IS_TIM_COMMUTATION_EVENT_INSTANCE()
    • +
    • IS_TIM_OCXREF_CLEAR_INSTANCE()
    • +
    • IS_TIM_CLOCKSOURCE_ETRMODE1_INSTANCE()
    • +
    • IS_TIM_CLOCKSOURCE_ETRMODE2_INSTANCE()
    • +
    • IS_TIM_REPETITION_COUNTER_INSTANCE()
    • +
    • IS_TIM_ENCODER_INTERFACE_INSTANCE()
    • +
    • IS_TIM_HALL_SENSOR_INTERFACE_INSTANCE()
    • +
    • IS_TIM_BREAK_INSTANCE()
    • +
  • +
  • CAN_IER register clean up: remove duplicated bit definitions
  • +
  • USB_OTG register: fix the wrong defined values for USB_OTG_GAHBCFG bits
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Add support of STM32F413xx and STM32F423xx devices +
      +
    • Add “stm32f413xx.h†and “stm32f423xx.h†files
    • +
    • Add startup files “startup_stm32f413xx.s†and “startup_stm32f423xx.s†for EWARM, MDK-ARM and SW4STM32 toolchains
    • +
    • Add Linker files “stm32f413xx_flash.icfâ€, “stm32f413xx_sram.icfâ€, “stm32f423xx_flash.icf†and “stm32f423xx_sram.icf†used within EWARM Workspaces
    • +
  • +
  • All header files +
      +
    • Use _Pos and _Mask macro for all Bit Definitions
    • +
    • Update LPTIM_OR Bit Definition
    • +
    • Update the defined frequencies by scale for USB exported constants
    • +
    • Add UID_BASE, FLASHSIZE_BASE and PACKAGE_BASE defines
    • +
    • Add new define DAC_CHANNEL2_SUPPORT to manage DAC channel2 support
    • +
    • Use new DAC1 naming
    • +
    • Rename PWR_CSR_UDSWRDY define to PWR_CSR_UDRDY in PWR_CSR register
    • +
    • Align Bit naming for EXTI_IMR and EXTI_EMR registers (ex: EXTI_IMR_MR0 –> EXTI_IMR_IM0)
    • +
    • Add new EXTI_IMR_IM define in EXTI_IMR register
    • +
    • Add missing DMA registers definition
    • +
    • Add macro to check SMBUS instance support
    • +
  • +
  • stm32f412cx.h, stm32f412zx.h, stm32f412vx.h, stm32f412rx.h files +
      +
    • Add missing SYSCFG register: CFGR2
    • +
  • +
  • stm32f405xx.h, stm32f407xx.h, stm32f427xx.h, stm32f429xx.h files +
      +
    • Remove HASH_RNG_IRQn in IRQn_Type enumeration
    • +
  • +
  • stm32f405xx.h, stm32f407xx.h, stm32f415xx.h, stm32f417xx.h files +
      +
    • Remove I2C FLTR register as not supported
    • +
  • +
  • stm32f407xx.h, stm32f417xx.h, stm32f427xx.h, stm32f429xx.h, stm32f437xx.h, stm32f439xx.h, stm32f469xx.h, stm32f479xx.h files +
      +
    • Add missing Bit Definition of ETH_MACDBGR register
    • +
  • +
  • system_stm32f4xx.c file +
      +
    • Add APBPrescTable declaration
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • stm32f412rx.h, stm32f412vx.h and stm32f412zx.h files: +
      +
    • Add QSPI1_V2_1L define to manage the QSPI DMA2 limitation
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Add support of STM32F412Cx, STM32F412Rx, STM32F412Vx and STM32F412Zx devices +
      +
    • Add “stm32f412Cx.hâ€, “stm32f412Rx.hâ€, “stm32f412Vx.h†and “stm32f412Zx.h†files
    • +
    • Add startup files “startup_stm32f412cx.sâ€, “startup_stm32f412rx.sâ€, “startup_stm32f412vx.s†and “startup_stm32f412zx.s†for EWARM, MDK-ARM and SW4STM32 toolchains
    • +
    • Add Linker files “stm32f412cx_flash.icfâ€, “stm32f412cx_sram.icfâ€, “stm32f412rx_flash.icfâ€, “stm32f412rx_sram.icfâ€, “stm32f412vx_flash.icfâ€, “stm32f412vx_sram.icfâ€, “stm32f412zx_flash.icf†and “stm32f412zx_sram.icf†used within EWARM Workspaces
    • +
  • +
  • Header files for all STM32 devices +
      +
    • Remove uint32_t cast and keep only Misra Cast (U) to avoid two types cast duplication
    • +
    • Correct some bits definition to be in line with naming used in the Reference Manual
    • +
    • WWDG_CR_Tx changed to WWDG_CR_T_x
    • +
    • WWDG_CFR_Wx changed to WWDG_CFR_W_x
    • +
    • WWDG_CFR_WDGTBx changed to WWDG_CFR_WDGTB_x
    • +
  • +
  • stm32f407xx.h, stm32f417xx.h, stm32f427xx.h, stm32f429xx.h, stm32f437xx.h, stm32f439xx.h, stm32f446xx.h, stm32f469xx.h, stm32f479xx.h files +
      +
    • Correct some bits definition to be in line with naming used in the Reference Manual
    • +
    • DCMI_RISR_x changed to DCMI_RIS_x
    • +
    • DCMI_RISR_OVF_RIS changed to DCMI_RIS_OVR_RIS
    • +
    • DCMI_IER_OVF_IE changed to DCMI_IER_OVR_IE
    • +
  • +
  • stm32f427xx.h, stm32f429xx.h, stm32f437xx.h, stm32f439xx.h, stm32f469xx.h, stm32f479xx.h, stm32f446xx.h files +
      +
    • Correct some bits definition to be in line with naming used in the Reference Manual
    • +
    • SAI_xFRCR_FSPO changed to SAI_xFRCR_FSPOL
    • +
    • Rename IS_SAI_BLOCK_PERIPH to IS_SAI_ALL_INSTANCE
    • +
  • +
  • stm32f410cx.h, stm32f410rx.h, stm32f410tx.h files and stm32f446xx.h +
      +
    • Remove FMPI2C_CR1_SWRST and FMPI2C_CR1_WUPEN Bit definition for I2C_CR1 register
    • +
  • +
  • stm32f407xx.h, stm32f417xx.h, stm32f427xx.h, stm32f437xx.h, stm32f439xx.h, stm32f469xx.h, stm32f479xx.h files +
      +
    • Add missing bits definitions for DMA2D_CR, DMA2D_FGPFCCR, DMA2D_BGPFCCR, DMA2D_OPFCCR registers
    • +
  • +
  • stm32f401xc.h, stm32f401xe.h, stm32f411xe.h files +
      +
    • Add missing RCC_DCKCFGR register in RCC_TypeDef structure
    • +
    • Add missing Bit definition for RCC_DCKCFGR register
    • +
  • +
  • system_stm32f4xx.c +
      +
    • Update SystemInit_ExtMemCtl() API to fix delay optimization problem with GCC compiler: index variable is declared as volatile
    • +
  • +
  • stm32f4xx.h +
      +
    • Rename __STM32F4xx_CMSIS_DEVICE_VERSION_xx defines to __STM32F4_CMSIS_VERSION_xx (MISRA-C 2004 rule 5.1)
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Header file for all STM32 devices +
      +
    • Rename ADC overrun flags definitions : ADC_CSR_DOVR1, ADC_CSR_DOVR2 and ADC_CSR_DOVR3 are replaced respectively by ADC_CSR_OVR1, ADC_CSR_OVR2 and ADC_CSR_OVR3 to be aligned with reference manuals
    • +
    • Add missing bits definitions for DAC : DAC_CR_DMAUDRIE1 and DAC_CR_DMAUDRIE2
    • +
    • Update CMSIS driver to be compliant with MISRA C 2004 rule 10.6
    • +
    • Remove the double definition of USB_OTG_HS_MAX_IN_ENDPOINTS and add a new one for USB_OTG_HS_MAX_OUT_ENDPOINTS
    • +
  • +
  • stm32f446xx.h, stm32f469xx.h, stm32f479xx.h files +
      +
    • Change the bit definition value of QUADSPI_CR_FTHRES
    • +
  • +
  • stm32f446xx.h, stm32f469xx.h, stm32f479xx.h, stm32f429xx.h, stm32f439xx.h files +
      +
    • Rename the LTDC_GCR_DTEN to LTDC_GCR_DEN in order to be aligned with the reference manual
    • +
    • Rename DCMI_MISR bit definitions to DCMI_MIS
    • +
    • Rename DCMI_ICR_OVF_ISC to DCMI_ICR_OVR_ISC
    • +
    • Add missing bits definitions for DCMI_ESCR, DCMI_ESUR, DCMI_CWSTRT, DCMI_CWSIZE, DCMI_DR registers
    • +
  • +
  • stm32f407xx.h, stm32f417xx.h, stm32f427xx.h, stm32f437xx.h files +
      +
    • Rename DCMI_MISR bit definitions to DCMI_MIS
    • +
    • Rename DCMI_ICR_OVF_ISC to DCMI_ICR_OVR_ISC
    • +
    • Add missing bits definitions for DCMI_ESCR, DCMI_ESUR, DCMI_CWSTRT, DCMI_CWSIZE, DCMI_DR registers
    • +
  • +
  • stm32f410cx.h, stm32f410rx.h, stm32f410tx.h files +
      +
    • Update the LPTIM SNGSTRT defined value
    • +
  • +
  • stm32f427xx.h, stm32f429xx.h, stm32f437xx.h, stm32f439xx.h, stm32f469xx.h, stm32f479xx.h files +
      +
    • Rename the DMA2D_IFSR bit definitions to DMA2D_IFCR
    • +
  • +
  • stm32f427xx.h, stm32f429xx.h, stm32f437xx.h, stm32f439xx.h, stm32f469xx.h, stm32f479xx.h, stm32f446xx.h files +
      +
    • Correct a wrong value of SAI_xCR2_CPL definition bit
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • system_stm32f4xx.c file +
      +
    • update SystemInit_ExtMemCtl() function implementation to allow the possibility of simultaneous use of DATA_IN_ExtSRAM and DATA_IN_ExtSDRAM
    • +
  • +
  • stm32f4xx.h file +
      +
    • add symbols for STM32F411xC devices
    • +
  • +
  • stm32f405xx.h, stm32f407xx.h, stm32f415xx.h, stm32f417xx.h files +
      +
    • add FSMC_BCRx_CPSIZE bits definitions
    • +
    • remove FSMC_BWTRx_CLKDIV and FSMC_BWTRx_DATLAT bits definitions
    • +
  • +
  • stm32f429xx.h, stm32f427xx.h, stm32f437xx.h files +
      +
    • add FMC_BCRx_CPSIZE bits definitions
    • +
    • remove FMC_BWTRx_CLKDIV and FMC_BWTRx_DATLAT bits definitions
    • +
  • +
  • stm32f446xx.h, stm32f469xx.h and stm32f479xx.h +
      +
    • update USB_OTG_GlobalTypeDef registers structure to remove ADP control registers
    • +
    • add USB_OTG_DOEPMSK_OTEPSPRM and USB_OTG_DOEPINT_OTEPSPR bits definitions
    • +
    • Remove ADP related bits definitions
    • +
    • add IS_PCD_ALL_INSTANCE() and IS_HCD_ALL_INSTANCE() macros
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • “stm32f469xx.hâ€, “stm32f479xx.h†+
      +
    • Update bits definition for DSI_WPCR and DSI_TCCR registers
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Add support of STM32F469xx and STM32F479xx devices +
      +
    • Add “stm32f469xx.h†and “stm32f479xx.h†files
    • +
    • Add startup files “startup_stm32f469xx.s†and “startup_stm32f479xx.s†for EWARM, MDK-ARM and SW4STM32 toolchains
    • +
    • Add Linker files “stm32f469xx_flash.icfâ€, “stm32f469xx_sram.icfâ€, “stm32f479xx_flash.icf†and “stm32f479xx_sram.icf†used within EWARM Workspaces
    • +
  • +
  • Add support of STM32F410xx devices +
      +
    • Add “stm32f410cx.hâ€, “stm32f410tx.h†and “stm32f410rx.h†files
    • +
    • Add startup files “startup_stm32f410cx.sâ€, “startup_stm32f410rx.s†and “startup_stm32f410tx.s†for EWARM, MDK-ARM and SW4STM32 toolchains
    • +
    • Add Linker files “stm32f410cx_flash.icfâ€, “stm32f410cx_sram.icfâ€, “stm32f410rx_flash.icfâ€, “stm32f410tx_sram.icfâ€, “stm32f410tx_flash.icfâ€, and “stm32f410rx_sram.icf†used within EWARM Workspaces
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • “stm32f405xx.hâ€, “stm32f407xx.hâ€, “stm32f415xx.h†and “stm32f417xx.h†+
      +
    • Update FSMC_BTRx_DATAST and FSMC_BWTRx_DATAST (where x can be 1, 2, 3 and 4) mask on 8bits instead of 4bits
    • +
  • +
  • “stm32f427xx.hâ€, “stm32f437xx.hâ€, “stm32f429xx.h†and “stm32f439xx.h†+
      +
    • Update the defined mask value for SAI_xSR_FLVL_2
    • +
  • +
  • “stm32f415xx.hâ€, “stm32f417xx.hâ€, “stm32f437xx.h†and “stm32f439xx.h†+
      +
    • HASH alignement with bits namming used in documentation
    • +
    • Rename HASH_IMR_DINIM to HASH_IMR_DINIE
    • +
    • Rename HASH_IMR_DCIM to HASH_IMR_DCIE
    • +
    • Rename HASH_STR_NBW to HASH_STR_NBW
    • +
  • +
  • system_stm32f4xx.c +
      +
    • Remove __IO on constant table declaration
    • +
    • Implement workaround to cover RCC limitation regarding peripheral enable delay
    • +
    • SystemInit_ExtMemCtl() update GPIO configuration when external SDRAM is used
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Header file for all STM32 devices +
      +
    • Update SRAM2, SRAM3 and BKPSRAM Bit-Banding base address defined values
    • +
    • Keep reference to SRAM3 only for STM32F42xx and STM32F43xx devices
    • +
    • Remove CCMDATARAM_BB_BASE: the CCM Data RAM region is not accessible via Bit-Banding
    • +
    • Update the RTC_PRER_PREDIV_S defined value to 0x00007FFF instead of 0x00001FFF
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Add support of STM32F446xx devices +
      +
    • Add “stm32f446xx.h†file
    • +
    • Add startup file “startup_stm32f446xx.s†for EWARM, MDK-ARM and TrueSTUDIO toolchains
    • +
    • Add Linker files “stm32f446xx_flash.icf†and “stm32f446xx_sram.icf†used within EWARM Workspaces
    • +
  • +
  • Header file for all STM32 devices +
      +
    • Add missing bits definition in the EXTI IMR, EMR, RTSR, FTSR, SWIER and PR registers
    • +
    • Update RCC_AHB1RSTR_OTGHRST bit definition
    • +
    • Update PWR_CR_VOS bits definition for STM32F40xx and STM32F41xx devices
    • +
    • update SAI_xCR1_MCKDIV bit definition
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • stm32f4xx.h +
      +
    • Add new constant definition STM32F4
    • +
  • +
  • system_stm32f4xx.c +
      +
    • Fix SDRAM configuration in SystemInit_ExtMemCtl(): change RowBitsNumber from 11 to 12 (for MT48LC4M32B2 available on STM324x9I_EVAL board)
    • +
  • +
  • Header file for all STM32 devices +
      +
    • Add missing bits definition for CAN, FMC and USB peripherals
    • +
    • GPIO_TypeDef: change the BSRR register definition, the two 16-bits definition BSRRH and BSRRL are merged in a single 32-bits definition BSRR
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Add support of STM32F411xExx devices +
      +
    • Add “stm32f411xe.h†file
    • +
    • Add startup file “startup_stm32f411xx.s†for EWARM, MDK-ARM and TrueSTUDIO toolchains
    • +
  • +
  • All header files +
      +
    • Add missing defines for GPIO LCKR Register
    • +
    • Add defines for memories base and end addresses: FLASH, SRAM, BKPSRAM and CCMRAM.
    • +
    • Add the following aliases for IRQ number and handler definition to ensure compatibility across the product lines of STM32F4 Series; +
        +
      • example for STM32F405xx.h

        +

        #define FMC_IRQn FSMC_IRQn #define FMC_IRQHandler FSMC_IRQHandler

      • +
      • and for STM32F427xx.h

        +

        #define FSMC_IRQn FMC_IRQn #define FSMC_IRQHandler FMC_IRQHandler

      • +
    • +
  • +
  • “stm32f401xc.h†and “stm32f401xe.hâ€: update to be in line with latest version of the Reference manual +
      +
    • Remove RNG registers structures and the corresponding bit definitions
    • +
    • Remove any occurrence to RNG (clock enable, clock reset,…)
    • +
    • Add the following bit definition for PWR CR registerAdd the following bit definition for PWR CR register +
        +
      • #define PWR_CR_ADCDC1 ((uint32_t)0x00002000)
      • +
      • #define PWR_CR_LPLVDS ((uint32_t)0x00000400)
      • +
      • #define PWR_CR_MRLVDS ((uint32_t)0x00000800)
      • +
    • +
  • +
  • “stm32f427xx.hâ€, “stm32f437xx.hâ€, “stm32f429xx.h†and “stm32f439xx.h†+
      +
    • Add a new legacy bit definition for PWR to be in line with latest version of the Reference manual +
        +
      • #define PWR_CR_LPUDS PWR_CR_LPLVDS
      • +
      • #define PWR_CR_MRUDS PWR_CR_MRLVDS
      • +
    • +
  • +
  • Update startup files for EWARM toolchain to cope with compiler enhancement of the V7.10 version
  • +
  • system_stm32f4xx.c +
      +
    • Remove dependency vs. the HAL, to allow using this file without the need to have the HAL drivers +
        +
      • Include stm32f4xx.h instead of stm32f4xx_hal.h
      • +
      • Add definition of HSE_VALUE and HSI_VALUE, if they are not yet defined in the compilation scope (these values are defined in stm32f4xx_hal_conf).
      • +
    • +
    • Use “__IO const†instead of “__Iâ€, to avoid any compilation issue when __cplusplus switch is defined
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Update based on STM32Cube specification
  • +
  • This version and later has to be used only with STM32CubeF4 based development
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Add support of STM32F401xExx devices
  • +
  • Update startup files “startup_stm32f401xx.s†for EWARM, MDK-ARM, TrueSTUDIO and Ride toolchains: Add SPI4 interrupt handler entry in the vector table
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • system_stm32f4xx.c : Update FMC SDRAM configuration (RBURST mode activation)
  • +
  • Update startup files “startup_stm32f427_437xx.s†and “startup_stm32f429_439xx.s†for TrueSTUDIO and Ride toolchains and maintain the old name of startup files for legacy purpose
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Add support of STM32F429/439xx and STM32F401xCxx devices
  • +
  • Update definition of STM32F427/437xx devices : extension of the features to include system clock up to 180MHz, dual bank Flash, reduced STOP Mode current, SAI, PCROP, SDRAM and DMA2D
  • +
  • stm32f4xx.h +
      +
    • Add the following device defines : +
        +
      • “#define STM32F40_41xxx†for all STM32405/415/407/417xx devices
      • +
      • “#define STM32F427_437xx†for all STM32F427/437xx devices
      • +
      • “#define STM32F429_439xx†for all STM32F429/439xx devices
      • +
      • “#define STM32F401xx†for all STM32F401xx devices
      • +
    • +
    • Maintain the old device define for legacy purpose
    • +
    • Update IRQ handler enumeration structure to support all STM32F4xx Family devices.
    • +
  • +
  • Add new startup files “startup_stm32f40_41xxx.sâ€,“startup_stm32f427_437xx.sâ€, “startup_stm32f429_439xx.s†and “startup_stm32f401xx.s†for all toolchains and maintain the old name for startup files for legacy purpose
  • +
  • system_stm32f4xx.c +
      +
    • Update the system configuration to support all STM32F4xx Family devices.
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Official release for STM32F427x/437x devices.
  • +
  • stm32f4xx.h +
      +
    • Update product define: replace “#define STM32F4XX†by “#define STM32F40XX†for STM32F40x/41x devices
    • +
    • Add new product define: “#define STM32F427X†for STM32F427x/437x devices.
    • +
  • +
  • Add new startup files “startup_stm32f427x.s†for all toolchains
  • +
  • rename startup files “startup_stm32f4xx.s†by “startup_stm32f40xx.s†for all toolchains
  • +
  • system_stm32f4xx.c +
      +
    • Prefetch Buffer enabled
    • +
    • Add reference to STM32F427x/437x devices and STM324x7I_EVAL board
    • +
    • SystemInit_ExtMemCtl() function +
        +
      • Add configuration of missing FSMC address and data lines
      • +
      • Change memory type to SRAM instead of PSRAM (PSRAM is available only on STM324xG-EVAL RevA) and update timing values
      • +
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • All source files: license disclaimer text update and add link to the License file on ST Internet.
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • All source files: update disclaimer to add reference to the new license agreement
  • +
  • stm32f4xx.h +
      +
    • Correct bit definition: RCC_AHB2RSTR_HSAHRST changed to RCC_AHB2RSTR_HASHRST
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • First official release for STM32F40x/41x devices
  • +
  • Add startup file for TASKING toolchain
  • +
  • system_stm32f4xx.c: driver’s comments update
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Official version (V1.0.0) Release Candidate2 for STM32F40x/41x devices
  • +
  • stm32f4xx.h +
      +
    • Add define for Cortex-M4 revision __CM4_REV
    • +
    • Correct RCC_CFGR_PPRE2_DIV16 bit (in RCC_CFGR register) value to 0x0000E000
    • +
    • Correct some bits definition to be in line with naming used in the Reference Manual (RM0090) +
        +
      • GPIO_OTYPER_IDR_x changed to GPIO_IDR_IDR_x
      • +
      • GPIO_OTYPER_ODR_x changed to GPIO_ODR_ODR_x
      • +
      • SYSCFG_PMC_MII_RMII changed to SYSCFG_PMC_MII_RMII_SEL
      • +
      • RCC_APB2RSTR_SPI1 changed to RCC_APB2RSTR_SPI1RST
      • +
      • DBGMCU_APB1_FZ_DBG_IWDEG_STOP changed to DBGMCU_APB1_FZ_DBG_IWDG_STOP
      • +
      • PWR_CR_PMODE changed to PWR_CR_VOS
      • +
      • PWR_CSR_REGRDY changed to PWR_CSR_VOSRDY
      • +
      • Add new define RCC_AHB1ENR_CCMDATARAMEN
      • +
      • Add new defines SRAM2_BASE, CCMDATARAM_BASE and BKPSRAM_BASE
      • +
    • +
    • GPIO_TypeDef structure: in the comment change AFR[2] address mapping to 0x20-0x24 instead of 0x24-0x28
    • +
  • +
  • system_stm32f4xx.c +
      +
    • SystemInit(): add code to enable the FPU
    • +
    • SetSysClock(): change PWR_CR_PMODE by PWR_CR_VOS
    • +
    • SystemInit_ExtMemCtl(): remove commented values
    • +
  • +
  • startup (for all compilers) +
      +
    • Delete code used to enable the FPU (moved to system_stm32f4xx.c file)
    • +
    • File’s header updated
    • +
  • +
+
+
+
+ +
+

Main Changes

+
    +
  • Official version (V1.0.0) Release Candidate1 for STM32F4xx devices
  • +
+
+
+
+

PXudVrQ8_Vz!58iV-M&0Gg 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�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 literal 0 HcmV?d00001 diff --git a/desktop/cores/sdl/SDL2.macos.arm/lib/libSDL2_test.a b/desktop/cores/sdl/SDL2.macos.arm/lib/libSDL2_test.a new file mode 100644 index 0000000000000000000000000000000000000000..c835c81914557b00b9eab848e94a82c1c371adc0 GIT binary patch literal 259552 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

V71oiN=~Xll?L_ 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%0D?cBR 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≫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 z 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

PXudVrQ8_Vz!58iV-M&0Gg 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�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 literal 0 HcmV?d00001 diff --git a/desktop/cores/sdl/SDL2.macos.arm/lib/libSDL2.a b/desktop/cores/sdl/SDL2.macos.arm/lib/libSDL2.a new file mode 100644 index 0000000000000000000000000000000000000000..e6bf3b81c88ea0e11b99de52efdd74dfe1262fab GIT binary patch literal 2460728 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 ?%ZQe?p-cvaF0(m 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%0D*KSTxMtP|Z&G3XOZb zF_6YZsU_SQmE)|wH`%GEW1N0nva`1?KiZXF$Jj6so&Eff$Q9{1(f#ECXVgaSiHcni z{Yn+%INJYA)`n+qal*4JQzxw1#NI{nW~MRAomusPjQryv=kn)6^Jj;#&)mVRv_QnT zPe59okjB@|i|7+?o2tEq*v(6g?w=oY_iM>a2Y29voy|RF9L8_AW{+PAd;9|Iag6Ri zGAw+B;wrW2!cq8Vf6h?mK88B8YjHF6(UPLnw7l7CiL)_`LBsevjKBA>cg_n#cDl3L z8?zP#ns?0S*U4VHyC#)R4c{Gope`%6KXP|iM+;l4M!949NCtBrflnEK*M6_GZW4TN z9$*9M@waQ7?aWgxlI%x__anIX_sQH1h8|@%6S@hy=|1S!gQ+c9fvj}e?pd)HKGVW| zLfkVpjy-#0_PoP5Hxioj`s>d|T5b-FOII9yhCQm4p$YYueB50jH;x21epBv*13P!T z&OK<7eRC+BF8QyIa<&g7*X%uI&dZZ+WO5H$Z5C;AcX0H>FXXlSoVgr@uTS`seBmda zgLg#tdwcK{Pu=OvAEiY~rfT{m)6s`3Fc9B+<*@9d$P z7rK_TzJ@fvnzWxs9+=DgSLtr@gxS05<(uE)$L>SQCqpokjM)_Igt41SK1pNmaFD&A zlqL4Ij=efsbR>;;o>HCF-(`OC%~8(kNC-?Ooi21Lc9Z7JTdbb?=zUpJ?+%O(=cG;u z4+*D-Q=O5FOESfuN}hQBgKvB@ZNwA5&c8eDp#>jTy|y5bnqGEy%IMqgUXVWZ?y(c5 z4oOR&ik$_d4>d}!y>9F|iS~vYd)Faj>beUTPaWxu+=9Klso@C@ZYjRn#yNY>WOGkg z_s&J~^Z3BSS?yyL=T2AWXlK9nK^`CDRAgYUoxHBH`)(*N|J~0y`*#zcQ%Rra*`G{( zW%gmO{nG8HivoFR?d;p^9PV@}pLbs5bp3$5+sT~e@0r)>;KzQ8t`p;&{S%PeJ;&MK zF~RBj9rLE|4rg9Clv@9Mkh#~5(FcBUL-?W2i=tf<4_^6#_?jP%vG=$0bDIAOr)38& zsMns&-z7UEp9+p`c{s)S@GbIM_r=bB$@(4hzU&9eR2W5f2a%2WPRhX;{ML5E*oXR) zmxsSPt&I0@GlC4sd=Gm{uctX5@QWKJ8oy-+CLWLA-$cLRm+kVC{CE&QZAo!nQCKPi z;p3fuwKGfdJ%3SuWQw;l)XS59-^2eVeS#CpxKFVD-8tm#F*~!~CC>gVFb=)(_4nKS zruyKfkkeF|;+XPJdzX0&wCaYNL#Zv{!1%IL>~s-qPLCP;7^ZlZ$B|{w-zpN9r!H-zj_RCQ zI(6$1XXMU6xIXZHw<(94h#$sh`xQ3wDf@R9P=>FgEYGJ*UrX7(hVp;)Zp!+0%0wn+ zn-uQRPPnC=cBDDr%x)w9wmaiztIW_>GVKZNxVkH?G81l^II5kt<{9eoQ{?mNVCDJw3bm z(rDeg)Q{8d-?Z;t&OH=xj^L-1<$UU_m+;F`>c_uHj_&_?*qOC(2zA&nXLW7Zsi0r8 zZ3c78F}f@{x6?xl2U-|qzCMDmY^U8B5geJmowlbn&Dl=G&1x;3(IP#yB_saHnUx(J zXY&0A)FHYDlfF;pfOeBd_Sl{NMqj(6NxycV#N7yXnUCqU|BtTyM{rYamw?QbdG?#E zznfZV!;HVvdfi@ltgqWGm#06JRgm`3k)M?m?GMKGcO^NOe?Kj{|3&J!je(5%YWh0a zABR>4W>St6e?w_AU)S8W+n3ow{9V#iyM94xdP_8r{rcYyi|%L5W>GV7c)`JWw8zMtot`LayL*_W@zA7Ny?8=AN$EifVd`@ys_t zXUwx);xE}fU?usSIQ~#|BtmJ;xHrmrl;+P50ib~qsN1k^o zE_t$a>ivQA{Cw5|9}lGE|6BJr7CjRfR^PGEsrWbAHhq`e=v3_7l)btQzitGpR6l{6 zFtcY$$b(1J)=W#Vq>{xJP6`Wm5Fl9n@hZ^v~W31V(otondLEN%PVtO>x%eBsntjT4e-|OlCdL2WY%s$#2i6 zI~9@8FcbgGr&Z(xoih3LJ<{ci#LFq-e^|P6`Hyi+_48Yyk$av!7-^{?J?1!}`kW{3 z&60U80IqPjd$lwv*us6mE!?A8wka5D$r&N{qg|zeV9Q0+yERX*dM)>4W#PRO*Z&~5 zqq4BZN!l|H{jZ@v5B;y9e=YjC+oMH~HZ40n^Oe%6>w=*zS3D9edTPq+JD#dtS6KM1 zXwly_tv~&old|W(!n^X(Xwkps{&L5)^|uvX^_S72|JAhNbpDU#PMsSFZF$WZy`>R* zkJdI8&fJarxv%XwR9{or8cg2OI^~x;D(cr3E(j!VSupvv9naU_UiiJ>kh14CgeN>2 zxhDNzoJ{lmwMo~c*EnfqHRvzMnVbII+{0Nlk=L^h=N`-29QkF|g2``WEy#H-Yo#+h z|ARnM{(F?I1s&OipSwG}_;V+-3m4oyV)24mPGL?4Yj4VPzw55Zr#{XMrnS5j7@7Vi zX&m@Xcg1_8@dDC#Yap$KINQH~bXM9P{II)XIq{$WcS{>*tNU{wtgLJuwG_q_vW2(qC(S2`m zRyRl=^&)<5ImnbZRCycekR zRR7s{ERsLJK{@+6UcB(o^p=sqVGHIYM}Hk9K0XW$r=LD-Pk?$VNZeoK zq^Hf}cdrvl&vAljIn*OLNpsT!)KfEv!-LcvtS3+&hwZt4h;#W{NlxaQL!!TKqs(q6 zy<((Wz|<+}#AUj#P9Z*tOAZ#Sr@pvR{oN0d(SiFq%cVLjhjhRX`$I!Y(o~P-IBERk z2l*u_JwP4C!pe==whoJuH)i0L!!IY8o)*Nt5+{&OKPYWRFe7b7a!L9~>ae3jhLdK) zO`4UE=e;z$Ipn6H^;><^rXA*>5bpC!Q;08xuz{k zw{019R$^QB4f5Mda)A&PJK2M+_Kr3xMef;YHF7i=F%3G<{sX$w0>#fqv$`Hdu+#}w=XaJMQGxdr=}d< zv95ky;d4&Nv_CbpMbG)#rBY{1St)JNcP5plSLYtg%8C3W>$|zH(Iy?sD$V`JtkTHg ztbB(yC=lWo%ul2IQisg=E_GyU+Hl&O;ik>mL?0rq%}KXy4r#FIRCmRZkdwJ{)9}?D z1?me9U)=^)f>mH#THTf1Us@H}v@&C#^GT~cxI1WC(f2cHMPINdt*A@SmR6UOR)5>H z?)2Z0R_Zg{eDN=L{NwFk>FG0!Cx6W(t(FFrR^$7m)w3q8#*GfFCLuM($gNLIlAEduTU`a_D~re_dHvLjNoHc4Z;; zKmG1#(LD4gp+67($>>j_EYvhDJw4^Wmri}n32d1~S!kScbVnooE5+}f#P4Qj$ew?O zr}ChDt+*2g`C{yc2w6_7yhf0wB=v1 z^OM_eE8HAPA`PC~v8;X>aUCjayeB-NCbBsFQ75JBQS$#kN3K@;RF=c{rbtQp(%ff> z&!dzD#qY7KoZNq)Y&@4$>RgylpFE#7DnIA#?7~SK!iy*I#zD@85sP!Ka0*8ZGySsF z2Wg{D(qk&a#^{qGpxiRpGf`EYAsX!`xOp1YrYm|v$(z2VTMEgyu2 z?m5ZW*P+dxL0VKdtv)^J{?e(`b6X~p7MpW_yd#(TWeRC=Thpzl8_~a6L~gkMsQ;Oj9`GDlm80+JIy^jV(O)##Q9J&hA;oT8{bNg#P|-N_Z{N< zDf0SQ^0>y}%)6)mr5$4y(Iy_bbK9Dq-L-SkrF;)3PXCy3`<1x)pX2tkkK2D8a5CH834A&*G-;Uns#_`V;Xp?E zR$Ja#H=LEt*z0-5UIok@Jwq8ePG9EhfuUt@+Oe1B`oheyy@@Q1JN_+{vFC|EIQ?^s z&)y6SumAaz-^kK9?fdkZhmns~9K5vUaq5u!pDmyc8D4ghdhI(ueRF8&`8S7_PK*qz z;a9r(yO)+e{K&AmnT4NT3OzR|V_50L?+$x5;QZ*{mUTgjhM-*a9R$iu@z%XQ&&YYqtsi_(P)0hWM{p` znil`haFW5j-%g-*&6W zwp%8BO}oWdE3VxlJ^sjctLgvqc8h${({3gAwp&}X&)#mGA}@RM2#*k--aJfxB(^A@ zd6;xt7qAbYr!Ht<{@@PE?e&KWT2#hApglWzb#&j8wBJrJoR-5c!dAEu!L&5y?9)ah zN7J96U44nNuQ>yiRmltDc1srX)0Ek_7~>wNJ*qeTm~>NSHST@U=fCRA-Zmnib|IfO z;sx3WwHN-f+|#f8-!#whSNQ4w5Az)B2#3GU-(u?gwDc{u&VP^iODCPn4_(r7Je0QQ zebQO!tvdP;zrpHg?)v`G3F`EE(!JWGdm43gnn`!o+-Xnxt{rF|S!)M3;J5dvtJ7>< zZS!G~&4;Eezo>jjS&qwxlcVd&hfO0sJ7T8t-wbeun-9xI&{v;6cA=XuOBd!pL77pW zG=59&&6CGDE2BKQsh>QlIgN|uH|p3lTgU!K`YoMy_-uabU>zyU-1U6+2&w<4`SHQ_ zO>~Ds?3GAnJvD`O)iCR;sjRcIPvWk0_S6Kohx61=#E3&~yu&;IdF!RbASsP|8prVdF7pd&SXeMlVt3Nd}3;Q>RbiY^D z9=>@4(33}-z*uJvZ3FEh@@O+a;)L^XWrT47VZDwp&nN8HvX*=edr3lOFNxfE9{K6q ztu?hEZNdV%#r_M=t?+8LkJ+^_Eq3=$vV3~&-S0{kZC z7Gmx%zzT2&coX<-@D}hpU={d1a2dDpIpr|Aw{yJ6SKji#6~w#PzXvV-oz{%#_wh$vFqL)7 zG4&(H?#z1PP}aWvfzi-OX3l>L>7@DiN9eo0$KH~(fU|!Kvb1-?(oGAXsi#`UkyVQN<)L&|^w6EiR_QlNU4y}F{`?G&=U)FudoMUAouW8&6cOQpjLKi`seO}`O zsV&tn%w&FJyxq?=*6insvY#t_D8hR2xcVsN;Wxq2=>y4UFG8Gr=yR4YxpG6}ym#l! z+x^6$oAGgjg;vk^#{`^6nyV2wk%%%31M{qALk3h%e5yKy|=Y{T_ zqfX4pzRb%j?~U7;rTk;^N=bCJ()c)eWFq~R{6Kp89O^3N1MceV$p>@T<1+gP4`+F4 zKAtksU;1`VU@sf{>Uz?cFexoQLf@0&rRzX4bZ?^Ududog+?w>e_}Zx>f~jfD8?XN0 zujcJe<$T?{)CKRdC+Q>3!0!1lWwzN{Z~^;|*u!@hW&6F0&7A7UcgIEd^PXw_rZ>AQ zn#sf2%EP4LyO-cE%7yAz&U>)$X5hM#HYU@WH+yeR(f(*(&rHT$&CD~;XAe${`lr_J z!BJYMT@p`wZ<^VAvy;6y)xk0ByK&DD`}f_nv+w3UyYJ?KKXi|K_oDrGoLL#5z7A4v zhp4}ksKDkDwtH*B*>_M7;ZM5iTxbrxBYWA+C zZew2!_0KHr3#=y34kIp@>yMj1{)YC>*!fV^4SzjJ>(}GzUm$H<^Bcd}8XVrDHG*yQ zskSee-ZG0hPR&8T#5%gnUP68%Pp=M8Cbci=ar!gXt+;gsZ`ulL-Nef43s zu-Bz%YWSgj>~-14`oeDZx;#%iFo$tF>-`l<-$*t{m`Xmy{u0hPtBvg-pU)>R=sTuw z&Qa?-=TfKQxAeir(g)Mr)_CH8bSoOq{*oQZ?sz$i{zxfvPLZD!_RKl8Uh|oj@u4i! z&l^wwKASqH4F83(e;<2OTiM@wpXqC+rr&4#n%B{n%l}Cp{mj%o^c`pqqD^0)JY@0K zz}U3Mi$~0S9Aq3wd3o zdd{x%Sqc6~`|T*Doo~1H;N5zkB$EpN+Xb zUHXrgT%Uiv;yRW0dr8aIH0?`tSui(bk|aev-I&Vv@^$~ zJ<8-GvnTl>oBpgl&Poks5l8I7I3JJ(3$A4SbArip zUorck{qd%`tlqq~JIR@qPJC^|tu*4Q9=9$Dq!V9h=w;M5*!a4Ha(~GS`7H~GuPx8# zw`|3|_j#xH2IB1JcaL9ufVnbfV&1Sv7z^cqZ;Rgi$feH1-x(HQzAA!VfcYwC;v>U` zIa4lt=~Lm$IIp$;IP>N7pFTWDKJ~}r3*=KT9T4LFTpo7yDm-%b05{W3K_y4tJaJ4VaK5_kh zqWk}d`unql#i0#X{sxAhuXlgk9xb$~LiylXRYuz1%cRUy^ z8rt;#viC0VQB~LC|2Z=unatz`1PDol2?huel#l?E)~ORf!7x=M0g+0R2x3CQOAr!3 z2$pMOCbo&Dih*0B#TIC|#TG2F{IB>cRoaV63nE&sVoPhRw9qRp!J_$p*E#!y$y02x zFLOSbtiASQ?e}}_wH7unN@RZ|Zd-ks#khU9sMzx$G_!}k6#pP}vXg+&s1ls;Rp6EI zMpY5Mc<90SDhZbabhG~AT}?K8kIrPRRL(q-7(;!W66Xb8phzyX!O;bI#C28IPch7c@+Omo(g2B6Uew6pNWMlDaF|HIeobZqz;P zC4L*`g_Q9Ksq0SKfYgz+fnq63>fdM+s%&p82u~eBnUA1s!zoJ#WvMgfS!G#4*~{Ki z)(}nh2jZnIzejs8`I6;rfo*&7G+*XbWyQ@S;V)RY$4C7~R4&dp>c6g7+H9^ba};g% zWwD>fx^?I)bvud{c^1v`WnNvjsCjXOukn<)6-wG(sXIvf@2s8B2aI;IgLd;>(%e2T z(mtv-({H5hP#)5D#7^3XF|ISG1-I*B_?m))tWD(FOo0mi!aj;+vh7vwoB@k}s(O|gtM=D))pP+&wTh$hQWU&fIs&Ls!0?iJiO8OE%pb zLOXYlxyp3E03G^OhxjA7iIb1EDIck9`^A2Yd!lle%`IPt@S{H_yhpk3?IQeH<-V&6 z_pKSGLs#y%k2T#lb`k!pazECE`$0KP=IhFRi@dPmdSih}xN`?!gga?7 z27^}lE}OyW;-mKIru_WPK*84liNarMLhlkFQTVx^7U&f@OT8o>6M7HP2#Lxs?GfnR z2PDc-4M3vsZ#AJIADkeg+~8EfD1NeWjiL>R+!S~r47n&&x$#~+N&-uiC-^6DpMv!Y zmMfUApie=!f|`OSpocMZpMv!YmMfUApie=!f&e32m~E@xfc%t$`XT-|+b-C`Y^Q8z z_^`;@gQmxW4}?-+i?zcsGhqi}1SnyYYY zTyhgem{k6iaadfM~t#S9O z3b)4bn-p%%E3*`C%_H!l82Pj2jZ+G@=82yP9+aQ`g42*Kc?Nfr2j;QXFJIx-Jo`iN zAZKk$J6y*x*!^?u@T2YgzuOK^ZHK?u4j<5t-hJ)dA8v=IwDWI@XhDibR>_jWf(a#- zC>Lx@SSnFeXE5BVi>w_r~E(_K?T96 zG(j4kErBd02F5%u-zSy(o1U3I#%+A2T$Vd%NtR3&q4&4SWpOk4$md~fk<94sq@TR;?pg+kdFCYNPEFjm_l1Ar z9%N{46*&y>O$@$5GwwJo*_BP))3Me*1`3~8$7SJN5)hRD3Y4PNkI znJ}iFH0*0BPD8FqAu>fPh_^5m2v0)c7_Cv#QFyh++k5bAWNmyq3z;$^)5YXVW+9bP z$lXx|_gjSvxC1HcdpGSbSLlSp-z{=x9^n4MOrP&OQ)uiq+(zKWvK6_e$caT}iTF!l zUmPyHBILP(`-gD&3ARBuH8P}+OMBAbH-DZx#}NzTayV1EMeby{GejnXnHG15wwtcc zl1$3NcV2iwbk4v98h1IIoxF{I%DQnYbi<+RgsyStCHBzee8rIOfqN_6eDpaAuSnsO zS(#fk_joJ5dUQGx8M0JW{Kz$x`0|k%+gUxWj}e+WvPV10Cb8s_NVyG=^a105t0+^} z%GD`$AAA@)`-psqB;NX%c{F9ABB!{K2D;0tNA5(=GesJ5pz&ABHb=^&wwyc*9fz)s zNs%yZ>;`vYH>eZ4D_ZSN$o&=ZK(B_nI^Qx}-{x5AYi@d9Z-!TVdK0`RC+4DWaH3Z8 zt1Gn{J91zdntmbiN;>M1yZbJDd_^JySWg^7*Nf$j%b;)4&N@l0{UALuC*QT{`@P@mb&Q7?-S4u_B#Z>>kN6prkon`Y4l0S z@7wgf)tpO9`&faWQsazTtNq}vA8{|Eb3bF_o(A26t3$SZXxCfy4~pz!;iHvX@STHIF~fJB&XgpZEa% zbufKF&b$wB|0CwBFz0ipExH?j>hzT@;7&9p$+_vL+`Ty&lcsVPnv&$&!{n-FOdn4{l7KswXqjuIi0dI_k89z*|&-?ErI1w zU|t$^yMiLGmi_rCq0Bfm>H?D=qzpY~K(ENJWnLPU3uH@QGR=f@{6PNvvTqXm3}ruB z*~cmv156@3{n^+ji+oktCm#W_*BW&Qh!;bjn81DD95ER3RR!Or+#7(5>7%wPoPKMB z+X&>y!O$}%{;QPz1|S2|DDJ_HbC6OXMc`jI&Y1#~`oqq! zrws_4BRJuVRinTVd!joz#6B%|mMI;zUf-fSY#VG9HivDaZ8QJR*v{HQ?9KKI_HuN8 z9SJ$c|Fa>@{BH_5#sA~IPV@?~?+V=$x;NBeI}~~ZS*y)qa?UB=b!0;J(bl&fgY1K? z_2ERpahG;3pX5jOoz{8}ITnWh?@@{JwcuC6BYVO4D0>4xpzuWH-=Z&*RxrKy1t%_R z{dWuDgZHsGOZl{Z8SU_^pdak+XotUrW3YQfJN!rO=(*duKhh5WZae&?cDS@R{#w6^ zcJ#~Jxxd@aeNsF3*V?%kwsTK!hu_)`-_$Pr`F8k@cDT8HmeF$}8#&~oDglj#exw=8 z?#wD#UbN!Qxzp~xbN-#n=N4Ygp88^tyquE7D@`F7h{08=&SI@t*VEH|FvwbfKlE`}Fq7?-tMe_*v#eyYlAy7VrhMc^)g~fUE3yRPj zwXAH}+~p;Alr1V)$evzeGPEFxk|v2sDCZX3QKHP=b{xs7MbYpID{zXWkjSvB&a@Uz`<06N{Xj z>pNJC39@OErL4Bwa!bjMQJ8H!n%i&wn_kk)qG;n~v_We$P3jZx%QMqGqus`LabKP{ z`r^*~Z{3&6WFnvJqm57DGAznZ4Rm}kKtP@bhqARiFNDuar^RoMX#Q5MZS%S-d49Ue~Oh= zqGA@^wcYN^W0FV2$91i>1V$!dj{f^w*d0a z_GZ@Khx++Ey$6Nl+;ZE}Jx`0gn?;E^ye;c}ogQ5*{Qlg%BSVGvmLIi@wc?)wPn{9J zpZETxVI=-Xk%q@fLj!3T*{f=5Bxz{i-G3x$h$0QqL1}o8G{_rWNe69t{2xh!n>6I< zy!xi>@^8!EvrpF=(@4w9@LBUFlJvwBKLIb|Y0~nt9vRsBp_!fwi+==9TwDD{j^s%r zoW$oM-rl4`%0bVVxb14%%gfYbv_C6zr_|dOT5nsX)|>9V)VH%dUW+WBnkI6AG+(3c zMqm7pC+`-SUj+%gLm7w68QwRrn!6<_rh2*_v-581zj^{^Hq=w~T;gM#sonrCXB)3mAzj1^yeJf-C5-A2J-r%=6mSH@UXyH_Jr?;AUn#zIZ*`ix!~LV220yV3iCa? zo&5v#w!ueBbzi2mGho?n&Rp&(y{Gw^|D7-b9=^)PCq&-I>vg-4_wkI#J}6z$d`|42 z@@1aG{^)|0o*({Z;s~*OmUs2%hP_euEbs4Em9A>OwRok+V~?vG0`1+<+5OSP!}k>3 z<9UX#M=`G`TH*Pjgu%SBXr-qa9u>N?q@%H2I-Fgk!`VqXoL!{D*|-B<{3mgA@Pqx{1U!-u~dKM$|XaUI2cWRTY99c{qWch*EF zyuT)C(=z$8!% znoV8Bx#9zw9hrgD4>H2ypONw$0;EoA1*c4F!AW!KZpt572=Tiq+hs#ujVDd1yU6o9 z$w%UMQ3mfIKY}!*-o;xF*2A%vnPwSnJ@z}K@0|-150@1U_k4%6iFt*XNt2jYVrGAM zx#MnpAyJ37EGF<&2z^fPRb226y1KqW**U5bQ zjc(@KM=o)`t-II<|LXZRC>^5vd$XG7DPbGsW0rd|HH8$S=tY+W2&dSf4D)|-+4 zUQuoGe@Qe4h2_-Trf8b=)+x>4{D0E68k}d50aC>IRdCt@w1H0Zi`=j@+uDajPNbW+ zk(`@lx_OJteQRc#)0eq4T+5VlD2~)>4sIV`i!{&04)V_QYrM(7oNdH9$7|oWX(M}<%2}S-x7*fdc=v6jQ<`QQ zF==+zMikB9HX{8aTJ0Ai*&jsFH)0smddvO*`30lDUmYJ4Sv~bM@@``o(_-LzScJ@( z-i!^s<=oBIyEuuq@~%tEI2|-LxM(Y+@fW@6AA_Cdn0AYtA2Oyrr_K-e73F*Ki$;6W z$UkR?HG|dJq3nsW8Q)}_c!0Jsg>mBDps}q>+Kg=-Xj2Yd(SDD%5Jf(sjC?Rgk&kWk z^FIXTL(V{H^PP;%TA$~TUGC$ou>JTf`sIT-IBT2@*WZ?3$~aIab<6v7<7`;!cfHl`k)JAc%l$*_yOhf%(WhKG(3ig64Sgw_ z3yYuaaQ6J+YUEQ))N5jMCynT@p@*D);GoJ=PX0k?CLPaBwKrUtVSjNzgjel@hPPg$R1GKC7n}Wtq89Oeut({?fmG&lUq|=PApF(@vLwjp9+MBwI zXw%-p|K|2)(wFwu4Si{Enr&dR)-N@qOGqVu8sAxyDAy2eKV`3#o7NrNg)c|GY7=yD z4^cLLF2)8=>v#n?jmt0jmUrR4KgM)#FyrY!e+f)W#@Cg*JVfDhci~eX!z0F+6z`N-` zYeMfCpoBkV!Z}Sqy!ua?&|6E6xmWd9o6swKW8%NUgmZ*{jH>mQo6y@tUSXEv3Q5=VEm!@Fx&q41IR%kQ~az?x`W_W zzkO2SRzJT_;a0zupyJ=^r}G4-LM%Ro84CAM>GHk8Px#3$DE^Am3U5|;G_=J1tim%D zZt+o^Q~oXdMTJKw`tK+l&56brrox9PJWb)!2<7{bpQPU!NB&pg*0}NiS-2V}b|~B$ z7oM3y^D zOoqD&)APFHWioxj*%bWh=#%p|rHvnxKCV4~Q(E`@O@*{`&284K4D<2^va8QM-gJ?1 zQ$7o4>@@hB7$c4EQu&*j(@pzUdyC=i0xHzsWP!Afx50vfZcXl|oN=k@!@Jnn+&WorQVx z?-qVT<7@xUY9slRufhbHn{t!%9fBU+R%jL~Cj*S(zsx0O^zy7&ei!vipCMc`xFkaS zHF-5nqL`9}J%NJxrKD)&C$uD6CQS^V;JZ{_Ceb6E4KGt3yi8wYpDKJ(_3SzFvO)|Q zt9kG)$$2Aly}?VxoE__AuayVCQ&Nz9Fi+D6^fC%>R6ctXjeW1IXRYv*VFx~^uJ%^K z|CBOc%ZNyxxDDQ;#s#eZ7Y@{V=dlOaj=Z&O_5j7bv_4hv`N%I4UZ1PEn~Y}v6~n$l z_<{Z(cZ&3x>ZjPB4TqN}3|<}$b1ZhOS!!4l7&2;EBk;aSs~G@12NWJE>e=A&d4g%~ zE2YJyNe+9U8Cj8TbZ+uKX1cr&8N^TaXCd(SYzPktYy#H7=kpkRK6%2w!o8zKXT8ut z?}gAYXmHL&dx*93-a>TAJW-0?pT{PQ*r3@fuZ3@8L-O&uYtgTf1aFDB?Zy~ zWBxV#K8a>J8r!AA*+n{>outFrMLL|x@Vd-J|4=f#FI!6IHGc&iA5u0BUsk|H>|tC zbsTdYdam!@1}=N~Rtr4c{McNwP{F()^TD?mZ+hl~xt{&Zw~s4+&-u&=GM4|6v3xaS z`FzImUow`@XDr{0c@txKcXPs<_<7@F=7eD~CouMx5l`m2?=WXrbHYGlPLR3a;RW-U z3;O;`&IQK!-#Xu4oC`w#N#+8p{r~zB+kd*T9yzlc;k=`N_q(=jy`R!* zbnc~WkMWLSDSVggmwIi}YC`{0s|jP@)OS6y0oXhBXYUlZgFO)Yr>5^{u#johtL?t? z*Rp>~RQo4n_1|&|eG|V!E_?*;RlKJ;z?k_!(E^VnJUsC3mB_k8?!KlEKFa%y%FPZ z=p9h|CSX|!VM|t*G@n7{ehl}@F*Zl#F6?5^6Iold#B z($&~Wx||=AE@wCCa(0p~XM~-&haRjusPa%zROpeit6*RCE_Z?N+wI%4_1LM|_l6|+ z@4a?Y)ra|qkz4EX-y7Dq^4@G$waw{@;LXE!+d#ulE`H?gL?!mQ;%99(dL^93`-CeQPU1;DU^Dku zmzk!qzq0E7(1Mhpx=(4X``56S{ncfTyYbgX9foh0eb!N1SitH>PegF8($21cI&W_$ z_sQ+;bmX?Qw|ib>l(eIl6-3WLd0(eFV5zw1uL=DkoRKVTqK`+sjb1)>k9}-Ne{E7&$EHGin{X zD`o61D=f`s?Dna#8(9`IM)r)|osHeU5}vky{Fog)_kPakJD<*&eGJ_%J#C@$dHldH zK3i>Jc28UAa!u#>@n@?o9RILqT7i>$(!f< zT5`9d`^>?)OoTfB;Y`I9!P!p5y;EE>FgI{+(`1V^{5bKG%^A@Y?49_Tf}cli4&!X+ zfPG+>e$dC~O5wgGd)>wJpV{2qXWN}qvpN5<_7ZjM1;0?Xq;$4xP{4-_jU{Fs#T#Y9 z+0G&G>S2fCjWQT=BtF6xkvfgKa11?Uoc}n~nNU~fKd=9#Wx!=VgDKky%}Y7|`S#_H zyRqi-(=1zmiTPQ1`QvW9=!($KCke2baJ9{1Td% z(tmz_`QvWfd--XW{pb?&^YrD9yKyr#FY}mLRuz<=SA)*C6o!g)*O8Bz zd$BXzi@gO;VAS`&7XLE$VyC$ii{Z{8M)C30Gw<|Xu-KzHdI$b!kJ=8W_0f8HM<{Ox z`)Qs}!<@$3tyAcJ6M1Ur_X`~OOOQ;pPb)uVyqDv>u#un3Z0}3ST@1ghlY3nk zeJ6sx!yDbkm*_i(=sVBh9{K$T<9E<^UZ(GyzIVdI9sAEB)qf(5{!=dfr!0SukGIF4 zhS_Rw=<*C?^X|^-KQZn}Bf?3)lYE%(@{u{n*dXtYN&BvpK3YaJZ!UZD=CXx+U1Piz zwD*m1YJGU$Q=O*91mWp*FIWQamBzctXoIIWiS_4mx?U`N%g+}Tc%XmGzaw~WNaAo4sFXdQgjn&OP4$qi0^C;`lFhIt`44q)@D82 zvCS=3Z7#}ab5Bryd8*ANl@*lI=E~6fUL)_3{~g?{HfM}ey<&_ux6f|MHZ6<}E?>qa z#wU@dATms27@z*ye*C;DQyxy_CWY&PALdUsXT-N&3}JVkE-Q!o9{Y1xF1L{-4B_d1O1I7 zx_2*3HQlQzRlW}4o3Aq6Pne*C`>~OxL!s%?!QFR_>CRj(UkCU4(Wd*^F5GM7AspYl zF5Fp97@xjsx~7*7(qke9YaVRmP25S8w8*E*R?{X$<>!7=|4-Y)_S%qd^m-)p(Xfq< z$HMEJn_QdC;y2&PT5^yvl^$a<`#p#lOac-sLEty#n_oCiE8aV2eNhJQI3v=N&ao&Ogh9-f0vlM!(;L z-bp-g8wl<*p?5>9QRe<~6M8*B-oE)WOz1tuqbR)kn@s4f2g2j#-)us!=%>Z2f0_xs z4j_O25EFX$_A%bo`}dg8TTR8Y7V=k_(0hRAkr@5^P3T<$j0P_;p?4Z^tjIGk;hbxE zcu3dsdrauv%%eKmfPbS2y`$-AF7S~i^h*D5f>)W)I|UdHE_ykD-fMvlaE}SS89?6H z`bV44I}%7#{uC2>lYzV=^t(;yjR8vfB1|}^o;FBQ{F_batp$=4f3*p{B|tipztDt+ zUKxV=r-1_vc?C4JC?7aY%8*w;S2pAoFl0ozv4;=UkXMilr0g`BiXpE6wZHIh;@t zdn1rpka#E&owtc+_`m;(16SZPsX7VQp?imh*)|NT7#3zbH0%ifHxI8Lj<%fPr})1s zWlsw75k_nn5oX&ra{ox=AdEgV8c7Nn$1{+U;Auj_!kKH&UK?h+;49C9=JnOr@eGb+YjE3e%-vPJhbr?oG3v1?wu#^B;$M=r7-we zAI;Rjt?_@c!W)za@oK#Pr_sol%TI9Y9rS+*PJzZ|h^wZ($WQPXg?~-qi3y=e+xtMJ>k`DgV~|{%z%cP`NYa81Y&2 z`!|*Q5#{by?$-SNTjhRSxu*&q)Sh+084#^^`Nt$Y1E}?`J_q-~{H*WG7=lm~jV@kv zry+Tw*b;4#)HW$TPfLs)B0di5i1&Ar$!Xjx3tLPxi0)M774MRx8n7m8{JqAoe z%dQBI7?U?%;U*lDH%{Ru93%R5O^;)hg9&lKfPjk(Q@{n$ArQl$Xu`BS3r}k!DFctD zgp4tHCQMThifu$@;L#M7F-AmXKsil8JR()2g9az$W{@1DW{A8E0-JI)#v@0=OqL1L zF5+eYQ;0@po(Vk)j#qG;f@2lTP%vG=F$$(xaCECf+UQTLZzw%&>}XTrP>{Z%tnTG; z#L2kmHd`KRBCVoL!!zw$|KDYrL_d)6DLf2Bel&68yHt4`+FaAW&=ZoLA{1(h)hCq4 zVbPKGiF_XBrS7uvar%K$=9zR(C?fguO?el`78&a@akNUn@c41^GaQ7MPsUOCd@Otz z9<*spj1C}cHEb?^EW^8%pHWRXpDMFB{2oF;&P?Fwm5jj!9&GMG#DbvG$LeX6ie-svr7<_^s2O#p`(wGIAg?9{D}4 zM;6Qbkdaq_^AmhM`58%Ej}uoC<`mKqe&V_5?RDGAnO>?Yg92!z;?IS@Jl$3NHsMmC zZ{^W-;&H?NiNCburjibkJ+hwkN}BTS)s~*XJc9T|hKNqNg^L;cCn#qJbgVKQygnM8 zPUUs1^U*P@)m%wAMc{Xx9#Nb{8D527@xM-Y6hF^dU``DaoTt@ zTOse7K+-AB`JTvgsp+B(S*GeHHLFmjdRxC5n69O zj<Y^^BTa_-P*~q{UT}ZQ$)A4LO-3&9JBf60K zEx5-sldz%-sXzLaZj~^Y2cT0)bRkJ!B^{0J(&6kP9nMbD;p`$E&fAnOq}z}g(x`MZ zlrc_Z@}BAq&apGmH}pBXFZ1(ZVkW)3!`rrTCZTvUv{akz3hUI)?-3-?Z z%tvhrQ;*{Q2JRv&Wv0Eq>kZszuAAvvf%zP7{m$XujC&mJ1MCA_&A1O(cdKhW=G`+B z`n)>Gp8lsnq3JR;?H=TNXe_#-Uge%V#ugFiuKtep)Ssct>R+k;9;2U%{vWFk{u$jn zJ$H`Is5_c z(JBW9%LlOab&j04?Lp+D-Nd}lc_I8zzVkn)zZZZHVLrDppWET7>czcl80&!y>{w-= zw`8Dkeq24<-c}YG=UFjdFI`w#1y9CD_VDfHyy?6SUgN^S7v-E)iJ!uy9`PgY@S9{t z$l6RpuRPGtyc-(f5BQmL`w3*ejRu}%uH}qd{59UpoFBbvar2~CMIKs&^9I6*+_T^6 zA%UBj^Tln8Ey6hi8XJ;dt^1SMGshcYUY}@$nFZYq$;awuF}KF8Ds4WEpM!*%#GG3- zbbnn)afxTkK+2D|td@*zkxOul^ghG-<4?@L!$ckdYm`E#CP&`J+D7uXc)|SE{59Zz zu}W{*?>0K@Mq<1y;e<@n+i6*avEwjQdBkVF;L}eLa zmm#0ve9^Mju*(R$ny|};{gSZQzcg_K`Y|ilkjK+QkJqgs?}@9{G;c0mjNF0i;c2=rEW2GAa!%9 zsKg`X*~Gpj8lLjg@TR|i?+`=I$zofy>wWmn7q7k9bq%@~o8cWl%z9?o+K_}5Ylmz) zifn?z=yF_vPP)VBbu6>@b&Z4X{0Fu~!_PhV*^Qqc;O8Fv`~W|j@N*A-9@ZjURcj&= zs_uIP`DuB+Zm`_<7{{bkcsqu0b>95?p4QhFk)xe!o{ZvbBC2 z#jNW0N0|H3#x8T*?Bk*WPicBU`u%|`KBa#DQrG={9(_26y4a9>xNX0uPsms?ll*T; zK3X@E`slxEh0*Wn6RrJz|AG=vNZh6M`vdGvy6g9{N9pOqJ$?AE7#IF2`tU*41CJxK zp@M$=b;iMy^o`TVF+K6O&f7ixw`UISnS)txeY$h7)ql6D{yT{^t<`@;Cb6u$fvxNA z2-ev*Gk1nY+#I->c`_Pkt-JBp=$Ez6s*>iLkweUS*_p!{==9LTb*yh%*WE8@*hAw5 z=HK7yp{?ugoL48Zeho3^-(|ys=ik18E!gLL za3t$)8JA@K<-ET-j&+)hPkoL2^)<#PS$B7se}7}Hp-V~kvSG*SnuFHR>8zp4NO$Yn zTCJg1%NlOzFY8vbrb}40x_NacYv?^eYiMIV-MYpq@m%O^4Q-~w*+n{>outFrMLIgH zp)at8zJ~H?p?qW=eb5#+wTgB0!)hJPTF3P;>*$iTepj45!quPkZv*FkUHM_ZV#xS9 zs3jPF?uU-ZVm`>4`F`jeT;q4m$In6JWj?GWx;CsyOxUm{HeueECr#aeowbgBnEKg7 z*)4M>1oAoSmHmXVZk9I08hJkJ=x+~uA%2;(9rxk*SG4HV`ILFpP~4*&i& zv&|g5hrVj2Fkv_Fh z`;Lh1@;)kG_ciKk5^|nE*1K(uBj>s`+`n-DkQLLfkA7{hbUo3Y4nJ1xMBP^wvc{P+ z0sH%6UFkOL=Uy9=p09;FCurK#tTlRq7MsQOT)0#B`BJmsZx`K1+3>gjJ>z(ud`KRi zCl5OLNeapbygRWI$cLSLgy6OTw|V3xguF~3FFw+K-zDbd(~d{pD~yDSxd;$l87wh0$) zBAhnSzt24K^0&8%%O6j7ZK8s4B8u@L+<04oJW@Vv7T$PuyYU6ru)`t328%ot|1{2 z_^ysI5C1>PdtBJxuLUi>8bxiCP;W>Ux{nEb~X>JpA5i z%on|;PM|$J>7L-406v#6=uViLgk6C<%Oy1IiQ4q9tg$E9-4k6tbS>m3I?{c#6=P1J zt@OnG_b+o9e@f1%If*m>DW}bL))h}e#rSqS3CJOi;vOQJF)D_8h~A7#I^&cZzJww0 zCBPTI)VH%dURz`9n*?71YyN817BydCTuSE7B7}PklOF+D#<;ye`1bq=vDk~g7a7Z* zVJw?bT2Ok_*)Q-XyS9BFW0{=$58x-d+1GJCodW;PBl!Cvx=M~Z2L!~ANAV;8%Vx5+ zJqq8!t*rCU+2Sf^v&I!ZgQMuzyp?tCYo#ljGws;3*3BGl@*lhb{{i>!f!pBq@ZiS% zO6*L+jf11 z*B_pOQuqww;4}C@#~nU{c;qlbt_#a~Y~5|H z+c2NwT>jkJ$b=8pCL4SPx57K{0el3v!bh+Y^HKIBM{$2cxd-zh#KDI!?$(4pe}>PX z|6r5P;3#|sz{tTZ%8hi#_)yKJ>`uK;`hwRU-{$cS9SbbzA_sxIrK9cz<^^x0D zA2~)J8ApGx?vSOAwCBZ%qmRVVM-o*Z*=Ub*CDKPWuAAkeuU$eP8Q5hX$)u0;SAAq$ z7k#9=I(f9GPK-7NZ^@Xh0U85&%yNNaX;_H2x=x4p91~>%0(RSv)R{lKZQ_)L3mU*(D$nIyJ z9mHJtCg*9$HE5&btg>`bX_CVk_?Z#{iKxNnJ`0^w7Xvq+0KqL%#s8Qmqd&i;XN)&`}7VO*<)o?o#RI<@d6E?=2P+D4reFnaCVUn=VP4V%39(v&gd+DMJXH6<@Syi zJN2D4j)e1T#u_@(AXOX6^3aw@%^|Pw@4v(yBG-S;^LY>go;vbPR90H^s-bn^hj_V${bD?6NiDFMclwu^rg0Xzp7e=czB2UmtS4<^ zULj`EB<7V$f6rx(yK$}-jc)kf$lK}Z^Rfr%>GP|aZ!KQw@z~>f`uu08&!g}Cmd87t z>zDV;^*wXFCzJK1@QVk}_a87%%UgrV%=aHKPfupPuVcQ?{X5L}*4?Ab_nV5#nC}z* zCFlE9{{-{B)%RszFYEjL#XWt#XMJDV)8{YWzM;(;{~y)ovw0sna(zoK`mb5@Yk{Ax z>z>~rob!n5blcJb?#`e8%Qtx=fL64uMZlNQzAtlTVelZV?My!2A|)TsJcC$2)K_{3ck@Zy!zXD@r`*R+ z048ZR_K4vLS@0H!eUdLLwE)?t|7`wwN4fMA@&&87t1lM$cS^osn=)Pfc*ip39^c75 zeovWptmCfAG=_F6?Q*cqwloR-ktwlS#!ugReELrZ*WaD$E}x#NebVyeu5)c?vgX>g z(d&JCXrql+a%VY-A9~@?;ygr}}ayM5jvf|FSjGav% zPH|}&k@Q))cZ;CUid1rka^t;9j~F=q`js*V93F2Ohq2=RheeFFQDANggo9<XDc(V|rgxEn=>t3AQE8x=myLHw-ypU<%e$YT$1 zfHzUr+k3eyy|rjn>pf5+=cvNZId1ie<{#2G&e;+x?fpHe2uSTERPTxu2bRMQA!AKz99%``M%D_I<>rW%72h zM)VtZ)h^_%Tz}fwQSRF-X%~kVtny^@w#vtw0%;dk`Tc;pTSoo-nS2Sq@R#+zt#y}9 z8Hw9W{EOQo!11(~nfOuVSDD_fEqoGsLZ=d%(iVQ8;yP!<^%`xV?td2x1= z7w7xLoeiJ$$F+rrjkeG?kVBcxR&C+9z30IB~Aw3SD>tCxHGqv(j~X$w7Vp{FhIW}#z>XDAan zOPPS4u&iyItr&X3E-da^Moz^lb-u#6)>Z0UOU_p&aITfgxt8$JE#fTY70yB|IXP*Z zmGtJUMCV-PE4cOIOa-1aLmo~T=PK5jigku!$;I2je)oaW1*JQ}<^069y`DX<@X*Oy z%>&5#dl2_Yypi0&8++5umpK#}kSEy7$~)bkAXfZ;l{peUC0nkDNgtY^WjwCw#V6vm z+(>B2zB_@R@JP?+oxQxrUGapx$Ay;;J|~fRaSZu1^LcY?ytU-b?K1eC`mS2myiDxj zvD;4=d2=Z;b>t0Z=&Je6M+@h9=E=KT;eY4+)Z%rMH2;=2yF0=|jWj>V9u)pUNpIuz z=r!YQP%L(3_Y=3!c@#M|=WLOc!fz)0ZLb%u^xRMU;gcAYsM4Nz8;1^JdJyS8s4hM`OEm z7<_M?q{G=sI-Fgk!+AgaUpt1rP@dPyF>KmLBoafADXc%v?czhLmby#VhUZ6Wo3%6&d>b03y^;BD>(Ei_f~ zN}l75yz@5qK`k=%L7U0@cA04!_hME(-@jl~Ydt?IW>wFRU>12cmpSgn5A1`si%gqZ zn>}FlRR?d`EZLUQ{`VJl?6b(|Ztb(bRA#Hs{wCh+v-SU^efE3OuRf!FR(Mzb8TzZm zhb{f}cu#-5)bXb=xUas%oc$bgtMpgv4r2r5IiERUgCjIh4sSa1lrhKl^w)pQIku<2 z{>^jtKXiVz`m3y6_X~fiycY=SuQI>R4_aSFs{Z;Ga};J{O?+ILLs%0(m>84Zi#4(6 z>fFct8oVBscN_DNU9&-pbS{(ij;wuI|H#^R1##@hUgY>0>t3}c-OsvL*2FJ~o2)fh z=UHpw!>scbvL==>M%Kh}>;Xg;&Hbc#K=DG)3l83fGR82M8f#+GYpjWPwx3^(ccPJ% z8;v#bvASbHYvS>&iLEyn##&jei8skwEcuspn^>Efa%noQi8ltViH-HKS}V!=*pf>_ z+Agk%&2%`sNJr~ikf3xpyGTceHSuxEYCdK4mU<8J5^q3vuvUHMUUMz`k}cBp4Dw`N zTKh%U0DGjd-hEITX!zNJA9?G+8q>7}KM$_?qAL?WBHwW#>t>onz;>u8EC30&C!hsSjE2K1^FUJnXQsCSJz-komUI z)Q2f|+K;g&-hdxto!rfu_%hRM(`Su!u-f}P7PKaQSgnaKbKI?Stn9}lD$V^kZ?*zk zE4t?u6MhE9Ytft9*c;vwkxL`|V6sns&z_VzlK%gDyS{xCd|ZF9k4hCjFy5~HJPlqj z;YF}%&g}a%$M}8xeuaElk;9SN%Q61fnA0#Da$1sorSml1c|E@kS}$Y;xSSI-yK_Rw zB-aFZzVelM)ibAA$Xy=$%(<($hN_*?MmOoTUQ zq63}=+U>9G_U(lqwd79JwTw%NOACn(q+VjWeoeY;4y`8SMBbitq;D#B{65lmJ?RVa zZRl&I?>mytkjbv6NaGgLNV{`QB8_*F#*@5PUq?EBZSS=`g>+s|Iz4>dgVw<0K-u8k1p|U6LeyJIz`^T6?FQoiN4T{zV#ghMx_I~zRbb{HN@-<2;-k zJguX96)BZ3hJzPElz7FX#vxP31ZuC{Ey8Em2RjP9P*svKS4mriz5~b%-Z1P?KxcMm7RQzX0zy^b7Bn=I}o z^k%?8djt4r6MEx-*MrBJ(0hT*%s^jbvkB)^k@!itSD4USPGTp5uQ8!_If+)&g6}nt!hgbBTSN16In_n6R|IMUE_>5ns^H}NV{-)fu*y{8ygF#4NJ=sf}C&wt#6-u=KC zl-E8J&N*_WsgL!L3B3n_DfmBNLhpXyaPWO5^zH>p_&p|^vkN#3_njv6)&qxvZ#JRV zoup~W;E5*mh730Kxr)+Oym*@v4Sk&cvnKSOWZcH9pEG-b-q8aYL%~Oy&|5u#`2f7i zgx->VroPug6M7HzHTAt7G@-Ylx2f-Ss|md_<|Gh4!i3(@(WbuFktXya7t+x8deVem zW=7Gk1@dyEOa4j@VKhnUbS3btawFE|b8Jpz>SJ7mH+*V;{e zuO1V6Gk_%3FM3&lUeU`+QvE3=^dmOi#|4>_n;<9SZRMI^zH#l{JTu(tpJ9AZ!n>E zIgqUSmzdCdJCG##XPMB@4=eF8W*PDAGvf`dSFl{cd1gi3yLXsQCGYN{13ko-hRdc|EJ`8D*#3Qo6>@aT)3G$qV73%#mKmZQy8 zuRvSuE`1OG59kN^Z)%G@VmpS0*arJfG{kPU*Yp2^y*vb6tRYqW-x#u)|9e9A@_%bc zgV^-i+$+qsrPo&e?+M+DM%kUAyZFB~v?0`Cs}J3Rd1L5i{%;7akpKNF`a`o{d4GrP zP`@Mn&|cfW7MorD_wc_qZetueV-uPZNW-9=gU}Z1u69H1%7!b^D?9v%+4|Bh8=ryY?$qAW^*QbWxufD3#5EP{RFhjPChc3+)X~n|9zA9Pj=Y$Ox}xm z=j2`d-#odV|Ho&XnBlPPo3S4av`sTk@qgpY%`^EwYctwt_k3yZmx$}j)nDd+esw-E z&fPheT+FYS&;P=TLef^;P#k7!UVLFOG?&#cgU*UmE5dAh%l4HKYEAVT?AO+>b=Z!t z*H5gcq|dC^&kEdnpWbjEaNqs<{`-LkEA>N_z++$2kADq#@*%zHA>f%u^s|ouFVyMf zn}8LM>s5~fYq#hdx9ARA^SAU1-vZycMc=grxZ&G+#kYaGw&{De3ErSrGywPP(D&{D zp4_Q7?G*PP=zD$uY}%!t+6AoLqi>Y{$z53?8zyr_e2cOf^rP0xltWWxh z;MO|&S%q8cZt*Ja);f^k*ubrIu@uk=U3t#$Mh7CmXueA17_-CCzh z{}lYV8v|b)X2Gp>?nl@OZmo+?D%@J<1|)xUd~Gg2zEFib6#jkO#63dcs}vri@Cgc! zRrpbbTl9V`>A}vU-1pc`e3HUfDLhW$e^TMy3V%Z3$qFx0c#6W)6h2bnVG19u@ZYHT zGZcQm!mm}hp;wnMK82^+&Gb!C_)ChOU*Sc{eVW1x6rQW_+Y~-a;kPLKc7@-l@O*`z zRP+lKUajyFh0jv>5{0`JzFgr?+0FEBQFxZZOBH@x`7c*^g~B%|e44^56fX0X)K`_l z|E$WZTH!|&UaRo$D}1BE?@{<>h2O65dW8?LoAkFT{8<%WgTfyb|NN|dQl|31Q@L+e z_%4Nu4q!>|UWGrb@I4B@PvQF%{-KKRkixeq{D{JrD*TwjMaYiOKd$g-g`ZIP?^XUz zD*S&G-lXuaDg2bePpR7P;fM&*7%;rR-eqYL?DeIe-yQTRuSzC+>16&|7R zXB8f!@Ou>=tMGXWk5l-6sq`c&T-HkxpIhNSR`Dk*{64|)lcMl!!9!TP%s10J1^R>e zSzkJaAXGDT0pEls zPF&iTODD>fvkaxm9;H<}ExlE}w5ev=qtr{|VCk7wdZ))=VQOd|NAAW2Co64?>BnQ~ zq)u0cRwdPz!DDHuPFIFjwbhovlc&^Grz^u4%fd2vlz!^*3X)VyVYO0VJ;rhoh-zt5 zUA0H4s~)c);Y^2Ceb%i8aUh(jz1pL+SC6-}S*N$EvsUV@+v~aZh@NXh@pZaoGE`ri zYOcirABG1@$u((dr|p^mCJ75#`mYJ9TnN(|2m=ZJ+f-tc2$MjoZtPZrr6jw9wro;k zhLIdtp)siM%Cu|7n0m3@#uscJJ31XZ(>ZPQ=y4etZuj^xY2yHVL3*@r>Ryj_A!Cc? zHX~$Ps~u}IkeK$?fA=3vlgxMWSvY$o*)Pdn!T2s!kGB6OrhlO)B)7h=^mxO-W4f6C zMep^ZWwN`n@Z9TXE%8^xO^&45+UhH3DdRj1E=)SwF)YhJ$S2L?j4xrC(Y!)_^Beobm`v*cl`+dPaQr` z8~mVl-q40@`_Qh}IMCG{j;`(u-b+3(!IybmlD70K=!@5_+i%Z%}%m`(p0T8|aU|W9YW__11C+Kfi8iRv+x+u}5dM=)8U&`@%uIF-JFb z0cKJzTi+V;AFFBQzxZ(VB+@mJ^i3e0S@5a;WUJh#6cNvL=;?kQ843>cNV9Lqu)@C2 zolAwJgS)o%@Q7OWzZd%f=It=w-lt@S-5b>kp^;i2Wz9k+#V(t7jmnR0MY4udtvdj-fye?Z$X z{oNOyynBJoOxJp7S>+d?{3?}w66v$-{{wqz8}AP9KSttzn=oeEh&J?@TmDuD)kP_0 zqb;C&ynynNd|xMe$T3UVSMrmxUr#(z?oxh&OSwpyOW7Ijlnq+Jm%a^@Ht+yH&aDhR z==8bjV)}xlDPOOw`x0?l^)6xDdPUvLpfu%Rj&e^L5zZa0le=1xH7alag-7~kc%r+&K3%bS_x(6wVYaRPq!e40#G^+!MRk#^h*@gg)*s zxYFH&Ler(p{#EP^eFVrt5nTjExhPpEfB&{O(`-MAI=YpYf^m*FNl4C=r|hrzSJH&&TC3|_|#*`SrAJ9aGgVd9X~7e)f?~e+(Y{ufo|=dGU_QK(YvAAY)?Bf`*lxWGsnJv_;^&vm@Q-V%AT=0 zVCj94F?$bl-(ciSS>^L+ap}M9T-L~3){nW&0;f4c zDfuCL4BZ(h)*}LrXIjS2wP~?a{948zSgW03t@ajsuc+^TE&gTpUZ+{Bonfu^N7ia5 zSgXCwTJ2Y?)qc%d?Idfp->_EuEo-$O&=&i#u42t`p0iHmr)f1DMi}pb+ppbZ&1SCa zw3_eGuaB%+T>6e9Ht>->YWoTHzhSg(kuNWD=GUn;o8|wj;(x(1kN6kAK;4PL85cR?Tn^R89Pn_0WDp6g#(3Vsg#w;O?bR+XY}!<47? zXWV4X7KZ)s$jO1_z>LV;z)|41$eG5Pjkp@y#pUcKE@wA!IbVaW$X@Gj@6Y>6bS(%j zkUeZIYqna}Z0P$)eJ5^a>RoER)>A$`SS{{ZIy zmu38UhB5ptr3XRECbxLSzwMZOjxqT~#^gV)oXs8qt~3AsMq`^U9b6e zj{bA*c4}*Xs%4}M)_RZlJm-I`DN~-VhFV_=~g1DghDjVX+S zO<~%37yD}m`|GCf*|$Zp$Bt%?9m5{GH+yUwdu%&hp3GX93*J`DHaxMkw@IqW~b zS*uBE&}zQGn#{Z(@tyzO0PVbtt!LA<##6{Ik7X||`knuWv9@AWNojGoBhUojLDP7x zF`x0)rPg+`=YC7hl#o$v@c;w}6I-;_Qyd-}5Wg!}3pHKIoi}^$!0O`Wroqr>Pb_=x z*RYd4cd;HDI4kDW%bN$VAD2CMii8_>pzc=IQ;#C+yVm9i6jEjmy?ZOIX@{|ZI_jkxr`iQsM z@7fc$UtqpIfh_+&a_;d5-W&hQ9<%)nx&R`ns~zxOR8Z~>MJqiHsgI!t z>t5@kjW!{}qo@7-RsE^aYFDyHmbPV$r9JIRTiVmE{_1hMwOvVjvf7r+(Vw+;CG)Aw ztHE<>PrK@AS6$4zmpl*lw5!Xz4zuRLp7E+@yt<70m7aM}t=oIXtDf=7IwQH9`@x=e z_35;$rtzF{ah@$_Z3pMI++MK@`XWEXy*>qpyt~=!JXDatO+1%^g$oF&V_rH zWpe(^y-JVYPhJkjneo|9?lInP8M}dZMP2dN8*(BGv<#=)*LWj*SySQ5x{f=vqNg8- z@0I8~k6z)v+3;q4m;1a1&V_eyE*u39{`WW=-pSc;Z|?t|;C%Q=&WG*jw%o!QG4gh* zU*LXF+&ght>g@?9)2s~$Ku=?{e`~9S6B4U zIdW4(&QFHfa$e((@y(CEeE6dUYdseK*Q12@aL0Sj85cMKfBhru-FBlZ`Wb=95dAZd zbFp~Nax-heYvD~7UiqhzUyYw3(EV!sf7p7Z)?z*~{7C$tfVD$jjeiP}mqTq@=6KE! zFEh=uA<+EET3gPMwGlb5uI-;AX+8SU#KU73jPvv(ox3q-EEwy_m9$_Ux*)?i%e>6# zXiVd-3xQBI-Iez-qr6Ld%KL9`OFiY?)4qGglb-S9a?MRW$0A*{%h8_t>egM zcm+pp5jperK$!=OhChWp#r@m0ny-FSt9kf7t)}+B zw3=~o^}M^4x3T6>XA zgxQOpbW3MfBJ1xu*5>i6S2h2LJNVW1#LADrR}X))Znixkkic3!VfE_fFR)K}6F1pU zygK~Vy4Ci6fp1{$xB8yu|5k3VfxkBVwYvZ24T$V9#BHU)vyxc389L{N9k1KFV5KLE zy+J?pZ9Gam1K?kImUw5|`&Ir0eD?5PqMJNEFaZDJ_IYR%SLN-{TRr@E-Ct}8#`|A! z+l$*~+}^}(GwJ=vmKZpP`Tu9{YXGAvuKnlU4P?pZ1_CS_gbg1;0@kbqO#0lq0aO~J zg(NDaUDx7dlYal zuosZdt-$(cs!MIFx&)o0Eq9oKWxBGIK|#%z>IO+SU>bMWZ>u73*2kmaR0(51L*G)f1miD6O6UAM}q%3 zagU?^vCIEHIZLPTzfZjC8~*3S|D5=r6RDSfCv;&tS8V39E#G_QXU_e-$46-Id)xPs z)`VG8M#eZSGfx2N{s5|C~j5T3q%pb+z*BfiU>{#O!js5Z-$J(0z=U!VAl${UV zUaqYP%1-LW(sR0EUJBo6gYYx_rKQiX>r5DQkX`Vf&cWJ){asl6Nt1O=2G);ZZ?|-8 z1ZjDHMpl9)D598QK4>w%D`@>0aK5F~mAG#H?52p4;V}+fe>|YVmFT#3>vIuVa}q3N zGHo(Le5EUKa=)$5S+XW4m`X>)P+W)A6j%3LhZUGJnm@Jah3S(M79*bKax5svf_a){ zS+f%?;H7XW!a8_{H;?$BFl$o6VSGeEuwaU|8)`C>5&=PIG?pcaX#kU`k`&m7PfxWCVhZt z4-3(BKcFpoM63h%KBwRNw!XM8N1u0P9IiJ8aeZ)oN1rVsZnrM|{x5Tx!tcKi>PKz* zepa6Z2kPcPTkpu;`0DEV^J13+T+oT>5g%rJJ#4u~Vxxi=8~yERt+{U9%XTUt5AT zi#yQ9CxTVDBf^0y9E7k*g|TL_4D;&y@H^q>`%birn5+MyMSebLXmSOuQ-h^$#^ z|IaY@{+~F_Cw3yvG^G7LFLzk%f!N%#?s)q*d??PYiX4c0355;tm3M~6gy~_iYw65a zEj0EL?<2_LJ}zt6+kI>{W$Yv0iKW4KN2OpNUEI$g^wa;E2E`_8gMxx_=H?>UJ7-rF z+k&x|D}`rK81F^wUsy}~w!ROWDAviGc^hfhowavs(sIkfq~#IlKb+$s`fnBb&z|)_ zLbRC`A2Nlmv17b0z+N`rW134Y&)G;B;Q_HQle5MG?!!6No= zrFQpabB^$`S0ve*Eq0uJiayHkyc;bVh>`S3w{avb1GX__*=HD2sY_QCgHqc*rS)g7QOsI*q&jB-iK$))ArmQ zefAalY(4txiv<(0M`T3t7qC6kzK>+LB-Cxwb*$P-z#@v(?-~PAHw&>dfoOo&XB9AwCy*zqE0ef{{O=M zlY40dxxp}yk?#8_YJuB?Qe|_#C+1Qnx_yF)8 zQ)m#^!?UrmdHT<>f7g2KYqTEw8euO@>_s^~wreu>qSRO-#@}TzT3E5K{xE!%s6Xz( zo*Kr!ou2)1A?@eo9{l~`?+^d;p~e^g^Wh(_{O=Fn?_dA((f;Sd(5*<%Itbm&*kj`C zcltvAJ6*T6TgNJgZuJm;AGOl4Xut1g7kTdY{WGsRmcLJ=4zF~^%cA#QN}tg8Z1u{P z@A6U4@uB}^!rrXYSbQA5h3xb#gnjfa>q>3iu#dh2-$MEL7AnBE&|-WG72;c{2;V}F z;9KZXd~F zgo^2#i0TL?oJ_Or3?P>f|Jg}afMUdmUd4ouwi;B?I# zY#y_;OYg31P7LH^VDxs|3_2TA>l765E7yW4HrE1Ipq9QG-()Y-HyP#$jlGRu#y6R< zx3MojffCI9JfT^G-qw1E`9b&#)Ed43--k3YkB;*l%dj3|ALhe8!1<01ftX(&=}HX3 z?+N%9eGdPk4EP)|_#92q?pRP1amRwpqbz-SGmHNn+O-<(N^`TtuNB*}u_om>_DlU8 z+V(j1OAW<(_ps5}ClzbnEm-d!fxS|7ta&$Ky?elHwyXozym!QUcZN0YgOK<50Cr{& z@*ckti2TR<0+Ijt5FmJt_XdLJ`1L^W9ghNn_v`ERm+x|aJF{iqgtIg7r#eIZdK7!A z8t>9SW86r}orXPCwc-=lQ}x6>a!*y-_nh_tq;)N1_tNmA?`SjDwG@Nr6O1*)GZFhs z7T`Pacld_Jo~1b7&UJ?NRecY0b4Qsb)|VXZ_tA5?Go~d3BOj#=fDBREz0h5b_B;68 zLZoFQOleCX)09TvwnzJY_*^#92B@+|;`amQCf4!8PgL zkE85t?EkqkD?8zKq*3@c2>(unixEBwUo5+hK)a z65Wtqgi$M#M3tuK>3Alm++?Q86% zOW_X(d}t%vz_`92*^*3J?AO>=m*TVge`s5Pe$S5n&^Cw#$9@te%lV|=JGP_P&-5tD zjJCgL%ht5mKkF=UINI-9y4L&wbgdg@eWKy}^-)%I!Y4yo=!2j$QFluFGs0NQ^KI3w z_Dj(x)Hk#*EcF%j8TB3Q5lek(UULl38uk%}|5#A5ujh3Xo>!w!>A5YK(Z1*Q9CG^j zIpoy;+(Mo`#<2^2ZZmNv_xbwt7|tgDclYTGJPXG7vh-{2u8$Zu=4WDkbnOnzv-2WXQZBAFKou~kQ?Z)@!@q2K#9@cIihYc`fB2WM3c+$&;9W?D;KC~KV zEgRo_)Bp9|_~v^b-+Y&1FS&>5af$cH*TW*irzTNB1ue-qZsQq^9SNrR4_j>cA{&``~OE~5K7N3_d zzvcFrbAjzK^;>R_AAh^ooB!Sgw#QrFZvASX|8}o8|C)bY*z*!W`M<^I_2-c+{!-?ZOE9neyw)ST0y+up*KVvwuP#(!tVgdNp~6^?UTsog zkM-y@2kkEh>-hh}+DFXWf9^4Fe}a56?tq^~3j8f@hu=jq{4aitGYLk$%nRk21bl=J zzYG)N1CZVU`8pzhApA1SFaOijawc^od@|@v>TdAKpf$h7{v*bj)IYFeEj`Ypwy86z zg>TM;kumqP!qm{izkLvEg=^24;$pEbyLJ$MOlIOA8OW z^M~xl)tWiT{6jX>D3j!3)hL!u`_5zn=^7PNa3=KY*z@BB?D?@AdwvwQ?D_HRN$lZ( z{+-ck&yN`%dwx*=R-=DgpHn_ljFmkw=328|ki!$#dz`yHNQG$+kO3-;F~vAf!_WKO zZ!0k-HhReKNj|i(Ba8g%@8^)2-WV5dHm8MTSkmOaNOF9574q{6&bdF?DXh2wz6Xae zKFoq|#O#jPyC=w%$hDwsrr*9G6XV3qdPsIY@KyM8tb_c#Msh=E+vDer$F8R=@aMm{ z_SuCoYR8u?ef;MC@u5FQA0Nh8@hQfNBN!__%VEpvF@7+dx6^&sIr<*&gJktSINx}2 zh{g+Jj6grT#|RoP`nHT0Cwi+ebTk@Y{*W~|L5v}Ie|U}|{_<2i`P;8noay96Uc`8I zyhYak`v3UQjX1mh=Wm`vrZ3}o_L>^cUTx=i=Fflb@!nhecpJ|~ppS1ReW)k&BIrla zz(vrD4n;vP@-fc+^~K946ZCF-2k*04c+b)MuD2eXeMn1LKx5WPjDfST2Nu2SPC`!- zVLZbpF$Rh-WaT8rKoN#apA=)@^$FsAh%u1fhyL>Y?ekrgA9@~79Q7D;>-o@(02cYz zV)=Y!tM88KuPwEi@m+Z=oYhXpx5u$yc4jdA*RO-``VZi@em#8FyA@&SKQ8sn