You need to enable JavaScript to run this app.
最新活动
大模型
产品
解决方案
定价
生态与合作
支持与服务
开发者
了解我们

ESP32-CAM图像保存至SD卡失败求助:SD卡挂载故障排查

解决ESP32-CAM SD卡挂载失败(Card Mount Failed)问题

我来帮你一步步排查这个问题,从硬件到代码逐一分析可能的原因和解决方案:

一、先检查硬件连接与SD卡本身

  • 接线与兼容性:确认ESP32-CAM和SD卡模块的接线是否正确(AI-Thinker模块默认SD_MMC引脚为GPIO12(D2)、GPIO13(D3)、GPIO14(CMD),电源必须接3.3V,严禁接5V)。如果是集成了SD卡槽的ESP32-CAM模块,检查卡槽是否接触良好,重新插拔SD卡试试。
  • SD卡规格:确保SD卡是32GB及以下的容量(超过32GB的FAT32格式化可能存在兼容性问题),并且是高速卡(Class 10以上最佳)。可以换一张已知能用的SD卡测试,排除卡本身的故障。
  • 电源供应:ESP32-CAM拍照时电流需求较高,USB供电可能不足导致SD卡供电异常。建议使用外接3.3V/500mA以上的电源模块供电。

二、代码层面的优化与调试

你的代码逻辑没问题,但可以通过调整SD_MMC初始化方式和增加调试信息来定位问题:

1. 修改SD卡初始化代码,增加调试信息

将原代码中的SD_MMC初始化部分替换为以下代码,这样能输出更详细的卡状态信息:

Serial.println("Starting SD Card");
// 尝试1线模式初始化(适配多数简易SD模块),如果不行可以把第二个参数改成false用4线模式
if(!SD_MMC.begin("/sdcard", true)){
    Serial.println("SD Card Mount Failed");
    // 打印详细卡类型信息,帮助排查
    uint8_t cardType = SD_MMC.cardType();
    Serial.print("Detected Card Type: ");
    switch(cardType){
        case CARD_NONE: Serial.println("No SD card attached"); break;
        case CARD_MMC: Serial.println("MMC card detected"); break;
        case CARD_SD: Serial.println("Standard SD card detected"); break;
        case CARD_SDHC: Serial.println("SDHC card detected"); break;
        default: Serial.println("Unknown card type");
    }
    return;
}

2. 添加延迟,给硬件上电缓冲时间

在相机初始化完成后,添加一点延迟再初始化SD卡,避免硬件还没准备好:

// Init Camera
esp_err_t err = esp_camera_init(&config);
if (err != ESP_OK) {
    Serial.printf("Camera init failed with error 0x%x", err);
    return;
}
delay(500); // 添加这行,给相机和SD卡模块上电缓冲时间

3. 检查GPIO12电平(关键!)

ESP32的GPIO12引脚电平会影响SD卡的识别模式,启动时如果GPIO12为高电平,可能导致SD卡挂载失败。对于AI-Thinker模块,GPIO12默认应该接下拉电阻,如果你的模块没有,建议手动添加一个10kΩ的下拉电阻到GND,确保启动时GPIO12为低电平。

三、完整修改后的代码参考

#include "esp_camera.h"
#include "Arduino.h"
#include "FS.h" // SD Card ESP32
#include "SD_MMC.h" // SD Card ESP32
#include "soc/soc.h" // Disable brownour problems
#include "soc/rtc_cntl_reg.h" // Disable brownour problems
#include "driver/rtc_io.h"
#include <EEPROM.h> // read and write from flash memory

// define the number of bytes you want to access
#define EEPROM_SIZE 1

// Pin definition for CAMERA_MODEL_AI_THINKER
#define PWDN_GPIO_NUM 32
#define RESET_GPIO_NUM -1
#define XCLK_GPIO_NUM 0
#define SIOD_GPIO_NUM 26
#define SIOC_GPIO_NUM 27
#define Y9_GPIO_NUM 35
#define Y8_GPIO_NUM 34
#define Y7_GPIO_NUM 39
#define Y6_GPIO_NUM 36
#define Y5_GPIO_NUM 21
#define Y4_GPIO_NUM 19
#define Y3_GPIO_NUM 18
#define Y2_GPIO_NUM 5
#define VSYNC_GPIO_NUM 25
#define HREF_GPIO_NUM 23
#define PCLK_GPIO_NUM 22

int pictureNumber = 0;

