| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Renesas 8bit timer | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Datasheet: RX62N Group, RX621 Group User's Manual: Hardware | 
					
						
							|  |  |  |  *            (Rev.1.40 R01UH0033EJ0140) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (c) 2019 Yoshinori Sato | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * SPDX-License-Identifier: GPL-2.0-or-later | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or modify it | 
					
						
							|  |  |  |  * under the terms and conditions of the GNU General Public License, | 
					
						
							|  |  |  |  * version 2 or later, as published by the Free Software Foundation. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope 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. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License along with | 
					
						
							|  |  |  |  * this program.  If not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "qemu/osdep.h"
 | 
					
						
							|  |  |  | #include "qemu/log.h"
 | 
					
						
							|  |  |  | #include "hw/irq.h"
 | 
					
						
							|  |  |  | #include "hw/registerfields.h"
 | 
					
						
							|  |  |  | #include "hw/qdev-properties.h"
 | 
					
						
							|  |  |  | #include "hw/timer/renesas_tmr.h"
 | 
					
						
							|  |  |  | #include "migration/vmstate.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | REG8(TCR, 0) | 
					
						
							|  |  |  |   FIELD(TCR, CCLR,  3, 2) | 
					
						
							|  |  |  |   FIELD(TCR, OVIE,  5, 1) | 
					
						
							|  |  |  |   FIELD(TCR, CMIEA, 6, 1) | 
					
						
							|  |  |  |   FIELD(TCR, CMIEB, 7, 1) | 
					
						
							|  |  |  | REG8(TCSR, 2) | 
					
						
							|  |  |  |   FIELD(TCSR, OSA,  0, 2) | 
					
						
							|  |  |  |   FIELD(TCSR, OSB,  2, 2) | 
					
						
							|  |  |  |   FIELD(TCSR, ADTE, 4, 2) | 
					
						
							|  |  |  | REG8(TCORA, 4) | 
					
						
							|  |  |  | REG8(TCORB, 6) | 
					
						
							|  |  |  | REG8(TCNT, 8) | 
					
						
							|  |  |  | REG8(TCCR, 10) | 
					
						
							|  |  |  |   FIELD(TCCR, CKS,   0, 3) | 
					
						
							|  |  |  |   FIELD(TCCR, CSS,   3, 2) | 
					
						
							|  |  |  |   FIELD(TCCR, TMRIS, 7, 1) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-19 22:32:40 +00:00
										 |  |  | #define CSS_EXTERNAL  0x00
 | 
					
						
							| 
									
										
										
										
											2021-02-19 22:32:39 +00:00
										 |  |  | #define CSS_INTERNAL  0x01
 | 
					
						
							| 
									
										
										
										
											2021-02-19 22:32:40 +00:00
										 |  |  | #define CSS_INVALID   0x02
 | 
					
						
							| 
									
										
										
										
											2021-02-19 22:32:39 +00:00
										 |  |  | #define CSS_CASCADING 0x03
 | 
					
						
							| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  | #define CCLR_A    0x01
 | 
					
						
							|  |  |  | #define CCLR_B    0x02
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const int clkdiv[] = {0, 1, 2, 8, 32, 64, 1024, 8192}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint8_t concat_reg(uint8_t *reg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return (reg[0] << 8) | reg[1]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void update_events(RTMRState *tmr, int ch) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint16_t diff[TMR_NR_EVENTS], min; | 
					
						
							|  |  |  |     int64_t next_time; | 
					
						
							|  |  |  |     int i, event; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (tmr->tccr[ch] == 0) { | 
					
						
							| 
									
										
										
										
											2022-10-24 15:28:02 +08:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (FIELD_EX8(tmr->tccr[ch], TCCR, CSS) == 0) { | 
					
						
							|  |  |  |         /* external clock mode */ | 
					
						
							|  |  |  |         /* event not happened */ | 
					
						
							| 
									
										
										
										
											2022-10-24 15:28:02 +08:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-02-19 22:32:39 +00:00
										 |  |  |     if (FIELD_EX8(tmr->tccr[0], TCCR, CSS) == CSS_CASCADING) { | 
					
						
							| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  |         /* cascading mode */ | 
					
						
							|  |  |  |         if (ch == 1) { | 
					
						
							|  |  |  |             tmr->next[ch] = none; | 
					
						
							| 
									
										
										
										
											2022-10-24 15:28:02 +08:00
										 |  |  |             return; | 
					
						
							| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  |         } | 
					
						
							|  |  |  |         diff[cmia] = concat_reg(tmr->tcora) - concat_reg(tmr->tcnt); | 
					
						
							|  |  |  |         diff[cmib] = concat_reg(tmr->tcorb) - concat_reg(tmr->tcnt); | 
					
						
							|  |  |  |         diff[ovi] = 0x10000 - concat_reg(tmr->tcnt); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         /* separate mode */ | 
					
						
							|  |  |  |         diff[cmia] = tmr->tcora[ch] - tmr->tcnt[ch]; | 
					
						
							|  |  |  |         diff[cmib] = tmr->tcorb[ch] - tmr->tcnt[ch]; | 
					
						
							|  |  |  |         diff[ovi] = 0x100 - tmr->tcnt[ch]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* Search for the most recently occurring event. */ | 
					
						
							|  |  |  |     for (event = 0, min = diff[0], i = 1; i < none; i++) { | 
					
						
							|  |  |  |         if (min > diff[i]) { | 
					
						
							|  |  |  |             event = i; | 
					
						
							|  |  |  |             min = diff[i]; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     tmr->next[ch] = event; | 
					
						
							|  |  |  |     next_time = diff[event]; | 
					
						
							|  |  |  |     next_time *= clkdiv[FIELD_EX8(tmr->tccr[ch], TCCR, CKS)]; | 
					
						
							|  |  |  |     next_time *= NANOSECONDS_PER_SECOND; | 
					
						
							|  |  |  |     next_time /= tmr->input_freq; | 
					
						
							|  |  |  |     next_time += qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL); | 
					
						
							|  |  |  |     timer_mod(&tmr->timer[ch], next_time); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int elapsed_time(RTMRState *tmr, int ch, int64_t delta) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int divrate = clkdiv[FIELD_EX8(tmr->tccr[ch], TCCR, CKS)]; | 
					
						
							|  |  |  |     int et; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     tmr->div_round[ch] += delta; | 
					
						
							|  |  |  |     if (divrate > 0) { | 
					
						
							|  |  |  |         et = tmr->div_round[ch] / divrate; | 
					
						
							|  |  |  |         tmr->div_round[ch] %= divrate; | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2023-07-14 14:32:24 +03:00
										 |  |  |         /* disable clock. so no update */ | 
					
						
							| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  |         et = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return et; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint16_t read_tcnt(RTMRState *tmr, unsigned size, int ch) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int64_t delta, now = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL); | 
					
						
							|  |  |  |     int elapsed, ovf = 0; | 
					
						
							|  |  |  |     uint16_t tcnt[2]; | 
					
						
							|  |  |  |     uint32_t ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     delta = (now - tmr->tick) * NANOSECONDS_PER_SECOND / tmr->input_freq; | 
					
						
							|  |  |  |     if (delta > 0) { | 
					
						
							|  |  |  |         tmr->tick = now; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-19 22:32:40 +00:00
										 |  |  |         switch (FIELD_EX8(tmr->tccr[1], TCCR, CSS)) { | 
					
						
							|  |  |  |         case CSS_INTERNAL: | 
					
						
							| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  |             /* timer1 count update */ | 
					
						
							|  |  |  |             elapsed = elapsed_time(tmr, 1, delta); | 
					
						
							|  |  |  |             if (elapsed >= 0x100) { | 
					
						
							|  |  |  |                 ovf = elapsed >> 8; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             tcnt[1] = tmr->tcnt[1] + (elapsed & 0xff); | 
					
						
							| 
									
										
										
										
											2021-02-19 22:32:40 +00:00
										 |  |  |             break; | 
					
						
							|  |  |  |         case CSS_INVALID: /* guest error to have set this */ | 
					
						
							|  |  |  |         case CSS_EXTERNAL: /* QEMU doesn't implement these */ | 
					
						
							|  |  |  |         case CSS_CASCADING: | 
					
						
							|  |  |  |             tcnt[1] = tmr->tcnt[1]; | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2021-03-30 14:05:34 +01:00
										 |  |  |         default: | 
					
						
							|  |  |  |             g_assert_not_reached(); | 
					
						
							| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  |         } | 
					
						
							|  |  |  |         switch (FIELD_EX8(tmr->tccr[0], TCCR, CSS)) { | 
					
						
							| 
									
										
										
										
											2021-02-19 22:32:39 +00:00
										 |  |  |         case CSS_INTERNAL: | 
					
						
							| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  |             elapsed = elapsed_time(tmr, 0, delta); | 
					
						
							|  |  |  |             tcnt[0] = tmr->tcnt[0] + elapsed; | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2021-02-19 22:32:39 +00:00
										 |  |  |         case CSS_CASCADING: | 
					
						
							| 
									
										
										
										
											2021-02-19 22:32:40 +00:00
										 |  |  |             tcnt[0] = tmr->tcnt[0] + ovf; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case CSS_INVALID: /* guest error to have set this */ | 
					
						
							|  |  |  |         case CSS_EXTERNAL: /* QEMU doesn't implement this */ | 
					
						
							|  |  |  |             tcnt[0] = tmr->tcnt[0]; | 
					
						
							| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2021-03-30 14:05:34 +01:00
										 |  |  |         default: | 
					
						
							|  |  |  |             g_assert_not_reached(); | 
					
						
							| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         tcnt[0] = tmr->tcnt[0]; | 
					
						
							|  |  |  |         tcnt[1] = tmr->tcnt[1]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (size == 1) { | 
					
						
							|  |  |  |         return tcnt[ch]; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         ret = 0; | 
					
						
							|  |  |  |         ret = deposit32(ret, 0, 8, tcnt[1]); | 
					
						
							|  |  |  |         ret = deposit32(ret, 8, 8, tcnt[0]); | 
					
						
							|  |  |  |         return ret; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint8_t read_tccr(uint8_t r) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint8_t tccr = 0; | 
					
						
							|  |  |  |     tccr = FIELD_DP8(tccr, TCCR, TMRIS, | 
					
						
							|  |  |  |                      FIELD_EX8(r, TCCR, TMRIS)); | 
					
						
							|  |  |  |     tccr = FIELD_DP8(tccr, TCCR, CSS, | 
					
						
							|  |  |  |                      FIELD_EX8(r, TCCR, CSS)); | 
					
						
							|  |  |  |     tccr = FIELD_DP8(tccr, TCCR, CKS, | 
					
						
							|  |  |  |                      FIELD_EX8(r, TCCR, CKS)); | 
					
						
							|  |  |  |     return tccr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint64_t tmr_read(void *opaque, hwaddr addr, unsigned size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     RTMRState *tmr = opaque; | 
					
						
							|  |  |  |     int ch = addr & 1; | 
					
						
							|  |  |  |     uint64_t ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (size == 2 && (ch != 0 || addr == A_TCR || addr == A_TCSR)) { | 
					
						
							|  |  |  |         qemu_log_mask(LOG_GUEST_ERROR, "renesas_tmr: Invalid read size 0x%" | 
					
						
							|  |  |  |                                        HWADDR_PRIX "\n", | 
					
						
							|  |  |  |                       addr); | 
					
						
							|  |  |  |         return UINT64_MAX; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     switch (addr & 0x0e) { | 
					
						
							|  |  |  |     case A_TCR: | 
					
						
							|  |  |  |         ret = 0; | 
					
						
							|  |  |  |         ret = FIELD_DP8(ret, TCR, CCLR, | 
					
						
							|  |  |  |                         FIELD_EX8(tmr->tcr[ch], TCR, CCLR)); | 
					
						
							|  |  |  |         ret = FIELD_DP8(ret, TCR, OVIE, | 
					
						
							|  |  |  |                         FIELD_EX8(tmr->tcr[ch], TCR, OVIE)); | 
					
						
							|  |  |  |         ret = FIELD_DP8(ret, TCR, CMIEA, | 
					
						
							|  |  |  |                         FIELD_EX8(tmr->tcr[ch], TCR, CMIEA)); | 
					
						
							|  |  |  |         ret = FIELD_DP8(ret, TCR, CMIEB, | 
					
						
							|  |  |  |                         FIELD_EX8(tmr->tcr[ch], TCR, CMIEB)); | 
					
						
							|  |  |  |         return ret; | 
					
						
							|  |  |  |     case A_TCSR: | 
					
						
							|  |  |  |         ret = 0; | 
					
						
							|  |  |  |         ret = FIELD_DP8(ret, TCSR, OSA, | 
					
						
							|  |  |  |                         FIELD_EX8(tmr->tcsr[ch], TCSR, OSA)); | 
					
						
							|  |  |  |         ret = FIELD_DP8(ret, TCSR, OSB, | 
					
						
							|  |  |  |                         FIELD_EX8(tmr->tcsr[ch], TCSR, OSB)); | 
					
						
							|  |  |  |         switch (ch) { | 
					
						
							|  |  |  |         case 0: | 
					
						
							|  |  |  |             ret = FIELD_DP8(ret, TCSR, ADTE, | 
					
						
							|  |  |  |                             FIELD_EX8(tmr->tcsr[ch], TCSR, ADTE)); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case 1: /* CH1 ADTE unimplement always 1 */ | 
					
						
							|  |  |  |             ret = FIELD_DP8(ret, TCSR, ADTE, 1); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return ret; | 
					
						
							|  |  |  |     case A_TCORA: | 
					
						
							|  |  |  |         if (size == 1) { | 
					
						
							|  |  |  |             return tmr->tcora[ch]; | 
					
						
							|  |  |  |         } else if (ch == 0) { | 
					
						
							|  |  |  |             return concat_reg(tmr->tcora); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-12-11 16:24:18 +01:00
										 |  |  |         /* fall through */ | 
					
						
							| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  |     case A_TCORB: | 
					
						
							|  |  |  |         if (size == 1) { | 
					
						
							|  |  |  |             return tmr->tcorb[ch]; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             return concat_reg(tmr->tcorb); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     case A_TCNT: | 
					
						
							|  |  |  |         return read_tcnt(tmr, size, ch); | 
					
						
							|  |  |  |     case A_TCCR: | 
					
						
							|  |  |  |         if (size == 1) { | 
					
						
							|  |  |  |             return read_tccr(tmr->tccr[ch]); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             return read_tccr(tmr->tccr[0]) << 8 | read_tccr(tmr->tccr[1]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         qemu_log_mask(LOG_UNIMP, "renesas_tmr: Register 0x%" HWADDR_PRIX | 
					
						
							|  |  |  |                                  " not implemented\n", | 
					
						
							|  |  |  |                       addr); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return UINT64_MAX; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void tmr_write_count(RTMRState *tmr, int ch, unsigned size, | 
					
						
							|  |  |  |                             uint8_t *reg, uint64_t val) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (size == 1) { | 
					
						
							|  |  |  |         reg[ch] = val; | 
					
						
							|  |  |  |         update_events(tmr, ch); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         reg[0] = extract32(val, 8, 8); | 
					
						
							|  |  |  |         reg[1] = extract32(val, 0, 8); | 
					
						
							|  |  |  |         update_events(tmr, 0); | 
					
						
							|  |  |  |         update_events(tmr, 1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void tmr_write(void *opaque, hwaddr addr, uint64_t val, unsigned size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     RTMRState *tmr = opaque; | 
					
						
							|  |  |  |     int ch = addr & 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (size == 2 && (ch != 0 || addr == A_TCR || addr == A_TCSR)) { | 
					
						
							|  |  |  |         qemu_log_mask(LOG_GUEST_ERROR, | 
					
						
							|  |  |  |                       "renesas_tmr: Invalid write size 0x%" HWADDR_PRIX "\n", | 
					
						
							|  |  |  |                       addr); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     switch (addr & 0x0e) { | 
					
						
							|  |  |  |     case A_TCR: | 
					
						
							|  |  |  |         tmr->tcr[ch] = val; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case A_TCSR: | 
					
						
							|  |  |  |         tmr->tcsr[ch] = val; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case A_TCORA: | 
					
						
							|  |  |  |         tmr_write_count(tmr, ch, size, tmr->tcora, val); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case A_TCORB: | 
					
						
							|  |  |  |         tmr_write_count(tmr, ch, size, tmr->tcorb, val); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case A_TCNT: | 
					
						
							|  |  |  |         tmr_write_count(tmr, ch, size, tmr->tcnt, val); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case A_TCCR: | 
					
						
							|  |  |  |         tmr_write_count(tmr, ch, size, tmr->tccr, val); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         qemu_log_mask(LOG_UNIMP, "renesas_tmr: Register 0x%" HWADDR_PRIX | 
					
						
							|  |  |  |                                  " not implemented\n", | 
					
						
							|  |  |  |                       addr); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const MemoryRegionOps tmr_ops = { | 
					
						
							|  |  |  |     .write = tmr_write, | 
					
						
							|  |  |  |     .read  = tmr_read, | 
					
						
							|  |  |  |     .endianness = DEVICE_LITTLE_ENDIAN, | 
					
						
							|  |  |  |     .impl = { | 
					
						
							|  |  |  |         .min_access_size = 1, | 
					
						
							|  |  |  |         .max_access_size = 2, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     .valid = { | 
					
						
							|  |  |  |         .min_access_size = 1, | 
					
						
							|  |  |  |         .max_access_size = 2, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void timer_events(RTMRState *tmr, int ch); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint16_t issue_event(RTMRState *tmr, int ch, int sz, | 
					
						
							|  |  |  |                         uint16_t tcnt, uint16_t tcora, uint16_t tcorb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint16_t ret = tcnt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (tmr->next[ch]) { | 
					
						
							|  |  |  |     case none: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case cmia: | 
					
						
							|  |  |  |         if (tcnt >= tcora) { | 
					
						
							|  |  |  |             if (FIELD_EX8(tmr->tcr[ch], TCR, CCLR) == CCLR_A) { | 
					
						
							|  |  |  |                 ret = tcnt - tcora; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (FIELD_EX8(tmr->tcr[ch], TCR, CMIEA)) { | 
					
						
							|  |  |  |                 qemu_irq_pulse(tmr->cmia[ch]); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (sz == 8 && ch == 0 && | 
					
						
							| 
									
										
										
										
											2021-02-19 22:32:39 +00:00
										 |  |  |                 FIELD_EX8(tmr->tccr[1], TCCR, CSS) == CSS_CASCADING) { | 
					
						
							| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  |                 tmr->tcnt[1]++; | 
					
						
							|  |  |  |                 timer_events(tmr, 1); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case cmib: | 
					
						
							|  |  |  |         if (tcnt >= tcorb) { | 
					
						
							|  |  |  |             if (FIELD_EX8(tmr->tcr[ch], TCR, CCLR) == CCLR_B) { | 
					
						
							|  |  |  |                 ret = tcnt - tcorb; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (FIELD_EX8(tmr->tcr[ch], TCR, CMIEB)) { | 
					
						
							|  |  |  |                 qemu_irq_pulse(tmr->cmib[ch]); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ovi: | 
					
						
							|  |  |  |         if ((tcnt >= (1 << sz)) && FIELD_EX8(tmr->tcr[ch], TCR, OVIE)) { | 
					
						
							|  |  |  |             qemu_irq_pulse(tmr->ovi[ch]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         g_assert_not_reached(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void timer_events(RTMRState *tmr, int ch) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint16_t tcnt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     tmr->tcnt[ch] = read_tcnt(tmr, 1, ch); | 
					
						
							| 
									
										
										
										
											2021-02-19 22:32:39 +00:00
										 |  |  |     if (FIELD_EX8(tmr->tccr[0], TCCR, CSS) != CSS_CASCADING) { | 
					
						
							| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  |         tmr->tcnt[ch] = issue_event(tmr, ch, 8, | 
					
						
							|  |  |  |                                     tmr->tcnt[ch], | 
					
						
							|  |  |  |                                     tmr->tcora[ch], | 
					
						
							|  |  |  |                                     tmr->tcorb[ch]) & 0xff; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         if (ch == 1) { | 
					
						
							| 
									
										
										
										
											2022-10-24 15:28:02 +08:00
										 |  |  |             return; | 
					
						
							| 
									
										
										
										
											2019-03-20 23:16:05 +09:00
										 |  |  |         } | 
					
						
							|  |  |  |         tcnt = issue_event(tmr, ch, 16, | 
					
						
							|  |  |  |                            concat_reg(tmr->tcnt), | 
					
						
							|  |  |  |                            concat_reg(tmr->tcora), | 
					
						
							|  |  |  |                            concat_reg(tmr->tcorb)); | 
					
						
							|  |  |  |         tmr->tcnt[0] = (tcnt >> 8) & 0xff; | 
					
						
							|  |  |  |         tmr->tcnt[1] = tcnt & 0xff; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     update_events(tmr, ch); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void timer_event0(void *opaque) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     RTMRState *tmr = opaque; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     timer_events(tmr, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void timer_event1(void *opaque) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     RTMRState *tmr = opaque; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     timer_events(tmr, 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void rtmr_reset(DeviceState *dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     RTMRState *tmr = RTMR(dev); | 
					
						
							|  |  |  |     tmr->tcr[0]   = tmr->tcr[1]   = 0x00; | 
					
						
							|  |  |  |     tmr->tcsr[0]  = 0x00; | 
					
						
							|  |  |  |     tmr->tcsr[1]  = 0x10; | 
					
						
							|  |  |  |     tmr->tcnt[0]  = tmr->tcnt[1]  = 0x00; | 
					
						
							|  |  |  |     tmr->tcora[0] = tmr->tcora[1] = 0xff; | 
					
						
							|  |  |  |     tmr->tcorb[0] = tmr->tcorb[1] = 0xff; | 
					
						
							|  |  |  |     tmr->tccr[0]  = tmr->tccr[1]  = 0x00; | 
					
						
							|  |  |  |     tmr->next[0]  = tmr->next[1]  = none; | 
					
						
							|  |  |  |     tmr->tick = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void rtmr_init(Object *obj) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     SysBusDevice *d = SYS_BUS_DEVICE(obj); | 
					
						
							|  |  |  |     RTMRState *tmr = RTMR(obj); | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     memory_region_init_io(&tmr->memory, OBJECT(tmr), &tmr_ops, | 
					
						
							|  |  |  |                           tmr, "renesas-tmr", 0x10); | 
					
						
							|  |  |  |     sysbus_init_mmio(d, &tmr->memory); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < ARRAY_SIZE(tmr->ovi); i++) { | 
					
						
							|  |  |  |         sysbus_init_irq(d, &tmr->cmia[i]); | 
					
						
							|  |  |  |         sysbus_init_irq(d, &tmr->cmib[i]); | 
					
						
							|  |  |  |         sysbus_init_irq(d, &tmr->ovi[i]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     timer_init_ns(&tmr->timer[0], QEMU_CLOCK_VIRTUAL, timer_event0, tmr); | 
					
						
							|  |  |  |     timer_init_ns(&tmr->timer[1], QEMU_CLOCK_VIRTUAL, timer_event1, tmr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const VMStateDescription vmstate_rtmr = { | 
					
						
							|  |  |  |     .name = "rx-tmr", | 
					
						
							|  |  |  |     .version_id = 1, | 
					
						
							|  |  |  |     .minimum_version_id = 1, | 
					
						
							|  |  |  |     .fields = (VMStateField[]) { | 
					
						
							|  |  |  |         VMSTATE_INT64(tick, RTMRState), | 
					
						
							|  |  |  |         VMSTATE_UINT8_ARRAY(tcnt, RTMRState, TMR_CH), | 
					
						
							|  |  |  |         VMSTATE_UINT8_ARRAY(tcora, RTMRState, TMR_CH), | 
					
						
							|  |  |  |         VMSTATE_UINT8_ARRAY(tcorb, RTMRState, TMR_CH), | 
					
						
							|  |  |  |         VMSTATE_UINT8_ARRAY(tcr, RTMRState, TMR_CH), | 
					
						
							|  |  |  |         VMSTATE_UINT8_ARRAY(tccr, RTMRState, TMR_CH), | 
					
						
							|  |  |  |         VMSTATE_UINT8_ARRAY(tcor, RTMRState, TMR_CH), | 
					
						
							|  |  |  |         VMSTATE_UINT8_ARRAY(tcsr, RTMRState, TMR_CH), | 
					
						
							|  |  |  |         VMSTATE_INT64_ARRAY(div_round, RTMRState, TMR_CH), | 
					
						
							|  |  |  |         VMSTATE_UINT8_ARRAY(next, RTMRState, TMR_CH), | 
					
						
							|  |  |  |         VMSTATE_TIMER_ARRAY(timer, RTMRState, TMR_CH), | 
					
						
							|  |  |  |         VMSTATE_END_OF_LIST() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Property rtmr_properties[] = { | 
					
						
							|  |  |  |     DEFINE_PROP_UINT64("input-freq", RTMRState, input_freq, 0), | 
					
						
							|  |  |  |     DEFINE_PROP_END_OF_LIST(), | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void rtmr_class_init(ObjectClass *klass, void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DeviceClass *dc = DEVICE_CLASS(klass); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     dc->vmsd = &vmstate_rtmr; | 
					
						
							|  |  |  |     dc->reset = rtmr_reset; | 
					
						
							|  |  |  |     device_class_set_props(dc, rtmr_properties); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const TypeInfo rtmr_info = { | 
					
						
							|  |  |  |     .name = TYPE_RENESAS_TMR, | 
					
						
							|  |  |  |     .parent = TYPE_SYS_BUS_DEVICE, | 
					
						
							|  |  |  |     .instance_size = sizeof(RTMRState), | 
					
						
							|  |  |  |     .instance_init = rtmr_init, | 
					
						
							|  |  |  |     .class_init = rtmr_class_init, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void rtmr_register_types(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     type_register_static(&rtmr_info); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type_init(rtmr_register_types) |