xref: /haiku/src/apps/soundrecorder/FileUtils.cpp (revision b8ded2f89783a220c7b3019d48266a682cc79158)
1 /******************************************************************************
2 /
3 /	File:			FileUtils.cpp
4 /
5 /   Description:	Utility functions for copying file data and attributes.
6 /
7 /	Copyright 1998-1999, Be Incorporated, All Rights Reserved
8 /
9 ******************************************************************************/
10 #include "FileUtils.h"
11 
12 #include <algorithm>
13 #include <new>
14 #include <stdio.h>
15 #include <string.h>
16 
17 #include <fs_attr.h>
18 
19 #include "AutoDeleter.h"
20 
21 
22 using std::nothrow;
23 
24 
25 status_t
CopyFileData(BFile & dst,BFile & src)26 CopyFileData(BFile& dst, BFile& src)
27 {
28 	struct stat src_stat;
29 	status_t err = src.GetStat(&src_stat);
30 	if (err != B_OK) {
31 		printf("couldn't get stat: %#010" B_PRIx32 "\n", err);
32 		return err;
33 	}
34 
35 	size_t bufSize = src_stat.st_blksize;
36 	if (bufSize == 0)
37 		bufSize = 32768;
38 
39 	char* buf = new (nothrow) char[bufSize];
40 	if (buf == NULL)
41 		return B_NO_MEMORY;
42 	ArrayDeleter<char> _(buf);
43 
44 	printf("copy data, bufSize = %ld\n", bufSize);
45 	// copy data
46 	while (true) {
47 		ssize_t bytes = src.Read(buf, bufSize);
48 		if (bytes > 0) {
49 			ssize_t result = dst.Write(buf, bytes);
50 			if (result != bytes) {
51 				fprintf(stderr, "Failed to write %ld bytes: %s\n", bytes,
52 					strerror((status_t)result));
53 				if (result < 0)
54 					return (status_t)result;
55 				else
56 					return B_IO_ERROR;
57 			}
58 		} else {
59 			if (bytes < 0) {
60 				fprintf(stderr, "Failed to read file: %s\n", strerror(
61 					(status_t)bytes));
62 				return (status_t)bytes;
63 			} else {
64 				// EOF
65 				break;
66 			}
67 		}
68 	}
69 
70 	// finish up miscellaneous stat stuff
71 	dst.SetPermissions(src_stat.st_mode);
72 	dst.SetOwner(src_stat.st_uid);
73 	dst.SetGroup(src_stat.st_gid);
74 	dst.SetModificationTime(src_stat.st_mtime);
75 	dst.SetCreationTime(src_stat.st_crtime);
76 
77 	return B_OK;
78 }
79 
80 
81 status_t
CopyAttributes(BNode & dst,BNode & src)82 CopyAttributes(BNode& dst, BNode& src)
83 {
84 	// copy attributes
85 	src.RewindAttrs();
86 	char attrName[B_ATTR_NAME_LENGTH];
87 	while (src.GetNextAttrName(attrName) == B_OK) {
88 		attr_info info;
89 		if (src.GetAttrInfo(attrName, &info) != B_OK) {
90 			fprintf(stderr, "Failed to read info for attribute '%s'\n",
91 				attrName);
92 			continue;
93 		}
94 		// copy one attribute in chunks of 4096 bytes
95 		size_t size = 4096;
96 		uint8 buffer[size];
97 		off_t offset = 0;
98 		ssize_t read = src.ReadAttr(attrName, info.type, offset, buffer,
99 			std::min((off_t)size, info.size));
100 		if (read < 0) {
101 			fprintf(stderr, "Error reading attribute '%s'\n", attrName);
102 			return (status_t)read;
103 		}
104 		// NOTE: Attributes of size 0 are perfectly valid!
105 		while (read >= 0) {
106 			ssize_t written = dst.WriteAttr(attrName, info.type, offset,
107 				buffer, read);
108 			if (written != read) {
109 				fprintf(stderr, "Error writing attribute '%s'\n", attrName);
110 				if (written < 0)
111 					return (status_t)written;
112 				else
113 					return B_IO_ERROR;
114 			}
115 			offset += read;
116 			read = src.ReadAttr(attrName, info.type, offset, buffer,
117 				std::min((off_t)size, info.size - offset));
118 			if (read < 0) {
119 				fprintf(stderr, "Error reading attribute '%s'\n", attrName);
120 				return (status_t)read;
121 			}
122 			if (read == 0)
123 				break;
124 		}
125 	}
126 
127 	return B_OK;
128 }
129 
130 
131 status_t
CopyFile(BFile & dst,BFile & src)132 CopyFile(BFile& dst, BFile& src)
133 {
134 	status_t err = CopyFileData(dst, src);
135 	if (err != B_OK)
136 		return err;
137 
138 	return CopyAttributes(dst, src);
139 }
140