void setup() {
    WRITE_PERI_REG(RTC_CNTL_BROWN_OUT_REG, 0); //disable brownout detector
    Serial.begin(115200);
    delay(1000); // 给串口和硬件上电缓冲

    camera_config_t config;
    config.ledc_channel = LEDC_CHANNEL_0;
    config.ledc_timer = LEDC_TIMER_0;
    config.pin_d0 = Y2_GPIO_NUM;
    config.pin_d1 = Y3_GPIO_NUM;
    config.pin_d2 = Y4_GPIO_NUM;
    config.pin_d3 = Y5_GPIO_NUM;
    config.pin_d4 = Y6_GPIO_NUM;
    config.pin_d5 = Y7_GPIO_NUM;
    config.pin_d6 = Y8_GPIO_NUM;
    config.pin_d7 = Y9_GPIO_NUM;
    config.pin_xclk = XCLK_GPIO_NUM;
    config.pin_pclk = PCLK_GPIO_NUM;
    config.pin_vsync = VSYNC_GPIO_NUM;
    config.pin_href = HREF_GPIO_NUM;
    config.pin_sscb_sda = SIOD_GPIO_NUM;
    config.pin_sscb_scl = SIOC_GPIO_NUM;
    config.pin_pwdn = PWDN_GPIO_NUM;
    config.pin_reset = RESET_GPIO_NUM;
    config.xclk_freq_hz = 20000000;
    config.pixel_format = PIXFORMAT_JPEG;

    if(psramFound()){
        config.frame_size = FRAMESIZE_UXGA; // FRAMESIZE_ + QVGA|CIF|VGA|SVGA|XGA|SXGA|UXGA
        config.jpeg_quality = 10;
        config.fb_count = 2;
    } else {
        config.frame_size = FRAMESIZE_SVGA;
        config.jpeg_quality = 12;
        config.fb_count = 1;
    }

    // Init Camera
    esp_err_t err = esp_camera_init(&config);
    if (err != ESP_OK) {
        Serial.printf("Camera init failed with error 0x%x", err);
        return;
    }
    delay(500); // 相机初始化后缓冲

    Serial.println("Starting SD Card");
    // 尝试1线模式初始化,若失败可改为false用4线模式
    if(!SD_MMC.begin("/sdcard", true)){
        Serial.println("SD Card Mount Failed");
        uint8_t cardType = SD_MMC.cardType();
        Serial.print("Detected Card Type: ");
        switch(cardType){
            case CARD_NONE: Serial.println("No SD card attached"); break;
            case CARD_MMC: Serial.println("MMC card detected"); break;
            case CARD_SD: Serial.println("Standard SD card detected"); break;
            case CARD_SDHC: Serial.println("SDHC card detected"); break;
            default: Serial.println("Unknown card type");
        }
        return;
    }

    uint8_t cardType = SD_MMC.cardType();
    if(cardType == CARD_NONE){
        Serial.println("No SD Card attached");
        return;
    }

    camera_fb_t * fb = NULL;
    // Take Picture with Camera
    fb = esp_camera_fb_get();
    if(!fb) {
        Serial.println("Camera capture failed");
        return;
    }

    // initialize EEPROM with predefined size
    EEPROM.begin(EEPROM_SIZE);
    pictureNumber = EEPROM.read(0) + 1;

    // Path where new picture will be saved in SD Card
    String path = "/picture" + String(pictureNumber) +".jpg";
    fs::FS &fs = SD_MMC;
    Serial.printf("Picture file name: %s\n", path.c_str());

    File file = fs.open(path.c_str(), FILE_WRITE);
    if(!file){
        Serial.println("Failed to open file in writing mode");
    } else {
        file.write(fb->buf, fb->len); // payload (image), payload length
        Serial.printf("Saved file to path: %s\n", path.c_str());
        EEPROM.write(0, pictureNumber);
        EEPROM.commit();
    }
    file.close();
    esp_camera_fb_return(fb);

    // Turns off the ESP32-CAM white on-board LED (flash) connected to GPIO 4
    pinMode(4, INPUT);
    digitalWrite(4, LOW);
    rtc_gpio_hold_dis(GPIO_NUM_4);

    delay(2000);
    Serial.println("Going to sleep now");
    delay(2000);
    esp_deep_sleep_start();
    Serial.println("This will never be printed");
}

void loop() {
}

按照上面的步骤逐一排查,应该能解决SD卡挂载失败的问题。

内容的提问来源于stack exchange,提问作者Rey

火山引擎 最新活动