본문으로 건너뛰기
피드

리눅스 커널에서 하드웨어/소프트웨어 인터페이스가 동작하는 원리

backend 약 4분
vote
0
댓글
북마크

리눅스에서 유저스페이스-커널 드라이버-하드웨어 간 인터페이스가 어떻게 동작하는지를 /dev 파일시스템, MMIO, 물리 주소 매핑까지 단계별로 설명하는 글. 커널 드라이버의 최하위 레벨은 결국 임베디드 베어메탈 프로그래밍이라는 점을 UART 드라이버 코드로 보여줌.

  • 1

    리눅스는 하드웨어를 /dev 가상 파일시스템의 파일로 노출하고 일반 시스콜로 접근 가능

  • 2

    드라이버와 하드웨어 간 통신은 대부분 MMIO(메모리 매핑 I/O)로 이뤄짐

  • 3

    물리 주소 매핑은 SoC 또는 마더보드의 디지털 로직이 구현

  • 결론부터 말하면, 핵심은 메모리임. 유저스페이스 앱 → 커널 드라이버 → 하드웨어 디바이스, 이 3단 스택이 전부임

  • 리눅스 커널은 하드웨어 디바이스를 /dev 가상 파일시스템의 파일로 노출함. 유저스페이스에서 open, read, write, mmap 같은 일반 시스콜을 그대로 쓸 수 있고, 디바이스별 특수 기능은 ioctl로 접근함

  • 이 파일들을 만드는 건 커널 드라이버인데, 드라이버의 존재 이유는 딱 하나 — 하드웨어를 추상화해서 OS나 유저스페이스가 쓸 수 있게 만드는 것임. I2C, SPI 같은 커널 내부 드라이버 프레임워크 위에 구현됨. /dev 파일에 접근하면 드라이버의 콜백 핸들러가 프로세스 컨텍스트에서 바로 실행됨

드라이버는 하드웨어랑 어떻게 통신하나

  • 요즘은 대부분 메모리 매핑 I/O(MMIO) 방식임. 디바이스 하드웨어가 특정 물리 주소에 "나타나는" 거고, CPU의 load/store 명령어로 디바이스가 정의한 "API"를 통해 통신함. 센서 데이터를 읽으려면 물리 주소를 읽기만 하면 되는 거임

  • MMIO 말고 다른 주요 인터페이스는 인터럽트임. 폴링 대신 인터럽트 기반 I/O 처리를 하는 건데, 드라이버가 특정 IRQ 번호에 핸들러를 등록하면 커널의 인터럽트 인프라가 호출해줌

  • MMIO 쓰는 건 PIC나 Arduino(AVR)에서 하는 베어메탈 프로그래밍이랑 거의 똑같음. 커널 드라이버의 최하위 레벨은 사실상 임베디드 베어메탈 프로그래밍이라는 거임

실제 코드로 보면

  • ARM 플랫폼의 UART(시리얼 포트) 드라이버 예시(amba-pl011.c)에서 진짜 마법이 일어나는 부분은 이거임:

    cr = readb(uap->port.membase + UART010_CR);

    디바이스 베이스 주소에 레지스터 오프셋을 더해서 읽는 것 뿐임. UART010_CR0x14로 정의된 컨트롤 레지스터임

  • 디바이스마다 레지스터가 몇 개에서 수십 개까지 다양함

물리 주소 매핑은 어떻게 구현되나

  • 디바이스가 특정 물리 주소에 매핑되는 건 CPU 바깥의 디지털 로직이 담당함. 임베디드 시스템에서는 SoC 위에, PC에서는 마더보드 위에 구현됨

  • CPU의 물리 인터페이스는 주소 버스, 데이터 버스, 컨트롤 버스로 구성됨. 주소 버스의 비트를 디지털 로직이 디코딩해서 해당 디바이스를 배타적으로 활성화(enable)함. 컨트롤 버스에서 읽기/쓰기 비트를 설정하고, 데이터 버스로 실제 데이터가 오감

  • 실제 구현 문서는 SoC 제조사 고객이 아니면 구하기 어렵지만, Tegra 2 SoC 같은 오래된 칩은 블록 다이어그램이 공개되어 있음. Motorola Atrix 4G 등에 탑재됐던 그 칩임

  • 고전적인 PIC 마이크로컨트롤러가 SoC 구조를 이해하기에 훨씬 접근성이 좋음. 하나의 MIPS 코어에 주변 장치들이 페리페럴 버스로 연결된 구조가 한눈에 보임. 심지어 "Peripheral Bridge"를 통해 I2C, SPI 같은 장치들이 두 번째 버스에 붙어있는 계층 구조까지 확인 가능함

커널 드라이버 개발이 신비로운 영역 같지만, 결국 물리 주소에 load/store 하는 임베디드 프로그래밍이라는 걸 코드 레벨로 보여주는 좋은 입문 글임.

댓글

댓글

댓글을 불러오는 중...

backend

Go에서 Rust로 옮길 때 진짜로 바뀌는 것들

이 글은 Go 백엔드 서비스를 Rust로 옮길 때 속도보다 컴파일 타임 보장, 런타임 트레이드오프, 개발자 경험이 더 중요하다고 설명한다. nil 패닉, 데이터 레이스, 에러 처리, 제네릭, 비동기 모델, 마이그레이션 전략까지 실무 관점에서 Go와 Rust를 길게 비교한다.

backend

Python 3.15에서 헤드라인은 못 탔지만 꽤 쓸만한 기능들

Python 3.15에는 lazy imports나 Tachyon profiler 같은 큰 기능 말고도 실무에서 바로 체감될 만한 작은 개선들이 들어가. TaskGroup 취소, 컨텍스트 매니저 데코레이터 개선, 스레드 안전 이터레이터처럼 평소 애매하게 불편했던 지점들이 꽤 깔끔해졌어.

backend

심평원, DUR부터 의료영상 심사까지 클라우드로 갈아엎는다

심평원이 정보시스템 클라우드 전환과 함께 병·의원 업무에 직접 닿는 DUR, 의료영상 AI 심사, 요양급여내역 조회 시스템을 고도화한다. 핵심은 설치형 프로그램 중심이던 연계를 웹과 API 기반으로 넓히고, 진료·청구 과정에서 실시간 확인과 자동 판독을 강화하는 쪽이다.

backend

윈도우 에러 코드 7번 ‘ERROR_ARENA_TRASHED’는 어디서 왔을까

ERROR_ARENA_TRASHED는 Win32에서 실제로 쓰이는 현대적 에러라기보다 MS-DOS 시절 메모리 관리 구조에서 넘어온 잔재야. MS-DOS가 메모리 블록 앞의 arena 시그니처를 훑다가 예상한 값이 아니면 ‘arena가 망가졌다’고 보고 이 에러를 냈다는 이야기야.

backend

C/C++ 컴파일러의 느슨한 메모리 동시성 버그를 자동으로 잡는 박사논문

C와 C++ 컴파일러에서 relaxed memory 동시성 버그를 찾는 자동 테스트 프레임워크를 다룬 박사논문이 공개됐어. Téléchat, Atomic-mixer 같은 도구로 소스 수준 동작과 컴파일된 프로그램 동작을 비교하고, LLVM과 GCC 툴체인에서 실제 버그를 찾아낸 내용이 핵심이야.