Turo Lamminen
2015-04-30 ea27b014a1f9da82bf41c67d3acfe615b8fe7b9b
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
#include <errno.h>
#include <stdbool.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
 
#include "mojodds.h"
 
 
// from http://graphics.stanford.edu/~seander/bithacks.html
static bool isPow2(unsigned int v) {
    return v && !(v & (v - 1));
}
 
 
static int ddsinfo(const char *filename) {
    printf("%s\n", filename);
 
    FILE *f = fopen(filename, "rb");
    if (!f) {
        printf("Error opening %s: %s (%d)\n", filename, strerror(errno), errno);
        return 1;
    }
 
    fseek(f, 0, SEEK_END);
    long size = ftell(f);
    fseek(f, 0, SEEK_SET);
 
    char *contents = malloc(size);
    size_t readbytes = fread(contents, 1, size, f);
    if (readbytes != size) {
        printf("Only got %u of %ld bytes: %s\n", (unsigned int) readbytes, size, strerror(errno));
        free(contents);
        fclose(f);
        return 2;
    }
 
    fclose(f);
 
    int isDDS = MOJODDS_isDDS(contents, size);
    printf("isDDS: %d\n", isDDS);
    if (isDDS) {
        const void *tex = NULL;
        unsigned long texlen = 0;
        unsigned int glfmt = 0, w = 0, h = 0, miplevels = 0;
        unsigned int cubemapfacelen = 0;
        MOJODDS_textureType textureType = MOJODDS_TEXTURE_NONE;
        int retval = MOJODDS_getTexture(contents, size, &tex, &texlen, &glfmt, &w, &h, &miplevels, &cubemapfacelen, &textureType);
        if (!retval) {
            printf("MOJODDS_getTexture failed\n");
            free(contents);
            return 3;
        }
 
        uintptr_t texoffset = ((const char *)(tex)) - contents;
        printf("texoffset: %u\n", (unsigned int)(texoffset));
        printf("texlen: %lu\n", texlen);
        printf("glfmt: 0x%x\n", glfmt);
        printf("width x height: %d x %d\n", w, h);
        printf("miplevels: %d\n", miplevels);
        printf("\n");
 
        for (unsigned int miplevel = 0; miplevel < miplevels; miplevel++) {
            const void *miptex = NULL;
            unsigned long miptexlen = 0;
            unsigned int mipW = 0, mipH = 0;
            retval = MOJODDS_getMipMapTexture(miplevel, glfmt, tex, texlen, w, h, &miptex, &miptexlen, &mipW, &mipH);
            if (!retval) {
                printf("MOJODDS_getMipMapTexture(%u) error: %d\n", miplevel, retval);
                continue;
            }
 
            uintptr_t miptexoffset = ((const char *)(miptex)) - ((const char *)(tex));
            bool npot = !(isPow2(mipW) || isPow2(mipH));
            printf("%4d x %4d  %s", mipW, mipH, npot ? "NPOT  " : "      ");
            printf("miptexoffset: %8u  ", (unsigned int)(miptexoffset));
            printf("miptexlen: %8lu\n", miptexlen);
        }
    }
 
    free(contents);
 
    return 0;
}
 
 
int main(int argc, char *argv[]) {
    if (argc < 2) {
        printf("Usage: %s DDS-file ...\n", argv[0]);
        return 0;
    }
 
    for (int i = 1; i < argc; i++) {
        ddsinfo(argv[i]);
    }
 
    return 0;
}