2010년 10월 22일 금요일

USB 포트로 장착된 장치 인식하기

요즘 진행하고 있는 프로젝트가 둘 있는데, 모두 하드웨어와 Interface하는 프로그램이다.
제작된 하드웨어가 컴퓨터에 연결되었을 때 이를 인식하여 Application level에서 일련의 동작을 수행하는 것으로, 하드웨어와 커널 중간에 Device Driver와 Filter Driver가 작동하고 있다.

이 때, 장치가 장착되었을 때와 탈착되었을 때를 알아야 이에 맞는 동작을 수행할 수 있어서 이런 Event를 잡아내는 것이 필요하며, Windows에서는 WM_DEVICECHANGE라는 메시지를 Broadcast하므로 이 메시지를 잡아 처리하면 알아낼 수 있다.

MSDN을 보면 함수 원형은
     afx_msg BOOL OnDeviceChange(UINT nEventType, DWORD dwData) ;
이고,
nEventType은 발생한 이벤트의 종류로써,
     DBT_DEVICEARRIVAL
     DBT_DEVICEQUERYREMOVE
     DBT_DEVICEQUERYERMOVEFAILD
     DBT_DEVICEREMOVEPENDING
     DBT_DEVICEREMOVECOMPLETE
     DBT_DEVICETYPESPECIFIC
     DBT_CONFIGCHANGED
등이 있다.
DWORD dwData는 각 event에 대한 자료(구조체)의 주소다.

MSDN 맨 아래 부분에 Note를 보면
This member function is called by the framework to allow your application to handle a Windows message.
...
이런 말이 써 있는데, 나를 비롯한 많은 사람들이 이 문장을 놓쳐서 고생했을 것으로 생각된다.
즉, Dialog base에서는 그냥 event에 대한 message handler를 작성하면 끝이지만 Doc/View 구조에서는 그렇게 하면 작동하지 않는다.

Dialog base에서는

XXXDlg.h
     afx_msg BOOL OnDeviceChange(UINT, DWORD);

XXXDlg.Cpp

BEGIN_MESSAGE_MAP(CAgentNewDlg, CDialog)
     //{{AFX_MSG_MAP(CAgentNewDlg)

     ...
     //}}AFX_MSG_MAP
     ON_MESSAGE(WM_BNB_NOTIFYTRAY, OnNotifyTray)
     ON_WM_DEVICECHANGE()
     ON_MESSAGE(WM_HOTKEY, OnHotKey)
END_MESSAGE_MAP()


...

BOOL CAgentNewDlg::OnDeviceChange(UINT nEventType, DWORD dwData)
{
   PDEV_BROADCAST_HDR hdr ;
   hdr = (PDEV_BROADCAST_HDR)dwData ;

    switch (nEventType)
    {
        case DBT_DEVICEARRIVAL:
            if(hdr->dbch_devicetype == DBT_DEVTYP_VOLUME)

                AfxMessageBox("Inserted!!") ;
        break ;

       case DBT_DEVICEREMOVECOMPLETE:
            if(hdr->dbch_devicetype != DBT_DEVTYP_VOLUME)
                 AfxMessageBox("Removed!!") ;
           break ;

    }
    return TRUE;
}
와 같이 동작시키면 된다.

위의 예제는 Volume에 대해서 장치를 장/탈착 할 경우, 동작한다.
가장 간단한 예로 USB를 장/탈착해보면 알 수 있다.

Doc/View 구조에서는

MainFrm.h
// Overrides
 // ClassWizard generated virtual function overrides
 //{{AFX_VIRTUAL(CMainFrame)
    public:
    virtual BOOL PreCreateWindow(CREATESTRUCT& cs);
    virtual void ActivateFrame(int nCmdShow = -1);
    protected:
    virtual LRESULT WindowProc(UINT message, WPARAM wParam, LPARAM lParam);
 //}}AFX_VIRTUAL


MainFrm.cpp
LRESULT CMainFrame::WindowProc(UINT message, WPARAM wParam, LPARAM lParam)
{
    if(WM_DEVICECHANGE == message)
    {

        상동...(Dialog base에서 OnDeviceChange 함수내의 이탤릭으로 된 부분)
    }

    return CFrameWnd::WindowProc(message, wParam, lParam);
}
과 같이 동작시킨다.

WindowProc은 ClassWizard에서 생성하면 쉽게 만들 수 있다.

