541 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			541 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|   | /*
 | ||
|  |  * 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); | ||
|  | } | ||
|  | 
 | ||
|  | static void npcm7xx_mft_hold_reset(Object *obj) | ||
|  | { | ||
|  |     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, | ||
|  |     .fields = (VMStateField[]) { | ||
|  |         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); |