Entrar com a conta da Microsoft
Entrar ou criar uma conta.
Olá,
Selecionar uma conta diferente.
Você tem várias contas
Escolha a conta com a qual você deseja entrar.

INTRODUÇÃO

Quando um dispositivo de armazenamento está conectado ao Windows, mesmo que resumidamente, o windows cria informações de registro para o dispositivo. Ao longo do tempo, o registro pode conter várias entradas para dispositivos que nunca serão usados novamente. Este artigo descreve como remover essas informações do registro do sistema.

É responsabilidade do software que estabelece a conexão entre o Windows limpar corretamente as informações para o dispositivo e o dispositivo de armazenamento. Esse processo é necessário porque o Windows não sabe quando um dispositivo de armazenamento é removido permanentemente ou temporariamente. Mas o software que estabelece a conexão normalmente sabe disso. Por exemplo, se o software de backup é montar números de unidades lógicas (LUNs) para fins de backup e, em seguida, desmontando os LUNs, seria a responsabilidade de software de backup para limpar as informações de LUN do Windows, pois o mesmo dispositivo de armazenamento não serão mais ser usada novamente pelo Windows.

Mais informações

Quando um novo dispositivo é conectado a um computador, registra as informações Windows sobre o dispositivo no registro do sistema. Para a maioria dos dispositivos, esse procedimento não representar um problema. No entanto, depois que um dispositivo de armazenamento é apresentado por um LUN por meio de um canal de fibra ou iSCSI, o dispositivo pode nunca ser encontrado novamente pelo computador. Por exemplo, um dispositivo pode ser identificado por um número de série ou por páginas de SCSI 0x80 e 0x83.

Nessa situação, o registro pode conter entradas para dispositivos que nunca podem aparecer novamente. Não só fazer essas entradas ocupam espaço no registro, essas entradas podem eventualmente causar problemas operacionais. Por exemplo, como índices para a funcionalidade Plug and Play usam valores decimais de quatro dígitos, um problema pode ocorrer quando o dispositivo 10,001 está conectado.

Para resolver essa limitação na funcionalidade Plug and Play, convém remover as informações de dispositivo do registro quando o dispositivo é uma unidade de disco rígido que não está mais presente. Você pode fazer isso usando o utilitário Microsoft DevNodeClean .

Como construir para Windows Server 2003, Windows Server 2008, Windows Server 2008 R2 e o Visual Studio 2005

Para limpar o registro para o GUID de classe do disco, no GUID_DEVCLASS_DISKDRIVE e para o GUID_DEVCLASS_VOLUME disco GUID de classe, siga estas etapas.

Observação: Recomendamos que você use o utilitário DevNodeClean para esta tarefa. As etapas a seguir e o exemplo de código na etapa 7 são fornecidas apenas para fins informativos.

  1. Chame a função SetupDiGetClassDevs para obter informações para a classe que está associada com o GUID.

  2. Chame a função SetupDiEnumDeviceInfo para obter informações da instância para cada dispositivo na classe atual.

  3. Chame a função CM_Get_DevNode_Status para ver se as informações do dispositivo atual representam um dispositivo ausente. Determine se o status da função é igual a CR_NO_SUCH_DEVINST ou a CR_NO_SUCH_VALUE.

  4. Opcionalmente, para um dispositivo ausente, chame a função CM_Get_Device_ID para obter a identificação de instância de dispositivo e para exibir a ID antes de remover as informações.

  5. Para o dispositivo ausente, use as informações de classe que você obteve na etapa 1 e as informações da instância que você obteve na etapa 2. Chame a função SetupDiCallClassInstaller (DIF_REMOVE...) para remover as informações do registro.

  6. Quando todos os dispositivos na classe atual foram tratados, chame a função SetupDiDestroyDeviceInfoList para limpar.

