blob: 1a3ff28b6037acb86e48571e2179ce9d71020ad9 [file] [log] [blame]
Andrew Molyneux516cab02014-12-24 20:07:05 +00001/*
2zip_source_win32w.c -- create data source from Windows file (Unicode)
3Copyright (C) 1999-2014 Dieter Baron and Thomas Klausner
4
5This file is part of libzip, a library to manipulate ZIP archives.
6The authors can be contacted at <libzip@nih.at>
7
8Redistribution and use in source and binary forms, with or without
9modification, are permitted provided that the following conditions
10are met:
111. Redistributions of source code must retain the above copyright
12notice, this list of conditions and the following disclaimer.
132. Redistributions in binary form must reproduce the above copyright
14notice, this list of conditions and the following disclaimer in
15the documentation and/or other materials provided with the
16distribution.
173. The names of the authors may not be used to endorse or promote
18products derived from this software without specific prior
19written permission.
20
21THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``AS IS'' AND ANY EXPRESS
22OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
23WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY
25DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
27GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
28INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
29IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
30OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
31IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32*/
33
34
35#include <errno.h>
36#include <stdio.h>
Andrew Molyneux516cab02014-12-24 20:07:05 +000037
Andrew Molyneuxfe8a5a82014-12-24 20:58:56 +000038#include "zipwin32.h"
Andrew Molyneux516cab02014-12-24 20:07:05 +000039#include "zipint.h"
40
Andrew Molyneux8c444d62014-12-24 21:09:16 +000041static void * _win32_strdup_w(const void *str);
Andrew Molyneux516cab02014-12-24 20:07:05 +000042static HANDLE _win32_open_w(_zip_source_win32_read_file_t *ctx);
43static HANDLE _win32_create_temp_w(_zip_source_win32_read_file_t *ctx, void **temp, zip_uint32_t value);
44static int _win32_rename_temp_w(_zip_source_win32_read_file_t *ctx);
45static int _win32_discard_source_w(_zip_source_win32_read_file_t *ctx);
46static int _win32_discard_temp_w(_zip_source_win32_read_file_t *ctx);
Andrew Molyneux516cab02014-12-24 20:07:05 +000047
48static _zip_source_win32_file_ops_t win32_ops_w = {
49 .op_strdup = _win32_strdup_w,
50 .op_open = _win32_open_w,
51 .op_create_temp = _win32_create_temp_w,
52 .op_rename_temp = _win32_rename_temp_w,
53 .op_discard_source = _win32_discard_source_w,
54 .op_discard_temp = _win32_discard_temp_w
55};
56
57ZIP_EXTERN zip_source_t *
58zip_source_win32w(zip_t *za, const wchar_t *fname, zip_uint64_t start, zip_int64_t len)
59{
60 if (za == NULL)
61 return NULL;
62
63 return zip_source_win32w_create(fname, start, len, &za->error);
64}
65
66
67ZIP_EXTERN zip_source_t *
68zip_source_win32w_create(const wchar_t *fname, zip_uint64_t start, zip_int64_t length, zip_error_t *error)
69{
70 if (fname == NULL || length < -1) {
71 zip_error_set(error, ZIP_ER_INVAL, 0);
72 return NULL;
73 }
74
75 return _zip_source_win32_handle_or_name(fname, INVALID_HANDLE_VALUE, start, length, 1, NULL, &win32_ops_w, error);
76}
77
78
79void *
80_win32_strdup_w(const void *str)
81{
82 return _wcsdup((const wchar_t *)str);
83}
84
85
86HANDLE
87_win32_open_w(_zip_source_win32_read_file_t *ctx)
88{
89 return CreateFileW(ctx->fname, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
90}
91
92
93HANDLE
94_win32_create_temp_w(_zip_source_win32_read_file_t *ctx, void **temp, zip_uint32_t value)
95{
96 int len;
97
98 len = wcslen((const wchar_t *)ctx->fname) + 10;
99 if (*temp == NULL) {
Andrew Molyneuxdef87292014-12-24 20:39:09 +0000100 if ((*temp = (char *)malloc(sizeof(wchar_t) * len)) == NULL) {
Andrew Molyneux516cab02014-12-24 20:07:05 +0000101 zip_error_set(&ctx->error, ZIP_ER_MEMORY, 0);
102 return INVALID_HANDLE_VALUE;
103 }
104 }
105 if (swprintf((wchar_t *)*temp, len, L"%s.%08x", (const wchar_t *)ctx->fname, value) != len - 1) {
106 return INVALID_HANDLE_VALUE;
107 }
108
109 return CreateFileW((const wchar_t *)*temp, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ, NULL, CREATE_NEW, FILE_ATTRIBUTE_NORMAL | FILE_ATTRIBUTE_TEMPORARY, NULL);
110}
111
112
113int
114_win32_rename_temp_w(_zip_source_win32_read_file_t *ctx)
115{
116 if (!MoveFileExW(ctx->tmpname, ctx->fname, MOVEFILE_REPLACE_EXISTING)) {
117 zip_error_set(&ctx->error, ZIP_ER_RENAME, _zip_set_win32_error(GetLastError(), &ctx->win32err));
118 return -1;
119 }
120 return 0;
121}
122
123
124int
125_win32_discard_source_w(_zip_source_win32_read_file_t *ctx)
126{
127 if (!DeleteFileW(ctx->fname)) {
128 zip_error_set(&ctx->error, ZIP_ER_REMOVE, _zip_set_win32_error(GetLastError(), &ctx->win32err));
129 return -1;
130 }
131 return 0;
132}
133
134
135int
136_win32_discard_temp_w(_zip_source_win32_read_file_t *ctx)
137{
138 DeleteFileW(ctx->tmpname);
139 return 0;
140}