| 
									
										
										
										
											2023-03-17 16:12:49 +07:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2020-06-29 18:40:23 +07:00
										 |  |  |  * The MIT License (MIT) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (c) 2020, Ha Thach (tinyusb.org) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This file is part of the TinyUSB stack. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-27 09:11:09 +07:00
										 |  |  | /* metadata:
 | 
					
						
							|  |  |  |    manufacturer: Espressif | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-03 15:50:52 +07:00
										 |  |  | #include "bsp/board_api.h"
 | 
					
						
							| 
									
										
										
										
											2020-11-27 22:31:47 +07:00
										 |  |  | #include "board.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-07 19:15:53 +07:00
										 |  |  | #include "esp_rom_gpio.h"
 | 
					
						
							| 
									
										
										
										
											2023-11-29 17:09:52 +07:00
										 |  |  | #include "esp_mac.h"
 | 
					
						
							| 
									
										
										
										
											2020-10-07 19:15:53 +07:00
										 |  |  | #include "hal/gpio_ll.h"
 | 
					
						
							| 
									
										
										
										
											2024-04-25 20:23:40 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-02 13:17:29 +07:00
										 |  |  | #include "driver/gpio.h"
 | 
					
						
							| 
									
										
										
										
											2023-09-26 19:09:36 +07:00
										 |  |  | #include "driver/uart.h"
 | 
					
						
							| 
									
										
										
										
											2024-09-25 17:18:41 +07:00
										 |  |  | #include "esp_private/periph_ctrl.h"
 | 
					
						
							| 
									
										
										
										
											2021-02-01 15:16:38 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef NEOPIXEL_PIN
 | 
					
						
							| 
									
										
										
										
											2020-11-27 22:31:47 +07:00
										 |  |  | #include "led_strip.h"
 | 
					
						
							| 
									
										
										
										
											2024-02-02 13:17:29 +07:00
										 |  |  | static led_strip_handle_t led_strip; | 
					
						
							| 
									
										
										
										
											2021-02-01 15:16:38 +07:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-06-29 18:40:23 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | #if CFG_TUH_ENABLED && CFG_TUH_MAX3421
 | 
					
						
							|  |  |  | #include "driver/spi_master.h"
 | 
					
						
							|  |  |  | static void max3421_init(void); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-06-29 18:40:23 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-25 17:18:41 +07:00
										 |  |  | static bool usb_init(void); | 
					
						
							| 
									
										
										
										
											2020-06-29 18:40:23 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // Implementation
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2021-02-01 15:16:38 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | // Initialize on-board peripherals : led, button, uart and USB
 | 
					
						
							|  |  |  | void board_init(void) { | 
					
						
							| 
									
										
										
										
											2021-02-01 15:16:38 +07:00
										 |  |  | #ifdef NEOPIXEL_PIN
 | 
					
						
							|  |  |  |   #ifdef NEOPIXEL_POWER_PIN
 | 
					
						
							|  |  |  |   gpio_reset_pin(NEOPIXEL_POWER_PIN); | 
					
						
							|  |  |  |   gpio_set_direction(NEOPIXEL_POWER_PIN, GPIO_MODE_OUTPUT); | 
					
						
							|  |  |  |   gpio_set_level(NEOPIXEL_POWER_PIN, NEOPIXEL_POWER_STATE); | 
					
						
							|  |  |  |   #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-29 18:40:23 +07:00
										 |  |  |   // WS2812 Neopixel driver with RMT peripheral
 | 
					
						
							| 
									
										
										
										
											2024-02-02 13:17:29 +07:00
										 |  |  |   led_strip_rmt_config_t rmt_config = { | 
					
						
							|  |  |  |       .clk_src = RMT_CLK_SRC_DEFAULT, // different clock source can lead to different power consumption
 | 
					
						
							|  |  |  |       .resolution_hz = 10 * 1000 * 1000,  // RMT counter clock frequency, default = 10 Mhz
 | 
					
						
							|  |  |  |       .flags.with_dma = false,        // DMA feature is available on ESP target like ESP32-S3
 | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   led_strip_config_t strip_config = { | 
					
						
							|  |  |  |       .strip_gpio_num = NEOPIXEL_PIN,           // The GPIO that connected to the LED strip's data line
 | 
					
						
							|  |  |  |       .max_leds = 1,                            // The number of LEDs in the strip,
 | 
					
						
							|  |  |  |       .led_pixel_format = LED_PIXEL_FORMAT_GRB, // Pixel format of your LED strip
 | 
					
						
							|  |  |  |       .led_model = LED_MODEL_WS2812,            // LED strip model
 | 
					
						
							|  |  |  |       .flags.invert_out = false,                // whether to invert the output signal
 | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2020-06-29 18:40:23 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-02 13:17:29 +07:00
										 |  |  |   ESP_ERROR_CHECK(led_strip_new_rmt_device(&strip_config, &rmt_config, &led_strip)); | 
					
						
							|  |  |  |   led_strip_clear(led_strip); // off
 | 
					
						
							| 
									
										
										
										
											2021-02-01 15:16:38 +07:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-06-29 18:40:23 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Button
 | 
					
						
							| 
									
										
										
										
											2022-01-16 13:24:36 +07:00
										 |  |  |   esp_rom_gpio_pad_select_gpio(BUTTON_PIN); | 
					
						
							| 
									
										
										
										
											2020-06-29 18:40:23 +07:00
										 |  |  |   gpio_set_direction(BUTTON_PIN, GPIO_MODE_INPUT); | 
					
						
							|  |  |  |   gpio_set_pull_mode(BUTTON_PIN, BUTTON_STATE_ACTIVE ? GPIO_PULLDOWN_ONLY : GPIO_PULLUP_ONLY); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-25 17:18:41 +07:00
										 |  |  | #if TU_CHECK_MCU(OPT_MCU_ESP32S2, OPT_MCU_ESP32S3, OPT_MCU_ESP32P4)
 | 
					
						
							|  |  |  |   usb_init(); | 
					
						
							| 
									
										
										
										
											2024-04-25 20:23:40 +07:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-11 16:07:33 +07:00
										 |  |  | #ifdef HIL_DEVICE_HOST_MUX_PIN
 | 
					
						
							|  |  |  |   gpio_reset_pin(HIL_DEVICE_HOST_MUX_PIN); | 
					
						
							|  |  |  |   gpio_set_direction(HIL_DEVICE_HOST_MUX_PIN, GPIO_MODE_OUTPUT); | 
					
						
							|  |  |  |   gpio_set_level(HIL_DEVICE_HOST_MUX_PIN, CFG_TUD_ENABLED ? HIL_DEVICE_STATE : (1-HIL_DEVICE_STATE)); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | #if CFG_TUH_ENABLED && CFG_TUH_MAX3421
 | 
					
						
							|  |  |  |   max3421_init(); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-10-07 19:15:53 +07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-06-29 18:40:23 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-25 20:23:40 +07:00
										 |  |  | #if TU_CHECK_MCU(OPT_MCU_ESP32S2, OPT_MCU_ESP32S3)
 | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-25 20:23:40 +07:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-06-29 18:40:23 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-15 16:07:44 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // Board porting API
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-29 17:09:52 +07:00
										 |  |  | size_t board_get_unique_id(uint8_t id[], size_t max_len) { | 
					
						
							|  |  |  |   // use factory default MAC as serial ID
 | 
					
						
							|  |  |  |   esp_efuse_mac_get_default(id); | 
					
						
							|  |  |  |   return 6; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | void board_led_write(bool state) { | 
					
						
							| 
									
										
										
										
											2021-02-01 15:16:38 +07:00
										 |  |  | #ifdef NEOPIXEL_PIN
 | 
					
						
							| 
									
										
										
										
											2024-02-02 13:17:29 +07:00
										 |  |  |   led_strip_set_pixel(led_strip, 0, state ? 0x08 : 0x00, 0x00, 0x00); | 
					
						
							|  |  |  |   led_strip_refresh(led_strip); | 
					
						
							| 
									
										
										
										
											2021-02-01 15:16:38 +07:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-06-29 18:40:23 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Get the current state of button
 | 
					
						
							|  |  |  | // a '1' means active (pressed), a '0' means inactive.
 | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | uint32_t board_button_read(void) { | 
					
						
							| 
									
										
										
										
											2020-06-29 18:40:23 +07:00
										 |  |  |   return gpio_get_level(BUTTON_PIN) == BUTTON_STATE_ACTIVE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Get characters from UART
 | 
					
						
							| 
									
										
										
										
											2023-10-04 18:00:32 +07:00
										 |  |  | int board_uart_read(uint8_t* buf, int len) { | 
					
						
							| 
									
										
										
										
											2024-11-26 22:07:28 +07:00
										 |  |  |   for (int i=0; i<len; i++) { | 
					
						
							|  |  |  |     int c = getchar(); | 
					
						
							|  |  |  |     if (c == EOF) { | 
					
						
							|  |  |  |       return i; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     buf[i] = (uint8_t) c; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return len; | 
					
						
							| 
									
										
										
										
											2020-06-29 18:40:23 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Send characters to UART
 | 
					
						
							| 
									
										
										
										
											2023-10-04 18:00:32 +07:00
										 |  |  | int board_uart_write(void const* buf, int len) { | 
					
						
							| 
									
										
										
										
											2024-11-26 22:07:28 +07:00
										 |  |  |   for (int i = 0; i < len; i++) { | 
					
						
							|  |  |  |     putchar(((char*) buf)[i]); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return len; | 
					
						
							| 
									
										
										
										
											2020-06-29 18:40:23 +07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-26 19:09:36 +07:00
										 |  |  | int board_getchar(void) { | 
					
						
							| 
									
										
										
										
											2024-11-26 22:07:28 +07:00
										 |  |  |   return getchar(); | 
					
						
							| 
									
										
										
										
											2023-09-26 19:09:36 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-25 17:18:41 +07:00
										 |  |  | //--------------------------------------------------------------------
 | 
					
						
							|  |  |  | // PHY Init
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if TU_CHECK_MCU(OPT_MCU_ESP32S2, OPT_MCU_ESP32S3, OPT_MCU_ESP32P4)
 | 
					
						
							|  |  |  | #if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 3, 0)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "esp_private/usb_phy.h"
 | 
					
						
							|  |  |  | #include "soc/usb_pins.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static usb_phy_handle_t phy_hdl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool usb_init(void) { | 
					
						
							|  |  |  |   // Configure USB PHY
 | 
					
						
							|  |  |  |   usb_phy_config_t phy_conf = { | 
					
						
							|  |  |  |     .controller = USB_PHY_CTRL_OTG, | 
					
						
							|  |  |  |     .target = USB_PHY_TARGET_INT, | 
					
						
							| 
									
										
										
										
											2024-11-07 16:37:33 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // maybe we can use USB_OTG_MODE_DEFAULT and switch using dwc2 driver
 | 
					
						
							|  |  |  | #if CFG_TUD_ENABLED
 | 
					
						
							| 
									
										
										
										
											2024-09-25 17:18:41 +07:00
										 |  |  |     .otg_mode = USB_OTG_MODE_DEVICE, | 
					
						
							| 
									
										
										
										
											2024-11-07 16:37:33 +07:00
										 |  |  | #elif CFG_TUH_ENABLED
 | 
					
						
							|  |  |  |     .otg_mode = USB_OTG_MODE_HOST, | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2025-01-20 22:47:32 +07:00
										 |  |  |     // https://github.com/hathach/tinyusb/issues/2943#issuecomment-2601888322
 | 
					
						
							|  |  |  |     // Set speed to undefined (auto-detect) to avoid timinng/racing issue with S3 with host such as macOS
 | 
					
						
							|  |  |  |     .otg_speed = USB_PHY_SPEED_UNDEFINED, | 
					
						
							| 
									
										
										
										
											2024-09-25 17:18:41 +07:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   usb_new_phy(&phy_conf, &phy_hdl); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "esp_private/usb_phy.h"
 | 
					
						
							|  |  |  | #include "hal/usb_hal.h"
 | 
					
						
							|  |  |  | #include "soc/usb_periph.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void configure_pins(usb_hal_context_t* usb) { | 
					
						
							|  |  |  |   /* usb_periph_iopins currently configures USB_OTG as USB Device.
 | 
					
						
							|  |  |  |    * Introduce additional parameters in usb_hal_context_t when adding support | 
					
						
							|  |  |  |    * for USB Host. */ | 
					
						
							|  |  |  |   for (const usb_iopin_dsc_t* iopin = usb_periph_iopins; iopin->pin != -1; ++iopin) { | 
					
						
							|  |  |  |     if ((usb->use_external_phy) || (iopin->ext_phy_only == 0)) { | 
					
						
							|  |  |  |       esp_rom_gpio_pad_select_gpio(iopin->pin); | 
					
						
							|  |  |  |       if (iopin->is_output) { | 
					
						
							|  |  |  |         esp_rom_gpio_connect_out_signal(iopin->pin, iopin->func, false, false); | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         esp_rom_gpio_connect_in_signal(iopin->pin, iopin->func, false); | 
					
						
							|  |  |  |         if ((iopin->pin != GPIO_MATRIX_CONST_ZERO_INPUT) && (iopin->pin != GPIO_MATRIX_CONST_ONE_INPUT)) { | 
					
						
							|  |  |  |           gpio_ll_input_enable(&GPIO, iopin->pin); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       esp_rom_gpio_pad_unhold(iopin->pin); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!usb->use_external_phy) { | 
					
						
							|  |  |  |     gpio_set_drive_capability(USBPHY_DM_NUM, GPIO_DRIVE_CAP_3); | 
					
						
							|  |  |  |     gpio_set_drive_capability(USBPHY_DP_NUM, GPIO_DRIVE_CAP_3); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool usb_init(void) { | 
					
						
							|  |  |  |   // USB Controller Hal init
 | 
					
						
							|  |  |  |   periph_module_reset(PERIPH_USB_MODULE); | 
					
						
							|  |  |  |   periph_module_enable(PERIPH_USB_MODULE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   usb_hal_context_t hal = { | 
					
						
							|  |  |  |     .use_external_phy = false // use built-in PHY
 | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   usb_hal_init(&hal); | 
					
						
							|  |  |  |   configure_pins(&hal); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // API: SPI transfer with MAX3421E, must be implemented by application
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | #if CFG_TUH_ENABLED && defined(CFG_TUH_MAX3421) && CFG_TUH_MAX3421
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static spi_device_handle_t max3421_spi; | 
					
						
							| 
									
										
										
										
											2023-09-27 11:30:18 +07:00
										 |  |  | SemaphoreHandle_t max3421_intr_sem; | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-26 19:09:36 +07:00
										 |  |  | static void IRAM_ATTR max3421_isr_handler(void* arg) { | 
					
						
							|  |  |  |   (void) arg; // arg is gpio num
 | 
					
						
							| 
									
										
										
										
											2023-09-27 11:30:18 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   BaseType_t xHigherPriorityTaskWoken = pdFALSE; | 
					
						
							|  |  |  |   xSemaphoreGiveFromISR(max3421_intr_sem, &xHigherPriorityTaskWoken); | 
					
						
							|  |  |  |   if (xHigherPriorityTaskWoken) { | 
					
						
							|  |  |  |     portYIELD_FROM_ISR(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-27 11:30:18 +07:00
										 |  |  | static void max3421_intr_task(void* param) { | 
					
						
							|  |  |  |   (void) param; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (1) { | 
					
						
							|  |  |  |     xSemaphoreTake(max3421_intr_sem, portMAX_DELAY); | 
					
						
							| 
									
										
										
										
											2023-09-27 18:09:19 +07:00
										 |  |  |     tuh_int_handler(BOARD_TUH_RHPORT, false); | 
					
						
							| 
									
										
										
										
											2023-09-27 11:30:18 +07:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | static void max3421_init(void) { | 
					
						
							|  |  |  |   // CS pin
 | 
					
						
							|  |  |  |   gpio_set_direction(MAX3421_CS_PIN, GPIO_MODE_OUTPUT); | 
					
						
							|  |  |  |   gpio_set_level(MAX3421_CS_PIN, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // SPI
 | 
					
						
							| 
									
										
										
										
											2023-10-04 18:00:32 +07:00
										 |  |  |   spi_bus_config_t buscfg = { | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  |       .miso_io_num = MAX3421_MISO_PIN, | 
					
						
							|  |  |  |       .mosi_io_num = MAX3421_MOSI_PIN, | 
					
						
							|  |  |  |       .sclk_io_num = MAX3421_SCK_PIN, | 
					
						
							|  |  |  |       .quadwp_io_num = -1, | 
					
						
							|  |  |  |       .quadhd_io_num = -1, | 
					
						
							|  |  |  |       .data4_io_num = -1, | 
					
						
							|  |  |  |       .data5_io_num = -1, | 
					
						
							|  |  |  |       .data6_io_num = -1, | 
					
						
							|  |  |  |       .data7_io_num = -1, | 
					
						
							| 
									
										
										
										
											2023-09-26 19:09:36 +07:00
										 |  |  |       .max_transfer_sz = 1024 | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2023-10-04 18:00:32 +07:00
										 |  |  |   ESP_ERROR_CHECK(spi_bus_initialize(MAX3421_SPI_HOST, &buscfg, SPI_DMA_CH_AUTO)); | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   spi_device_interface_config_t max3421_cfg = { | 
					
						
							|  |  |  |       .mode = 0, | 
					
						
							| 
									
										
										
										
											2024-04-25 20:23:40 +07:00
										 |  |  |       .clock_speed_hz = 20000000, // S2/S3 can work with 26 Mhz, but esp32 seems only work up to 20 Mhz
 | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  |       .spics_io_num = -1, // manual control CS
 | 
					
						
							|  |  |  |       .queue_size = 1 | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2023-10-04 18:00:32 +07:00
										 |  |  |   ESP_ERROR_CHECK(spi_bus_add_device(MAX3421_SPI_HOST, &max3421_cfg, &max3421_spi)); | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Interrupt pin
 | 
					
						
							| 
									
										
										
										
											2023-09-27 11:30:18 +07:00
										 |  |  |   max3421_intr_sem = xSemaphoreCreateBinary(); | 
					
						
							| 
									
										
										
										
											2023-10-04 18:00:32 +07:00
										 |  |  |   xTaskCreate(max3421_intr_task, "max3421 intr", 2048, NULL, configMAX_PRIORITIES - 2, NULL); | 
					
						
							| 
									
										
										
										
											2023-09-27 11:30:18 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-26 19:09:36 +07:00
										 |  |  |   gpio_set_direction(MAX3421_INTR_PIN, GPIO_MODE_INPUT); | 
					
						
							|  |  |  |   gpio_set_intr_type(MAX3421_INTR_PIN, GPIO_INTR_NEGEDGE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   gpio_install_isr_service(0); | 
					
						
							|  |  |  |   gpio_isr_handler_add(MAX3421_INTR_PIN, max3421_isr_handler, NULL); | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void tuh_max3421_int_api(uint8_t rhport, bool enabled) { | 
					
						
							|  |  |  |   (void) rhport; | 
					
						
							|  |  |  |   if (enabled) { | 
					
						
							|  |  |  |     gpio_intr_enable(MAX3421_INTR_PIN); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     gpio_intr_disable(MAX3421_INTR_PIN); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void tuh_max3421_spi_cs_api(uint8_t rhport, bool active) { | 
					
						
							|  |  |  |   (void) rhport; | 
					
						
							|  |  |  |   gpio_set_level(MAX3421_CS_PIN, active ? 0 : 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-04 18:00:32 +07:00
										 |  |  | bool tuh_max3421_spi_xfer_api(uint8_t rhport, uint8_t const* tx_buf, uint8_t* rx_buf, size_t xfer_bytes) { | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  |   (void) rhport; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-04 18:00:32 +07:00
										 |  |  |   if (tx_buf == NULL) { | 
					
						
							| 
									
										
										
										
											2023-09-27 12:33:36 +07:00
										 |  |  |     // fifo read, transmit rx_buf as dummy
 | 
					
						
							|  |  |  |     tx_buf = rx_buf; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-04 18:00:32 +07:00
										 |  |  |   // length in bits
 | 
					
						
							|  |  |  |   size_t const len_bits = xfer_bytes << 3; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  |   spi_transaction_t xact = { | 
					
						
							| 
									
										
										
										
											2023-10-04 18:00:32 +07:00
										 |  |  |       .length = len_bits, | 
					
						
							|  |  |  |       .rxlength = rx_buf ? len_bits : 0, | 
					
						
							| 
									
										
										
										
											2023-09-25 16:53:11 +07:00
										 |  |  |       .tx_buffer = tx_buf, | 
					
						
							|  |  |  |       .rx_buffer = rx_buf | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ESP_ERROR_CHECK(spi_device_transmit(max3421_spi, &xact)); | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 |