Observação: Em alguns cenários, talvez você precise limpar o registro, não apenas para o GUID_DEVCLASS_DISKDRIVE e o GUID_DEVCLASS_VOLUME do disco classe GUIDs, mas também para o GUID_DEVCLASS_SCSIADAPTER e o GUID_DEVCLASS_VOLUMESNAPSHOT do disco GUIDs de classe. Para fazer isso, você deve alterar a definição de DiskClassesToClean no código de exemplo a seguir.

O seguinte aplicativo do console Win32 é um exemplo de um aplicativo que limpa o registro. Para usar este aplicativo, siga estas etapas.

A Microsoft fornece exemplos de programação somente para ilustração, sem garantia expressa ou implícita. Isso inclui, mas não está limitado a, garantias implícitas de comercialização ou adequação a um propósito específico. Este artigo presume que você esteja familiarizado com a linguagem de programação que está sendo demonstrada e com as ferramentas usadas para criar e depurar procedimentos. Os engenheiros de suporte da Microsoft podem ajudar a explicar a funcionalidade de um determinado procedimento. No entanto, eles não modificarão esses exemplos para fornecer funcionalidades adicionais ou construir procedimentos para atender às suas necessidades específicas.

  1. No Microsoft Visual Studio 2005, clique em novo no menu arquivo e, em seguida, clique em Project.

  2. Expanda Visual C++e, em seguida, clique em Win32.

  3. Clique em Aplicativo do Console Win32, digite Limpeza da caixa de texto nome e, em seguida, clique em Okey.

  4. Clique em Concluir na caixa de diálogo Assistente de aplicativos Win32 .

  5. No Solution Explorer, expanda Arquivos de origem, clique com o botão Cleanup.cppe, em seguida, clique em View Code.

  6. Localize o código a seguir:

    int _tmain(int argc, _TCHAR* argv[])
    {
    return 0;
    }
    
  7. Substitua o código que você encontrou na etapa 6 com o código a seguir.

    /**************************************************************************************************/     
    /*                                                                                                */     
    /* Copyright (c) 2007 Microsoft Corporation.  All Rights Reserved                                 */     
    /*                                                                                                */     
    /**************************************************************************************************/     
    
    #pragma warning( disable : 4201 ) // nonstandard extension used : nameless strut/union
    
    #include <windows.h>
    #include <stdlib.h>
    #include <stdio.h>
    #include <stddef.h>
    #include <tchar.h>
    #include <setupapi.h>
    #include <cfgmgr32.h>
    #include <initguid.h>
    #include <devguid.h>
    
    #define SIZECHARS(x) (sizeof((x))/sizeof(TCHAR))
    #define arraysize(p) (sizeof(p)/sizeof((p)[0]))
    
    CONST GUID *DiskClassesToClean[2] = {
        &GUID_DEVCLASS_DISKDRIVE,
        &GUID_DEVCLASS_VOLUME
    };
    
    /**************************************************************************************************/
    /*                                                                                                */
    /* The user must be member of Administrator group and must have backup and restore permissions         */
    /* (SE_BACKUP_NAME and SE_RESTORE_NAME). No check for these is performed in this example.              */
    /*                                                                                                */
    /**************************************************************************************************/
    int
    __cdecl
    main(
         IN int    ArgC,
         IN char * pArgV[]
        )
    {
        HDEVINFO DeviceInfoSet;
        SP_DEVINFO_DATA DeviceInfoData;
        ULONG DevicesRemoved = 0,
              i,
              MemberIndex,
              Status, 
              Problem,
              ulClassesToCleanIdx;
        BOOL bDoRemove = TRUE;
        CONFIGRET cr;
        TCHAR DeviceInstanceId[MAX_DEVICE_ID_LEN];
        OSVERSIONINFO osvi;
        const GUID ** ClassesToClean;
    
        //
        // Parse parameters.
        //
        for (i = 1; i < (ULONG)ArgC; i++) {
            //
            // Check for help.
            //
            if ( (lstrcmpi(pArgV[i], TEXT("-?")) == 0) ||
                    (lstrcmpi(pArgV[i], TEXT("/?")) == 0) ){
    
                printf("\nCleanUp will remove phantom storage device nodes from this machine.\n\n");
                printf("Usage:  CleanUp \n");
                printf("\twhere /n displays but does not remove the phantom devnodes.\n");
                printf("\nBackup and Restore privileges are required to run this utility.\n");
                return 0;
            }
    
            //
            // Check for -n, which means just list the devices that we would remove.
            //
            if ( (lstrcmpi(pArgV[i], TEXT("-n")) == 0) ||
                 (lstrcmpi(pArgV[i], TEXT("/n")) == 0) ) {
                bDoRemove = FALSE;
            }
        }
    
        //
        // Run only on Windows XP/2003 (version 5.1) or later.
        //
    
        ZeroMemory(&osvi, sizeof(OSVERSIONINFO));
        osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
    
        if (!GetVersionEx(&osvi)) {
            printf("CleanUp:  Unable to verify Windows version, exiting...\n");
            return -1;
        }
    
        if ((osvi.dwMajorVersion == 5) &&
            ((osvi.dwMinorVersion == 1) || (osvi.dwMinorVersion == 2))) {
        }
        else if (osvi.dwMajorVersion>=6) {
        }
        else
        {
            printf("CleanUp:  This utility is  designed to run on Windows XP/2003 and later\n");
            return -1;
        }
    
        ClassesToClean = DiskClassesToClean;
        ulClassesToCleanIdx = arraysize(DiskClassesToClean);
    
        for (i=0; (i<ulClassesToCleanIdx) && (bDoRemove); i++) {
    
            DeviceInfoSet = SetupDiGetClassDevs(ClassesToClean[i],
                                                NULL,
                                                NULL,
                                                0
                                                );
    
            if (INVALID_HANDLE_VALUE!=DeviceInfoSet) {
    
                DeviceInfoData.cbSize = sizeof(DeviceInfoData);
                MemberIndex = 0;
    
                while (SetupDiEnumDeviceInfo(DeviceInfoSet,
                                             MemberIndex++,
                                             &DeviceInfoData
                                             )) {
    
                    //
                    // Determine whether this device is a phantom.
                    //
                    cr = CM_Get_DevNode_Status(&Status,
                                               &Problem,
                                               DeviceInfoData.DevInst,
                                               0
                                               );
    
                    if ((cr == CR_NO_SUCH_DEVINST) ||
                        (cr == CR_NO_SUCH_VALUE)) {
                        //
                        // This is a phantom. Now get the DeviceInstanceId so we
                        // can display this as output, then delete the phantom if requested.
                        //
                        if (CM_Get_Device_ID(DeviceInfoData.DevInst,
                                             DeviceInstanceId,
                                             SIZECHARS(DeviceInstanceId),
                                             0) == CR_SUCCESS) {
    
                            if (bDoRemove) {
                                printf("DevNodePhantomCleaner:  %s will be removed.\n",
                                       DeviceInstanceId);
    
                                //
                                // Call DIF_REMOVE to remove the device's hardware
                                // and software registry keys.
                                //
                                if (SetupDiCallClassInstaller(DIF_REMOVE,
                                                              DeviceInfoSet,
                                                              &DeviceInfoData
                                                              )) {
                                    DevicesRemoved++;
                                } else {
                                    printf("CleanUp:  Error 0x%X removing phantom\n",
                                           GetLastError);
                                }
                            } else {
                                printf("CleanUp:  %s would have been removed.\n",
                                       DeviceInstanceId);
                            }
                        }
                    }
                }
    
                SetupDiDestroyDeviceInfoList(DeviceInfoSet);
            }
        }
    
        return DevicesRemoved;
    }
  8. Clique no menu Debug e, em seguida, clique em Iniciar depuração.

