| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  *  ASPEED LPC Controller | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  Copyright (C) 2017-2018 IBM Corp. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This code is licensed under the GPL version 2 or later.  See | 
					
						
							|  |  |  |  * the COPYING file in the top-level directory. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "qemu/osdep.h"
 | 
					
						
							|  |  |  | #include "qemu/log.h"
 | 
					
						
							|  |  |  | #include "qemu/error-report.h"
 | 
					
						
							|  |  |  | #include "hw/misc/aspeed_lpc.h"
 | 
					
						
							|  |  |  | #include "qapi/error.h"
 | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | #include "qapi/visitor.h"
 | 
					
						
							|  |  |  | #include "hw/irq.h"
 | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | #include "hw/qdev-properties.h"
 | 
					
						
							|  |  |  | #include "migration/vmstate.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define TO_REG(offset) ((offset) >> 2)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define HICR0                TO_REG(0x00)
 | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | #define   HICR0_LPC3E        BIT(7)
 | 
					
						
							|  |  |  | #define   HICR0_LPC2E        BIT(6)
 | 
					
						
							|  |  |  | #define   HICR0_LPC1E        BIT(5)
 | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | #define HICR1                TO_REG(0x04)
 | 
					
						
							|  |  |  | #define HICR2                TO_REG(0x08)
 | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | #define   HICR2_IBFIE3       BIT(3)
 | 
					
						
							|  |  |  | #define   HICR2_IBFIE2       BIT(2)
 | 
					
						
							|  |  |  | #define   HICR2_IBFIE1       BIT(1)
 | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | #define HICR3                TO_REG(0x0C)
 | 
					
						
							|  |  |  | #define HICR4                TO_REG(0x10)
 | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | #define   HICR4_KCSENBL      BIT(2)
 | 
					
						
							|  |  |  | #define IDR1                 TO_REG(0x24)
 | 
					
						
							|  |  |  | #define IDR2                 TO_REG(0x28)
 | 
					
						
							|  |  |  | #define IDR3                 TO_REG(0x2C)
 | 
					
						
							|  |  |  | #define ODR1                 TO_REG(0x30)
 | 
					
						
							|  |  |  | #define ODR2                 TO_REG(0x34)
 | 
					
						
							|  |  |  | #define ODR3                 TO_REG(0x38)
 | 
					
						
							|  |  |  | #define STR1                 TO_REG(0x3C)
 | 
					
						
							|  |  |  | #define   STR_OBF            BIT(0)
 | 
					
						
							|  |  |  | #define   STR_IBF            BIT(1)
 | 
					
						
							|  |  |  | #define   STR_CMD_DATA       BIT(3)
 | 
					
						
							|  |  |  | #define STR2                 TO_REG(0x40)
 | 
					
						
							|  |  |  | #define STR3                 TO_REG(0x44)
 | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | #define HICR5                TO_REG(0x80)
 | 
					
						
							|  |  |  | #define HICR6                TO_REG(0x84)
 | 
					
						
							|  |  |  | #define HICR7                TO_REG(0x88)
 | 
					
						
							|  |  |  | #define HICR8                TO_REG(0x8C)
 | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | #define HICRB                TO_REG(0x100)
 | 
					
						
							|  |  |  | #define   HICRB_IBFIE4       BIT(1)
 | 
					
						
							|  |  |  | #define   HICRB_LPC4E        BIT(0)
 | 
					
						
							|  |  |  | #define IDR4                 TO_REG(0x114)
 | 
					
						
							|  |  |  | #define ODR4                 TO_REG(0x118)
 | 
					
						
							|  |  |  | #define STR4                 TO_REG(0x11C)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum aspeed_kcs_channel_id { | 
					
						
							|  |  |  |     kcs_channel_1 = 0, | 
					
						
							|  |  |  |     kcs_channel_2, | 
					
						
							|  |  |  |     kcs_channel_3, | 
					
						
							|  |  |  |     kcs_channel_4, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const enum aspeed_lpc_subdevice aspeed_kcs_subdevice_map[] = { | 
					
						
							|  |  |  |     [kcs_channel_1] = aspeed_lpc_kcs_1, | 
					
						
							|  |  |  |     [kcs_channel_2] = aspeed_lpc_kcs_2, | 
					
						
							|  |  |  |     [kcs_channel_3] = aspeed_lpc_kcs_3, | 
					
						
							|  |  |  |     [kcs_channel_4] = aspeed_lpc_kcs_4, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct aspeed_kcs_channel { | 
					
						
							|  |  |  |     enum aspeed_kcs_channel_id id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int idr; | 
					
						
							|  |  |  |     int odr; | 
					
						
							|  |  |  |     int str; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct aspeed_kcs_channel aspeed_kcs_channel_map[] = { | 
					
						
							|  |  |  |     [kcs_channel_1] = { | 
					
						
							|  |  |  |         .id = kcs_channel_1, | 
					
						
							|  |  |  |         .idr = IDR1, | 
					
						
							|  |  |  |         .odr = ODR1, | 
					
						
							|  |  |  |         .str = STR1 | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     [kcs_channel_2] = { | 
					
						
							|  |  |  |         .id = kcs_channel_2, | 
					
						
							|  |  |  |         .idr = IDR2, | 
					
						
							|  |  |  |         .odr = ODR2, | 
					
						
							|  |  |  |         .str = STR2 | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     [kcs_channel_3] = { | 
					
						
							|  |  |  |         .id = kcs_channel_3, | 
					
						
							|  |  |  |         .idr = IDR3, | 
					
						
							|  |  |  |         .odr = ODR3, | 
					
						
							|  |  |  |         .str = STR3 | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     [kcs_channel_4] = { | 
					
						
							|  |  |  |         .id = kcs_channel_4, | 
					
						
							|  |  |  |         .idr = IDR4, | 
					
						
							|  |  |  |         .odr = ODR4, | 
					
						
							|  |  |  |         .str = STR4 | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct aspeed_kcs_register_data { | 
					
						
							|  |  |  |     const char *name; | 
					
						
							|  |  |  |     int reg; | 
					
						
							|  |  |  |     const struct aspeed_kcs_channel *chan; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct aspeed_kcs_register_data aspeed_kcs_registers[] = { | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         .name = "idr1", | 
					
						
							|  |  |  |         .reg = IDR1, | 
					
						
							|  |  |  |         .chan = &aspeed_kcs_channel_map[kcs_channel_1], | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         .name = "odr1", | 
					
						
							|  |  |  |         .reg = ODR1, | 
					
						
							|  |  |  |         .chan = &aspeed_kcs_channel_map[kcs_channel_1], | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         .name = "str1", | 
					
						
							|  |  |  |         .reg = STR1, | 
					
						
							|  |  |  |         .chan = &aspeed_kcs_channel_map[kcs_channel_1], | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         .name = "idr2", | 
					
						
							|  |  |  |         .reg = IDR2, | 
					
						
							|  |  |  |         .chan = &aspeed_kcs_channel_map[kcs_channel_2], | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         .name = "odr2", | 
					
						
							|  |  |  |         .reg = ODR2, | 
					
						
							|  |  |  |         .chan = &aspeed_kcs_channel_map[kcs_channel_2], | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         .name = "str2", | 
					
						
							|  |  |  |         .reg = STR2, | 
					
						
							|  |  |  |         .chan = &aspeed_kcs_channel_map[kcs_channel_2], | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         .name = "idr3", | 
					
						
							|  |  |  |         .reg = IDR3, | 
					
						
							|  |  |  |         .chan = &aspeed_kcs_channel_map[kcs_channel_3], | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         .name = "odr3", | 
					
						
							|  |  |  |         .reg = ODR3, | 
					
						
							|  |  |  |         .chan = &aspeed_kcs_channel_map[kcs_channel_3], | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         .name = "str3", | 
					
						
							|  |  |  |         .reg = STR3, | 
					
						
							|  |  |  |         .chan = &aspeed_kcs_channel_map[kcs_channel_3], | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         .name = "idr4", | 
					
						
							|  |  |  |         .reg = IDR4, | 
					
						
							|  |  |  |         .chan = &aspeed_kcs_channel_map[kcs_channel_4], | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         .name = "odr4", | 
					
						
							|  |  |  |         .reg = ODR4, | 
					
						
							|  |  |  |         .chan = &aspeed_kcs_channel_map[kcs_channel_4], | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         .name = "str4", | 
					
						
							|  |  |  |         .reg = STR4, | 
					
						
							|  |  |  |         .chan = &aspeed_kcs_channel_map[kcs_channel_4], | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { }, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct aspeed_kcs_register_data * | 
					
						
							|  |  |  | aspeed_kcs_get_register_data_by_name(const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const struct aspeed_kcs_register_data *pos = aspeed_kcs_registers; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (pos->name) { | 
					
						
							|  |  |  |         if (!strcmp(pos->name, name)) { | 
					
						
							|  |  |  |             return pos; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         pos++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct aspeed_kcs_channel * | 
					
						
							|  |  |  | aspeed_kcs_get_channel_by_register(int reg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const struct aspeed_kcs_register_data *pos = aspeed_kcs_registers; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (pos->name) { | 
					
						
							|  |  |  |         if (pos->reg == reg) { | 
					
						
							|  |  |  |             return pos->chan; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         pos++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void aspeed_kcs_get_register_property(Object *obj, | 
					
						
							|  |  |  |                                              Visitor *v, | 
					
						
							|  |  |  |                                              const char *name, | 
					
						
							|  |  |  |                                              void *opaque, | 
					
						
							|  |  |  |                                              Error **errp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const struct aspeed_kcs_register_data *data; | 
					
						
							|  |  |  |     AspeedLPCState *s = ASPEED_LPC(obj); | 
					
						
							|  |  |  |     uint32_t val; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     data = aspeed_kcs_get_register_data_by_name(name); | 
					
						
							|  |  |  |     if (!data) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!strncmp("odr", name, 3)) { | 
					
						
							|  |  |  |         s->regs[data->chan->str] &= ~STR_OBF; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     val = s->regs[data->reg]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     visit_type_uint32(v, name, &val, errp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool aspeed_kcs_channel_enabled(AspeedLPCState *s, | 
					
						
							|  |  |  |                                        const struct aspeed_kcs_channel *channel) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (channel->id) { | 
					
						
							|  |  |  |     case kcs_channel_1: return s->regs[HICR0] & HICR0_LPC1E; | 
					
						
							|  |  |  |     case kcs_channel_2: return s->regs[HICR0] & HICR0_LPC2E; | 
					
						
							|  |  |  |     case kcs_channel_3: | 
					
						
							|  |  |  |         return (s->regs[HICR0] & HICR0_LPC3E) && | 
					
						
							|  |  |  |                     (s->regs[HICR4] & HICR4_KCSENBL); | 
					
						
							|  |  |  |     case kcs_channel_4: return s->regs[HICRB] & HICRB_LPC4E; | 
					
						
							|  |  |  |     default: return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool | 
					
						
							|  |  |  | aspeed_kcs_channel_ibf_irq_enabled(AspeedLPCState *s, | 
					
						
							|  |  |  |                                    const struct aspeed_kcs_channel *channel) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!aspeed_kcs_channel_enabled(s, channel)) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (channel->id) { | 
					
						
							|  |  |  |     case kcs_channel_1: return s->regs[HICR2] & HICR2_IBFIE1; | 
					
						
							|  |  |  |     case kcs_channel_2: return s->regs[HICR2] & HICR2_IBFIE2; | 
					
						
							|  |  |  |     case kcs_channel_3: return s->regs[HICR2] & HICR2_IBFIE3; | 
					
						
							|  |  |  |     case kcs_channel_4: return s->regs[HICRB] & HICRB_IBFIE4; | 
					
						
							|  |  |  |     default: return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void aspeed_kcs_set_register_property(Object *obj, | 
					
						
							|  |  |  |                                              Visitor *v, | 
					
						
							|  |  |  |                                              const char *name, | 
					
						
							|  |  |  |                                              void *opaque, | 
					
						
							|  |  |  |                                              Error **errp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const struct aspeed_kcs_register_data *data; | 
					
						
							|  |  |  |     AspeedLPCState *s = ASPEED_LPC(obj); | 
					
						
							|  |  |  |     uint32_t val; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     data = aspeed_kcs_get_register_data_by_name(name); | 
					
						
							|  |  |  |     if (!data) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!visit_type_uint32(v, name, &val, errp)) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (strncmp("str", name, 3)) { | 
					
						
							|  |  |  |         s->regs[data->reg] = val; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!strncmp("idr", name, 3)) { | 
					
						
							|  |  |  |         s->regs[data->chan->str] |= STR_IBF; | 
					
						
							|  |  |  |         if (aspeed_kcs_channel_ibf_irq_enabled(s, data->chan)) { | 
					
						
							|  |  |  |             enum aspeed_lpc_subdevice subdev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             subdev = aspeed_kcs_subdevice_map[data->chan->id]; | 
					
						
							|  |  |  |             qemu_irq_raise(s->subdevice_irqs[subdev]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void aspeed_lpc_set_irq(void *opaque, int irq, int level) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     AspeedLPCState *s = (AspeedLPCState *)opaque; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (level) { | 
					
						
							|  |  |  |         s->subdevice_irqs_pending |= BIT(irq); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         s->subdevice_irqs_pending &= ~BIT(irq); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     qemu_set_irq(s->irq, !!s->subdevice_irqs_pending); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | static uint64_t aspeed_lpc_read(void *opaque, hwaddr offset, unsigned size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     AspeedLPCState *s = ASPEED_LPC(opaque); | 
					
						
							|  |  |  |     int reg = TO_REG(offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (reg >= ARRAY_SIZE(s->regs)) { | 
					
						
							|  |  |  |         qemu_log_mask(LOG_GUEST_ERROR, | 
					
						
							|  |  |  |                       "%s: Out-of-bounds read at offset 0x%" HWADDR_PRIx "\n", | 
					
						
							|  |  |  |                       __func__, offset); | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  |     switch (reg) { | 
					
						
							|  |  |  |     case IDR1: | 
					
						
							|  |  |  |     case IDR2: | 
					
						
							|  |  |  |     case IDR3: | 
					
						
							|  |  |  |     case IDR4: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         const struct aspeed_kcs_channel *channel; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         channel = aspeed_kcs_get_channel_by_register(reg); | 
					
						
							|  |  |  |         if (s->regs[channel->str] & STR_IBF) { | 
					
						
							|  |  |  |             enum aspeed_lpc_subdevice subdev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             subdev = aspeed_kcs_subdevice_map[channel->id]; | 
					
						
							|  |  |  |             qemu_irq_lower(s->subdevice_irqs[subdev]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         s->regs[channel->str] &= ~STR_IBF; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  |     return s->regs[reg]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void aspeed_lpc_write(void *opaque, hwaddr offset, uint64_t data, | 
					
						
							|  |  |  |                              unsigned int size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     AspeedLPCState *s = ASPEED_LPC(opaque); | 
					
						
							|  |  |  |     int reg = TO_REG(offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (reg >= ARRAY_SIZE(s->regs)) { | 
					
						
							|  |  |  |         qemu_log_mask(LOG_GUEST_ERROR, | 
					
						
							|  |  |  |                       "%s: Out-of-bounds write at offset 0x%" HWADDR_PRIx "\n", | 
					
						
							|  |  |  |                       __func__, offset); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     switch (reg) { | 
					
						
							|  |  |  |     case ODR1: | 
					
						
							|  |  |  |     case ODR2: | 
					
						
							|  |  |  |     case ODR3: | 
					
						
							|  |  |  |     case ODR4: | 
					
						
							|  |  |  |         s->regs[aspeed_kcs_get_channel_by_register(reg)->str] |= STR_OBF; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  |     s->regs[reg] = data; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const MemoryRegionOps aspeed_lpc_ops = { | 
					
						
							|  |  |  |     .read = aspeed_lpc_read, | 
					
						
							|  |  |  |     .write = aspeed_lpc_write, | 
					
						
							|  |  |  |     .endianness = DEVICE_LITTLE_ENDIAN, | 
					
						
							|  |  |  |     .valid = { | 
					
						
							|  |  |  |         .min_access_size = 1, | 
					
						
							|  |  |  |         .max_access_size = 4, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void aspeed_lpc_reset(DeviceState *dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct AspeedLPCState *s = ASPEED_LPC(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  |     s->subdevice_irqs_pending = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  |     memset(s->regs, 0, sizeof(s->regs)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s->regs[HICR7] = s->hicr7; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void aspeed_lpc_realize(DeviceState *dev, Error **errp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     AspeedLPCState *s = ASPEED_LPC(dev); | 
					
						
							|  |  |  |     SysBusDevice *sbd = SYS_BUS_DEVICE(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     sysbus_init_irq(sbd, &s->irq); | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  |     sysbus_init_irq(sbd, &s->subdevice_irqs[aspeed_lpc_kcs_1]); | 
					
						
							|  |  |  |     sysbus_init_irq(sbd, &s->subdevice_irqs[aspeed_lpc_kcs_2]); | 
					
						
							|  |  |  |     sysbus_init_irq(sbd, &s->subdevice_irqs[aspeed_lpc_kcs_3]); | 
					
						
							|  |  |  |     sysbus_init_irq(sbd, &s->subdevice_irqs[aspeed_lpc_kcs_4]); | 
					
						
							|  |  |  |     sysbus_init_irq(sbd, &s->subdevice_irqs[aspeed_lpc_ibt]); | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     memory_region_init_io(&s->iomem, OBJECT(s), &aspeed_lpc_ops, s, | 
					
						
							|  |  |  |             TYPE_ASPEED_LPC, 0x1000); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     sysbus_init_mmio(sbd, &s->iomem); | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     qdev_init_gpio_in(dev, aspeed_lpc_set_irq, ASPEED_LPC_NR_SUBDEVS); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void aspeed_lpc_init(Object *obj) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     object_property_add(obj, "idr1", "uint32", aspeed_kcs_get_register_property, | 
					
						
							|  |  |  |                         aspeed_kcs_set_register_property, NULL, NULL); | 
					
						
							|  |  |  |     object_property_add(obj, "odr1", "uint32", aspeed_kcs_get_register_property, | 
					
						
							|  |  |  |                         aspeed_kcs_set_register_property, NULL, NULL); | 
					
						
							|  |  |  |     object_property_add(obj, "str1", "uint32", aspeed_kcs_get_register_property, | 
					
						
							|  |  |  |                         aspeed_kcs_set_register_property, NULL, NULL); | 
					
						
							|  |  |  |     object_property_add(obj, "idr2", "uint32", aspeed_kcs_get_register_property, | 
					
						
							|  |  |  |                         aspeed_kcs_set_register_property, NULL, NULL); | 
					
						
							|  |  |  |     object_property_add(obj, "odr2", "uint32", aspeed_kcs_get_register_property, | 
					
						
							|  |  |  |                         aspeed_kcs_set_register_property, NULL, NULL); | 
					
						
							|  |  |  |     object_property_add(obj, "str2", "uint32", aspeed_kcs_get_register_property, | 
					
						
							|  |  |  |                         aspeed_kcs_set_register_property, NULL, NULL); | 
					
						
							|  |  |  |     object_property_add(obj, "idr3", "uint32", aspeed_kcs_get_register_property, | 
					
						
							|  |  |  |                         aspeed_kcs_set_register_property, NULL, NULL); | 
					
						
							|  |  |  |     object_property_add(obj, "odr3", "uint32", aspeed_kcs_get_register_property, | 
					
						
							|  |  |  |                         aspeed_kcs_set_register_property, NULL, NULL); | 
					
						
							|  |  |  |     object_property_add(obj, "str3", "uint32", aspeed_kcs_get_register_property, | 
					
						
							|  |  |  |                         aspeed_kcs_set_register_property, NULL, NULL); | 
					
						
							|  |  |  |     object_property_add(obj, "idr4", "uint32", aspeed_kcs_get_register_property, | 
					
						
							|  |  |  |                         aspeed_kcs_set_register_property, NULL, NULL); | 
					
						
							|  |  |  |     object_property_add(obj, "odr4", "uint32", aspeed_kcs_get_register_property, | 
					
						
							|  |  |  |                         aspeed_kcs_set_register_property, NULL, NULL); | 
					
						
							|  |  |  |     object_property_add(obj, "str4", "uint32", aspeed_kcs_get_register_property, | 
					
						
							|  |  |  |                         aspeed_kcs_set_register_property, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const VMStateDescription vmstate_aspeed_lpc = { | 
					
						
							|  |  |  |     .name = TYPE_ASPEED_LPC, | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  |     .version_id = 2, | 
					
						
							|  |  |  |     .minimum_version_id = 2, | 
					
						
							| 
									
										
										
										
											2023-12-21 14:16:21 +11:00
										 |  |  |     .fields = (const VMStateField[]) { | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  |         VMSTATE_UINT32_ARRAY(regs, AspeedLPCState, ASPEED_LPC_NR_REGS), | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  |         VMSTATE_UINT32(subdevice_irqs_pending, AspeedLPCState), | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  |         VMSTATE_END_OF_LIST(), | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Property aspeed_lpc_properties[] = { | 
					
						
							|  |  |  |     DEFINE_PROP_UINT32("hicr7", AspeedLPCState, hicr7, 0), | 
					
						
							|  |  |  |     DEFINE_PROP_END_OF_LIST(), | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void aspeed_lpc_class_init(ObjectClass *klass, void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DeviceClass *dc = DEVICE_CLASS(klass); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     dc->realize = aspeed_lpc_realize; | 
					
						
							|  |  |  |     dc->reset = aspeed_lpc_reset; | 
					
						
							|  |  |  |     dc->desc = "Aspeed LPC Controller", | 
					
						
							|  |  |  |     dc->vmsd = &vmstate_aspeed_lpc; | 
					
						
							|  |  |  |     device_class_set_props(dc, aspeed_lpc_properties); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const TypeInfo aspeed_lpc_info = { | 
					
						
							|  |  |  |     .name = TYPE_ASPEED_LPC, | 
					
						
							|  |  |  |     .parent = TYPE_SYS_BUS_DEVICE, | 
					
						
							|  |  |  |     .instance_size = sizeof(AspeedLPCState), | 
					
						
							|  |  |  |     .class_init = aspeed_lpc_class_init, | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  |     .instance_init = aspeed_lpc_init, | 
					
						
							| 
									
										
										
										
											2021-03-09 12:01:28 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void aspeed_lpc_register_types(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     type_register_static(&aspeed_lpc_info); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type_init(aspeed_lpc_register_types); |