Files
VesiScan-Basic-firmware-test/project/ble_peripheral/ble_app_bladder_patch/pulse_gen.c
jhChun 4881c7f937 - Piezo 6ch 측정 + 센서(배터리, IMU, 온도) 측정: mbb 명령어 추가
- Flash Memory Piezo 측정 파라미터 추가

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-18 13:54:06 +09:00

183 lines
7.5 KiB
C

/*******************************************************************************
* @file pulse_gen.c
* @brief PWM Pulse Generator (Build-safe version)
* @author CandyPops Co. / ??? ????
* @version V1.0.1
* @date 2025-10-13
******************************************************************************/
/*******************************************************************************
* PWM 펄스 생성기
*
* === 현재 상태 ===
* SKIP_PWM 매크로가 정의되어 있어 PWM 기능이 비활성화됨.
* init/start/stop 함수는 모두 Stub으로,
* 로그만 출력하고 실제 PWM 동작은 하지 않는다.
* 빌드 호환성을 위해 함수 인터페이스만 유지.
*
* === 실제 구현 (SKIP_PWM 미정의 시) ===
* nrfx_pwm 드라이버 + app_timer를 사용하여 3핀(P28/P29/P30) PWM 제어.
* PWM 인스턴스 0, 16MHz 클럭, TOP=8, 개별 채널 로드 모드.
* 6단계 시퀀스로 채널별 듀티 사이클 조절.
* app_timer(10ms)로 주기적 반복 재생.
*
* === 비활성화 사유 ===
* 현재 프로젝트에서는 피에조 구동을 dr_piezo 드라이버가 담당하므로
* 이 PWM 펄스 생성기는 사용하지 않음.
******************************************************************************/
#define SKIP_PWM // ? ?? ???? PWM ?? ???
#include "pulse_gen.h"
#include "debug_print.h"
#include "app_timer.h"
#include "nrf_log.h"
#ifdef SKIP_PWM
/* ========================================================================== */
/* PWM 비활성화 모드: Stub 구현 (로그만 출력, 실제 동작 없음) */
/* ========================================================================== */
/* Stub 초기화: 아무 동작 없이 성공 반환 */
ret_code_t pulse_gen_init(void)
{
DBG_PRINTF("[PWM] skipped init\r\n");
return NRF_SUCCESS;
}
/* Stub 시작: 아무 동작 없음 */
void pulse_gen_start(void)
{
DBG_PRINTF("[PWM] skipped start\r\n");
}
/* Stub 정지: 아무 동작 없음 */
void pulse_gen_stop(void)
{
DBG_PRINTF("[PWM] skipped stop\r\n");
}
#else /* ==================================================================== */
/* 실제 PWM 구현 코드 (SKIP_PWM 미정의 시에만 컴파일됨) */
/* ========================================================================== */
#include "nrfx_pwm.h"
/* --- 설정값 --- */
#define PULSE_PIN_1 28 /* PWM 출력 핀 1 (P0.28) */
#define PULSE_PIN_2 29 /* PWM 출력 핀 2 (P0.29) */
#define PULSE_PIN_3 30 /* PWM 출력 핀 3 (P0.30) */
#define PULSE_DELAY_MS 10 /* 반복 재생 간격 (ms) */
#define PWM_INSTANCE_ID 0 /* PWM 하드웨어 인스턴스 번호 */
#define PWM_TOP_VALUE 8 /* PWM 카운터 최대값 (듀티 사이클 분해능) */
#define SEQUENCE_LENGTH 6 /* PWM 시퀀스 단계 수 */
static nrfx_pwm_t m_pwm = NRFX_PWM_INSTANCE(PWM_INSTANCE_ID); /* PWM 인스턴스 */
APP_TIMER_DEF(m_pulse_delay_timer_id); /* 반복 재생용 앱 타이머 */
static volatile bool m_pulses_running = false; /* 펄스 생성 동작 중 플래그 */
/* PWM 시퀀스 데이터: 6단계에 걸쳐 3채널의 듀티 사이클을 개별 설정 */
static nrf_pwm_values_individual_t m_pulse_seq_values[SEQUENCE_LENGTH];
static nrf_pwm_sequence_t const m_pulse_sequence =
{
.values.p_individual = m_pulse_seq_values,
.length = NRF_PWM_VALUES_LENGTH(m_pulse_seq_values),
.repeats = 0, /* 각 단계 반복 없음 */
.end_delay = 0 /* 시퀀스 종료 후 추가 지연 없음 */
};
/* -------------------------------------------------------------------------- */
/* 내부 핸들러 함수 */
/* -------------------------------------------------------------------------- */
/*
* PWM 시퀀스 값 초기화
* 6단계에 걸쳐 CH0/CH1/CH2의 듀티 사이클을 교대 설정:
* 단계0: CH0=100%, CH1=0, CH2=0
* 단계1: CH0=0, CH1=100%, CH2=0
* 단계2: CH0=100%, CH1=0, CH2=0
* 단계3: CH0=0, CH1=100%, CH2=0
* 단계4: CH0=0, CH1=0, CH2=0 (모두 OFF)
* 단계5: CH0=0, CH1=0, CH2=100%
*/
static void prepare_pulse_sequence(void)
{
m_pulse_seq_values[0].channel_0 = PWM_TOP_VALUE; m_pulse_seq_values[0].channel_1 = 0; m_pulse_seq_values[0].channel_2 = 0;
m_pulse_seq_values[1].channel_0 = 0; m_pulse_seq_values[1].channel_1 = PWM_TOP_VALUE; m_pulse_seq_values[1].channel_2 = 0;
m_pulse_seq_values[2].channel_0 = PWM_TOP_VALUE; m_pulse_seq_values[2].channel_1 = 0; m_pulse_seq_values[2].channel_2 = 0;
m_pulse_seq_values[3].channel_0 = 0; m_pulse_seq_values[3].channel_1 = PWM_TOP_VALUE; m_pulse_seq_values[3].channel_2 = 0;
m_pulse_seq_values[4].channel_0 = 0; m_pulse_seq_values[4].channel_1 = 0; m_pulse_seq_values[4].channel_2 = 0;
m_pulse_seq_values[5].channel_0 = 0; m_pulse_seq_values[5].channel_1 = 0; m_pulse_seq_values[5].channel_2 = PWM_TOP_VALUE;
}
/* 타이머 타임아웃 핸들러: m_pulses_running이면 PWM 시퀀스 1회 재생 후 타이머 재시작 */
static void pulse_delay_timeout_handler(void * p_context)
{
if (m_pulses_running)
{
(void)nrfx_pwm_simple_playback(&m_pwm, &m_pulse_sequence, 1, NRFX_PWM_FLAG_STOP);
(void)app_timer_start(m_pulse_delay_timer_id, APP_TIMER_TICKS(PULSE_DELAY_MS), NULL);
}
}
/* -------------------------------------------------------------------------- */
/* 공개 함수 */
/* -------------------------------------------------------------------------- */
/*
* PWM 초기화: PWM 인스턴스 설정 + 앱 타이머 생성
* PWM: 16MHz 클럭, UP 카운트 모드, TOP=8, 개별 채널 로드
* 타이머: 단발(SINGLE_SHOT) 모드, 10ms 후 시퀀스 재생
*/
ret_code_t pulse_gen_init(void)
{
ret_code_t err_code;
prepare_pulse_sequence();
nrfx_pwm_config_t const config =
{
.output_pins =
{
PULSE_PIN_1,
PULSE_PIN_2,
PULSE_PIN_3,
NRFX_PWM_PIN_NOT_USED
},
.irq_priority = NRFX_PWM_DEFAULT_CONFIG_IRQ_PRIORITY,
.base_clock = NRF_PWM_CLK_16MHz,
.count_mode = NRF_PWM_MODE_UP,
.top_value = PWM_TOP_VALUE,
.load_mode = NRF_PWM_LOAD_INDIVIDUAL,
.step_mode = NRF_PWM_STEP_AUTO
};
err_code = nrfx_pwm_init(&m_pwm, &config, NULL);
if (err_code != NRF_SUCCESS)
return err_code;
err_code = app_timer_create(&m_pulse_delay_timer_id,
APP_TIMER_MODE_SINGLE_SHOT,
pulse_delay_timeout_handler);
return err_code;
}
/* 펄스 생성 시작: 플래그 설정 후 타임아웃 핸들러를 직접 호출하여 즉시 첫 재생 */
void pulse_gen_start(void)
{
if (m_pulses_running) return; /* 이미 동작 중이면 무시 */
m_pulses_running = true;
pulse_delay_timeout_handler(NULL); /* 첫 번째 재생 즉시 시작 */
}
/* 펄스 생성 정지: 플래그 해제 + 타이머 정지 + PWM 즉시 정지 */
void pulse_gen_stop(void)
{
m_pulses_running = false;
app_timer_stop(m_pulse_delay_timer_id);
nrfx_pwm_stop(&m_pwm, true); /* true = 즉시 정지 (현재 시퀀스 중단) */
}
#endif /* SKIP_PWM */