win32 GetConsoleMode()错误代码6

时间:2015-11-02 11:15:16

标签: c++ winapi cmake

我正在关注另一个帖子Reading a password from std::cin中的示例,您可以获取标准输入的句柄,获取当前控制台模式,更改模式以抑制输入上的回显。出于某种原因,当我调用GetStdHandle()时,它返回一个有效的句柄,但是当我调用GetConsoleMode()时它失败并返回错误代码6.我在cmake项目中使用它。是否有任何调试标志我应该为此按预期工作?还有其他人遇到过吗?

void set_stdin_echo(bool enable) {
    HANDLE hStdin = GetStdHandle(STD_INPUT_HANDLE);
    if (hStdin == INVALID_HANDLE_VALUE) {
        std::cout << "invalid handle " << GetLastError() << std::endl;
        return;
    }
    else if (hStdin == NULL) {
        std::cout << "no handle associated" << std::endl;
        return;
    }

    DWORD mode = 0;
    if (GetConsoleMode(hStdin, &mode) == 0) {
        std::cout << "Could not get console mode" << GetLastError() << std::endl;
    }

    std::cout << mode << std::endl;
    if (!enable)
        mode &= ~ENABLE_ECHO_INPUT;
    else
        mode |= ENABLE_ECHO_INPUT;
    std::cout << mode << std::endl;

    if (SetConsoleMode(hStdin, mode) == 0) {
        std::cout << "Could not set input mode" << std::endl;
        std::cout << GetLastError() << std::endl;
    }
}

编辑:使用此代码复制。

的CMakeLists.txt

cmake_minimum_required (VERSION 2.6)
project(tester)

if(CMAKE_COMPILER_IS_GNUCXX)
    set(CMAKE_CXX_FLAGS "-g -w -Wall -pedantic-errors -std=c++11")
elseif(MSVC)
    set(CMAKE_CXX_FLAGS "/EHsc")
    set(CMAKE_CXX_FLAGS_DEBUG "/EHsc /MTd")
    set(CMAKE_CXX_FLAGS_RELEASE "/EHsc /MT")
endif()

set(SOURCE_FILES "main.cpp")

add_executable(tester ${SOURCE_FILES})

的main.cpp

#include <iostream>
#include <string>
#include <windows.h>

void set_stdin_echo(bool enable) {
    DWORD error;
    HANDLE hStdin = GetStdHandle(STD_INPUT_HANDLE);
    error = GetLastError();
    if (hStdin == INVALID_HANDLE_VALUE) {
        std::cout << "invalid handle " << error << std::endl;
        return;
    }
    else if (hStdin == NULL) {
        std::cout << "no handle associated" << std::endl;
        return;
    }

    DWORD mode = 0;
    if (GetConsoleMode(hStdin, &mode) == 0) {
        error = GetLastError();
        std::cout << "Could not get console mode" << error << std::endl;
    }

    std::cout << mode << std::endl;
    if (!enable)
        mode &= ~ENABLE_ECHO_INPUT;
    else
        mode |= ENABLE_ECHO_INPUT;
    std::cout << mode << std::endl;

    if (SetConsoleMode(hStdin, mode) == 0) {
        error = GetLastError();
        std::cout << "Could not set input mode" << std::endl;
        std::cout << error << std::endl;
    }
}

int main() {
    std::string input;
    std::cout << "password: ";
    set_stdin_echo(false);
    std::cin >> input;
    set_stdin_echo(true);

    return 0;
}

Visual Studio 12.在某个目录中调用它们。我得到“无法获得控制台模式6”,暗示GetLastError()返回6。

mkdir build
cmake ..
msbuild.exe tester.sln
./Debug/tester.exe

编辑:似乎只有在从命令行运行时才会失败。在visual studio调试器的visual studio中,它成功了。仍然没有足够的视觉工作室知识来确定发生了什么。我只想创建一个可以从命令行运行并隐藏其输入的简单exe。

编辑:清洁并在调用后asap获取GetLastError。仍然没有变化。

1 个答案:

答案 0 :(得分:4)

此错误6表示“无效句柄”,表示您传递给GetConsoleMode的句柄不是控制台句柄。即STDIN已从某处重定向。

为确保您拥有当前控制台的句柄,无论是否从其他位置重定向STDIN,请使用具有特殊名称CreateFile的{​​{1}},这将打开控制台(如果存在)。< / p>

示例:

"CONIN$"