Click here to Skip to main content
Click here to Skip to main content

Enumerate Installed Devices Using Setup API

By , 16 Mar 2004
 

Sample Image - EnumDevices.jpg

Introduction

Windows has a rich collection of APIs to get useful information about installed devices. In this article, I will show how you can enumerate devices on a machine using Setup API.

The program has a simple GUI: only a list control that shows all of the installed devices. All the information about a device (such as name and its icon) is grabbed from Windows Setup API.

Setup API

The Setup application programming interface (API) provides a set of functions that your setup application can call to perform installation operations or get several information about installed devices, their class and also their GUID (a unique identifier for every device).

The application requires the following APIs (description of these APIs was taken from MSDN):

WINSETUPAPI BOOL WINAPI SetupDiGetClassImageList(
        OUT CLASSIMAGELIST_DATA ClassImageListData);

The SetupDiGetClassImageList function builds an image list that contains bitmaps for every installed class and returns the list in a data structure.

WINSETUPAPI BOOL WINAPI SetupDiDestroyClassImageList(
        IN PSP_CLASSIMAGELIST_DATA ClassImageListData);

The SetupDiDestroyClassImageList function destroys a class image list that was built by a call to SetupDiGetClassImageList.

WINSETUPAPI BOOL WINAPI SetupDiGetClassImageIndex(
        IN PSP_CLASSIMAGELIST_DATA ClassImageListData, 
        IN LPGUID ClassGuid, 
        OUT PINT ImageIndex);

The SetupDiGetClassImageIndex function retrieves the index within the class image list of a specified class.

CMAPI CONFIGRET WINAPI CM_Connect_Machine(
            IN PCTSTR  UNCServerName,
            OUT PHMACHINE  phMachine);

CMAPI CONFIGRET WINAPI CM_Disconnect_Machine(
            IN HMACHINE  hMachine);

The CM_Connect_Machine function creates a connection to a local or remote machine. The CM_Disconnect_Machine function removes a connection to a local or remote machine.

CMAPI CONFIGRET WINAPI CM_Locate_DevNode_Ex(
        OUT PDEVINST  pdnDevInst,
        IN DEVINSTID  pDeviceID,  OPTIONAL
        IN ULONG  ulFlags,
        IN HMACHINE  hMachine);

The CM_Locate_DevNode_Ex function obtains a device instance handle to the device node that is associated with a specified device instance identifier, on a local or a remote machine.

pDeviceID represents a device instance identifier. If this value is NULL, or if it points to a zero-length string, the function supplies a device instance handle to the device node at the top of the device tree. The PnP Manager maintains a device tree that keeps track of the devices in the system. The following figure shows the device tree for a sample system configuration. The device tree contains information about the devices present on the system. The PnP Manager builds this tree when the machine boots, using information from drivers and other components, and updates the tree as devices are added or removed. Each node of the device tree is called a device node, or devnode. A devnode consists of the device objects for the device's drivers, plus internal information maintained by the system. Therefore, there is a devnode for each device stack.

Device Tree

CMAPI CONFIGRET WINAPI CM_Get_Child_Ex(
        OUT PDEVINST  pdnDevInst,
        IN DEVINST  dnDevInst,
        IN ULONG  ulFlags,
        IN HMACHINE  hMachine);

The CM_Get_Child_Ex function is used to retrieve a device instance handle to the first child node of a specified device node, in a local or a remote machine's device tree.

CMAPI CONFIGRET WINAPI CM_Get_Sibling_Ex(
        OUT PDEVINST  pdnDevInst,
        IN DEVINST  DevInst,
        IN ULONG  ulFlags,
        IN HMACHINE  hMachine);

The CM_Get_Sibling_Ex function obtains a device instance handle to the next sibling node of a specified device node, in a local or a remote machine's device tree.

Solution

According to the above functions, we should first establish a connection to our computer (local or remote), then get the top node of the device tree (Root device). After this, we should enumerate the installed devices by a simple loop. The application uses three function: EnumDevices(), RetrieveSubNodes() and GetDeviceName(). Following is the source of them:

void EnumDevices()
{
    TCHAR LocalComputer[MAX_PATH];
    DWORD Size = MAX_PATH - 2;
    
    GetComputerName(LocalComputer + 2, &Size);
    LocalComputer[0] = _T('\\');
    LocalComputer[1] = _T('\\');
    
    CONFIGRET cr;
    cr = CM_Connect_Machine(LocalComputer, &m_hMachine);
    
    if (cr != CR_SUCCESS)
    {
        TCHAR Text[MAX_PATH];
        wsprintf(Text, _T("Machine Connection failed, cr= %lx(hex)\n"), cr);
    ::MessageBox(m_hWnd, Text, _T("Error"), MB_ICONSTOP | MB_OK);
        return;
    }    
    
    //Set Image List 
    m_ImageListData.cbSize = sizeof(m_ImageListData);
    SetupDiGetClassImageList(&m_ImageListData);

    m_ImageList.Attach(m_ImageListData.ImageList);
    m_Devices.SetImageList(&m_ImageList, LVSIL_SMALL);
    m_Devices.SetImageList(&m_ImageList, LVSIL_NORMAL);

    DEVNODE dnRoot;
    CM_Locate_DevNode_Ex(&dnRoot, NULL, 0, m_hMachine);
    
    DEVNODE dnFirst;
    CM_Get_Child_Ex(&dnFirst, dnRoot, 0, m_hMachine);    

    RetrieveSubNodes(dnRoot, NULL, dnFirst);

    CM_Disconnect_Machine(m_hMachine);
}

void RetrieveSubNodes(DEVINST parent, DEVINST sibling, DEVNODE dn)
{
    DEVNODE dnSibling, dnChild;
    
    do
    {
        CONFIGRET cr = CM_Get_Sibling_Ex(&dnSibling, dn, 0, m_hMachine);
        
        if (cr != CR_SUCCESS)
        dnSibling = NULL;
        
    TCHAR GuidString[MAX_GUID_STRING_LEN];
    ULONG Size = sizeof(GuidString);
        
    cr = CM_Get_DevNode_Registry_Property_Ex(dn, CM_DRP_CLASSGUID, NULL,
        GuidString, &Size, 0,  m_hMachine);
        
    if (cr == CR_SUCCESS)
    {
        GUID Guid;
        GuidString[MAX_GUID_STRING_LEN - 2] = _T('\0');
        UuidFromString(&GuidString[1], &Guid);
        int Index;
        if (SetupDiGetClassImageIndex(&m_ImageListData, &Guid, &Index))
        {
                CString DeviceName=GetDeviceName(dn);
        m_Devices.InsertItem(m_Devices.GetItemCount(), DeviceName, Index);
        }
    }
        
    cr = CM_Get_Child_Ex(&dnChild, dn, 0, m_hMachine);
    if (cr == CR_SUCCESS)
    {
        RetrieveSubNodes(dn, NULL, dnChild);
    }
        
    dn = dnSibling;
        
    } while (dn != NULL);
}

CString GetDeviceName(DEVNODE DevNode)
{
    CString    strType;
    CString    strValue;
    CString DisplayName;
    LPTSTR    Buffer;
    
    int  BufferSize = MAX_PATH + MAX_DEVICE_ID_LEN;
    ULONG  BufferLen = BufferSize * sizeof(TCHAR);
    
    Buffer  = strValue.GetBuffer(BufferSize);
    if (CR_SUCCESS == CM_Get_DevNode_Registry_Property_Ex(DevNode,
        CM_DRP_FRIENDLYNAME, NULL,
        Buffer, &BufferLen, 0, m_hMachine))
    {
    DisplayName = Buffer;
    }
    else
    {
    BufferLen = BufferSize * sizeof(TCHAR);
        
    if (CR_SUCCESS == CM_Get_DevNode_Registry_Property_Ex(DevNode,
        CM_DRP_DEVICEDESC, NULL,
        Buffer, &BufferLen, 0, m_hMachine))
    {
        DisplayName = Buffer;
    }
    else
    {
        DisplayName=_T("Unknown Device");
    }
    }
    
    return DisplayName;
}

m_Devices is a list control, m_ImageList is an image list of device icons and m_hMachine is a handle of the connected machine.

Enjoy!

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here

About the Author

A. Riazi
Iran (Islamic Republic Of) Iran (Islamic Republic Of)
Member
I was born in Shiraz, a very beautiful famous city in Iran. I started programming when I was 12 years old with GWBASIC. Since now, I worked with various programming languages from Basic, Foxpro, C/C++, Visual Basic, Pascal to MATLAB and now Visual C++.
I graduated from Iran University of Science & Technology in Communication Eng., and now work as a system programmer for a telecommunication industry.
I wrote several programs and drivers for Synthesizers, Power Amplifiers, GPIB, GPS devices, Radio cards, Data Acqusition cards and so many related devices.
I'm author of several books like Learning C (primary and advanced), Learning Visual Basic, API application for VB, Teach Yourself Object Oriented Programming (OOP) and etc.
I'm winner of January, May, August 2003 and April 2005 best article of month competetion, my articles are:

