| // Copyright 2022 The Pigweed Authors |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); you may not |
| // use this file except in compliance with the License. You may obtain a copy of |
| // the License at |
| // |
| // https://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| // WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| // License for the specific language governing permissions and limitations under |
| // the License. |
| #include "app_common/common.h" |
| #include "board.h" |
| #include "fsl_iopctl.h" |
| #include "pin_mux.h" |
| #include "pw_display_driver_mipi/display_driver.h" |
| #include "pw_mipi_dsi_mcuxpresso/device.h" |
| #include "pw_status/try.h" |
| |
| using pw::Status; |
| using pw::color::color_rgb565_t; |
| using pw::display::Display; |
| using pw::display_driver::DisplayDriverMipiDsi; |
| using pw::framebuffer::FramebufferRgb565; |
| using pw::mipi::dsi::MCUXpressoDevice; |
| |
| namespace { |
| |
| // Framebuffer addresses in on-board PSRAM. |
| constexpr uint32_t kBuffer0Addr = 0x28000000U; |
| constexpr uint32_t kBuffer1Addr = 0x28200000U; |
| constexpr video_pixel_format_t kPixelFormat = kVIDEO_PixelFormatRGB565; |
| constexpr uint16_t kBufferStrideBytes = |
| FRAMEBUFFER_WIDTH * pw::mipi::dsi::kBytesPerPixel; |
| constexpr pw::coordinates::Size<int> kDisplaySize = {LCD_WIDTH, LCD_HEIGHT}; |
| |
| const pw::framebuffer::pool::PoolData s_fb_pool_data = { |
| .fb_addr = |
| { |
| reinterpret_cast<pw::color::color_rgb565_t*>(kBuffer0Addr), |
| reinterpret_cast<pw::color::color_rgb565_t*>(kBuffer1Addr), |
| nullptr, |
| }, |
| .num_fb = 2, |
| .size = {LCD_WIDTH, LCD_HEIGHT}, |
| .row_bytes = kBufferStrideBytes, |
| .start = {FRAMEBUFFER_START_X, FRAMEBUFFER_START_Y}, |
| }; |
| |
| MCUXpressoDevice s_mipi_device(s_fb_pool_data, |
| {.width = LCD_WIDTH, .height = LCD_HEIGHT}, |
| kPixelFormat); |
| DisplayDriverMipiDsi s_display_driver( |
| { |
| .mipi_device = s_mipi_device, |
| }, |
| kDisplaySize); |
| Display s_display(s_display_driver, kDisplaySize); |
| |
| void InitMipiPins(void) { |
| constexpr uint32_t kPwmModeFunc = |
| (IOPCTL_PIO_FUNC0 | IOPCTL_PIO_PUPD_DI | IOPCTL_PIO_PULLDOWN_EN | |
| IOPCTL_PIO_INBUF_EN | IOPCTL_PIO_SLEW_RATE_NORMAL | |
| IOPCTL_PIO_FULLDRIVE_DI | IOPCTL_PIO_ANAMUX_DI | IOPCTL_PIO_PSEDRAIN_DI | |
| IOPCTL_PIO_INV_DI); |
| IOPCTL_PinMuxSet(IOPCTL, BOARD_MIPI_BL_PORT, BOARD_MIPI_BL_PIN, kPwmModeFunc); |
| |
| constexpr uint32_t kPwrEnModeFunc = |
| (IOPCTL_PIO_FUNC0 | IOPCTL_PIO_PUPD_DI | IOPCTL_PIO_PULLDOWN_EN | |
| IOPCTL_PIO_INBUF_EN | IOPCTL_PIO_SLEW_RATE_NORMAL | |
| IOPCTL_PIO_FULLDRIVE_DI | IOPCTL_PIO_ANAMUX_DI | IOPCTL_PIO_PSEDRAIN_DI | |
| IOPCTL_PIO_INV_DI); |
| IOPCTL_PinMuxSet( |
| IOPCTL, BOARD_MIPI_POWER_PORT, BOARD_MIPI_POWER_PIN, kPwrEnModeFunc); |
| |
| constexpr uint32_t kPort2Pin18ModeFunc = |
| (IOPCTL_PIO_FUNC0 | IOPCTL_PIO_PUPD_EN | IOPCTL_PIO_PULLDOWN_EN | |
| IOPCTL_PIO_INBUF_EN | IOPCTL_PIO_SLEW_RATE_NORMAL | |
| IOPCTL_PIO_FULLDRIVE_DI | IOPCTL_PIO_ANAMUX_DI | IOPCTL_PIO_PSEDRAIN_DI | |
| IOPCTL_PIO_INV_DI); |
| IOPCTL_PinMuxSet(IOPCTL, 3U, 18U, kPort2Pin18ModeFunc); |
| |
| constexpr uint32_t kResetBModeFunc = |
| (IOPCTL_PIO_FUNC0 | IOPCTL_PIO_PUPD_DI | IOPCTL_PIO_PULLDOWN_EN | |
| IOPCTL_PIO_INBUF_EN | IOPCTL_PIO_SLEW_RATE_NORMAL | |
| IOPCTL_PIO_FULLDRIVE_DI | IOPCTL_PIO_ANAMUX_DI | IOPCTL_PIO_PSEDRAIN_DI | |
| IOPCTL_PIO_INV_DI); |
| IOPCTL_PinMuxSet( |
| IOPCTL, BOARD_MIPI_RST_PORT, BOARD_MIPI_RST_PIN, kResetBModeFunc); |
| } |
| |
| } // namespace |
| |
| // static |
| Status Common::Init() { |
| InitMipiPins(); |
| BOARD_InitPsRam(); |
| |
| GPIO_PortInit(GPIO, BOARD_MIPI_POWER_PORT); |
| GPIO_PortInit(GPIO, BOARD_MIPI_BL_PORT); |
| GPIO_PortInit(GPIO, BOARD_MIPI_RST_PORT); |
| GPIO_PortInit(GPIO, BOARD_MIPI_TE_PORT); |
| |
| BOARD_BootClockRUN(); |
| |
| PW_TRY(s_mipi_device.Init()); |
| return s_display_driver.Init(); |
| } |
| |
| // static |
| pw::display::Display& Common::GetDisplay() { return s_display; } |