summaryrefslogtreecommitdiffstatshomepage
path: root/windows/msvc/dirent.c
blob: e050432a1707deffd40c8e13fdc7657ceda9352b (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
/*
* This file is part of the MicroPython project, http://micropython.org/
*
* The MIT License (MIT)
*
* Copyright (c) 2015 Damien P. George
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/

#include "dirent.h"
#include <errno.h>
#include <Windows.h>

typedef struct DIR {
    HANDLE findHandle;
    WIN32_FIND_DATA findData;
    struct dirent result;
} DIR;

DIR *opendir(const char *name) {
    if (!name || !*name) {
        errno = ENOENT;
        return NULL;
    }

    DIR *dir = malloc(sizeof(DIR));
    if (!dir) {
      errno = ENOMEM;
      return NULL;
    }
    dir->result.d_ino = 0;
    dir->result.d_name = NULL;
    dir->findHandle = INVALID_HANDLE_VALUE;

    const size_t nameLen = strlen(name);
    char *path = malloc(nameLen + 3); // allocate enough for adding "/*"
    if (!path) {
      free(dir);
      errno = ENOMEM;
      return NULL;
    }
    strcpy(path, name);

    // assure path ends with wildcard
    const char lastChar = path[nameLen - 1];
    if (lastChar != '*') {
        const char *appendWC = (lastChar != '/' && lastChar != '\\') ? "/*" : "*";
        strcat(path, appendWC);
    }

    // init
    dir->findHandle = FindFirstFile(path, &dir->findData);
    free(path);
    if (dir->findHandle == INVALID_HANDLE_VALUE) {
      free(dir);
      errno = ENOENT;
      return NULL;
    }
    return dir;
}

int closedir(DIR *dir) {
    if (dir) {
        FindClose(dir->findHandle);
        free(dir);
        return 0;
    } else {
        errno = EBADF;
        return -1;
    }
}

struct dirent *readdir(DIR *dir) {
    if (!dir) {
        errno = EBADF;
        return NULL;
    }

    // first pass d_name is NULL so use result from FindFirstFile in opendir, else use FindNextFile
    if (!dir->result.d_name || FindNextFile(dir->findHandle, &dir->findData)) {
        dir->result.d_name = dir->findData.cFileName;
        return &dir->result;
    }

    return NULL;
}