Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 7개 있습니다.)

PC에 연결해 동작하는 자신만의 USB 장치 만들어 보기

컴퓨터를 사용하다가, ISA, PCI 등의 인터페이스 슬롯이 비어 있는 것을 보면 한 번쯤은 PC와 연결할 수 있는 장치를 만들어 보고 싶어질 때가 있습니다. 하지만 전기/전자 지식이 전혀 없어 그저 꿈만 같은 이야기일 뿐인데요, 그러던 것이 이제는 세상이 좋아져서 저 같은 응용 프로그래머도 쉽게 만들 수 있는 환경이 제공되고 있습니다.

이를 위해 Arduino나 Raspberry PI 등의 보드를 써도 됩니다. 이것들도 사실 따지고 보면 ISA, PCI 등의 인터페이스가 아닌 USB 인터페이스를 통해 연결한 장치나 다름없기 때문입니다. 하지만 그런 것들은 이미 다음의 글에서도 충분히 해봤기 때문에,

Raspberry Pi Zero(OTG)를 다른 컴퓨터에 연결해 가상 키보드 및 마우스로 쓰는 방법 (절대 좌표, 상대 좌표, 휠)
; https://www.sysnet.pe.kr/2/0/11369

.NET Micro Framework - 넷두이노 플러스
; https://www.sysnet.pe.kr/2/0/1392

Thinary Electronic - ATmega328PB 아두이노 호환 보드의 개발 환경 구성
; https://www.sysnet.pe.kr/2/0/11597

New NodeMcu v3 아두이노 마이크로 호환 보드의 기본 개발 환경 구성
; https://www.sysnet.pe.kr/2/0/11595

이번에는 좀 더 아래 수준으로 내려가 FTDI 부품을 이용해 보겠습니다. 실습을 위해 제가 가지고 있는 것은 다음의 부품입니다.

SparkFun FTDI Basic Breakout - 3.3V (DEV-09873)
; https://www.sparkfun.com/products/9873

[그림 1: FTDI 부품]
ftdi_device_0.jpg

이 부품은, USB 인터페이스를 가지면서 장치와의 통신은 COM 포트를 이용한 직렬 방식으로 처리할 수 있도록 해줍니다. 게다가, USB 케이블로 연결되는 장점 외에도 이를 장치로 인식하는 device driver가 기본 제공됩니다. 그러니까, 그냥 FTDI 장치를 가지고 있는 USB 케이블로 연결만 해주면, 처음에는 잠시 "Other devices"에 "FT232R USB UART"라고 떠 있다가 Ports와 Universal Serial Bus controllers 범주에 2개의 장치가 자동으로 다음과 같이 인식됩니다.

ftdi_device_1.png

"USB Serial Port (COM4)", "USB Serial Converter"로 되어 있는데, 다시 말해 이 장치는 USB 인터페이스로 통신은 하지만 결국 COM4에 해당하는 직렬 포트로 연결된 장치로 인식된 것입니다. 이와 함께 "USB Serial Port"의 속성창을 보면 "Port Settings"에 다음과 같이 직렬 통신에 관련된 프로토콜을 설정할 수 있습니다.

ftdi_device_2.png

직렬 장치라고 하니, 이제 일반적인 Serial 통신 클래스를 사용하면,

Serial Comms in C# for Beginners
; https://www.codeproject.com/Articles/678025/Serial-Comms-in-Csharp-for-Beginners

해당 장치와 데이터를 주고받는 것이 가능합니다. 예를 들어, C#으로 다음과 같이 만들면 현재 PC에 연결된 직렬 장치를 모두 열람할 수 있습니다.

using System;
using System.IO.Ports;

namespace ConsoleApp1
{
    class Program
    {
        static void Main(string[] args)
        {
            foreach (string portName in SerialPort.GetPortNames())
            {
                Console.WriteLine(portName);
            }
        }
    }
}

/* 출력 결과 - 장치 관리자의 "Ports" 화면에 있는 COM 포트 나열
COM1
COM5
COM6
COM4
*/

또한 다음과 같이 열고 닫을 수 있습니다.