Como construir para o Windows Server 2012 e o Visual Studio 2012

Para criar para o Windows Server 2012 e o Microsoft Visual Studio 2012, siga estas etapas.

Observação: Recomendamos que você use o utilitário DevNodeClean para esta tarefa. As etapas a seguir e o exemplo de código na etapa 7 são fornecidas apenas para fins informativos.

  1. No Microsoft Visual Studio 2012, clique em novo no menu arquivo e, em seguida, clique em Project.

  2. Na caixa de diálogo New Project , digite Limpeza no campo nome e, em seguida, clique duas vezes no Projeto do Win32.

  3. No Win32 Application Wizard, clique em Avançar.

  4. Em tipo de aplicativo, clique para selecionar o Aplicativo de Consolee, em seguida, clique em Concluir.

  5. No Solution Explorer, expanda Arquivos de origem, clique com o botão Cleanup.cppe, em seguida, clique em View Code.

  6. Localize o código a seguir:

    int _tmain(int argc, _TCHAR* argv[])
    {
    return 0;
    }
    
  7. Substitua o código que você encontrou na etapa 6 com o código a seguir.

    //DevPhantomClnr.cpp : Defines the entry point for the console application.
    // 
     
    #include "stdafx.h"
    
    /**************************************************************************************************/     
    /*                                                                                                */     
    /* Copyright (c) 2007 Microsoft Corporation.  All Rights Reserved                                 */     
    /*                                                                                                */     
    /**************************************************************************************************/     
    
    #pragma warning( disable : 4201 ) // nonstandard extension used : nameless strut/union
    
    #include <windows.h>
    #include <stdlib.h>
    #include <stdio.h>
    #include <stddef.h>
    #include <tchar.h>
    #include <setupapi.h>
    #include <cfgmgr32.h>
    #include <initguid.h>
    #include <devguid.h>
    
    #define SIZECHARS(x) (sizeof((x))/sizeof(TCHAR))
    #define arraysize(p) (sizeof(p)/sizeof((p)[0]))
    
    CONST GUID *DiskClassesToClean[2] = {
        &GUID_DEVCLASS_DISKDRIVE,
        &GUID_DEVCLASS_VOLUME
    };
    
    /**************************************************************************************************/
    /*                                                                                                */
    /* The user must be member of Administrator group and must have backup and restore permissions         */
    /* (SE_BACKUP_NAME and SE_RESTORE_NAME). No check for these is performed in this example.              */
    /*                                                                                                */
    /**************************************************************************************************/
    int
    __cdecl
    main(
         IN int    ArgC,
         IN LPCWSTR pArgV[]
        )
    {
        HDEVINFO DeviceInfoSet;
        SP_DEVINFO_DATA DeviceInfoData;
        ULONG DevicesRemoved = 0,
              i,
              MemberIndex,
              Status, 
              Problem,
              ulClassesToCleanIdx;
        BOOL bDoRemove = TRUE;
        CONFIGRET cr;
        TCHAR DeviceInstanceId[MAX_DEVICE_ID_LEN];
        OSVERSIONINFO osvi;
        const GUID ** ClassesToClean;
    
        //
        // Parse parameters.
        //
        for (i = 1; i < (ULONG)ArgC; i++) {
            //
            // Check for help.
            //
            if ( (lstrcmpi(pArgV[i], L"-?") == 0) ||
                    (lstrcmpi(pArgV[i], L"/?") == 0) ){
    
                printf("\nDevNodePhantomCleaner will remove phantom storage device nodes from this machine.\n\n");
                printf("Usage:  nDevNodePhantomCleaner \n");
                printf("\tWhere /n displays but does not remove the phantom devnodes.\n");
                printf("\nBackup and Restore privileges are required to run this utility.\n");
                return 0;
            }
    
            //
            // Check for -n, which means just list the devices that we would remove.
            //
            if ( (lstrcmpi(pArgV[i], L"-n") == 0) ||
                 (lstrcmpi(pArgV[i], L"/n") == 0) ) {
                bDoRemove = FALSE;
            }
        }
    
        //
        // Run only on Windows XP/2003 (version 5.1) or later.
        //
    
        ZeroMemory(&osvi, sizeof(OSVERSIONINFO));
        osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
    
        if (!GetVersionEx(&osvi)) {
            printf("DevNodePhantomCleaner:  Unable to verify Windows version, exiting...\n");
            return -1;
        }
    
        if ((osvi.dwMajorVersion == 5) &&
            ((osvi.dwMinorVersion == 1) || (osvi.dwMinorVersion == 2))) {
        
        // 5.1 || 5.2
    
        }
    
        else if (osvi.dwMajorVersion>=6) {
        
        //Nothing special on 6.x
    
        }
    
        else
        {
            printf("DevNodePhantomCleaner:  This utility is  designed to run on Windows XP/2003 and later\n");
            return -1;
        }
    
        ClassesToClean = DiskClassesToClean;
        ulClassesToCleanIdx = arraysize(DiskClassesToClean);
    
        for (i=0; (i<ulClassesToCleanIdx) && (bDoRemove); i++) {
    
            DeviceInfoSet = SetupDiGetClassDevs(ClassesToClean[i],
                                                NULL,
                                                NULL,
                                                0
                                                );
    
            if (INVALID_HANDLE_VALUE!=DeviceInfoSet) {
    
                DeviceInfoData.cbSize = sizeof(DeviceInfoData);
                MemberIndex = 0;
    
                while (SetupDiEnumDeviceInfo(DeviceInfoSet,
                                             MemberIndex++,
                                             &DeviceInfoData
                                             )) {
    
                    //
                    // Determine whether this device is a phantom.
                    //
                    cr = CM_Get_DevNode_Status(&Status,
                                               &Problem,
                                               DeviceInfoData.DevInst,
                                               0
                                               );
    
                    if ((cr == CR_NO_SUCH_DEVINST) ||
                        (cr == CR_NO_SUCH_VALUE)) {
                        //
                        // This is a phantom. Now get the DeviceInstanceId so we
                        // can display this as output, then delete the phantom if requested.
                        //
                        if (CM_Get_Device_ID(DeviceInfoData.DevInst,
                                             DeviceInstanceId,
                                             SIZECHARS(DeviceInstanceId),
                                             0) == CR_SUCCESS) {
    
                            if (bDoRemove) {
                                printf("DevNodePhantomCleaner:  %ws will be removed.\n",
                                       DeviceInstanceId);
    
                                //
                                // Call DIF_REMOVE to remove the device's hardware
                                // and software registry keys.
                                //
                                if (SetupDiCallClassInstaller(DIF_REMOVE,
                                                              DeviceInfoSet,
                                                              &DeviceInfoData
                                                              )) {
                                    DevicesRemoved++;
                                } else {
                                    printf("DevNodePhantomCleaner:  Error 0x%x removing phantom\n",
                                           GetLastError());
                                }
                            } else {
                                printf("DevNodePhantomCleaner:  %ws would have been removed.\n",
                                       DeviceInstanceId);
                            }
                        }
                    }
                }
    
                SetupDiDestroyDeviceInfoList(DeviceInfoSet);
            }
        }
    
        return DevicesRemoved;
    }
  8. No Solution Explorer, clique com o botão Limpare, em seguida, clique em Propriedades.

  9. Expanda Configuration Properties, expanda vinculadore, em seguida, clique em entrada.

  10. Selecione Dependências adicionais, clique na seta para baixo e, em seguida, selecione Editar.

  11. Na caixa de diálogo Dependências adicionais , tipo setupapi.lib e cfgmgr32.lib.

  12. Clique em OK duas vezes.

  13. Compile o projeto.

Precisa de mais ajuda?

Quer mais opções

Explore os benefícios da assinatura, procure cursos de treinamento, saiba como proteger seu dispositivo e muito mais.

As comunidades ajudam você a fazer e responder perguntas, fazer comentários e ouvir especialistas com conhecimento avançado.

Essas informações foram úteis?

Qual é o seu grau de satisfação com a qualidade do idioma?
O que afetou sua experiência?
Ao pressionar enviar, seus comentários serão usados para aprimorar os produtos e serviços da Microsoft. Seu administrador de TI poderá coletar esses dados. Política de Privacidade.

Agradecemos seus comentários!

×