GIC
GIC
GIC is ‘Generic Interrupt Contoller’. 요약을 보려면, GIC summary 참조 하세요.
GIC는 ARM에서 제공하는 ARM 전용 인터럽트 컨트롤러 이다.
이 문서는 GICv3를 기준으로 서술합니다.
소프트웨어 예시는 대표적으로 Linux Kernel이 있습니다.
관련 내용은 IRQ management in Linux kernel을 참조하세요.
Introduction
Interrupt Controller란,
말 그대로 인터럽트를 제어하는 장치이다.
외부 장치로부터 interrupt 신호를 수신하면,
적절한 PE(Core)에 전달하는 역할을 한다.
GIC를 구성하는 요소는 다음과 같다.
1. Distributor
2. Redistributor
3. CPU-INTERFACE
4. ITS (optional)
인터럽트 종류는 크게 4가지가 있다.
1. SGIs
2. PPIs
3. SPIs
4. LPIs
GIC components
크게
Distributor , (SPIs route)
Redistributor , (SGIs, PPIs, LPIs route)
CPU-INTERFACE , (interface of CPU)
ITS , (Translate LPIs)
가 있다.
Distributor
디스트리뷰터는 on-chip device나 external device에서
요청하는 SPI를 분배하는 역할을 한다.
Redistributor
Re-디스트리뷰터는 SGIs 와 PPIs 핸들링 역할을 한다.
LPIs의 메세지 기반의 인터럽트가 번역된 정보가 저장된
메모리에 접근해서 정보를 읽고 분배하는 역할을 한다.
CPU-INTERFACE
cpu interface는 GIC가 core로 보내는 인터럽트 신호(IRQ/FIQ)를 Core에 전달하는 역할,
Core의 Ack 신호를 GIC에 전달하는 역할을 한다.
즉, GIC와 Core 사이의 소통을 위한 장치라고 보면 된다.
GICv2에서는 GIC내의 components 중 하나였으나, GICv3부터 ARMv8의 시스템 레지스터가 되었다.
그래서 관련 레지스터들은 memory-map되어 있지 않다.
ARM의 시스템 레지스터는 아키텍처에서 인코딩된 주소로 매핑되어 있다.
Trace32 디버거에서 레지스터 접근을 위해 SPR: class를 사용해야 한다.
ITS (optional)
메세지 기반의 인터럽트의 메세지를 번역하는 역할을 한다.
GICv3 부터 지원하지만, 필수 요소는 아니다.
인터럽트 메세지를 DT / ITT / CT 를 참조하여 통해 번역한다.
DT: Device Table (MSI -> Device ID)
ITT: Interrupt Translation Table (MSI -> ICID)
CT: Collection Table (ICID -> Translate Interrupt)
번역된 정보를 system memory에 적재한다.
이 정보를 실제 커널 코드 분석을 통해 확인해야 한다.
Interrupts
인터럽트는 하던 작업을 중단하고, 바로 처리해야 하는 것
이라고 여기면 된다.
인터럽트의 종류는 매우 많기 때문에 이를 관리해줄 컨트롤러가 필요하다.
이를 인터럽트 컨트롤러라고 하며, GIC는 ARM에서 제공하는 인터럽트 컨트롤러 이다.
인터럽트는 GIC와 Core의 신호에 따라 상태가 결정된다.
‘Inactive’, ‘Pending’ ‘Active’ ‘Active & Pending’
인터럽트 type에 따라 인터럽트 상태 변화가 결정되며, 인터럽트 타입은 다음과 같다.
- Level sensitive
- Edge trigger
인터럽트 타입은 선택사항이지만, SGI와 LPI는 Edge trigger가 대부분이다.
Level sensitive에 한해서, SPI는 Active HIGH, PPI는 <span style-“color:red; background-color:#f5f5f5”>Active LOW</span> 이다.
Picture [1] GIC level sensitive
level sensitive는 pending 상태가 어떻게 정의되는지 잘 안보이지만,
peripheral -> gic 신호가 HIGH라면 ‘pending 상태가 존재한다’라고 보면 된다.
Picture [2] GIC edge trigger
edge trigger는 인터럽트가 처리 중일 때, 인터럽트 신호가 들어오면
Active & Pending 상태라고 보면 된다.
실제 인터럽트의 life-cycle에서 해당 상태에 따라 상태값을 저장하는 레지스터가 존재한다.
이 부분은 GIC register set을 참조하면 된다.
인터럽트는 그룹으로 관리된다.
인터럽트 그룹은 SPIs는
GICD_IGROUPRn 및 GICD_IGRPMODRn 레지스터 값, 그리고
인터럽트 그룹은 SGIs와 PPIs는
GICR_IGROUPR0 및 GICR_IGRPMODR0 레지스터 값, 그리고
GICD_CTLR.DS 레지스터 값에 따라 결정된다.
GICD_CTLR.DS == 0
| GICD_IGROUPR | GICD_IGRPMODR | Interrupt Group |
|————–|—————|—————–|
| 0 | 0 | G0S |
| 0 | 1 | G1S |
| 1 | 0 | NSG1 |
| 1 | 1 | x |
GICD_CTLR.DS == 1
| GICD_IGROUPR | Interrupt Group |
|————–|—————–|
| 0 | G0 |
| 1 | G1 |
인터럽트 핸들링관련 레지스터들은 그룹에 따라 구분되어 있다.
인터럽트의 종류에는 SGIs, PPIs, SPIs, LPIs가 있다.
이 다음부터는 각 인터럽트 종류별 특성에 대해 서술한다.
SGIs
‘Software Generated Interrupt’ 이다.
명령어를 통해 발생하는 인터럽트 이지만,
GIC에서 IRQ 신호를 Core에 보내면서 익셉션을 유발하기 때문에,
하드웨어 인터럽트로 분류된다.
SGIs는 인터럽트 번호 0 ~ 15번을 사용한다.
PPIs
‘Private Processor Interrupt’ 이다.
각 코어마다 연결되어 있는 Redistributor를 통해 분배된다.
대표적으로 타이머가 있다.
PPIs는 인터럽트 번호 16 ~ 31번을 사용한다.
PPIs는 인터럽트 번호 16(16 + 0)번 부터 시작한다.
(16(16 + 0)이라고 표현한 이유는 SPIs 설명을 보면 알 수 있다.)
PPIs는 SoC마다 제각각이지만, 주로 사용되는 번호가 몇가지 존재한다.
ARM Developer documents에서 den0094c 을 참조하면 된다.
해당 문서에서 설명하는 PPIs assignments 중에서
가상 인터럽트(for vPE)를 처리하기 위한 GIC의 고유 PPI는 25(16 + 9)를 사용한다.
커널 디바이스 트리에서 GIC 정의부분을 보면 다음과 같이 표현된 경우가 많다.
gic: gic@19960000 {
compatible = "arm,gic-v3";
regs = <0x0 0x19960000 0x0 0x10000>,
<0x0 0x19980000 0x0 0x20000>;
interrupt-controller;
interrupts = <GIC_PPI 9 IRQ_TYPE_LEVEL_HIGH>;
...
}
SPIs
Shared Peripheral Interrupts 이다.
흔히 ‘인터럽트가 발생했다’ 라고할 때 인터럽트가 SPI 이다.
어느 코어에서든 처리가 가능한 인터럽트 이며, on-chip 이나 external device 대부분 이에 해당된다.
SPIs는 인터럽트 번호 32 ~ 1019번을 사용한다.
SPIs는 인터럽트 번호 32(32 + 0)번 부터 시작한다.
32(32 + 0)이라고 표현한 이유는 예시를 보면 알 수 있다.
ARMv8기반 칩 중 하나의 예(여기서 X칩 이라고 하겠다.)를 가져왔다.
X칩의 인터럽트 테이블은 다음과 같다.
| 0 | EXTINT0
| 1 | EXTINT1
| 2 | EXTINT2
...
| 51 | IREQ_GPSBS
| 52 | IREQ_I2CM0
| 53 | IREQ_I2CM1
...
0번부터 시작해서 인터럽트 종류가 정해져 있다.
실제 인터럽트 번호는 표에 나와있는 인터럽트 번호에 SPI의 오프셋인 32를 더해주어야 한다.(단, 칩 설계마다 다를 수 있음)
예를 들어, I2CM0의 인터럽트 번호는 ‘(52 + 32) = 84’ 이다.
$ cat /proc/interrupts
CPU0 CPU1 CPU2 CPU3 CPU4 CPU5 CPU6 CPU7
...
63: 0 0 0 0 0 0 0 0 GICv3 84 Level .....i2c0
...
이 84번 i2c 인터럽트를 실제 Trace32로 확인할 수 있다.
Trace32로 인터럽트 학인하는 예제
(이미지의 워터마크는 회사의 Trace32 라이센스를 사용했기 때문에 어쩔 수 없음)
인터럽트의 life-cycle은
Generated - Distribute - Deliver - Pending - Activate - Priority Drop - Deactivation
이다.인터럽트 Active 상태일 때, 브레이크 포인트를 적절하게 설정하여
ISACTIVR, ICACTIVER 레지스터를 확인한다.
Pending 상태를 확인하는 방법은 T32 사용으로는 난해하다.
pending -> active 상태 넘어가는 기준은, core가 IAR 레지스터를 읽는 시점이며,
읽는 순간 ACK 신호 발생 및, 인터럽트 번호 식별이 이루어 지기 때문이다.
먼저, X칩의 GIC 레지스터 주소는 다음과 같다.
0x17600000 | Main Cluster GIC
예시로 드는 칩은 GIC-600 이 탑재되어 있어서, GIC 레지스터 셋은 GIC-600 을 참조한다.Base address는 0x17600000 이고,
ISACTIVR의 offset은 0x300 , ICACTIVER의 offset은 0x380 이다.
I2C 인터럽트 핸들러 진입 flow에서 break point를 걸고, 레지스터 dump를 했다.
다음 명령어를 통해 dump를 한다.
B:: d.dump a:0x17600300 /sl
i2c0 사용하여 실제 84번 인터럽트가 active 되는지 확인한다.
$ i2cdetect -y 0
picture [SPI - T32] Trace32 View I_ACTIVER
정확히 84번 인터럽트가 업데이트 되는 것을 확인할 수 있다.
이 상항에서 콘솔 로그는 다음과 같다.
[ERROR][I2C] transfer timeout, addr: a, result: 0
[ERROR][I2C] transfer timeout, addr: b, result: 0
[ERROR][I2C] transfer timeout, addr: c, result: 0
...
~~Trace32로 i2c 핸들러에 break point를 걸었기 때문에 인터럽트 처리 도중 멈추게 되었다.~~정확히는, ISPENDR 도 함께 확인해야 한다.
왜냐하면, edge-trigger type 은 인터럽트 핸들러 진입 시, ISACTIVR는 set 되지 않는다.
당연하게도, level-sensitive type 과 인터럽트 상태 머신이 다르기 때문이다.
커널 디바이스트리로 인터럽트 타입을 변경하고 테스트를 하면 위의 사진과 다른 모습을 확인할 수 있다.
한가지 더 고려할 부분이 있다. 위 이미지에서는 ICACTIVER도 함께 set 되어 있는데,
정확히는 clear 레지스터가 set이 되는 것을 조건으로 인터럽트 상태가 변화한다.
이는 ISPENDR / ICPENDR 역시 마찬가지 이다.
다만 함께 값이 변하는 것 처럼 보이는 것은, 예시 칩이 mirroring으로 설계 되어 있거나,
인터럽트 핸들러에서 clear 비트를 건들게 설계되어 있을 수 있다.
T32 사용 시 주의사항으로,
커널 elf 올리고 Trace32를 사용할 경우에는
반드시 WDT(watchdog timer)를 disable 하고 사용해야 한다.
커널 입장에서는 칩이 비정상 멈춤 상태로 인식하고 WDT로 강제로 Reset 하기 때문이다.
interrupt life-cycle 확인
인터럽트 life-cycle을 확인하는 방법은
해당 인터럽트의 enable 레지스터와 pending 레지스터가 모두 ‘1’인 상태여야 하고,
ICC_HPPIRn_EL1 레지스터 값이 유효한 레지스터 ID를 가지고 있을 때,
ICC_IARn_EL1 레지스터를 직접 읽었을 경우,
해당 인터럽트의 active 레지스터가 0에서 1로 set 된다.
인터럽트가 active -> end 상태가 되려면,
eoimode 에 따라, 0이면 ICC_EOIRn_ELx 레지스터에
ICC_IARn_EL1 에서 마지막으로 읽은 유효한 인터럽트 ID값을 쓰면, 인터럽트 우선순위와 상태가 함께 클리어되고,
eoimode 이 1이면, 인터럽트 우선순위만 클리어 된다.
이때, ICC_DIRn_ELx 에 해당 인터럽트 id를 쓰면 인터럽트 상태도 클리어 된다.
익셉션 레벨에 따라 라우팅 옵션을 바꾸기 위해선, ARMv8a architecture의 SCR_EL3 레지스터를 참조하면 된다.
n은 인터럽트 그룹이다.
LPIs
2가지 보안상태를 지원하는 환경에서,
LPIs 는 항상 Non-secure Group 1 이다.
2가지 보안상태란,(GICD_CTLR.DS=0)
첫번째는, Group 0 와 Group 1으로 나누는 것,
두번째는, Secure Group 1과 Non-secure Group 1 으로 나누기 때문에
2가지 보안상태를 지원하고, 2가지 보안상태라고 한다.
LPIs는 무조건 Edge-Trigger 방식을 지원한다.
각 인터럽트의 pending 정보는 레지스터가 아닌 메모리 내의
테이블에 저장되며 이 테이블은 Redistributor에 저장된 레지스터에 의해 참조된다.
메모리에 인터럽트의 정보를 저장하는 이유는 LPIs는 다른 종류의 인터럽트들 보다
훨씬 개수가 많기 때문이다.
위 내용은 kernel 이나 tf-a 코드를 통해 반드시 확인해야 한다.
LPIs 파트는 GIC가 ITS를 지원한다고 가정하고 작성하겠습니다.
LPIs는 ITS를 사용하여 장치의 EventID를 LPI INTID로 변환한다.
최소 8192개의 LPI를 지원하기 때문에,
이 때문에 각 인터럽트의 구성 및 각 인터럽트의 대기 정보는 레지스터가 아닌 메모리의 테이블에 저장되며,
테이블은 Redistributors에 있는 레지스터가 가리킨다.
LPIs 구성테이블은 GICR_PROPBASER 으로 결정된다.
LPIs pending 테이블은 GICR_PENDBASER 으로 결정된다.
SPIs가 상태 레지스터로 표현되었던 pending이 LPIs는 메모리에 해당 EventID가 저장되어 표현된다.
GIC가 서로 다른 LPI 구성 테이블을 가리키는 Redistributor를 지원하는지 여부는
구현 정의(IMPLEMENTATION DEFINED)
즉, 구현에 따라 GICR_PROPBASER 가 Redistributor마다 다를 수 있다.
그러나 서로 다른 LPIs 구성 테이블을 지원하지 않는 경우에는 모두 같은 테이블을 가리킨다.
Redistributor 중에 GICR_CTLR.EnableLPIs == 1인 것은, GICR_PROPBASER 가 가리키는
데이터를 캐시할 수 있다.
GICR_PENDBASER 레지스터는 물리적 LPI에 대한 LPI 대기 테이블의 기본 주소를 제공한다.
LPIs pending 테이블은 각 LPI가 현재 pending 상태인지 아닌지를 나타내는 비트를 메모리에 저장하는 테이블이다.
각 LPI는 특정 번호(INTID)를 가지며, 그 번호에 해당하는 비트가 LPI 대기 테이블에 저장된다.
비트값이 1 이면 pending 상태를 나타낸다.
Address | Data | Info |
---|---|---|
0x0001 | 00000000 | INTID 00 ~ 07 pending |
0x0002 | 00000000 | INTID 08 ~ 15 pending |
0x0003 | 00000000 | INTID 16 ~ 23 pending |
0x0004 | 00000000 | INTID 24 ~ 31 pending |
… | … | … |
pending table 역시 Redistributor마다 다른 테이블을 가리킬 수 있다.
구현에 따라 다르며, pending table은 2개이상이 될 수 있다는 뜻이다.
ITS는 LPIs를 특정 Redistributor에 전달하는 역할을 한다.
이때 LPIs는 EventID를 통해 특정 Redistributor에 매핑되고, 이 Redistributor는 해당 인터럽트를 처리한다.
Redistributor마다 담당하는 LPI가 고정적으로 할당되어 있으며,
이 Redistributor는 자신의 프로세서에 해당하는 LPI들의 대기 상태와 우선순위를 관리한다.
LPIs 사용 조건 확인
LPI를 사용하기 위해 확인해야 하는 checklist는 다음과 같다.
/* LPI configs */
1. GICD_CTLR.DS == ? (0 or 1)
` 0: Nonsecure Group 1
1: Group 1
2. GICR_CTLR.EnableLPIs == 1
3. GICR_TYPER.CommonLPIAff (Optional)
/* Address infomation */
1. GITS_BASER (memory structure)
2. GITS_CBASER (ITS command queue)
/* LPI tables */
1. GICR_PROPBASER
2. GICR_PENDBASER