using System;
using System.IO.Ports;

namespace ConsoleApp1
{
    class Program
    {
        static void Main(string[] args)
        {
            SerialPort comPort = OpenCOM(4); // 4번 == 장치 관리자에서 확인한 번호.

            Console.WriteLine(comPort.IsOpen); // True
            Console.ReadLine();

            comPort.Close();
            Console.WriteLine(comPort.IsOpen); // False
        }

        private static SerialPort OpenCOM(int portNumber)
        {
            SerialPort comPort = new SerialPort();

            // 장치 관리자의 Port Settings에 따라,
            comPort.PortName = $"COM{portNumber}";
            comPort.BaudRate = 9600;
            comPort.DataBits = 8;
            comPort.StopBits = StopBits.One;
            comPort.Handshake = Handshake.None;
            comPort.Parity = Parity.None;
            comPort.Open();

            return comPort;  
        }
    }
}

하지만, 우리가 연결한 FTDI 부품은 아무것도 하지 않는, 단순히 USB 연결을 Serial 인터페이스로 변환해 주는 역할만 하기 때문에 그림 1에 보면 장치 후면으로 (사진으로는 안 보이지만) 차례대로 DTR, RXI, TXO, 3V3, CTS, GND에 해당하는 연결 핀에 신호만 On/Off할 수 있습니다. 그중에서 간단하게 DTR 핀에 신호를 보내는 정도는 C#에서 SerialPort의 DtrEnabled 속성을 true로 바꿔주면 됩니다.

comPort.DtrEnable = true;

DTR 핀이 ON되었다는 것은 스위치가 연결된 것에 불과합니다. 즉, DTR로 전압이 가해지는 것은 아니고 단지 끊겼던 선이 연결된 것입니다. 따라서, DTR - GND 간의 전압 차는 없습니다. 전압을 인가하려면 3V3 핀에 연결하면 되는데, 3V3 핀은 언제나 3.3V 전압이 전달되고 있는 핀입니다. 그러니까, 아무 동작을 안 해도 USB 케이블로 FTDI 장치에 연결하면 3V3 - GND 간의 전압 차는 항상 3.3V가 있습니다. 달리 말하면, 3V3 - GND 핀에 LED를 연결해 두면 항상 불이 들어온다는 식인데, 이를 이용해 여러분들도 FTDI, 모터와 날개만 있으면 "USB 선풍기" 같은 것들을 만들 수 있는 것입니다. (근사하게 3D 프린터로 케이스를 만들면 더 좋겠지요. ^^)

그럼, DTR - 3V3 핀에 LED의 -/+ 극을 연결하면 어떻게 될까요? 이때는, DTR 신호가 OFF이면 회로가 단락 된 상태이기 때문에 LED에 불이 들어오지 않습니다. 반면 ON이 되면 회로가 연결된 상태로 바뀌기 때문에 DTR - 3V3 핀 간에는 3.3V 전압 차가 발생하고 그 사이에 연결된 LED에 불이 들어옵니다.




아쉽게도 제가 그 흔하디흔한 LED 부품이 없습니다. 그래서, DTR의 신호를 받아서 내장 LED를 켜줄 다음의 보드를 이용해 간단하게 실습해봤습니다.

New NodeMcu v3 아두이노 호환 보드의 기본 개발 환경 구성
; https://www.sysnet.pe.kr/2/0/11595

New NodeMcu v3 아두이노 호환 보드의 내장 LED 및 입력 핀 사용법
; https://www.sysnet.pe.kr/2/0/11605

동작 방식은 이렇습니다. DTR이 ON 되면 보드의 내장 LED를 켜고, DTR이 Off되면 내장 LED를 끄는 것입니다. 보드에서 신호를 받을 핀은 풀업 동작 모드를 활성화시킨 GPIO4 (D2) 핀으로 정하고 그럼 다음과 같이 배선을 하면 됩니다.

ftdi_device_3.png

간단하죠. ^^

