bluray.c 5.33 KB
Newer Older
Accident's avatar
 
Accident committed
1 2 3 4 5 6 7 8 9 10 11
#if HAVE_CONFIG_H
#include "config.h"
#endif

#if HAVE_MALLOC_H
#include <malloc.h>
#endif

#if HAVE_STDLIB_H
#include <stdlib.h>
#endif
cRTrn13's avatar
cRTrn13 committed
12

cRTrn13's avatar
cRTrn13 committed
13
#include <dlfcn.h>
cRTrn13's avatar
cRTrn13 committed
14
#include <stdio.h>
cRTrn13's avatar
cRTrn13 committed
15

cRTrn13's avatar
cRTrn13 committed
16
#include "bluray.h"
cRTrn13's avatar
cRTrn13 committed
17
#include "util/macro.h"
cRTrn13's avatar
cRTrn13 committed
18
#include "util/logging.h"
cRTrn13's avatar
cRTrn13 committed
19

cRTrn13's avatar
cRTrn13 committed
20 21
BLURAY *bd_open(const char* device_path, const char* keyfile_path)
{
22
    BLURAY *bd = calloc(1, sizeof(BLURAY));
cRTrn13's avatar
cRTrn13 committed
23

cRTrn13's avatar
cRTrn13 committed
24 25 26
    if (device_path) {
        strncpy(bd->device_path, device_path, 100);

cRTrn13's avatar
cRTrn13 committed
27 28 29 30
        bd->aacs = NULL;
        bd->h_libaacs = NULL;
        bd->fp = NULL;

cRTrn13's avatar
Fix  
cRTrn13 committed
31
        if (keyfile_path) {
cRTrn13's avatar
cRTrn13 committed
32
            if ((bd->h_libaacs = dlopen("libaacs.so", RTLD_LAZY))) {
cRTrn13's avatar
cRTrn13 committed
33 34 35
                fptr_p_void fptr;
                uint8_t *vid;

cRTrn13's avatar
cRTrn13 committed
36
                DEBUG(DBG_BLURAY, "Downloaded libaacs (0x%08x)\n", bd->h_libaacs);
cRTrn13's avatar
cRTrn13 committed
37

cRTrn13's avatar
cRTrn13 committed
38 39 40 41 42 43 44 45 46 47 48 49 50
                if ((fptr = dlsym(bd->h_libaacs, "aacs_open"))) {
                    if ((bd->aacs = fptr(device_path, keyfile_path))) {
                        if ((fptr = dlsym(bd->h_libaacs, "aacs_get_vid"))) {
                            vid = fptr(bd->aacs);
                        }

                        if ((bd->h_libbdplus = dlopen("libbdplus.so", RTLD_LAZY))) {
                            DEBUG(DBG_BLURAY, "Downloaded libbdplus (0x%08x)\n", bd->h_libbdplus);

                            if ((fptr = dlsym(bd->h_libbdplus, "bdplus_open"))) {
                                if (!(bd->bdplus = fptr(device_path, keyfile_path, vid))) {
                                    DEBUG(DBG_BLURAY, "libbdplus ret NULL!\n");
                                }
cRTrn13's avatar
Fix  
cRTrn13 committed
51
                            }
cRTrn13's avatar
cRTrn13 committed
52 53 54 55 56 57 58
                        } else {
                            DEBUG(DBG_BLURAY, "libbdplus not present!\n");
                        }
                    } else {
                        DEBUG(DBG_BLURAY, "libaacs ret NULL!\n");
                    }
                }
cRTrn13's avatar
cRTrn13 committed
59
            } else {
cRTrn13's avatar
cRTrn13 committed
60
                DEBUG(DBG_BLURAY, "libaacs not present!\n");
cRTrn13's avatar
cRTrn13 committed
61 62 63 64 65
            }
        } else {
            DEBUG(DBG_BLURAY, "No keyfile provided. You will not be able to make use of crypto functionality (0x%08x)\n", bd);
        }

cRTrn13's avatar
Fix  
cRTrn13 committed
66
        DEBUG(DBG_BLURAY, "BLURAY initialized! (0x%08x)\n", bd);
cRTrn13's avatar
cRTrn13 committed
67
    } else {
cRTrn13's avatar
cRTrn13 committed
68 69
        X_FREE(bd);

cRTrn13's avatar
cRTrn13 committed
70
        DEBUG(DBG_BLURAY, "No device path provided!\n");
cRTrn13's avatar
cRTrn13 committed
71 72 73 74 75 76 77
    }

    return bd;
}

void bd_close(BLURAY *bd)
{
cRTrn13's avatar
cRTrn13 committed
78 79 80 81 82 83 84 85 86 87 88 89 90 91
    if (bd->h_libaacs) {
        if (bd->aacs) {
            fptr_p_void fptr = dlsym(bd->h_libaacs, "aacs_close");
            fptr(bd->aacs);
        }

        dlclose(bd->h_libaacs);
    }

    if (bd->h_libbdplus) {
        if (bd->bdplus) {
            fptr_p_void fptr = dlsym(bd->h_libbdplus, "bdplus_close");
            fptr(bd->bdplus);
        }
cRTrn13's avatar
cRTrn13 committed
92

93 94
        dlclose(bd->h_libaacs);
    }
cRTrn13's avatar
cRTrn13 committed
95

cRTrn13's avatar
cRTrn13 committed
96 97 98
    if (bd->fp) {
        file_close(bd->fp);
    }
cRTrn13's avatar
cRTrn13 committed
99 100

    DEBUG(DBG_BLURAY, "BLURAY destroyed! (0x%08x)\n", bd);
cRTrn13's avatar
cRTrn13 committed
101 102

    X_FREE(bd);
cRTrn13's avatar
cRTrn13 committed
103
}
cRTrn13's avatar
cRTrn13 committed
104

