입출력 작업을 수행하려면 CPU와 장치 컨트롤러가 정보를 주고받아야 한다.
그 방법으로는 무엇이 있을까?

프로그램 입출력은 기본적으로 프로그램 속 명령어로 입출력 장치를 제어하는 방법이다. CPU가 프로그램 속 명령어를 실행하는 과정에서 입출력 명령어를 만나면 CPU는 입출력장치에 연결된 장치 컨트롤러와 상호작용하며 입출력 작업을 수행한다.![]
메모리에 저장된 하드 디스크를 백업한다는 말은 하드 디스크에 새로운 정보를 쓴다는 말과 같다.

cpu -> 하드디스크 컨트롤러의 제어 레지스터에 쓰기 명령을 보냄
하드 디스크 컨트롤러는 하드 디스크 상태를 확인한다.

하드 디스크가 준비된 상태라면 하드 디스크 컨트롤러는 상태 레지스터에 준비되었다고 표시한다.
1.CPU는 상태 레지스터를 주기적으로 읽어보며 하드 디스크의 준비 여부를 확인한다.
2.1 하드 디스크가 준비됐음을 CPU가 알게 되면 백업할 메모리의 정보를 데이터 레지스터에 쓴다.
2.2 아직 백업 작업(쓰기 작업)이 끝나지 않았다면 1번부터 반복.
3. 쓰기가 끝났다면 작업을 종료

...그런데... CPU는 입출력장치들의 주소를 어떻게 아는 걸까? 정확히 말해 CPU는 장치 컨트롤러의 레지스터들을 어떻게 아는 걸까?
-> 메모리 맵 입출력, 고립형 입출력 방식으로 !
메모리 맵 입출력은 메모리에 접근하기 위한 주소 공간과 입출력 장치에 접근하기 위한 주소 공간을 하나의 주소 공간으로 간주하는 방법이다.

EX) 가령 1,024개의 주소를 표현할 수 있는 컴퓨터가 있을 때 1,024개 전부 메모리 주소를 표현하는 데 사용하지 않는다. 512개는 메모리 주
소를, 512개는 장치 컨트롤러의 레지스터를 표현하기 위해 사용한다.
<주소 공간 일부를 예를 들어 설명>
ex) CPU는 ‘517번지를 읽어 들여라’라는 명령어로 키보드 상태를 읽을 수 있다.
ex) ‘518번지에 a를 써라’라는 명령어로 하드 디스크 컨트롤러의 데이터 레지스터로 데이터를 보낼 수 있다.
메모리 맵 입출력 방식에서 CPU는 메모리의 주소들이나 장치 컨트롤러의 레지스터들이나 모두 똑같이 메모리 주소를 대하듯 하면 된다는 점이다. 그래서 메모리에 접근하는 명령어와 입출력장치에 접근하는 명령어는 굳이 다를 필요가 없다.
CPU가 ‘517번지를 읽어라’라는 명령어를 실행했을 때 517번지가 메모리상의 주소를 가리킨다면 CPU는 메모리 517번지에 저장된 정보를 읽어 들일 것이고, 517번지가 프린터 컨트롤러의 상태 레지스터를 가리킨다면 CPU는 프린터의 상태를 확인할 수 있을 것이니까.
고립형 입출력은 메모리를 위한 주소 공간과 입출력장치를 위한 주소 공간을 분리하는 방법이다.
가령 1,024개의 주소 공간을 가진 컴퓨터가 있다고 가정해보자.
아래 그림처럼 제어 버스에 ‘메모리 읽기/쓰기’ 선 이외에 ‘입출력장치 읽기/쓰기’ 선이 따로 있다면 메모리에도 1,024개의 주소 공간을 활용하고, 입출력장치도 1,024개의 주소 공간을 활용할 수 있다. CPU가 메모리 읽기/쓰기 선이 활성화되는 명령어를 실행할 때는 메모리에 접근하고, 입출력장
치 읽기/쓰기 선이 활성화되는 명령어를 실행할 때는 장치 컨트롤러에 접근하기 때문이다.

고립형 입출력 방식에서 cpu는 입출력장치에 접근하기 위해 메모리에 접근하는 명령어와는 다른 (입출력 읽기/쓰기 선을 활성화시키는) 입출력 명령어를 사용한다. 메모리에 접근하는 명령어와 입출력장치에 접근하는 명령어는 굳이 다를 필요가 없었던 메모리 맵 입출력과 대조적!!

입출력 장치에 의한 하드웨어 인터럽트는 정확히 말하자면 장치 컨트롤러에 의해 발생한다.
이렇게 인터럽트를 기반으로 하는 입출력을 인터럽트 기반 입출력이라고 한다.
폴링인터럽트와 자주 비교되는 개념 중 폴링이란, 입출력 장치의 상태는 어떤지, 처리할 데이터가 있는지를 주기적으로 확인하는 방식폴링 방식은 당연히 인터럽트 방식보다 CPU부담이 더 크다. 인터럽트를 활용하면 CPU가 인터럽트 요청을 받을 때까지 다른 일을 할 수 있기 때문이다.
여러 입출력 장치에서 인터럽트가 동시에 발생한 경우


우선순위를 반영하여 다중 인터럽트를 처리하는 방법에는 여러 가지가 있지만, 많은 컴퓨터에서는
프로그래머블 인터럽트 컨트롤러(PIC)라는 하드웨어를 사용한다.