이렇게 하면 DTR 신호가 없을 때 단락 된 것과 같고, 따라서 D2는 HIGH 상태에 머무르게 됩니다. 이때는 LED를 끌 것이므로 아두이노 스케치에서는 다음과 같이 처리할 수 있습니다.

const int buttonPin = 4; // GPIO4 - D2
const int ledPin = 2;    // GPIO2 - D4 - builtin led

int buttonState = 0;

void setup() {
  pinMode(ledPin, OUTPUT);
  pinMode(buttonPin, INPUT_PULLUP);
}

void LedOn() {
    digitalWrite(ledPin, LOW); // nodemcu 보드에서 내장 LED는 LOW 신호일 때 점등
}

void LedOff() {
    digitalWrite(ledPin, HIGH);
}

void loop() {
  buttonState = digitalRead(buttonPin);

  if (buttonState == HIGH) { // 스위치가 열렸으면,
    LedOff();
  } else { // 스위치가 닫혔으면,             
    LedOn();
  }

  delay(1000);
}

자, 그럼 C# 프로그램에서 DtrEnable 속성을 바꿔주는 것에 따라,

comPort.DtrEnable = true; // LED 점등

comPort.DtrEnable = false; // LED 점멸

LED가 동작하는 것을 확인할 수 있습니다. 어떠세요? 아주 초보적이지만 자신만의 USB 디바이스를 갖게 된 것입니다.

