编程语言
首页 > 编程语言> > C/C++ Select端口扫描 [收集/转载]

C/C++ Select端口扫描 [收集/转载]

作者:互联网

#define _CRT_SECURE_NO_DEPRECATE
#define _CRT_NONSTDC_NO_DEPRECATE
#define _WINSOCK_DEPRECATED_NO_WARNINGS
#define _AFXDLL

#include <afxext.h>
#include <winsock.h>

// 编译时需使用的库
#pragma comment(lib,"wsock32.lib")

// select()成员定义
#define ZERO (fd_set *)0

// 变量定义
int maxth, scanok, scannum;
int portip, hoststart, hoststop, startport, endport;
//定义了开始I和结束P地址,开始和结束端口
long searchnum, searched;
void usage(char*);
// 定义显示使用方法函数
void playx(int);
// 定义状态提示函数
void setip2(char*);
// 定义设置IP函数
void customport(char*, char*, char*);
// 定义自定义扫描端口函数
void portscannow(int);
// 定义端口扫描扫描

// 主程序
int main(int argc, char* argv[])
{
    WSADATA wsadata;
    // 清屏
    system("cls.exe");

    // 检查输入
    if ((argc < 3) || (argc > 4))
    {
        // 显示帮助提示
        usage(argv[0]);
        return-1;
    }

    // 检测是否为port扫描
    if (!(stricmp(strlwr(argv[1]), "-p") == 0))
    {
        usage(argv[0]);
        return-1;
    }

    // 程序初始化
    //如果初始化错误
    if (WSAStartup(MAKEWORD(1, 1), &wsadata) != 0)
    {
        printf("\r\nWsatartup error");
        //出错信息
        return-1;
    }

    // 端口扫描参数转换
    // 如果参数为三个
    if (argc == 3)
    {
        // 直接设置IP
        setip2(argv[2]);
    }
    // 如果参数为四个
    else
        if (argc == 4)
        {
            // 进入定制端口扫描处理
            customport(argv[0], argv[2], argv[3]);
        }
    // 参数过多显示帮助
        else
        {
            usage(argv[0]);
            return-1;
        }

    // 扫描端口开始
    portscannow(argc);

    WSACleanup();
    return 0;
}

// 帮助提示函数
void usage(char* prog)
{
    printf("Usage: %s <Option>", prog);
    printf("\r\n\n <Option>:");
    printf("\r\n -p [ Port|StartPort-EndPort ] < HostName|IP|StartIP-EndIP >");
    printf("\r\n\n Example: ");
    printf("\r\n %s -p 192.168.0.1", prog);
    printf("\r\n %s -p 192.168.0.1-192.168.0.254", prog);
    printf("\r\n %s -p 21-80 192.168.0.1", prog);
    printf("\r\n %s -p 21-80 192.168.0.1-192.168.0.254\r\n", prog);
    return;
}

// 进度提示
void playx(int play = 0)
{
    // 进度条
    const char* plays[12] =
    {
        " | ",
        " / ",
        " - ",
        " // ",
        " | ",
        " / ",
        " - ",
        " // ",
        " | ",
        " / ",
        " - ",
        " // ",
    }
    ;

    if (searchnum != 0)
    {
        for (int i = 0; i <= 3; i++)
        {
            printf(" =%s= %d%s Completed.  \r", plays[i], searched * 100 / (searchnum + 1), "%");
            Sleep(5);
        }
    }
    else
    {
        printf(" =%s=\r", plays[play]);
        //显示进度
        Sleep(10);
    }
}

// 设置IP
void setip2(char* cp)
{
    int host;
    struct hostent* testhost;
    const char* startip = "", * endip = "";

    // 判断是否为 192.168.0.1-192.168.0.254 形式的输入
    if (strstr(cp, "-") && strlen(cp) > 15 && strlen(cp) < 32)
    {
        // 提取出结束IP
        endip = strchr(cp, '-') + 1;
        // 提取出开始IP
        strncpy((char *)startip, cp, strlen(cp) - strlen(strchr(cp, '-')));

        // 给控制要扫描IP段的变量赋值
        hoststart = ntohl(inet_addr(startip));
        hoststop = ntohl(inet_addr(endip));
    }
    else
    {
        // 取得输入的主机地址
        testhost = gethostbyname(startip);

        // 如果地址不存在
        if (!testhost)
        {
            WSACleanup();
            printf("\r\nCan't get ip of: %s", cp);
            exit(-1);
        }
        // 给控制要扫描IP段的变量赋值
        memcpy(&host, testhost->h_addr, 4);
        hoststop = hoststart = ntohl(host);
    }
}

// 测试线程是否已满
void TestThread(int thread = 200)
{
    for (;;)
    {
        playx();

        // 测试线程是否已满
        if (maxth > thread)
            Sleep(100);
        else break;
    }
    return;
}

// 等待线程结束函数
void WaitThreadEnd()
{
    // 延时
    Sleep(6000);

    // 显示等待提示
    printf("\r     \r\n");
    printf(" Wait ( %d )Thread end...\r\n", maxth);

    for (;;)
    {
        // 判断所有线程是否已经结束
        if (maxth > 0)
        {
            // 延时等待线程序结束
            Sleep(100);
            playx();
            continue;
        }
        else break;
    }
    printf("\r\n");
    return;
}

// 定制端口扫描参数定义
void customport(char* cp, char* cp2, char* cp3)
{
    int intport;
    char* checker;

    // 处理要扫描的端口
    // 扫描开始端口变量赋值
    startport = atoi(cp2);
    // 扫描结束端口变量赋值
    endport = atoi(cp2);

    // 判断是否 21-80 形式
    if (strstr(cp2, "-"))
    {
        intport = atoi(checker = strchr(cp2, '-') + 1);
        if (intport > 0 && intport < 65536)
            // 扫描结束端口变量赋值
            endport = intport;
    }

    // 端口大小判断
    if (startport < 0 || startport>65536 || endport < 0 || endport>65535)
    {
        usage(cp);
        exit(-1);
    }
    // 处理ip地址
    setip2(cp3);
}