PIC는 여러 장치 컨트롤러에 연결되어 장치 컨트롤러에서 보낸 하드웨어 인터럽트 요청들의 우선순위를 판별한 뒤 CPU에 지금 처리해야 할 하드웨어 인터럽트는 무엇인지를 알려주는 장치이다.

PIC의 다중 인터럽트 처리 과정은 아래와 같다.
1. PIC가 장치 컨트롤러에서 인터럽트 요청 신호(들)를 받아들인다.
2. PIC는 인터럽트 우선순위를 판단한 -> CPU에 처리해야 할 인터럽트 요청 신호 전송
3. CPU는 PIC에 인터럽트 확인 신호 전송
4. PIC는 데이터 버스를 통해 CPU에 인터럽트 벡터를 전송
5. CPU는 인터럽트 벡터를 통해 인터럽트 요청의 주체를 알게 되고, 해당 장치의 인터럽트 서비스 루틴 실행
일반적으로 더 많고 복잡한 장치들의 인터럽트를 관리하기 위해 아래와 같이 PIC를 두 개 이상 계층적으로 구성함.

PIC가 무시할 수 없는 NMI까지 우선순위를 판별하지 않는다.NMI는 우선순위가 가장 높아 판별이 불필요하기 때문이다. PIC가 우선순위를 조정해 주는 인터럽트는 인터럽트 비트를 통해 막을 수 있는 하드웨어 인터럽트이다.
프로그램 기반 입출력과 인터럽트 기반 입출력의 공통점
입출력 장치와 메모리 사이에 전송되는 모든 데이터가 반드시 CPU를 거친다면 CPU는 입출력장치를 위한 연산 때문에 시간을 뺏기게 된다.
그래서 입출력장치와 메모리가 CPU를 거치지 않고도 상호작용할 수 있는 입출력 방식인 DMA(Direct MEmory Access)가 등장하였다.
DMA 입출력을 하기 위해서는 시스템 버스에 연결된 DMA 컨트롤러라는 하드웨어가 필요하다.


1. CPU 는 DMA 컨트롤러에 입출력 장치의 주소, 수행할 연산(읽기/쓰기), 읽거나 쓸 메모리의 주소등과 같은 정보로 입출력 작업을 명령함.

2. DMA 컨트롤러는 CPU 대신 장치 컨트롤러와 상호작용하며 입출력 작업을 수행. 이떄 DMA 컨트롤러는 필요한 경우 메모리에 직접 접근하여 정보를 읽거나 쓴다.

3. 입출력 작업이 끝나면 DMA 컨트롤러는 CPU에 인터럽트를 걸어 작업이 끝났음을 알린다.
CPU는 DMA 컨트롤러에게 입출력 작업 명령을 내리고, 인터럽트만 받으면 되기 때문에 작업 부담을 훨씬 줄일 수 있다. 즉, CPU는 오로지 입출력의 시작과 끝에만 관여하면 된다.
하지만 DMA 컨트롤러는 시스템 버스로 메모리에 직접 접근하는데, 시스템 버스는 공용 자원이기 떄문에 동시 사용이 불가능하다. 따라서 CPU가 시스템 버스를 사용할 때 DMA 컨트롤러는 시스템 버스를 사용할 수 없고, DMA 컨트롤러가 시스템 버스를 사용할 때는 CPU가 시스템 버스를 사용할 수 없다.
그래서 아래의 두 가지 방법을 사용한다.

위와 같은 DMA의 시스템 버스 이용을 사이클 스틸링이라고 부른다.
CPU, 메모리, DMA 컨트롤러, 장치 컨트롤러가 모두 같은 버스를 공유하는 구성에서는 DMA를 위해 한 번 메모리에 접근할 때마다 시스템 버스를 두 번 사용하게 되는 부작용이 있다.
이 경우...
1. 메모리에서 DMA 컨트롤러로 데이터를 가져오기 위해 시스템 버스를 한 번 사용하고,
2. DMA 컨트롤러의 데이터를 장치 컨트롤러로 옮기기 위해 시스템 버스를 또 한 번 사용한다.

DMA를 위해 시스템 버스를 너무 자주 사용하면 그만큼 CPU가 시스템 버스를 이용하지 못한다. 이 문제는 DMA 컨트롤러와 장치 컨트롤러들을 입출력 버스라는 별도의 버스에 연결하여 해결할 수 있다.
위 그림과 같이 장치 컨트롤러들이 시스템 버스가 아닌 입출력 버스로 DMA 컨트롤러에 연결된다면 DMA 컨트롤러와 장치 컨트롤러가 서로 데이터를 전송할 때는 시스템 버스를 이용할 필요가 없으므로 시스템 버스의 사용 빈드롤 줄일 수 있다.
현대 컴퓨터에는 입출력 버스가 있다.
즉, 대부분의 입출력장치(장치 컨트롤러)는 시스템 버스가 아닌 입출력 버스와 연결된다.
입출력 버스에는 PCI(Peripheral Compoent Interconnect)버스, PCI Express(PCIe) 버스등 여러 종류가 있다.
아래의 그림은 여러 입출력장치들을 PCIe 버스와 연결해 주는 통로인 PCIe 슬롯이다.

우리가 사용하는 거의 모든 입출력장치들은 이렇게 입출력 버스와 연결되는 통로를 통해 시스템 버스를 타고 CPU와 정보를 주고받는다.