(첨부 파일은 이 글에 실린 다이어그램의 배열과 C# 프로젝트 파일입니다.)




그나저나, New NodeMcu v3 보드가 브레드보드에 붙여서 사용하기에는 영 좋지 않은 핀 배치를 가지고 있습니다. 다음과 같이 장착할 수 있는데, 우측 핀 배열과 좌측 핀 배열 사이의 공간이 정확히 8개의 기판 배열을 점유하고 있어서 배선을 위한 공간이 전혀 없게 됩니다.

ftdi_device_4.png

그래서 그림에서 보는 바와 같이 보드 하단의 영역에 미리 배선을 밖으로 빼놓는 작업을 해놔야 하는... 이런 말도 안 되는, 비상식적인 일을 해야 합니다. ^^;;;




참고로 다음의 영상을 보면,

Make Your Own USB Device Using FTDI 
; https://www.youtube.com/watch?v=mJa39CLbWBk

FTDI device driver를 직접 설치하는 방법을 알려주고 있습니다. 제 경우에는 그냥 디바이스가 인식되었다고 했는데, 만약 자신의 PC에 FTDI 장치를 USB 케이블로 연결했을 때 장치 관리자에 자동으로 "USB Serial Port (COM4)", "USB Serial Converter" 장치로 인식하지 않는다면, 다음의 사이트에 방문해,

D2XX Direct Drivers
; http://www.ftdichip.com/Drivers/D2XX.htm

페이지 하단에 있는 device driver를 다운로드하면 됩니다. 예를 들어, 64비트 윈도우 운영체제라면 다음의 파일을 다운로드해 장치를 인식시켜 주면 됩니다.

Windows* 2017-08-30 ver 2.12.28  
; http://www.ftdichip.com/Drivers/CDM/CDM%20v2.12.28%20WHQL%20Certified.zip

게다가 제 글에서는 .NET BCL의 System.IO.Ports.Serial 타입을 사용했는데 좀 더 저수준으로 제어하고 싶다면 동영상에서 소개하고 있는 다음의 라이브러리를 사용해도 됩니다.

Code Examples
; http://www.ftdichip.com/Support/SoftwareExamples/CodeExamples.htm

Code Examples - C#
; http://www.ftdichip.com/Support/SoftwareExamples/CodeExamples/CSharp.htm

FTD2XX_NET.DLL
; http://www.ftdichip.com/Support/SoftwareExamples/CodeExamples/CSharp/FTD2XX_NET_v1.1.0.zip

CodeExamples/CSharp/FTD2XX_NET.zip
; http://www.ftdichip.com/Support/SoftwareExamples/CodeExamples/CSharp/FTD2XX_NET.zip




이 글에서 사용한 FTDI Basic 부품은 miniUSB를 사용하는데, 이 케이블이 요즘에는 잘 사용하지 않습니다. 따라서 동일한 부품인데 microUSB 케이블을 사용하는 다음의 부품이 있으니 실습하실 때 참고하시면 됩니다.

SparkFun FT231X Breakout
; https://www.sparkfun.com/products/13263




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 7/10/2021]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 16  17  18  19  20  [21]  22  23  24  25  26  27  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
13094정성태7/6/20225981오류 유형: 816. Golang - "short write" 오류 원인
13093정성태7/5/20226908.NET Framework: 2029. C# - HttpWebRequest로 localhost 접속 시 2초 이상 지연
13092정성태7/3/20227835.NET Framework: 2028. C# - HttpWebRequest의 POST 동작 방식파일 다운로드1
13091정성태7/3/20226675.NET Framework: 2027. C# - IPv4, IPv6를 모두 지원하는 서버 소켓 생성 방법
13090정성태6/29/20225800오류 유형: 815. PyPI에 업로드한 패키지가 반영이 안 되는 경우
13089정성태6/28/20226291개발 환경 구성: 646. HOSTS 파일 변경 시 Edge 브라우저에 반영하는 방법
13088정성태6/27/20225418개발 환경 구성: 645. "Developer Command Prompt for VS 2022" 명령행 환경의 폰트를 바꾸는 방법
13087정성태6/23/20228348스크립트: 41. 파이썬 - FastAPI / uvicorn 호스팅 환경에서 asyncio 사용하는 방법 [1]
13086정성태6/22/20227785.NET Framework: 2026. C# 11 - 문자열 보간 개선 2가지파일 다운로드1
13085정성태6/22/20227851.NET Framework: 2025. C# 11 - 원시 문자열 리터럴(raw string literals)파일 다운로드1
13084정성태6/21/20226463개발 환경 구성: 644. Windows - 파이썬 2.7을 msi 설치 없이 구성하는 방법
13083정성태6/20/20227027.NET Framework: 2024. .NET 7에 도입된 GC의 메모리 해제에 대한 segment와 region의 차이점 [2]
13082정성태6/19/20226059.NET Framework: 2023. C# - Process의 I/O 사용량을 보여주는 GetProcessIoCounters Win32 API파일 다운로드1
13081정성태6/17/20226128.NET Framework: 2022. C# - .NET 7 Preview 5 신규 기능 - System.IO.Stream ReadExactly / ReadAtLeast파일 다운로드1
13080정성태6/17/20226757개발 환경 구성: 643. Visual Studio 2022 17.2 버전에서 C# 11 또는 .NET 7.0 preview 적용
13079정성태6/17/20224544오류 유형: 814. 파이썬 - Error: The file/path provided (...) does not appear to exist
13078정성태6/16/20226552.NET Framework: 2021. WPF - UI Thread와 Render Thread파일 다운로드1
13077정성태6/15/20226888스크립트: 40. 파이썬 - PostgreSQL 환경 구성
13075정성태6/15/20225842Linux: 50. Linux - apt와 apt-get의 차이 [2]
13074정성태6/13/20226142.NET Framework: 2020. C# - NTFS 파일에 사용자 정의 속성값 추가하는 방법파일 다운로드1
13073정성태6/12/20226345Windows: 207. Windows Server 2022에 도입된 WSL 2
13072정성태6/10/20226629Linux: 49. Linux - ls 명령어로 출력되는 디렉터리 색상 변경 방법
13071정성태6/9/20227215스크립트: 39. Python에서 cx_Oracle 환경 구성
13070정성태6/8/20227035오류 유형: 813. Windows 11에서 입력 포커스가 바뀌는 문제 [1]
13069정성태5/26/20229257.NET Framework: 2019. C# - .NET에서 제공하는 3가지 Timer 비교 [2]
13068정성태5/24/20227717.NET Framework: 2018. C# - 일정 크기를 할당하는 동안 GC를 (가능한) 멈추는 방법 [1]파일 다운로드1
... 16  17  18  19  20  [21]  22  23  24  25  26  27  28  29  30  ...