2021-03-11 21:08:52 +03:00
|
|
|
/*
|
|
|
|
* Nuvoton NPCM7xx MFT Module
|
|
|
|
*
|
|
|
|
* Copyright 2021 Google LLC
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License as published by the
|
|
|
|
* Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
|
|
* for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "qemu/osdep.h"
|
|
|
|
#include "hw/irq.h"
|
|
|
|
#include "hw/qdev-clock.h"
|
|
|
|
#include "hw/qdev-properties.h"
|
|
|
|
#include "hw/misc/npcm7xx_mft.h"
|
|
|
|
#include "hw/misc/npcm7xx_pwm.h"
|
|
|
|
#include "hw/registerfields.h"
|
|
|
|
#include "migration/vmstate.h"
|
|
|
|
#include "qapi/error.h"
|
|
|
|
#include "qapi/visitor.h"
|
|
|
|
#include "qemu/bitops.h"
|
|
|
|
#include "qemu/error-report.h"
|
|
|
|
#include "qemu/log.h"
|
|
|
|
#include "qemu/module.h"
|
|
|
|
#include "qemu/timer.h"
|
|
|
|
#include "qemu/units.h"
|
|
|
|
#include "trace.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Some of the registers can only accessed via 16-bit ops and some can only
|
|
|
|
* be accessed via 8-bit ops. However we mark all of them using REG16 to
|
|
|
|
* simplify implementation. npcm7xx_mft_check_mem_op checks the access length
|
|
|
|
* of memory operations.
|
|
|
|
*/
|
|
|
|
REG16(NPCM7XX_MFT_CNT1, 0x00);
|
|
|
|
REG16(NPCM7XX_MFT_CRA, 0x02);
|
|
|
|
REG16(NPCM7XX_MFT_CRB, 0x04);
|
|
|
|
REG16(NPCM7XX_MFT_CNT2, 0x06);
|
|
|
|
REG16(NPCM7XX_MFT_PRSC, 0x08);
|
|
|
|
REG16(NPCM7XX_MFT_CKC, 0x0a);
|
|
|
|
REG16(NPCM7XX_MFT_MCTRL, 0x0c);
|
|
|
|
REG16(NPCM7XX_MFT_ICTRL, 0x0e);
|
|
|
|
REG16(NPCM7XX_MFT_ICLR, 0x10);
|
|
|
|
REG16(NPCM7XX_MFT_IEN, 0x12);
|
|
|
|
REG16(NPCM7XX_MFT_CPA, 0x14);
|
|
|
|
REG16(NPCM7XX_MFT_CPB, 0x16);
|
|
|
|
REG16(NPCM7XX_MFT_CPCFG, 0x18);
|
|
|
|
REG16(NPCM7XX_MFT_INASEL, 0x1a);
|
|
|
|
REG16(NPCM7XX_MFT_INBSEL, 0x1c);
|
|
|
|
|
|
|
|
/* Register Fields */
|
|
|
|
#define NPCM7XX_MFT_CKC_C2CSEL BIT(3)
|
|
|
|
#define NPCM7XX_MFT_CKC_C1CSEL BIT(0)
|
|
|
|
|
|
|
|
#define NPCM7XX_MFT_MCTRL_TBEN BIT(6)
|
|
|
|
#define NPCM7XX_MFT_MCTRL_TAEN BIT(5)
|
|
|
|
#define NPCM7XX_MFT_MCTRL_TBEDG BIT(4)
|
|
|
|
#define NPCM7XX_MFT_MCTRL_TAEDG BIT(3)
|
|
|
|
#define NPCM7XX_MFT_MCTRL_MODE5 BIT(2)
|
|
|
|
|
|
|
|
#define NPCM7XX_MFT_ICTRL_TFPND BIT(5)
|
|
|
|
#define NPCM7XX_MFT_ICTRL_TEPND BIT(4)
|
|
|
|
#define NPCM7XX_MFT_ICTRL_TDPND BIT(3)
|
|
|
|
#define NPCM7XX_MFT_ICTRL_TCPND BIT(2)
|
|
|
|
#define NPCM7XX_MFT_ICTRL_TBPND BIT(1)
|
|
|
|
#define NPCM7XX_MFT_ICTRL_TAPND BIT(0)
|
|
|
|
|
|
|
|
#define NPCM7XX_MFT_ICLR_TFCLR BIT(5)
|
|
|
|
#define NPCM7XX_MFT_ICLR_TECLR BIT(4)
|
|
|
|
#define NPCM7XX_MFT_ICLR_TDCLR BIT(3)
|
|
|
|
#define NPCM7XX_MFT_ICLR_TCCLR BIT(2)
|
|
|
|
#define NPCM7XX_MFT_ICLR_TBCLR BIT(1)
|
|
|
|
#define NPCM7XX_MFT_ICLR_TACLR BIT(0)
|
|
|
|
|
|
|
|
#define NPCM7XX_MFT_IEN_TFIEN BIT(5)
|
|
|
|
#define NPCM7XX_MFT_IEN_TEIEN BIT(4)
|
|
|
|
#define NPCM7XX_MFT_IEN_TDIEN BIT(3)
|
|
|
|
#define NPCM7XX_MFT_IEN_TCIEN BIT(2)
|
|
|
|
#define NPCM7XX_MFT_IEN_TBIEN BIT(1)
|
|
|
|
#define NPCM7XX_MFT_IEN_TAIEN BIT(0)
|
|
|
|
|
|
|
|
#define NPCM7XX_MFT_CPCFG_GET_B(rv) extract8((rv), 4, 4)
|
|
|
|
#define NPCM7XX_MFT_CPCFG_GET_A(rv) extract8((rv), 0, 4)
|
|
|
|
#define NPCM7XX_MFT_CPCFG_HIEN BIT(3)
|
|
|
|
#define NPCM7XX_MFT_CPCFG_EQEN BIT(2)
|
|
|
|
#define NPCM7XX_MFT_CPCFG_LOEN BIT(1)
|
|
|
|
#define NPCM7XX_MFT_CPCFG_CPSEL BIT(0)
|
|
|
|
|
|
|
|
#define NPCM7XX_MFT_INASEL_SELA BIT(0)
|
|
|
|
#define NPCM7XX_MFT_INBSEL_SELB BIT(0)
|
|
|
|
|
|
|
|
/* Max CNT values of the module. The CNT value is a countdown from it. */
|
|
|
|
#define NPCM7XX_MFT_MAX_CNT 0xFFFF
|
|
|
|
|
|
|
|
/* Each fan revolution should generated 2 pulses */
|
|
|
|
#define NPCM7XX_MFT_PULSE_PER_REVOLUTION 2
|
|
|
|
|
|
|
|
typedef enum NPCM7xxMFTCaptureState {
|
|
|
|
/* capture succeeded with a valid CNT value. */
|
|
|
|
NPCM7XX_CAPTURE_SUCCEED,
|
|
|
|
/* capture stopped prematurely due to reaching CPCFG condition. */
|
|
|
|
NPCM7XX_CAPTURE_COMPARE_HIT,
|
|
|
|
/* capture fails since it reaches underflow condition for CNT. */
|
|
|
|
NPCM7XX_CAPTURE_UNDERFLOW,
|
|
|
|
} NPCM7xxMFTCaptureState;
|
|
|
|
|
|
|
|
static void npcm7xx_mft_reset(NPCM7xxMFTState *s)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* Only registers PRSC ~ INBSEL need to be reset. */
|
|
|
|
for (i = R_NPCM7XX_MFT_PRSC; i <= R_NPCM7XX_MFT_INBSEL; ++i) {
|
|
|
|
s->regs[i] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void npcm7xx_mft_clear_interrupt(NPCM7xxMFTState *s, uint8_t iclr)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Clear bits in ICTRL where corresponding bits in iclr is 1.
|
|
|
|
* Both iclr and ictrl are 8-bit regs. (See npcm7xx_mft_check_mem_op)
|
|
|
|
*/
|
|
|
|
s->regs[R_NPCM7XX_MFT_ICTRL] &= ~iclr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the CPCFG's condition should be triggered during count down from
|
|
|
|
* NPCM7XX_MFT_MAX_CNT to src if compared to tgt, return the count when
|
|
|
|
* the condition is triggered.
|
|
|
|
* Otherwise return -1.
|
|
|
|
* Since tgt is uint16_t it must always <= NPCM7XX_MFT_MAX_CNT.
|
|
|
|
*/
|
|
|
|
static int npcm7xx_mft_compare(int32_t src, uint16_t tgt, uint8_t cpcfg)
|
|
|
|
{
|
|
|
|
if (cpcfg & NPCM7XX_MFT_CPCFG_HIEN) {
|
|
|
|
return NPCM7XX_MFT_MAX_CNT;
|
|
|
|
}
|
|
|
|
if ((cpcfg & NPCM7XX_MFT_CPCFG_EQEN) && (src <= tgt)) {
|
|
|
|
return tgt;
|
|
|
|
}
|
|
|
|
if ((cpcfg & NPCM7XX_MFT_CPCFG_LOEN) && (tgt > 0) && (src < tgt)) {
|
|
|
|
return tgt - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Compute CNT according to corresponding fan's RPM. */
|
|
|
|
static NPCM7xxMFTCaptureState npcm7xx_mft_compute_cnt(
|
|
|
|
Clock *clock, uint32_t max_rpm, uint32_t duty, uint16_t tgt,
|
|
|
|
uint8_t cpcfg, uint16_t *cnt)
|
|
|
|
{
|
|
|
|
uint32_t rpm = (uint64_t)max_rpm * (uint64_t)duty / NPCM7XX_PWM_MAX_DUTY;
|
|
|
|
int32_t count;
|
|
|
|
int stopped;
|
|
|
|
NPCM7xxMFTCaptureState state;
|
|
|
|
|
|
|
|
if (rpm == 0) {
|
|
|
|
/*
|
|
|
|
* If RPM = 0, capture won't happen. CNT will continue count down.
|
|
|
|
* So it's effective equivalent to have a cnt > NPCM7XX_MFT_MAX_CNT
|
|
|
|
*/
|
|
|
|
count = NPCM7XX_MFT_MAX_CNT + 1;
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* RPM = revolution/min. The time for one revlution (in ns) is
|
|
|
|
* MINUTE_TO_NANOSECOND / RPM.
|
|
|
|
*/
|
|
|
|
count = clock_ns_to_ticks(clock, (60 * NANOSECONDS_PER_SECOND) /
|
|
|
|
(rpm * NPCM7XX_MFT_PULSE_PER_REVOLUTION));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count > NPCM7XX_MFT_MAX_CNT) {
|
|
|
|
count = -1;
|
|
|
|
} else {
|
|
|
|
/* The CNT is a countdown value from NPCM7XX_MFT_MAX_CNT. */
|
|
|
|
count = NPCM7XX_MFT_MAX_CNT - count;
|
|
|
|
}
|
|
|
|
stopped = npcm7xx_mft_compare(count, tgt, cpcfg);
|
|
|
|
if (stopped == -1) {
|
|
|
|
if (count == -1) {
|
|
|
|
/* Underflow */
|
|
|
|
state = NPCM7XX_CAPTURE_UNDERFLOW;
|
|
|
|
} else {
|
|
|
|
state = NPCM7XX_CAPTURE_SUCCEED;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
count = stopped;
|
|
|
|
state = NPCM7XX_CAPTURE_COMPARE_HIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count != -1) {
|
|
|
|
*cnt = count;
|
|
|
|
}
|
|
|
|
trace_npcm7xx_mft_rpm(clock->canonical_path, clock_get_hz(clock),
|
|
|
|
state, count, rpm, duty);
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Capture Fan RPM and update CNT and CR registers accordingly.
|
|
|
|
* Raise IRQ if certain contidions are met in IEN.
|
|
|
|
*/
|
|
|
|
static void npcm7xx_mft_capture(NPCM7xxMFTState *s)
|
|
|
|
{
|
|
|
|
int irq_level = 0;
|
|
|
|
NPCM7xxMFTCaptureState state;
|
|
|
|
int sel;
|
|
|
|
uint8_t cpcfg;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If not mode 5, the behavior is undefined. We just do nothing in this
|
|
|
|
* case.
|
|
|
|
*/
|
|
|
|
if (!(s->regs[R_NPCM7XX_MFT_MCTRL] & NPCM7XX_MFT_MCTRL_MODE5)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Capture input A. */
|
|
|
|
if (s->regs[R_NPCM7XX_MFT_MCTRL] & NPCM7XX_MFT_MCTRL_TAEN &&
|
|
|
|
s->regs[R_NPCM7XX_MFT_CKC] & NPCM7XX_MFT_CKC_C1CSEL) {
|
|
|
|
sel = s->regs[R_NPCM7XX_MFT_INASEL] & NPCM7XX_MFT_INASEL_SELA;
|
|
|
|
cpcfg = NPCM7XX_MFT_CPCFG_GET_A(s->regs[R_NPCM7XX_MFT_CPCFG]);
|
|
|
|
state = npcm7xx_mft_compute_cnt(s->clock_1,
|
|
|
|
sel ? s->max_rpm[2] : s->max_rpm[0],
|
|
|
|
sel ? s->duty[2] : s->duty[0],
|
|
|
|
s->regs[R_NPCM7XX_MFT_CPA],
|
|
|
|
cpcfg,
|
|
|
|
&s->regs[R_NPCM7XX_MFT_CNT1]);
|
|
|
|
switch (state) {
|
|
|
|
case NPCM7XX_CAPTURE_SUCCEED:
|
|
|
|
/* Interrupt on input capture on TAn transition - TAPND */
|
|
|
|
s->regs[R_NPCM7XX_MFT_CRA] = s->regs[R_NPCM7XX_MFT_CNT1];
|
|
|
|
s->regs[R_NPCM7XX_MFT_ICTRL] |= NPCM7XX_MFT_ICTRL_TAPND;
|
|
|
|
if (s->regs[R_NPCM7XX_MFT_IEN] & NPCM7XX_MFT_IEN_TAIEN) {
|
|
|
|
irq_level = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NPCM7XX_CAPTURE_COMPARE_HIT:
|
|
|
|
/* Compare Hit - TEPND */
|
|
|
|
s->regs[R_NPCM7XX_MFT_ICTRL] |= NPCM7XX_MFT_ICTRL_TEPND;
|
|
|
|
if (s->regs[R_NPCM7XX_MFT_IEN] & NPCM7XX_MFT_IEN_TEIEN) {
|
|
|
|
irq_level = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NPCM7XX_CAPTURE_UNDERFLOW:
|
|
|
|
/* Underflow - TCPND */
|
|
|
|
s->regs[R_NPCM7XX_MFT_ICTRL] |= NPCM7XX_MFT_ICTRL_TCPND;
|
|
|
|
if (s->regs[R_NPCM7XX_MFT_IEN] & NPCM7XX_MFT_IEN_TCIEN) {
|
|
|
|
irq_level = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
g_assert_not_reached();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Capture input B. */
|
|
|
|
if (s->regs[R_NPCM7XX_MFT_MCTRL] & NPCM7XX_MFT_MCTRL_TBEN &&
|
|
|
|
s->regs[R_NPCM7XX_MFT_CKC] & NPCM7XX_MFT_CKC_C2CSEL) {
|
|
|
|
sel = s->regs[R_NPCM7XX_MFT_INBSEL] & NPCM7XX_MFT_INBSEL_SELB;
|
|
|
|
cpcfg = NPCM7XX_MFT_CPCFG_GET_B(s->regs[R_NPCM7XX_MFT_CPCFG]);
|
|
|
|
state = npcm7xx_mft_compute_cnt(s->clock_2,
|
|
|
|
sel ? s->max_rpm[3] : s->max_rpm[1],
|
|
|
|
sel ? s->duty[3] : s->duty[1],
|
|
|
|
s->regs[R_NPCM7XX_MFT_CPB],
|
|
|
|
cpcfg,
|
|
|
|
&s->regs[R_NPCM7XX_MFT_CNT2]);
|
|
|
|
switch (state) {
|
|
|
|
case NPCM7XX_CAPTURE_SUCCEED:
|
|
|
|
/* Interrupt on input capture on TBn transition - TBPND */
|
|
|
|
s->regs[R_NPCM7XX_MFT_CRB] = s->regs[R_NPCM7XX_MFT_CNT2];
|
|
|
|
s->regs[R_NPCM7XX_MFT_ICTRL] |= NPCM7XX_MFT_ICTRL_TBPND;
|
|
|
|
if (s->regs[R_NPCM7XX_MFT_IEN] & NPCM7XX_MFT_IEN_TBIEN) {
|
|
|
|
irq_level = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NPCM7XX_CAPTURE_COMPARE_HIT:
|
|
|
|
/* Compare Hit - TFPND */
|
|
|
|
s->regs[R_NPCM7XX_MFT_ICTRL] |= NPCM7XX_MFT_ICTRL_TFPND;
|
|
|
|
if (s->regs[R_NPCM7XX_MFT_IEN] & NPCM7XX_MFT_IEN_TFIEN) {
|
|
|
|
irq_level = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NPCM7XX_CAPTURE_UNDERFLOW:
|
|
|
|
/* Underflow - TDPND */
|
|
|
|
s->regs[R_NPCM7XX_MFT_ICTRL] |= NPCM7XX_MFT_ICTRL_TDPND;
|
|
|
|
if (s->regs[R_NPCM7XX_MFT_IEN] & NPCM7XX_MFT_IEN_TDIEN) {
|
|
|
|
irq_level = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
g_assert_not_reached();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
trace_npcm7xx_mft_capture(DEVICE(s)->canonical_path, irq_level);
|
|
|
|
qemu_set_irq(s->irq, irq_level);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Update clock for counters. */
|
|
|
|
static void npcm7xx_mft_update_clock(void *opaque, ClockEvent event)
|
|
|
|
{
|
|
|
|
NPCM7xxMFTState *s = NPCM7XX_MFT(opaque);
|
|
|
|
uint64_t prescaled_clock_period;
|
|
|
|
|
|
|
|
prescaled_clock_period = clock_get(s->clock_in) *
|
|
|
|
(s->regs[R_NPCM7XX_MFT_PRSC] + 1ULL);
|
|
|
|
trace_npcm7xx_mft_update_clock(s->clock_in->canonical_path,
|
|
|
|
s->regs[R_NPCM7XX_MFT_CKC],
|
|
|
|
clock_get(s->clock_in),
|
|
|
|
prescaled_clock_period);
|
|
|
|
/* Update clock 1 */
|
|
|
|
if (s->regs[R_NPCM7XX_MFT_CKC] & NPCM7XX_MFT_CKC_C1CSEL) {
|
|
|
|
/* Clock is prescaled. */
|
|
|
|
clock_update(s->clock_1, prescaled_clock_period);
|
|
|
|
} else {
|
|
|
|
/* Clock stopped. */
|
|
|
|
clock_update(s->clock_1, 0);
|
|
|
|
}
|
|
|
|
/* Update clock 2 */
|
|
|
|
if (s->regs[R_NPCM7XX_MFT_CKC] & NPCM7XX_MFT_CKC_C2CSEL) {
|
|
|
|
/* Clock is prescaled. */
|
|
|
|
clock_update(s->clock_2, prescaled_clock_period);
|
|
|
|
} else {
|
|
|
|
/* Clock stopped. */
|
|
|
|
clock_update(s->clock_2, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
npcm7xx_mft_capture(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint64_t npcm7xx_mft_read(void *opaque, hwaddr offset, unsigned size)
|
|
|
|
{
|
|
|
|
NPCM7xxMFTState *s = NPCM7XX_MFT(opaque);
|
|
|
|
uint16_t value = 0;
|
|
|
|
|
|
|
|
switch (offset) {
|
|
|
|
case A_NPCM7XX_MFT_ICLR:
|
|
|
|
qemu_log_mask(LOG_GUEST_ERROR,
|
|
|
|
"%s: register @ 0x%04" HWADDR_PRIx " is write-only\n",
|
|
|
|
__func__, offset);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
value = s->regs[offset / 2];
|
|
|
|
}
|
|
|
|
|
|
|
|
trace_npcm7xx_mft_read(DEVICE(s)->canonical_path, offset, value);
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void npcm7xx_mft_write(void *opaque, hwaddr offset,
|
|
|
|
uint64_t v, unsigned size)
|
|
|
|
{
|
|
|
|
NPCM7xxMFTState *s = NPCM7XX_MFT(opaque);
|
|
|
|
|
|
|
|
trace_npcm7xx_mft_write(DEVICE(s)->canonical_path, offset, v);
|
|
|
|
switch (offset) {
|
|
|
|
case A_NPCM7XX_MFT_ICLR:
|
|
|
|
npcm7xx_mft_clear_interrupt(s, v);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case A_NPCM7XX_MFT_CKC:
|
|
|
|
case A_NPCM7XX_MFT_PRSC:
|
|
|
|
s->regs[offset / 2] = v;
|
|
|
|
npcm7xx_mft_update_clock(s, ClockUpdate);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
s->regs[offset / 2] = v;
|
|
|
|
npcm7xx_mft_capture(s);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool npcm7xx_mft_check_mem_op(void *opaque, hwaddr offset,
|
|
|
|
unsigned size, bool is_write,
|
|
|
|
MemTxAttrs attrs)
|
|
|
|
{
|
|
|
|
switch (offset) {
|
|
|
|
/* 16-bit registers. Must be accessed with 16-bit read/write.*/
|
|
|
|
case A_NPCM7XX_MFT_CNT1:
|
|
|
|
case A_NPCM7XX_MFT_CRA:
|
|
|
|
case A_NPCM7XX_MFT_CRB:
|
|
|
|
case A_NPCM7XX_MFT_CNT2:
|
|
|
|
case A_NPCM7XX_MFT_CPA:
|
|
|
|
case A_NPCM7XX_MFT_CPB:
|
|
|
|
return size == 2;
|
|
|
|
|
|
|
|
/* 8-bit registers. Must be accessed with 8-bit read/write.*/
|
|
|
|
case A_NPCM7XX_MFT_PRSC:
|
|
|
|
case A_NPCM7XX_MFT_CKC:
|
|
|
|
case A_NPCM7XX_MFT_MCTRL:
|
|
|
|
case A_NPCM7XX_MFT_ICTRL:
|
|
|
|
case A_NPCM7XX_MFT_ICLR:
|
|
|
|
case A_NPCM7XX_MFT_IEN:
|
|
|
|
case A_NPCM7XX_MFT_CPCFG:
|
|
|
|
case A_NPCM7XX_MFT_INASEL:
|
|
|
|
case A_NPCM7XX_MFT_INBSEL:
|
|
|
|
return size == 1;
|
|
|
|
|
|
|
|
default:
|
|
|
|
/* Invalid registers. */
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void npcm7xx_mft_get_max_rpm(Object *obj, Visitor *v, const char *name,
|
|
|
|
void *opaque, Error **errp)
|
|
|
|
{
|
|
|
|
visit_type_uint32(v, name, (uint32_t *)opaque, errp);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void npcm7xx_mft_set_max_rpm(Object *obj, Visitor *v, const char *name,
|
|
|
|
void *opaque, Error **errp)
|
|
|
|
{
|
|
|
|
NPCM7xxMFTState *s = NPCM7XX_MFT(obj);
|
|
|
|
uint32_t *max_rpm = opaque;
|
|
|
|
uint32_t value;
|
|
|
|
|
|
|
|
if (!visit_type_uint32(v, name, &value, errp)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
*max_rpm = value;
|
|
|
|
npcm7xx_mft_capture(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void npcm7xx_mft_duty_handler(void *opaque, int n, int value)
|
|
|
|
{
|
|
|
|
NPCM7xxMFTState *s = NPCM7XX_MFT(opaque);
|
|
|
|
|
|
|
|
trace_npcm7xx_mft_set_duty(DEVICE(s)->canonical_path, n, value);
|
|
|
|
s->duty[n] = value;
|
|
|
|
npcm7xx_mft_capture(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct MemoryRegionOps npcm7xx_mft_ops = {
|
|
|
|
.read = npcm7xx_mft_read,
|
|
|
|
.write = npcm7xx_mft_write,
|
|
|
|
.endianness = DEVICE_LITTLE_ENDIAN,
|
|
|
|
.valid = {
|
|
|
|
.min_access_size = 1,
|
|
|
|
.max_access_size = 2,
|
|
|
|
.unaligned = false,
|
|
|
|
.accepts = npcm7xx_mft_check_mem_op,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
static void npcm7xx_mft_enter_reset(Object *obj, ResetType type)
|
|
|
|
{
|
|
|
|
NPCM7xxMFTState *s = NPCM7XX_MFT(obj);
|
|
|
|
|
|
|
|
npcm7xx_mft_reset(s);
|
|
|
|
}
|
|
|
|
|
2024-04-12 19:08:07 +03:00
|
|
|
static void npcm7xx_mft_hold_reset(Object *obj, ResetType type)
|
2021-03-11 21:08:52 +03:00
|
|
|
{
|
|
|
|
NPCM7xxMFTState *s = NPCM7XX_MFT(obj);
|
|
|
|
|
|
|
|
qemu_irq_lower(s->irq);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void npcm7xx_mft_init(Object *obj)
|
|
|
|
{
|
|
|
|
NPCM7xxMFTState *s = NPCM7XX_MFT(obj);
|
|
|
|
SysBusDevice *sbd = SYS_BUS_DEVICE(obj);
|
|
|
|
DeviceState *dev = DEVICE(obj);
|
|
|
|
|
|
|
|
memory_region_init_io(&s->iomem, obj, &npcm7xx_mft_ops, s,
|
|
|
|
TYPE_NPCM7XX_MFT, 4 * KiB);
|
|
|
|
sysbus_init_mmio(sbd, &s->iomem);
|
|
|
|
sysbus_init_irq(sbd, &s->irq);
|
|
|
|
s->clock_in = qdev_init_clock_in(dev, "clock-in", npcm7xx_mft_update_clock,
|
|
|
|
s, ClockUpdate);
|
|
|
|
s->clock_1 = qdev_init_clock_out(dev, "clock1");
|
|
|
|
s->clock_2 = qdev_init_clock_out(dev, "clock2");
|
|
|
|
|
|
|
|
for (int i = 0; i < NPCM7XX_PWM_PER_MODULE; ++i) {
|
|
|
|
object_property_add(obj, "max_rpm[*]", "uint32",
|
|
|
|
npcm7xx_mft_get_max_rpm,
|
|
|
|
npcm7xx_mft_set_max_rpm,
|
|
|
|
NULL, &s->max_rpm[i]);
|
|
|
|
}
|
|
|
|
qdev_init_gpio_in_named(dev, npcm7xx_mft_duty_handler, "duty",
|
|
|
|
NPCM7XX_MFT_FANIN_COUNT);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateDescription vmstate_npcm7xx_mft = {
|
|
|
|
.name = "npcm7xx-mft-module",
|
|
|
|
.version_id = 0,
|
|
|
|
.minimum_version_id = 0,
|
2023-12-21 06:16:21 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2021-03-11 21:08:52 +03:00
|
|
|
VMSTATE_CLOCK(clock_in, NPCM7xxMFTState),
|
|
|
|
VMSTATE_CLOCK(clock_1, NPCM7xxMFTState),
|
|
|
|
VMSTATE_CLOCK(clock_2, NPCM7xxMFTState),
|
|
|
|
VMSTATE_UINT16_ARRAY(regs, NPCM7xxMFTState, NPCM7XX_MFT_NR_REGS),
|
|
|
|
VMSTATE_UINT32_ARRAY(max_rpm, NPCM7xxMFTState, NPCM7XX_MFT_FANIN_COUNT),
|
|
|
|
VMSTATE_UINT32_ARRAY(duty, NPCM7xxMFTState, NPCM7XX_MFT_FANIN_COUNT),
|
|
|
|
VMSTATE_END_OF_LIST(),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
static void npcm7xx_mft_class_init(ObjectClass *klass, void *data)
|
|
|
|
{
|
|
|
|
ResettableClass *rc = RESETTABLE_CLASS(klass);
|
|
|
|
DeviceClass *dc = DEVICE_CLASS(klass);
|
|
|
|
|
|
|
|
dc->desc = "NPCM7xx MFT Controller";
|
|
|
|
dc->vmsd = &vmstate_npcm7xx_mft;
|
|
|
|
rc->phases.enter = npcm7xx_mft_enter_reset;
|
|
|
|
rc->phases.hold = npcm7xx_mft_hold_reset;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const TypeInfo npcm7xx_mft_info = {
|
|
|
|
.name = TYPE_NPCM7XX_MFT,
|
|
|
|
.parent = TYPE_SYS_BUS_DEVICE,
|
|
|
|
.instance_size = sizeof(NPCM7xxMFTState),
|
|
|
|
.class_init = npcm7xx_mft_class_init,
|
|
|
|
.instance_init = npcm7xx_mft_init,
|
|
|
|
};
|
|
|
|
|
|
|
|
static void npcm7xx_mft_register_type(void)
|
|
|
|
{
|
|
|
|
type_register_static(&npcm7xx_mft_info);
|
|
|
|
}
|
|
|
|
type_init(npcm7xx_mft_register_type);
|