그런데, 문제는 DDK Sample 중 USBView를 통해서 보면 확인가능하겠지만, PnP Manager에서 위 Message를 한 번만 Broadcasting하는 것이 아니라는 사실이다.
이거때매 완전좌절모드로...ㅠㅠ
꽁수를 써서 한 번만 동작하도록도 해보고, 다른 짓거리도 해 보았지만, 워낙 상황이 다양해서 모두 대처할 수 없어 폐인같은 삶을 지속하다가 한 줄기 단비와도 같은 함수 이름 하나를 듣게 되었다...
그 날의 감동이란..
외주업체에 Windows는 마지막 Message를 잡아 처리하는데 왜 나는 안되느냐며 떼를 썻고, 그 업체 직원들이 지인을 통해 수소문한 결과, SHChangeNotifyRegister라는 API를 알아왔다..^^

이 함수는 이름 앞에 붙은 접두어에서도 알 수 있듯이, Shell 함수이다.
A, B라는 메시지가 발생했을 때, 나한테 C라는 메시지를 날려라 라고 등록하는 함수로써, 이 함수가 성공하면, C라는 메시지에 대한 이벤트 처리만 하면 된다는 것이다.
보통 이 함수는 탐색기 같은 것 만들때, 폴더의 변화를 알아오거나 하는 경우 많이 사용하고, 알고 있던 함수였으나, 이런 오묘한 기능마저 하는 줄은 상상도 못했다.
역시 MSDN을 외우라는 건가..ㅠㅠ
원형은 다음과 같이 생겼다.
ULONG SHChangeNotifyRegister( 
    HWND hwnd,
    int fSources,
    LONG fEvents,
    UINT wMsg,     int cEntries,
    SHChangeNotifyEntry *pfsne
);


여기서, fEvent에
SHCNE_DRIVEADD | SHCNE_DRIVEREMOVED 를 조합해서 넣으면, 장치의 장/탈착을 알 수 있다.
이 것이 WM_DEVICECHANGE와 다른 점은 장/탈착되는 시점이 아닌 mount되는 시점을 안 다는 것이다.
따라서, 그냥 장착되었다는 것만 알 필요가 있다면 요놈이 더 편리하다.
그르나 바뜨, 이 것도 커널버전에 따라 적용 안될 수 있고, Platform SDK가 있어야 하는 등의 제한 사항은 있다.
MSDN 참조..ㅡㅡ;;

헤더에
    afx_msg LRESULT OnMediaChanged(WPARAM wParam, LPARAM lParam) ;

CPP에

    ON_MESSAGE(WM_USER_MEDIACHANGED, OnMediaChanged)
...

BOOL CNtfyDlg::OnInitDialog()
{
    HWND hWnd = GetSafeHwnd() ;
    LPITEMIDLIST ppidl ;
    if(NOERROR == SHGetSpecialFolderLocation(hWnd, CSIDL_DESKTOP, &ppidl))
    {
        SHChangeNotifyEntry shCNE ;
        shCNE.pidl = ppidl ;
        shCNE.fRecursive = TRUE ;

        m_ulSHChangeNotifyRegister = SHChangeNotifyRegister(hWnd,
            SHCNE_DISKEVENTS,
            SHCNE_DRIVEADD  | SHCNE_DRIVEREMOVED,
            WM_USER_MEDIACHANGED,
            1,
            &shCNE) ;

     }
     else
         ASSERT(0) ;

     ...
}

LRESULT CNtfyDlg::OnMediaChanged(WPARAM wParam, LPARAM lParam)
{
    SHNOTIFYSTRUCT* pshns = (SHNOTIFYSTRUCT*)wParam ;
    CString strPath, strMsg ;
    switch(lParam)
    {
    case SHCNE_DRIVEADD  :
        strPath = GetPathFromPIDL(pshns->dwItem1) ;
        if(FALSE == strPath.IsEmpty())
        {
            strMsg.Format("Media inserted into %s", strPath) ;
            AfxMessageBox(strMsg) ;
        }
        break ;


    case SHCNE_DRIVEREMOVED :
        strPath = GetPathFromPIDL(pshns->dwItem1) ;
        if(FALSE == strPath.IsEmpty())
       {
            strMsg.Format("Media removed into %s", strPath) ;
            AfxMessageBox(strMsg) ;
       }
       break ;
     }

      return NULL ;
}

위 소스는 CodeProject에서 가져와 나에게 맞도록 메시지 정도만 수정한 것이며, 원문을 보고 싶다면 여기(Using the shell to receive notification of removable media being inserted or removed.)를 클릭!!

샘플은 냉중에...^^

 

출처 http://andy9273.egloos.com/

댓글 없음:

댓글 쓰기