cRTrn13's avatar
cRTrn13 committed
105
uint64_t bd_seek(BLURAY *bd, uint64_t pos)
cRTrn13's avatar
cRTrn13 committed
106 107 108 109
{
    if (pos < bd->s_size) {
        bd->s_pos = pos - (pos % 6144);

cRTrn13's avatar
cRTrn13 committed
110
        file_seek(bd->fp, bd->s_pos, SEEK_SET);
111

cRTrn13's avatar
cRTrn13 committed
112 113 114 115 116
        if (bd->h_libbdplus && bd->bdplus) {
            fptr_p_void fptr = dlsym(bd->h_libbdplus, "bdplus_seek");
            fptr(bd->bdplus, bd->s_pos);
        }

117
        DEBUG(DBG_BLURAY, "Seek to %ld (0x%08x)\n", bd->s_pos, bd);
cRTrn13's avatar
cRTrn13 committed
118 119 120 121 122 123 124
    }

    return bd->s_pos;
}

int bd_read(BLURAY *bd, unsigned char *buf, int len)
{
125 126 127 128 129 130 131
    if (bd->fp) {
        DEBUG(DBG_BLURAY, "Reading unit [%d bytes] at %ld... (0x%08x)\n", len, bd->s_pos, bd);

        if (len + bd->s_pos < bd->s_size) {
            int read_len;

            if ((read_len = file_read(bd->fp, buf, len))) {
cRTrn13's avatar
cRTrn13 committed
132
                if (bd->h_libaacs && bd->aacs) {
133 134 135 136 137 138
                    if ((bd->libaacs_decrypt_unit = dlsym(bd->h_libaacs, "aacs_decrypt_unit"))) {
                        if (!bd->libaacs_decrypt_unit(bd->aacs, buf, len, bd->s_pos)) {
                            DEBUG(DBG_BLURAY, "Unable decrypt unit! (0x%08x)\n", bd);

                            return 0;
                        }
cRTrn13's avatar
cRTrn13 committed
139 140 141 142 143

                        if (bd->h_libbdplus && bd->bdplus) {
                            fptr_p_void fptr = dlsym(bd->h_libbdplus, "bdplus_fixup");
                            fptr(bd->bdplus, buf, len);
                        }
144
                    }
cRTrn13's avatar
cRTrn13 committed
145 146
                }

147
                bd->s_pos += len;
cRTrn13's avatar
cRTrn13 committed
148

149 150 151 152
                DEBUG(DBG_BLURAY, "%d bytes read OK! (0x%08x)\n", read_len, bd);

                return read_len;
            }
cRTrn13's avatar
cRTrn13 committed
153 154 155
        }
    }

156 157
    DEBUG(DBG_BLURAY, "No valid title selected! (0x%08x)\n", bd->s_pos);

cRTrn13's avatar
cRTrn13 committed
158 159
    return 0;
}
cRTrn13's avatar
cRTrn13 committed
160

cRTrn13's avatar
cRTrn13 committed
161 162 163 164 165 166 167 168 169 170 171 172 173
int bd_select_title(BLURAY *bd, uint64_t title)
{
    char f_name[100];

    memset(f_name, 0, sizeof(f_name));
    snprintf(f_name, 100, "%s/BDMV/STREAM/%05ld.m2ts", bd->device_path, title);

    bd->s_size = 0;
    bd->s_pos = 0;

    if ((bd->fp = file_open(f_name, "rb"))) {
        file_seek(bd->fp, 0, SEEK_END);
        if ((bd->s_size = file_tell(bd->fp))) {
cRTrn13's avatar
cRTrn13 committed
174 175 176 177 178
            if (bd->h_libbdplus && bd->bdplus) {
                fptr_p_void fptr = dlsym(bd->h_libbdplus, "bdplus_set_title");
                fptr(bd->bdplus, title);
            }

179 180 181
            bd_seek(bd, 0);

            DEBUG(DBG_BLURAY, "Title %s selected! (0x%08x)\n", f_name, bd);
cRTrn13's avatar
cRTrn13 committed
182 183 184

            return 1;
        }
185 186

        DEBUG(DBG_BLURAY, "Title %s empty! (0x%08x)\n", f_name, bd);
cRTrn13's avatar
cRTrn13 committed
187 188
    }

189 190
    DEBUG(DBG_BLURAY, "Unable to select title %s! (0x%08x)\n", f_name, bd);

cRTrn13's avatar
cRTrn13 committed
191 192
    return 0;
}
193 194 195 196 197

uint64_t bd_get_title_size(BLURAY *bd)
{
    return bd->s_size;
}
cRTrn13's avatar
cRTrn13 committed
198 199 200 201 202

uint64_t bd_tell(BLURAY *bd)
{
    return bd->s_pos;
}