// 端口扫描函数
UINT portscan(LPVOID port)
{
    int addr = portip;
    // 取得要扫描的地址
    int sock;
    struct fd_set mask;
    struct timeval timeout;
    struct sockaddr_in server;
    unsigned long flag = 1;

    // 创建一个sock
    sock = socket(AF_INET, SOCK_STREAM, 0);

    // 创建sock失败处理
    if (sock == INVALID_SOCKET)
    {
        printf("\r\nSock Error:%s", WSAGetLastError());
        maxth--;
        return-1;
    }

    // 给sock成员赋值
    server.sin_family = AF_INET;
    server.sin_addr.s_addr = htonl(addr);
    // 要扫描的地址
    server.sin_port = htons(short(port));
    // 要扫描的端口

    // 显示进度
    playx();

    // 调用ioctlsocket()设置套接字为非阻塞模式
    if (ioctlsocket(sock, FIONBIO, &flag) != 0)
    {
        // 设置失败处理
        printf("\r\nSock Error:%s", WSAGetLastError());
        closesocket(sock);
        maxth--;
        return-1;
    }

    // 调用connect()连接远程主机端口
    connect(sock, (struct sockaddr*)&server, sizeof(server));

    timeout.tv_sec = 18;
    // 超时限制为18秒
    timeout.tv_usec = 0;
    FD_ZERO(&mask);
    // 清空集合mask
    FD_SET(sock, &mask);
    // 将sock放入集合mask中

    // 用select() 处理扫描结果
    switch (select(sock + 1, ZERO, &mask, ZERO, &timeout))
    {
    case-1:
    {
        printf("\r\nSelect() error");
        maxth--;
        return-1;
    }

    // sock超时处理
    case 0:
    {
        maxth--;
        closesocket(sock);
        return-1;
    }

    default:
        if (FD_ISSET(sock, &mask))
        {
            // 禁止sock发送和接受数据
            shutdown(sock, 0);

            // 设置输出结果格式
            printf(" [Found:] %s Port: %d open.\r\n", inet_ntoa(server.sin_addr), ntohs(server.sin_port));

            // 关闭sock
            closesocket(sock);
            scanok++;
            maxth--;
            return 1;
        }
    }
    return 0;
}

// 扫描开始主函数
void portscannow(int xp)
{
    int sport;
    char* timenow, timebuf[32];

    // 定义默认扫描的端口
    const char* ports[32] = {"80","443","445"};

    // 显示扫描开始的时间
    timenow = _strtime(timebuf);
    printf("\r\nPortScan Start Time: %s\r\n\n", timenow);

    // 计数器初始化.
    int i = 0;
    maxth = 0;
    scanok = 0;
    scannum = 0;
    searched = 0;

    // 计算要扫描的端口数量
    searchnum = hoststop - hoststart + 1;
    if (xp == 3)
        searchnum = searchnum * 32;
    if (xp == 4)
        searchnum = searchnum * (endport - startport + 1);

    // 端口扫描开始
    for (portip = hoststart; portip <= hoststop; portip++, scannum++)
    {
        // *.*.*.0和*.*.*.255 地址处理
        if ((portip % 256) == 0 || (portip % 256) == 255)
        {
            if (xp == 3)
                searchnum = searchnum - 32;
            if (xp == 4)
                searchnum = searchnum - (endport - startport + 1);
            scannum--;
            playx();
            continue;
        }

        if (i > 11)i = 0;
        // 默认端口扫描
        // scan 192.168.0.1
        // scan 192.168.0.1-192.168.0.254
        if (xp == 3)
        {
            for (sport = 0; sport < 32; sport++, maxth++, searched++)
            {
                // 测试当前线程是否大于180
                TestThread(180);
                // 产生新的线程处理端口扫描
                CWinThread* pthread = AfxBeginThread(portscan, LPVOID(atoi((char*)ports[sport])));
                //延时
                Sleep(120);
            }
        }

        // 自定义端口扫描
        // scan -p 21 192.168.0.1
        // scan -p 21-80 192.168.0.1-192.168.0.254
        if (xp == 4)
        {
            // 计算要扫描的端口
            sport = endport - startport;
            if (sport > 500)
            {
                // 扫描自定义的端口
                for (sport = startport; sport <= endport; sport++, maxth++, searched++)
                {
                    TestThread(2000);
                    // 产生新的线程处理端口扫描
                    CWinThread* pthread = AfxBeginThread(portscan, LPVOID(sport));
                    // 延时
                    Sleep(10);
                }
            }
            else
            {
                // 扫描自定义的端口
                for (sport = startport; sport <= endport; sport++, maxth++, searched++)
                {
                    // 测试当前线程是否大于250
                    TestThread(250);
                    // 产生新的线程处理端口扫描
                    CWinThread* pthread = AfxBeginThread(portscan, LPVOID(sport));
                    // 延时
                    Sleep(100);
                    playx();
                }
            }
        }
    }

    // 等待所有的线程结束
    WaitThreadEnd();

    // 显示端口扫描结束时间
    timenow = _strtime(timebuf);
    printf("\r\nPortScan End Time: %s", timenow);
    printf("\r\nScan %d Hosts completed. Open %d Ports!\r\n", scannum, scanok);
}

代码来源: https://blog.csdn.net/n4ry/article/details/675657

标签:return,端口扫描,扫描,sock,C++,char,int,printf,Select
来源: https://www.cnblogs.com/LyShark/p/15087272.html