You can see list of my articles, by clicking here


Sign Up to vote   Poor Excellent
Add a reason or comment to your vote: x
Votes of 3 or less require a comment

Comments and Discussions

 
You must Sign In to use this message board.
Search this forum  
    Spacing  Noise  Layout  Per page   
GeneralMy vote of 5membermanoj kumar choubey26 Feb '12 - 19:55 
Nice
QuestionLicense for this codemembermikejurka30 Apr '09 - 5:36 
Hello,
 
Thanks for this excellent article! I was wondering what license the code is offered under. Could I use it in a proprietary commercial product?
 
Michael J
GeneralFinding a specific device...memberSharath Madhuranath14 Apr '09 - 1:50 
Hi... First off its a great article and is something along the lines of what I was looking for. My question is, if i need to find a particular device.. in my case a Printer from this tree, how would i go about doing it?
 
Any help is appreciated. Am not much into device drivers or programming at this level.
 
-
Sharath
Generalserial number of a connected mass storage devicemembermadddi9 Mar '09 - 7:56 
Hi,
 
gud stuff
But is it possible to get the serial numebr or the product and vendor id of the connected mass storage device.
 
I tried to get the job done by using
CeateFile() and DeviceIoControl (using IOCTL_sTORAGE_MEDIA_SERIAL_NUMBER)
given the drive letter of the drive.
 
CreateFile is successful and I get a valid handle.
But DeviceIoControl fails with the error code -ERROR_ACCESS_VIOLATION (0X32).
 
Help appriciated
Thanks in advance
QuestionCan you tell me how to enumerate 3G device from PCmemberalert091927 Jul '08 - 23:30 
Hello,Riazi
First ,I want to enumerate 3Gor(WWWAN) devices from PC and
second, I want to disable or enable those device and third
control thems connection like connect and disconnect
 
how can I do this, can I know if the PC connect to the Internet via 3G device and I can now it is 3G device
because I found the 3G device show information from PC only to know its enumerate a NIC card or modem
I can not know it is a 3G devices
 
I search from google about 1 months everyday,but my job need to complete it but I still have no idea,
can you give me some hint or sample code or reference
my develop tools is viusal stduio 6.0
my develop enviorment is windows xp
 
my email is : alert0919@gmail.com

best regards
Eric
QuestionCM_Get_DevNode_Registry_Property_Ex not safememberIngeer13 Nov '07 - 13:25 
I checked MSDN which says 'Do not use CM_Get_DevNode_Registry_Property_Ex, instead, use SetupDiGetDeviceRegistryProperty', another function takes different input parameters for reaching the properties. Any idea how we gonna bridge to that?
QuestionHow to detect if a network card is disabledmembersamaruf9 May '07 - 1:05 
I have used your code, but it only shows the network cards that are currently present and active. I'd like to be able to the tell the difference between a network card that is present but disabled and one that is active.
 
Thank you
 
Shakil
Generalhard disk serialmembernicolas20027 Apr '07 - 13:34 

Hello,
 
Using setupapi can I get hard disk serial info or smbios info?
GeneralRe: hard disk serialmemberA. Riazi30 Apr '07 - 23:31 
Hi,
Just see my other article: System Information[^]
 
Best regards,
A. Riazi

GeneralRe: hard disk serialmembernicolas2002 May '07 - 4:39 

 
Hello, first of all, thanx for your reply.
 
I've watched some of that article and I think with DeviceIoControl and IOCTL_STORAGE_GET_MEDIA_SERIAL_NUMBER I could get the serial. I also need detailed info of all the devices, but, mandatory info is the cpu uuid, bios serial (well, I'm going to use smbios for the bios info) and hard disks serials. That's why I thought of using something like setupAPI, although that doesn't give me any serials, but I can get al the GUID and was hoping that with something else I could get the devices details.
 
I've found a software that does what I need, MSI:
http://www.mitec.cz/msi.html
 
I need to get the same hardware info. For now I've been using WMI, but I really need to use the Win32 API, as it has to run (preferably) as guest on all windows SOs, starting on win98 or winME.
 
So, I'd really apreciate any pointers anybody can give me.
 
Thank you.

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

Permalink | Advertise | Privacy | Mobile
Web03 | 2.6.130523.1 | Last Updated 17 Mar 2004
Article Copyright 2004 by A. Riazi
Everything else Copyright © CodeProject, 1999-2013
Terms of Use
Layout: fixed | fluid