| 
									
										
										
										
											2023-03-17 16:12:49 +07:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  |  * The MIT License (MIT) | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-10-06 16:54:52 +01:00
										 |  |  |  * Copyright 2021 Bridgetek Pte Ltd | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-03 15:50:52 +07:00
										 |  |  | #include "bsp/board_api.h"
 | 
					
						
							| 
									
										
										
										
											2021-12-08 10:03:33 +00:00
										 |  |  | #include "board.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | #include <ft900.h>
 | 
					
						
							| 
									
										
										
										
											2022-11-28 17:41:00 +00:00
										 |  |  | #include <registers/ft900_registers.h>
 | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-23 21:46:40 +07:00
										 |  |  | #if CFG_TUD_ENABLED
 | 
					
						
							| 
									
										
										
										
											2022-11-29 18:49:23 +00:00
										 |  |  | int8_t board_ft9xx_vbus(void); // Board specific implementation of VBUS detection for USB device.
 | 
					
						
							|  |  |  | extern void ft9xx_usbd_pm_ISR(uint16_t pmcfg); // Interrupt handler for USB device power management
 | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  | #ifdef BOARD_GPIO_REMOTE_WAKEUP
 | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | void gpio_ISR(void); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | void timer_ISR(void); | 
					
						
							|  |  |  | volatile unsigned int timer_ms = 0; | 
					
						
							|  |  |  | void board_pm_ISR(void); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define WELCOME_MSG "\x1B[2J\x1B[H" \
 | 
					
						
							| 
									
										
										
										
											2021-10-06 17:13:44 +01:00
										 |  |  |                     "MM900EVxB board\r\n" | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Initialize on-board peripherals : led, button, uart and USB
 | 
					
						
							|  |  |  | void board_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	sys_reset_all(); | 
					
						
							| 
									
										
										
										
											2022-11-30 11:46:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  |     // Enable the UART Device.
 | 
					
						
							|  |  |  |     sys_enable(sys_device_uart0); | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  |     // Set BOARD_UART GPIO function pins for TXD and RXD.
 | 
					
						
							|  |  |  | #ifdef BOARD_GPIO_UART_TX
 | 
					
						
							|  |  |  |     gpio_function(BOARD_GPIO_UART_TX, pad_uart0_txd); /* UART0 TXD */ | 
					
						
							| 
									
										
										
										
											2022-11-30 11:46:13 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  | #ifdef BOARD_GPIO_UART_RX
 | 
					
						
							|  |  |  |     gpio_function(BOARD_GPIO_UART_RX, pad_uart0_rxd); /* UART0 RXD */ | 
					
						
							| 
									
										
										
										
											2022-11-30 11:46:13 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  |     uart_open(BOARD_UART,                             /* Device */ | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  |               1,                                 /* Prescaler = 1 */ | 
					
						
							|  |  |  |               UART_DIVIDER_19200_BAUD,           /* Divider = 1302 */ | 
					
						
							|  |  |  |               uart_data_bits_8,                  /* No. Data Bits */ | 
					
						
							|  |  |  |               uart_parity_none,                  /* Parity */ | 
					
						
							|  |  |  |               uart_stop_bits_1);                 /* No. Stop Bits */ | 
					
						
							|  |  |  | 	// Print out a welcome message.
 | 
					
						
							|  |  |  |     // Use sizeof to avoid pulling in strlen unnecessarily.
 | 
					
						
							|  |  |  |     board_uart_write(WELCOME_MSG, sizeof(WELCOME_MSG)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  | #ifdef BOARD_GPIO_LED
 | 
					
						
							| 
									
										
										
										
											2023-03-17 16:12:49 +07:00
										 |  |  |     gpio_function(BOARD_GPIO_LED, pad_func_0); | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  |     gpio_idrive(BOARD_GPIO_LED, pad_drive_12mA); | 
					
						
							|  |  |  |     gpio_dir(BOARD_GPIO_LED, pad_dir_output); | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2021-10-06 16:50:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  | #ifdef BOARD_GPIO_BUTTON
 | 
					
						
							|  |  |  |     gpio_function(BOARD_GPIO_BUTTON, pad_func_0); | 
					
						
							|  |  |  |     // Pull up if active low. Down if active high.
 | 
					
						
							|  |  |  |     gpio_pull(BOARD_GPIO_BUTTON, (BOARD_GPIO_BUTTON_STATE_ACTIVE == 0)?pad_pull_pullup:pad_pull_pulldown); | 
					
						
							|  |  |  |     gpio_dir(BOARD_GPIO_BUTTON, pad_dir_input); | 
					
						
							| 
									
										
										
										
											2022-11-30 11:46:13 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | 	sys_enable(sys_device_timer_wdt); | 
					
						
							|  |  |  | 	/* Timer A = 1ms */ | 
					
						
							|  |  |  | 	timer_prescaler(timer_select_a, 1000); | 
					
						
							|  |  |  | 	timer_init(timer_select_a, 100, timer_direction_down, timer_prescaler_select_on, timer_mode_continuous); | 
					
						
							|  |  |  | 	timer_enable_interrupt(timer_select_a); | 
					
						
							|  |  |  | 	timer_start(timer_select_a); | 
					
						
							| 
									
										
										
										
											2021-10-06 16:50:55 +01:00
										 |  |  | 	interrupt_attach(interrupt_timers, (int8_t)interrupt_timers, timer_ISR); | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Setup VBUS detect GPIO. If the device is connected then this
 | 
					
						
							|  |  |  |     // will set the MASK_SYS_PMCFG_DEV_DETECT_EN bit in PMCFG.
 | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  |     gpio_interrupt_disable(BOARD_USBD_VBUS_DTC_PIN); | 
					
						
							|  |  |  |     gpio_function(BOARD_USBD_VBUS_DTC_PIN, pad_vbus_dtc); | 
					
						
							|  |  |  |     gpio_pull(BOARD_USBD_VBUS_DTC_PIN, pad_pull_pulldown); | 
					
						
							|  |  |  |     gpio_dir(BOARD_USBD_VBUS_DTC_PIN, pad_dir_input); | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     interrupt_attach(interrupt_0, (int8_t)interrupt_0, board_pm_ISR); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  | #ifdef BOARD_GPIO_REMOTE_WAKEUP
 | 
					
						
							|  |  |  |     // Configuring GPIO pin to wakeup.
 | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  |     // Set up the wakeup pin.
 | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  |     gpio_dir(BOARD_GPIO_REMOTE_WAKEUP, pad_dir_input); | 
					
						
							|  |  |  |     gpio_pull(BOARD_GPIO_REMOTE_WAKEUP, pad_pull_pullup); | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Attach an interrupt handler.
 | 
					
						
							|  |  |  |     interrupt_attach(interrupt_gpio, (uint8_t)interrupt_gpio, gpio_ISR); | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  |     gpio_interrupt_enable(BOARD_GPIO_REMOTE_WAKEUP, gpio_int_edge_falling); | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  | 	uart_disable_interrupt(BOARD_UART, uart_interrupt_tx); | 
					
						
							|  |  |  | 	uart_disable_interrupt(BOARD_UART, uart_interrupt_rx); | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Enable all peripheral interrupts.
 | 
					
						
							|  |  |  |     interrupt_enable_globally(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     TU_LOG1("MM900EV1B board setup complete\r\n"); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void timer_ISR(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (timer_is_interrupted(timer_select_a)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         timer_ms++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  | #ifdef BOARD_GPIO_REMOTE_WAKEUP
 | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | void gpio_ISR(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  |     if (gpio_is_interrupted(BOARD_GPIO_REMOTE_WAKEUP)) | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Power management ISR */ | 
					
						
							|  |  |  | void board_pm_ISR(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint16_t pmcfg = SYS->PMCFG_H; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined(__FT930__)
 | 
					
						
							|  |  |  |     if (pmcfg & MASK_SYS_PMCFG_SLAVE_PERI_IRQ_PEND) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // Clear d2xx hw engine wakeup.
 | 
					
						
							|  |  |  |         SYS->PMCFG_H = MASK_SYS_PMCFG_SLAVE_PERI_IRQ_PEND; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     if (pmcfg & MASK_SYS_PMCFG_PM_GPIO_IRQ_PEND) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // Clear GPIO wakeup pending.
 | 
					
						
							|  |  |  |         SYS->PMCFG_H = MASK_SYS_PMCFG_PM_GPIO_IRQ_PEND; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined(__FT900__)
 | 
					
						
							|  |  |  |     // USB device power management interrupts.
 | 
					
						
							|  |  |  |     if (pmcfg & (MASK_SYS_PMCFG_DEV_CONN_DEV | | 
					
						
							|  |  |  |               MASK_SYS_PMCFG_DEV_DIS_DEV | | 
					
						
							|  |  |  |               MASK_SYS_PMCFG_HOST_RST_DEV | | 
					
						
							|  |  |  |               MASK_SYS_PMCFG_HOST_RESUME_DEV) | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-02-23 21:46:40 +07:00
										 |  |  | #if CFG_TUD_ENABLED
 | 
					
						
							| 
									
										
										
										
											2022-11-29 18:49:23 +00:00
										 |  |  |         ft9xx_usbd_pm_ISR(pmcfg); | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-23 21:46:40 +07:00
										 |  |  | #if CFG_TUD_ENABLED
 | 
					
						
							| 
									
										
										
										
											2022-11-29 18:49:23 +00:00
										 |  |  | int8_t board_ft9xx_vbus(void) | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  | 	return gpio_read(BOARD_USBD_VBUS_DTC_PIN); | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // Board porting API
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Turn LED on or off
 | 
					
						
							|  |  |  | void board_led_write(bool state) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  | #ifdef BOARD_GPIO_LED
 | 
					
						
							|  |  |  |     gpio_write(BOARD_GPIO_LED, (state == 0)?(BOARD_GPIO_LED_STATE_ON?0:1):BOARD_GPIO_LED_STATE_ON); | 
					
						
							| 
									
										
										
										
											2022-11-29 18:49:23 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Get the current state of button
 | 
					
						
							|  |  |  | // a '1' means active (pressed), a '0' means inactive.
 | 
					
						
							|  |  |  | uint32_t board_button_read(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-11-30 11:46:13 +00:00
										 |  |  |     uint32_t state = 0; | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  | #ifdef BOARD_GPIO_BUTTON
 | 
					
						
							|  |  |  |     state = (gpio_read(BOARD_GPIO_BUTTON) == BOARD_GPIO_BUTTON_STATE_ACTIVE)?1:0; | 
					
						
							| 
									
										
										
										
											2022-11-30 11:46:13 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |     return state; | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Get characters from UART
 | 
					
						
							|  |  |  | int board_uart_read(uint8_t *buf, int len) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-11-30 11:46:13 +00:00
										 |  |  |     int r = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  | #ifdef BOARD_UART
 | 
					
						
							|  |  |  |     if (uart_rx_has_data(BOARD_UART)) | 
					
						
							| 
									
										
										
										
											2022-11-30 11:46:13 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  |         r = uart_readn(BOARD_UART, (uint8_t *)buf, len); | 
					
						
							| 
									
										
										
										
											2022-11-30 11:46:13 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return r; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Send characters to UART
 | 
					
						
							|  |  |  | int board_uart_write(void const *buf, int len) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  |     int r = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef BOARD_UART
 | 
					
						
							| 
									
										
										
										
											2021-12-08 09:34:29 +00:00
										 |  |  | #pragma GCC diagnostic push
 | 
					
						
							|  |  |  | #pragma GCC diagnostic ignored "-Wcast-qual" // uart_writen does not have const for buffer parameter.
 | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  |     r = uart_writen(BOARD_UART, (uint8_t *)((const void *)buf), len); | 
					
						
							| 
									
										
										
										
											2021-12-08 09:34:29 +00:00
										 |  |  | #pragma GCC diagnostic pop
 | 
					
						
							| 
									
										
										
										
											2022-11-30 16:18:07 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return r; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Get current milliseconds
 | 
					
						
							|  |  |  | uint32_t board_millis(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t safe_ms; | 
					
						
							| 
									
										
										
										
											2023-03-17 16:12:49 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-05 13:54:47 +01:00
										 |  |  |     CRITICAL_SECTION_BEGIN | 
					
						
							|  |  |  |     safe_ms = timer_ms; | 
					
						
							|  |  |  |     CRITICAL_SECTION_END | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return safe_ms; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-11-28 17:41:00 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Restart the program
 | 
					
						
							|  |  |  | // Called in the event of a watchdog timeout
 | 
					
						
							|  |  |  | void chip_reboot(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // SOFT reset
 | 
					
						
							|  |  |  |     __asm__("call 0"); | 
					
						
							|  |  |  |  #if 0
 | 
					
						
							|  |  |  |     // HARD reset
 | 
					
						
							|  |  |  |     // Initiates data transfer from Flash Memory to Data Memory (DBG_CMDF2D3)
 | 
					
						
							|  |  |  |     // followed by a system reboot
 | 
					
						
							|  |  |  |  	dbg_memory_copy(0xfe, 0, 0, 255); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } |