1 /*
2 * Copyright (c) 2020-2021 Huawei Device Co., Ltd.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16 #include "plic.h"
17 #include "soc.h"
18 #include "los_reg.h"
19 #include "los_arch_interrupt.h"
20 #include "los_debug.h"
21
22 #ifdef __cplusplus
23 #if __cplusplus
24 extern "C" {
25 #endif
26 #endif
27
OsMachineExternalInterrupt(VOID *arg)28 STATIC VOID OsMachineExternalInterrupt(VOID *arg)
29 {
30 volatile UINT32 *plicReg = (volatile UINT32 *)(PLIC_REG_BASE + 0x4);
31 UINT32 irqNum, saveIrqNum;
32
33 READ_UINT32(irqNum, plicReg);
34 saveIrqNum = irqNum;
35
36 if ((irqNum >= OS_RISCV_CUSTOM_IRQ_VECTOR_CNT) || (irqNum == 0)) {
37 HalHwiDefaultHandler((VOID *)irqNum);
38 }
39
40 irqNum += RISCV_SYS_MAX_IRQ;
41
42 g_hwiForm[irqNum].pfnHook(g_hwiForm[irqNum].uwParam);
43
44 WRITE_UINT32(saveIrqNum, plicReg);
45 }
46
47 #define OS_PLIC_MAX ((OS_RISCV_CUSTOM_IRQ_VECTOR_CNT >> 5) + 1)
48
PlicIrqInitnull49 VOID PlicIrqInit(VOID)
50 {
51 volatile UINT32 *plicPrioReg = (volatile UINT32 *)PLIC_PRIO_BASE;
52 volatile UINT32 *plicEnReg = (volatile UINT32 *)PLIC_ENABLE_BASE;
53 volatile UINT32 *plicReg = (volatile UINT32 *)PLIC_REG_BASE;
54 INT32 i;
55 UINT32 ret;
56
57 for (i = 0; i < OS_PLIC_MAX; i++) {
58 WRITE_UINT32(0x0, plicEnReg);
59 plicEnReg++;
60 }
61
62 for (i = 0; i < OS_RISCV_CUSTOM_IRQ_VECTOR_CNT; i++) {
63 WRITE_UINT32(0x0, plicPrioReg);
64 plicPrioReg++;
65 }
66
67 WRITE_UINT32(0, plicReg);
68
69 ret = LOS_HwiCreate(RISCV_MACH_EXT_IRQ, 0x1, 0, OsMachineExternalInterrupt, 0);
70 if (ret != LOS_OK) {
71 PRINT_ERR("Create machine external failed! ret : 0x%x\n", ret);
72 }
73 }
74
PlicIrqSetPrio(UINT32 vector, UINT32 pri)75 VOID PlicIrqSetPrio(UINT32 vector, UINT32 pri)
76 {
77 volatile UINT32 *plicReg = (volatile UINT32 *)PLIC_PRIO_BASE;
78
79 plicReg += (vector - RISCV_SYS_MAX_IRQ);
80 WRITE_UINT32(pri, plicReg);
81 }
82
PlicIrqEnable(UINT32 vector)83 VOID PlicIrqEnable(UINT32 vector)
84 {
85 UINT32 irqValue;
86 UINT32 locIrq = vector - RISCV_SYS_MAX_IRQ;
87 volatile UINT32 *plicReg = (volatile UINT32 *)PLIC_ENABLE_BASE;
88
89 plicReg += (locIrq >> 5); /* 5: The PLIC interrupt controls the bit width */
90 READ_UINT32(irqValue, plicReg);
91 irqValue |= (1 << (locIrq & 31)); /* 31: plic irq mask */
92 WRITE_UINT32(irqValue, plicReg);
93 }
94
PlicIrqDisable(UINT32 vector)95 VOID PlicIrqDisable(UINT32 vector)
96 {
97 UINT32 irqValue;
98 UINT32 locIrq = vector - RISCV_SYS_MAX_IRQ;
99 volatile UINT32 *plicReg = (volatile UINT32 *)PLIC_ENABLE_BASE;
100
101 plicReg += (locIrq >> 5); /* 5: The PLIC interrupt controls the bit width */
102 READ_UINT32(irqValue, plicReg);
103 irqValue &= ~(1 << (locIrq & 31)); /* 31: plic irq mask */
104 WRITE_UINT32(irqValue, plicReg);
105 }
106
107 #ifdef __cplusplus
108 #if __cplusplus
109 }
110 #endif /* __cplusplus */
111 #endif /* __cplusplus */
112
113