* Implemented persistence storage of attributes and name changes - it's able to

read the files the BeOS cdda wrote, but it handles attributes completely different.
* All attributes are now stored per CD.
* There are some special shared attributes between all CDs (or a specific device)
  to improve your Tracker experience.
* An existing buffer is now used to fill a read request.
* Lots of other small changes here and there.


git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@21201 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Axel Dörfler 2007-05-22 09:23:00 +00:00
parent 37593af284
commit f77a87bfa4
2 changed files with 539 additions and 90 deletions

View File

@ -607,7 +607,27 @@ read_cdda_data(int fd, off_t offset, void *data, size_t length,
{ {
if (bufferOffset >= 0 && bufferOffset <= offset + length if (bufferOffset >= 0 && bufferOffset <= offset + length
&& bufferOffset + bufferSize > offset) { && bufferOffset + bufferSize > offset) {
// TODO: fill request from buffer if (offset >= bufferOffset) {
// buffer reaches into the beginning of the request
off_t dataOffset = offset - bufferOffset;
size_t bytes = min_c(bufferSize - dataOffset, length);
if (user_memcpy(data, (uint8 *)buffer + dataOffset, bytes) < B_OK)
return B_BAD_ADDRESS;
data = (void *)((uint8 *)data + bytes);
length -= bytes;
offset += bytes;
} else if (offset < bufferOffset
&& offset + length < bufferOffset + bufferSize) {
// buffer overlaps at the end of the request
off_t dataOffset = bufferOffset - offset;
size_t bytes = length - dataOffset;
if (user_memcpy((uint8 *)data + dataOffset, buffer, bytes) < B_OK)
return B_BAD_ADDRESS;
length -= bytes;
}
// we don't handle the case we would need to split the request
} }
while (length > 0) { while (length > 0) {

View File

@ -5,8 +5,10 @@
#include "cdda.h" #include "cdda.h"
#include "cddb.h"
#include "Lock.h" #include "Lock.h"
#include <FindDirectory.h>
#include <fs_info.h> #include <fs_info.h>
#include <fs_interface.h> #include <fs_interface.h>
#include <KernelExport.h> #include <KernelExport.h>
@ -66,49 +68,63 @@ struct wav_header {
riff_chunk data; riff_chunk data;
} _PACKED; } _PACKED;
enum attr_mode {
kDiscIDAttributes,
kSharedAttributes,
kDeviceAttributes
};
class Volume { class Volume {
public: public:
Volume(mount_id id); Volume(mount_id id);
~Volume(); ~Volume();
status_t InitCheck(); status_t InitCheck();
mount_id ID() const { return fID; } mount_id ID() const { return fID; }
Inode &RootNode() const { return *fRootNode; } uint32 DiscID() const { return fDiscID; }
Inode &RootNode() const { return *fRootNode; }
status_t Mount(const char* device); status_t Mount(const char* device);
int Device() const { return fDevice; } int Device() const { return fDevice; }
vnode_id GetNextNodeID() { return fNextID++; } vnode_id GetNextNodeID() { return fNextID++; }
const char *Name() const { return fName; } const char *Name() const { return fName; }
status_t SetName(const char *name); status_t SetName(const char *name);
Semaphore &Lock(); Semaphore &Lock();
Inode *Find(vnode_id id); Inode *Find(vnode_id id);
Inode *Find(const char *name); Inode *Find(const char *name);
Inode *FirstEntry() const { return fFirstEntry; } Inode *FirstEntry() const { return fFirstEntry; }
off_t NumBlocks() const { return fNumBlocks; } off_t NumBlocks() const { return fNumBlocks; }
size_t BufferSize() const { return 32 * kFrameSize; } size_t BufferSize() const { return 32 * kFrameSize; }
// TODO: for now // TODO: for now
static void DetermineName(cdtext &text, char *name, size_t length); static void DetermineName(cdtext &text, char *name, size_t length);
private: private:
Inode *_CreateNode(Inode *parent, const char *name, Inode *_CreateNode(Inode *parent, const char *name,
off_t start, off_t frames, int32 type); off_t start, off_t frames, int32 type);
int _OpenAttributes(int mode,
enum attr_mode attrMode = kDiscIDAttributes);
void _RestoreAttributes();
void _StoreAttributes();
void _RestoreSharedAttributes();
void _StoreSharedAttributes();
Semaphore fLock; Semaphore fLock;
int fDevice; int fDevice;
mount_id fID; mount_id fID;
Inode *fRootNode; uint32 fDiscID;
vnode_id fNextID; Inode *fRootNode;
char *fName; vnode_id fNextID;
off_t fNumBlocks; char *fName;
off_t fNumBlocks;
// root directory contents - we don't support other directories // root directory contents - we don't support other directories
Inode *fFirstEntry; Inode *fFirstEntry;
}; };
class Attribute : public DoublyLinkedListLinkImpl<Attribute> { class Attribute : public DoublyLinkedListLinkImpl<Attribute> {
@ -123,10 +139,12 @@ class Attribute : public DoublyLinkedListLinkImpl<Attribute> {
status_t ReadAt(off_t offset, uint8 *buffer, size_t *_length); status_t ReadAt(off_t offset, uint8 *buffer, size_t *_length);
status_t WriteAt(off_t offset, const uint8 *buffer, size_t *_length); status_t WriteAt(off_t offset, const uint8 *buffer, size_t *_length);
void Truncate(); void Truncate();
status_t SetSize(off_t size);
const char *Name() const { return fName; } const char *Name() const { return fName; }
size_t Size() const { return fSize; } size_t Size() const { return fSize; }
type_code Type() const { return fType; } type_code Type() const { return fType; }
uint8 *Data() const { return fData; }
private: private:
char *fName; char *fName;
@ -165,8 +183,10 @@ class Inode {
{ return fFrameCount * kFrameSize /* + WAV header */; } { return fFrameCount * kFrameSize /* + WAV header */; }
Attribute *FindAttribute(const char *name) const; Attribute *FindAttribute(const char *name) const;
status_t AddAttribute(Attribute *attribute, bool overwrite);
status_t AddAttribute(const char *name, type_code type, status_t AddAttribute(const char *name, type_code type,
const uint8 *data = 0, size_t length = 0); bool overwrite, const uint8 *data = 0,
size_t length = 0);
status_t AddAttribute(const char *name, type_code type, status_t AddAttribute(const char *name, type_code type,
const char *string); const char *string);
status_t AddAttribute(const char *name, int32 value); status_t AddAttribute(const char *name, int32 value);
@ -176,6 +196,9 @@ class Inode {
void RemoveAttrCookie(attr_cookie *cookie); void RemoveAttrCookie(attr_cookie *cookie);
void RewindAttrCookie(attr_cookie *cookie); void RewindAttrCookie(attr_cookie *cookie);
AttributeList::ConstIterator Attributes() const
{ return fAttributes.GetIterator(); }
const wav_header *WAVHeader() const { return &fWAVHeader; } const wav_header *WAVHeader() const { return &fWAVHeader; }
Inode *Next() const { return fNext; } Inode *Next() const { return fNext; }
@ -220,6 +243,209 @@ struct file_cookie {
void *buffer; void *buffer;
}; };
static const uint32 kMaxAttributeSize = 65536;
static const uint32 kMaxAttributes = 64;
// #pragma mark helper functions
/*!
Determines if the attribute is shared among all devices or among
all CDs in a specific device.
We use this to share certain Tracker attributes.
*/
static bool
is_special_attribute(const char *name, attr_mode attrMode)
{
if (attrMode == kDeviceAttributes) {
static const char *kAttributes[] = {
"_trk/windframe",
"_trk/pinfo",
"_trk/pinfo_le",
NULL,
};
for (int32 i = 0; kAttributes[i]; i++) {
if (!strcmp(name, kAttributes[i]))
return true;
}
} else if (attrMode == kSharedAttributes) {
static const char *kAttributes[] = {
"_trk/columns",
"_trk/columns_le",
"_trk/viewstate",
"_trk/viewstate_le",
NULL,
};
for (int32 i = 0; kAttributes[i]; i++) {
if (!strcmp(name, kAttributes[i]))
return true;
}
}
return false;
}
static void
write_line(int fd, const char *line)
{
if (line == NULL)
line = "";
size_t length = strlen(line);
write(fd, line, length);
write(fd, "\n", 1);
}
static void
write_attributes(int fd, Inode *inode, attr_mode attrMode = kDiscIDAttributes)
{
// count attributes
AttributeList::ConstIterator iterator = inode->Attributes();
uint32 count = 0;
while (iterator.HasNext()) {
Attribute *attribute = iterator.Next();
if (attrMode == kDiscIDAttributes
|| is_special_attribute(attribute->Name(), attrMode))
count++;
}
// we're artificially limiting the attribute count per inode
if (count > kMaxAttributes)
count = kMaxAttributes;
count = B_HOST_TO_BENDIAN_INT32(count);
write(fd, &count, sizeof(uint32));
// write attributes
iterator.Rewind();
while (iterator.HasNext()) {
Attribute *attribute = iterator.Next();
if (attrMode != kDiscIDAttributes
&& !is_special_attribute(attribute->Name(), attrMode))
continue;
uint32 type = B_HOST_TO_BENDIAN_INT32(attribute->Type());
write(fd, &type, sizeof(uint32));
uint8 length = strlen(attribute->Name());
write(fd, &length, 1);
write(fd, attribute->Name(), length);
uint32 size = B_HOST_TO_BENDIAN_INT32(attribute->Size());
write(fd, &size, sizeof(uint32));
if (size != 0)
write(fd, attribute->Data(), attribute->Size());
if (--count == 0)
break;
}
}
static bool
read_line(int fd, char *line, size_t length)
{
bool first = true;
size_t pos = 0;
char c;
while (read(fd, &c, 1) == 1) {
first = false;
if (c == '\n')
break;
if (pos < length)
line[pos] = c;
pos++;
}
if (pos >= length - 1)
pos = length - 1;
line[pos] = '\0';
return !first;
}
static bool
read_attributes(int fd, Inode *inode)
{
uint32 count;
if (read(fd, &count, sizeof(uint32)) != (ssize_t)sizeof(uint32))
return false;
count = B_BENDIAN_TO_HOST_INT32(count);
dprintf("inode %s read %lu attrs\n", inode->Name(), count);
if (count > kMaxAttributes)
return false;
for (uint32 i = 0; i < count; i++) {
char name[B_ATTR_NAME_LENGTH + 1];
uint32 type, size;
uint8 length;
if (read(fd, &type, sizeof(uint32)) != (ssize_t)sizeof(uint32)
|| read(fd, &length, 1) != 1
|| read(fd, name, length) != length
|| read(fd, &size, sizeof(uint32)) != (ssize_t)sizeof(uint32))
return false;
type = B_BENDIAN_TO_HOST_INT32(type);
size = B_BENDIAN_TO_HOST_INT32(size);
name[length] = '\0';
dprintf(" type %08lx, size %lu, name %s\n", type, size, name);
Attribute *attribute = new Attribute(name, type);
if (attribute->SetSize(size) != B_OK
|| inode->AddAttribute(attribute, true) != B_OK) {
delete attribute;
} else
read(fd, attribute->Data(), size);
}
return true;
}
static void
fill_stat_buffer(Volume *volume, Inode *inode, Attribute *attribute,
struct stat &stat)
{
stat.st_dev = volume->ID();
stat.st_ino = inode->ID();
if (attribute != NULL) {
stat.st_size = attribute->Size();
stat.st_mode = S_ATTR | 0666;
stat.st_type = attribute->Type();
} else {
stat.st_size = inode->Size();
stat.st_mode = inode->Type();
stat.st_type = 0;
}
stat.st_nlink = 1;
stat.st_blksize = 2048;
stat.st_uid = inode->UserID();
stat.st_gid = inode->GroupID();
stat.st_atime = time(NULL);
stat.st_mtime = stat.st_ctime = inode->ModificationTime();
stat.st_crtime = inode->CreationTime();
}
// #pragma mark - Volume class
Volume::Volume(mount_id id) Volume::Volume(mount_id id)
: :
@ -232,19 +458,22 @@ Volume::Volume(mount_id id)
fNumBlocks(0), fNumBlocks(0),
fFirstEntry(NULL) fFirstEntry(NULL)
{ {
// create the root vnode
fRootNode = _CreateNode(NULL, "", 0, 0, S_IFDIR | 0777);
} }
Volume::~Volume() Volume::~Volume()
{ {
_StoreAttributes();
_StoreSharedAttributes();
close(fDevice); close(fDevice);
// put_vnode on the root to release the ref to it // put_vnode on the root to release the ref to it
if (fRootNode) if (fRootNode)
put_vnode(ID(), fRootNode->ID()); put_vnode(ID(), fRootNode->ID());
delete fRootNode;
Inode *inode, *next; Inode *inode, *next;
for (inode = fFirstEntry; inode != NULL; inode = next) { for (inode = fFirstEntry; inode != NULL; inode = next) {
@ -259,8 +488,7 @@ Volume::~Volume()
status_t status_t
Volume::InitCheck() Volume::InitCheck()
{ {
if (fLock.InitCheck() < B_OK if (fLock.InitCheck() < B_OK)
|| fRootNode == NULL)
return B_ERROR; return B_ERROR;
return B_OK; return B_OK;
@ -297,6 +525,19 @@ Volume::Mount(const char* device)
return status; return status;
} }
fDiscID = compute_cddb_disc_id(*toc);
// create the root vnode
fRootNode = _CreateNode(NULL, "", 0, 0, S_IFDIR | 0777);
if (fRootNode == NULL)
status = B_NO_MEMORY;
if (status >= B_OK)
status = publish_vnode(ID(), fRootNode->ID(), fRootNode);
if (status < B_OK) {
free(toc);
return status;
}
cdtext text; cdtext text;
if (read_cdtext(fDevice, text) < B_OK) if (read_cdtext(fDevice, text) < B_OK)
dprintf("CDDA: no CD-Text found.\n"); dprintf("CDDA: no CD-Text found.\n");
@ -328,10 +569,12 @@ Volume::Mount(const char* device)
} else } else
snprintf(title, sizeof(title), "%02ld.wav", track); snprintf(title, sizeof(title), "%02ld.wav", track);
// remove '/' from title // remove '/' and '\n' from title
for (int32 j = 0; title[j]; j++) { for (int32 j = 0; title[j]; j++) {
if (title[j] == '/') if (title[j] == '/')
title[j] = '-'; title[j] = '-';
else if (title[j] == '\n')
title[j] = ' ';
} }
totalFrames += frames; totalFrames += frames;
@ -356,6 +599,9 @@ Volume::Mount(const char* device)
inode->AddAttribute("BEOS:TYPE", B_MIME_STRING_TYPE, "audio/x-wav"); inode->AddAttribute("BEOS:TYPE", B_MIME_STRING_TYPE, "audio/x-wav");
} }
_RestoreSharedAttributes();
_RestoreAttributes();
free(toc); free(toc);
// determine volume title // determine volume title
@ -391,11 +637,19 @@ Volume::_CreateNode(Inode *parent, const char *name, off_t start, off_t frames,
} }
if (S_ISREG(type)) { if (S_ISREG(type)) {
inode->SetNext(fFirstEntry); // we need to order it by track for compatibility with BeOS' cdda
fFirstEntry = inode; Inode *last = NULL, *current = fFirstEntry;
while (current != NULL) {
last = current;
current = current->Next();
}
if (last)
last->SetNext(inode);
else
fFirstEntry = inode;
} }
publish_vnode(ID(), inode->ID(), inode);
return inode; return inode;
} }
@ -444,7 +698,168 @@ Volume::SetName(const char *name)
} }
// #pragma mark - /*!
Opens the file that contains the volume and inode titles as well as all
of their attributes.
The attributes are stored in files below B_USER_SETTINGS_DIRECTORY/cdda.
*/
int
Volume::_OpenAttributes(int mode, enum attr_mode attrMode)
{
char* path = (char*)malloc(B_PATH_NAME_LENGTH);
if (path == NULL)
return -1;
bool create = (mode & O_WRONLY) != 0;
if (find_directory(B_USER_SETTINGS_DIRECTORY, -1, create, path,
B_PATH_NAME_LENGTH) != B_OK) {
free(path);
return -1;
}
strlcat(path, "/cdda", B_PATH_NAME_LENGTH);
if (create)
mkdir(path, 0755);
if (attrMode == kDiscIDAttributes) {
char id[64];
snprintf(id, sizeof(id), "/%08lx", fDiscID);
strlcat(path, id, B_PATH_NAME_LENGTH);
} else if (attrMode == kDeviceAttributes) {
uint32 length = strlen(path);
char *device = path + length;
if (ioctl(fDevice, B_GET_PATH_FOR_DEVICE, device,
B_PATH_NAME_LENGTH - length) < B_OK) {
free(path);
return B_ERROR;
}
device++;
// replace slashes in the device path
while (device[0]) {
if (device[0] == '/')
device[0] = '_';
device++;
}
} else
strlcat(path, "/shared", B_PATH_NAME_LENGTH);
dprintf("PATH: %s\n", path);
int fd = open(path, mode | (create ? O_CREAT | O_TRUNC : 0), 0644);
free(path);
return fd;
}
/*!
Reads the attributes, if any, that belong to the CD currently being
mounted.
*/
void
Volume::_RestoreAttributes()
{
int fd = _OpenAttributes(O_RDONLY);
if (fd < 0)
return;
char line[B_FILE_NAME_LENGTH];
if (!read_line(fd, line, B_FILE_NAME_LENGTH)) {
close(fd);
return;
}
dprintf("VOLUME %s\n", line);
SetName(line);
for (Inode *inode = fFirstEntry; inode != NULL; inode = inode->Next()) {
if (!read_line(fd, line, B_FILE_NAME_LENGTH))
break;
inode->SetName(line);
dprintf("INODE %s\n", line);
}
if (read_attributes(fd, fRootNode)) {
for (Inode *inode = fFirstEntry; inode != NULL; inode = inode->Next()) {
if (!read_attributes(fd, inode))
break;
}
}
close(fd);
}
void
Volume::_StoreAttributes()
{
int fd = _OpenAttributes(O_WRONLY);
if (fd < 0)
return;
write_line(fd, Name());
for (Inode *inode = fFirstEntry; inode != NULL; inode = inode->Next()) {
write_line(fd, inode->Name());
}
write_attributes(fd, fRootNode);
for (Inode *inode = fFirstEntry; inode != NULL; inode = inode->Next()) {
write_attributes(fd, inode);
}
close(fd);
}
/*!
Restores the attributes, if any, that are shared between CDs; some are
stored per device, others are stored for all CDs no matter which device.
*/
void
Volume::_RestoreSharedAttributes()
{
// device attributes overwrite shared attributes
int fd = _OpenAttributes(O_RDONLY, kSharedAttributes);
if (fd >= 0) {
read_attributes(fd, fRootNode);
close(fd);
}
fd = _OpenAttributes(O_RDONLY, kDeviceAttributes);
if (fd >= 0) {
read_attributes(fd, fRootNode);
close(fd);
}
}
void
Volume::_StoreSharedAttributes()
{
// write shared and device specific settings
int fd = _OpenAttributes(O_WRONLY, kSharedAttributes);
if (fd >= 0) {
write_attributes(fd, fRootNode, kSharedAttributes);
close(fd);
}
fd = _OpenAttributes(O_WRONLY, kDeviceAttributes);
if (fd >= 0) {
write_attributes(fd, fRootNode, kDeviceAttributes);
close(fd);
}
}
// #pragma mark - Attribute class
Attribute::Attribute(const char *name, type_code type) Attribute::Attribute(const char *name, type_code type)
@ -490,7 +905,7 @@ Attribute::ReadAt(off_t offset, uint8 *buffer, size_t *_length)
if (offset < 0) if (offset < 0)
return B_BAD_VALUE; return B_BAD_VALUE;
if (offset >= length) { if (offset >= fSize) {
*_length = 0; *_length = 0;
return B_OK; return B_OK;
} }
@ -505,6 +920,10 @@ Attribute::ReadAt(off_t offset, uint8 *buffer, size_t *_length)
} }
/*!
Writes to the attribute and enlarges it as needed.
An attribute has a maximum size of 65536 bytes for now.
*/
status_t status_t
Attribute::WriteAt(off_t offset, const uint8 *buffer, size_t *_length) Attribute::WriteAt(off_t offset, const uint8 *buffer, size_t *_length)
{ {
@ -515,8 +934,8 @@ Attribute::WriteAt(off_t offset, const uint8 *buffer, size_t *_length)
// we limit the attribute size to something reasonable // we limit the attribute size to something reasonable
off_t end = offset + length; off_t end = offset + length;
if (end > 65536) { if (end > kMaxAttributeSize) {
end = 65536; end = kMaxAttributeSize;
length = end - offset; length = end - offset;
} }
if (offset > end) { if (offset > end) {
@ -545,6 +964,7 @@ Attribute::WriteAt(off_t offset, const uint8 *buffer, size_t *_length)
} }
//! Removes all data from the attribute.
void void
Attribute::Truncate() Attribute::Truncate()
{ {
@ -554,7 +974,30 @@ Attribute::Truncate()
} }
// #pragma mark - /*!
Resizes the data part of an attribute to the requested amount \a size.
An attribute has a maximum size of 65536 bytes for now.
*/
status_t
Attribute::SetSize(off_t size)
{
if (size > kMaxAttributeSize)
return E2BIG;
uint8 *data = (uint8 *)realloc(fData, size);
if (data == NULL)
return B_NO_MEMORY;
if (fSize < size)
memset(data + fSize, 0, size - fSize);
fData = data;
fSize = size;
return B_OK;
}
// #pragma mark - Inode class
Inode::Inode(Volume *volume, Inode *parent, const char *name, off_t start, Inode::Inode(Volume *volume, Inode *parent, const char *name, off_t start,
@ -623,7 +1066,9 @@ Inode::InitCheck()
status_t status_t
Inode::SetName(const char* name) Inode::SetName(const char* name)
{ {
if (name == NULL || !name[0]) if (name == NULL || !name[0]
|| strchr(name, '/') != NULL
|| strchr(name, '\n') != NULL)
return B_BAD_VALUE; return B_BAD_VALUE;
name = strdup(name); name = strdup(name);
@ -655,21 +1100,15 @@ Inode::FindAttribute(const char *name) const
status_t status_t
Inode::AddAttribute(const char *name, type_code type, Inode::AddAttribute(Attribute *attribute, bool overwrite)
const uint8 *data, size_t length)
{ {
if (FindAttribute(name) != NULL) Attribute *oldAttribute = FindAttribute(attribute->Name());
return B_NAME_IN_USE; if (oldAttribute != NULL) {
if (!overwrite)
return B_NAME_IN_USE;
Attribute *attribute = new Attribute(name, type); fAttributes.Remove(oldAttribute);
status_t status = attribute != NULL ? B_OK : B_NO_MEMORY; delete oldAttribute;
if (status == B_OK)
status = attribute->InitCheck();
if (status == B_OK && data != NULL && length != 0)
status = attribute->WriteAt(0, data, &length);
if (status < B_OK) {
delete attribute;
return status;
} }
fAttributes.Add(attribute); fAttributes.Add(attribute);
@ -679,12 +1118,32 @@ Inode::AddAttribute(const char *name, type_code type,
status_t status_t
Inode::AddAttribute(const char *name, type_code type, Inode::AddAttribute(const char *name, type_code type,
const char *string) bool overwrite, const uint8 *data, size_t length)
{
Attribute *attribute = new Attribute(name, type);
status_t status = attribute != NULL ? B_OK : B_NO_MEMORY;
if (status == B_OK)
status = attribute->InitCheck();
if (status == B_OK && data != NULL && length != 0)
status = attribute->WriteAt(0, data, &length);
if (status == B_OK)
status = AddAttribute(attribute, overwrite);
if (status < B_OK) {
delete attribute;
return status;
}
return B_OK;
}
status_t
Inode::AddAttribute(const char *name, type_code type, const char *string)
{ {
if (string == NULL) if (string == NULL)
return NULL; return NULL;
return AddAttribute(name, type, (const uint8 *)string, return AddAttribute(name, type, true, (const uint8 *)string,
strlen(string)); strlen(string));
} }
@ -692,8 +1151,8 @@ Inode::AddAttribute(const char *name, type_code type,
status_t status_t
Inode::AddAttribute(const char *name, int32 value) Inode::AddAttribute(const char *name, int32 value)
{ {
return AddAttribute(name, B_INT32_TYPE, (const uint8 *)&value, return AddAttribute(name, B_INT32_TYPE, true,
sizeof(int32)); (const uint8 *)&value, sizeof(int32));
} }
@ -748,38 +1207,6 @@ Inode::RewindAttrCookie(attr_cookie *cookie)
} }
// #pragma mark -
void
fill_stat_buffer(Volume *volume, Inode *inode, Attribute *attribute,
struct stat &stat)
{
stat.st_dev = volume->ID();
stat.st_ino = inode->ID();
if (attribute != NULL) {
stat.st_size = attribute->Size();
stat.st_mode = S_ATTR | 0666;
stat.st_type = attribute->Type();
} else {
stat.st_size = inode->Size();
stat.st_mode = inode->Type();
stat.st_type = 0;
}
stat.st_nlink = 1;
stat.st_blksize = 2048;
stat.st_uid = inode->UserID();
stat.st_gid = inode->GroupID();
stat.st_atime = time(NULL);
stat.st_mtime = stat.st_ctime = inode->ModificationTime();
stat.st_crtime = inode->CreationTime();
}
// #pragma mark - Module API // #pragma mark - Module API
@ -1398,6 +1825,8 @@ cdda_create_attr(fs_volume _volume, fs_vnode _node, const char *name,
return status; return status;
} else if ((openMode & O_EXCL) == 0) { } else if ((openMode & O_EXCL) == 0) {
attribute->SetType(type); attribute->SetType(type);
if ((openMode & O_TRUNC) != 0)
attribute->Truncate();
} else } else
return B_FILE_EXISTS; return B_FILE_EXISTS;